chiark / gitweb /
Now building multiple versions of apks from specified commits
authorCiaran Gultnieks <ciaran@ciarang.com>
Thu, 11 Nov 2010 22:34:39 +0000 (22:34 +0000)
committerCiaran Gultnieks <ciaran@ciarang.com>
Thu, 11 Nov 2010 22:34:39 +0000 (22:34 +0000)
build.py
metadata.py [new file with mode: 0644]
metadata/com.eleybourn.bookcatalogue.txt
metadata/org.fdroid.fdroid.txt
update.py

index 40a8fc7a529fa70042022a442d73e0a60973255b..d89ed1f6308511f3ee28356ff7da9750bad8e872 100644 (file)
--- a/build.py
+++ b/build.py
@@ -30,6 +30,8 @@ from optparse import OptionParser
 #Read configuration...
 execfile('config.py')
 
+execfile('metadata.py')
+
 # Parse command line...
 parser = OptionParser()
 parser.add_option("-v", "--verbose", action="store_true", default=False,
@@ -37,79 +39,7 @@ parser.add_option("-v", "--verbose", action="store_true", default=False,
 (options, args) = parser.parse_args()
 
 # Get all apps...
-apps = []
-
-for metafile in glob.glob(os.path.join('metadata','*.txt')):
-
-    thisinfo = {}
-
-    # Get metadata...
-    thisinfo['id'] = metafile[9:-4]
-    print "Reading metadata for " + thisinfo['id']
-    thisinfo['description'] = ''
-    thisinfo['summary'] = ''
-    thisinfo['license'] = 'Unknown'
-    thisinfo['web'] = ''
-    thisinfo['source'] = ''
-    thisinfo['tracker'] = ''
-    thisinfo['disabled'] = None
-    thisinfo['marketversion'] = ''
-    thisinfo['marketvercode'] = '0'
-    thisinfo['repotype'] = ''
-    thisinfo['repo'] = ''
-    f = open(metafile, 'r')
-    mode = 0
-    for line in f.readlines():
-        line = line.rstrip('\r\n')
-        if len(line) == 0:
-            pass
-        elif mode == 0:
-            index = line.find(':')
-            if index == -1:
-                print "Invalid metadata in " + metafile + " at:" + line
-                sys.exit(1)
-            field = line[:index]
-            value = line[index+1:]
-            if field == 'Description':
-                mode = 1
-            elif field == 'Summary':
-                thisinfo['summary'] = value
-            elif field == 'Source Code':
-                thisinfo['source'] = value
-            elif field == 'License':
-                thisinfo['license'] = value
-            elif field == 'Web Site':
-                thisinfo['web'] = value
-            elif field == 'Issue Tracker':
-                thisinfo['tracker'] = value
-            elif field == 'Disabled':
-                thisinfo['disabled'] = value
-            elif field == 'Market Version':
-                thisinfo['marketversion'] = value
-            elif field == 'Market Version Code':
-                thisinfo['marketvercode'] = value
-            elif field == 'Repo Type':
-                thisinfo['repotype'] = value
-            elif field == 'Repo':
-                thisinfo['repo'] = value
-            else:
-                print "Unrecognised field " + field
-                sys.exit(1)
-        elif mode == 1:
-            if line == '.':
-                mode = 0
-            else:
-                if len(line) == 0:
-                    thisinfo['description'] += '\n\n'
-                else:
-                    if (not thisinfo['description'].endswith('\n') and
-                        len(thisinfo['description']) > 0):
-                        thisinfo['description'] += ' '
-                    thisinfo['description'] += line
-    if len(thisinfo['description']) == 0:
-        thisinfo['description'] = 'No description available'
-
-    apps.append(thisinfo)
+apps = read_metadata()
 
 unsigned_dir = 'unsigned'
 if os.path.exists(unsigned_dir):
@@ -137,35 +67,70 @@ for app in apps:
             print "Invalid repo type " + app['repotype'] + " in " + app['id']
             sys.exit(1)
 
-        # Generate (or update) the ant build file, build.xml...
-        if subprocess.call(['android','update','project','-p','.'],
-                cwd=build_dir) != 0:
-            print "Failed to update project"
-            sys.exit(1)
+        for thisbuild in app['builds']:
+
+            print "Building version " + thisbuild['version']
 
-        # If the app has ant set up to sign the release, we need to switch
-        # that off, because we want the unsigned apk...
-        if os.path.exists(os.path.join(build_dir, 'build.properties')):
-            if subprocess.call(['sed','-i','s/^key.store/#/',
-                'build.properties'], cwd=build_dir) !=0:
-                print "Failed to amend build.properties"
+            if app['repotype'] == 'git':
+                if subprocess.call(['git','checkout',thisbuild['commit']],
+                        cwd=build_dir) != 0:
+                    print "Git checkout failed"
+                    sys.exit(1)
+            else:
+                print "Invalid repo type " + app['repotype']
                 sys.exit(1)
 
-        # Build the release...
-        p = subprocess.Popen(['ant','release'], cwd=build_dir, 
-                stdout=subprocess.PIPE)
-        output = p.communicate()[0]
-        print output
-        if p.returncode != 0:
-            print "Build failed"
-            sys.exit(1)
+            # Generate (or update) the ant build file, build.xml...
+            if subprocess.call(['android','update','project','-p','.'],
+                    cwd=build_dir) != 0:
+                print "Failed to update project"
+                sys.exit(1)
+
+            # If the app has ant set up to sign the release, we need to switch
+            # that off, because we want the unsigned apk...
+            if os.path.exists(os.path.join(build_dir, 'build.properties')):
+                if subprocess.call(['sed','-i','s/^key.store/#/',
+                    'build.properties'], cwd=build_dir) !=0:
+                    print "Failed to amend build.properties"
+                    sys.exit(1)
+
+            # Build the release...
+            p = subprocess.Popen(['ant','release'], cwd=build_dir, 
+                    stdout=subprocess.PIPE)
+            output = p.communicate()[0]
+            print output
+            if p.returncode != 0:
+                print "Build failed"
+                sys.exit(1)
+
+            # Find the apk name in the output...
+            src = re.match(r".*^.*Creating (\S+) for release.*$.*", output,
+                    re.S|re.M).group(1)
+            src = os.path.join(os.path.join(build_dir, 'bin'), src)
+
+            # By way of a sanity check, make sure the version and version
+            # code in our new apk match what we expect...
+            p = subprocess.Popen([aapt_path,'dump','badging',
+               src], stdout=subprocess.PIPE)
+            output = p.communicate()[0]
+            vercode = None
+            version = None
+            for line in output.splitlines():
+                if line.startswith("package:"):
+                    pat = re.compile(".*versionCode='([0-9]*)'.*")
+                    vercode = re.match(pat, line).group(1)
+                    pat = re.compile(".*versionName='([^']*)'.*")
+                    version = re.match(pat, line).group(1)
+            if (version != thisbuild['version'] or
+                    vercode != thisbuild['vercode']):
+                print "Unexpected version/version code in output"
+                sys.exit(1)
 
-        # Find the apk name in the output...
-        src = re.match(r".*^.*Creating (\S+) for release.*$.*", output,
-                re.S|re.M).group(1)
-        dest = os.path.join(unsigned_dir, app['id'] + '.apk')
-        shutil.copyfile(os.path.join( os.path.join(build_dir, 'bin'),
-            src), dest)
+            # Copy the unsigned apk to our 'unsigned' directory to be
+            # dealt with later...
+            dest = os.path.join(unsigned_dir, app['id'] + '_' +
+                    thisbuild['vercode'] + '.apk')
+            shutil.copyfile(src, dest)
 
 print "Finished."
 
diff --git a/metadata.py b/metadata.py
new file mode 100644 (file)
index 0000000..e7ebf41
--- /dev/null
@@ -0,0 +1,106 @@
+# -*- coding: UTF-8 -*-
+#
+# metadata.py - part of the FDroid server tools
+# Copyright (C) 2010, Ciaran Gultnieks, ciaran@ciarang.com
+#
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU Affero General Public License for more details.
+#
+# You should have received a copy of the GNU Affero General Public License
+# along with this program.  If not, see <http://www.gnu.org/licenses/>.
+
+def read_metadata():
+
+    apps = []
+
+    for metafile in glob.glob(os.path.join('metadata','*.txt')):
+
+        thisinfo = {}
+
+        # Get metadata...
+        thisinfo['id'] = metafile[9:-4]
+        print "Reading metadata for " + thisinfo['id']
+        thisinfo['description'] = ''
+        thisinfo['summary'] = ''
+        thisinfo['license'] = 'Unknown'
+        thisinfo['web'] = ''
+        thisinfo['source'] = ''
+        thisinfo['tracker'] = ''
+        thisinfo['disabled'] = None
+        thisinfo['marketversion'] = ''
+        thisinfo['marketvercode'] = '0'
+        thisinfo['repotype'] = ''
+        thisinfo['repo'] = ''
+        thisinfo['builds'] = []
+        f = open(metafile, 'r')
+        mode = 0
+        for line in f.readlines():
+            line = line.rstrip('\r\n')
+            if len(line) == 0:
+                pass
+            elif mode == 0:
+                index = line.find(':')
+                if index == -1:
+                    print "Invalid metadata in " + metafile + " at:" + line
+                    sys.exit(1)
+                field = line[:index]
+                value = line[index+1:]
+                if field == 'Description':
+                    mode = 1
+                elif field == 'Summary':
+                    thisinfo['summary'] = value
+                elif field == 'Source Code':
+                    thisinfo['source'] = value
+                elif field == 'License':
+                    thisinfo['license'] = value
+                elif field == 'Web Site':
+                    thisinfo['web'] = value
+                elif field == 'Issue Tracker':
+                    thisinfo['tracker'] = value
+                elif field == 'Disabled':
+                    thisinfo['disabled'] = value
+                elif field == 'Market Version':
+                    thisinfo['marketversion'] = value
+                elif field == 'Market Version Code':
+                    thisinfo['marketvercode'] = value
+                elif field == 'Repo Type':
+                    thisinfo['repotype'] = value
+                elif field == 'Repo':
+                    thisinfo['repo'] = value
+                elif field == 'Build Version':
+                    parts = value.split(",")
+                    if len(parts) != 3:
+                        print "Invalid build format: " + value
+                        sys.exit(1)
+                    thisbuild = {}
+                    thisbuild['version'] = parts[0]
+                    thisbuild['vercode'] = parts[1]
+                    thisbuild['commit'] = parts[2]
+                    thisinfo['builds'].append(thisbuild)
+                else:
+                    print "Unrecognised field " + field
+                    sys.exit(1)
+            elif mode == 1:
+                if line == '.':
+                    mode = 0
+                else:
+                    if len(line) == 0:
+                        thisinfo['description'] += '\n\n'
+                    else:
+                        if (not thisinfo['description'].endswith('\n') and
+                            len(thisinfo['description']) > 0):
+                            thisinfo['description'] += ' '
+                        thisinfo['description'] += line
+        if len(thisinfo['description']) == 0:
+            thisinfo['description'] = 'No description available'
+
+        apps.append(thisinfo)
+
+    return apps
index da1595a74726258ad874a281af31370d4be705f9..af503a081282361543c848188f44fbbc3d9a6f8f 100644 (file)
@@ -11,3 +11,5 @@ Market Version:3.0.1
 Market Version Code:37
 Repo Type:git
 Repo:git://github.com/eleybourn/Book-Catalogue.git
+Build Version:3.0.1,36,b876c6df82c7e195ec5d
+Build Version:3.0,35,448858bad8b397974db7
index c37980419ab31c9ef89521efdf00413a3ce9dc35..90727154e65600fd6bfd08084c2c4b76bf930609 100644 (file)
@@ -9,3 +9,6 @@ it is?
 .
 Repo Type:git
 Repo:git://gitorious.org/f-droid/fdroidclient.git
+Build Version:0.12,3,651696a49be2cd7db5ce6a2fa8185e31f9a20035
+Build Version:0.13,4,4f677285cc2cf0b7e7feb5c9acf61791bec15fbc
+
index af43b9ccec3cc1d7b51b97df7a983daf4fe3b345..851c102c0633bcacd2a92b5a28df8c9d9dc99db3 100644 (file)
--- a/update.py
+++ b/update.py
@@ -30,6 +30,8 @@ from optparse import OptionParser
 #Read configuration...
 execfile('config.py')
 
+execfile('metadata.py')
+
 # Parse command line...
 parser = OptionParser()
 parser.add_option("-c", "--createmeta", action="store_true", default=False,
@@ -126,70 +128,7 @@ for apkfile in glob.glob(os.path.join('repo','*.apk')):
     apks.append(thisinfo)
 
 # Get all apps...
-apps = []
-
-for metafile in glob.glob(os.path.join('metadata','*.txt')):
-
-    thisinfo = {}
-
-    # Get metadata...
-    thisinfo['id'] = metafile[9:-4]
-    print "Reading metadata for " + thisinfo['id']
-    thisinfo['description'] = ''
-    thisinfo['summary'] = ''
-    thisinfo['license'] = 'Unknown'
-    thisinfo['web'] = ''
-    thisinfo['source'] = ''
-    thisinfo['tracker'] = ''
-    thisinfo['disabled'] = None
-    thisinfo['marketversion'] = ''
-    thisinfo['marketvercode'] = '0'
-    f = open(metafile, 'r')
-    mode = 0
-    for line in f.readlines():
-        line = line.rstrip('\r\n')
-       if len(line) == 0:
-            pass
-        elif mode == 0:
-            index = line.find(':')
-            if index == -1:
-                print "Invalid metadata in " + metafile + " at:" + line
-                sys.exit(1)
-            field = line[:index]
-            value = line[index+1:]
-            if field == 'Description':
-                mode = 1
-            elif field == 'Summary':
-                thisinfo['summary'] = value
-            elif field == 'Source Code':
-                thisinfo['source'] = value
-            elif field == 'License':
-                thisinfo['license'] = value
-            elif field == 'Web Site':
-                thisinfo['web'] = value
-            elif field == 'Issue Tracker':
-                thisinfo['tracker'] = value
-            elif field == 'Disabled':
-                thisinfo['disabled'] = value
-            elif field == 'Market Version':
-                thisinfo['marketversion'] = value
-            elif field == 'Market Version Code':
-                thisinfo['marketvercode'] = value
-        elif mode == 1:
-            if line == '.':
-                mode = 0
-            else:
-                if len(line) == 0:
-                    thisinfo['description'] += '\n\n'
-                else:
-                    if (not thisinfo['description'].endswith('\n') and
-                        len(thisinfo['description']) > 0):
-                        thisinfo['description'] += ' '
-                    thisinfo['description'] += line
-    if len(thisinfo['description']) == 0:
-        thisinfo['description'] = 'No description available'
-
-    apps.append(thisinfo)
+apps = read_metadata()
 
 # Some information from the apks needs to be applied up to the application
 # level. When doing this, we use the info from the most recent version's apk.