chiark / gitweb /
update: use only 7 chars of SHA256 for non-APK version name
[fdroidserver.git] / fdroidserver / verify.py
1 #!/usr/bin/env python3
2 #
3 # verify.py - part of the FDroid server tools
4 # Copyright (C) 2013, Ciaran Gultnieks, ciaran@ciarang.com
5 #
6 # This program is free software: you can redistribute it and/or modify
7 # it under the terms of the GNU Affero General Public License as published by
8 # the Free Software Foundation, either version 3 of the License, or
9 # (at your option) any later version.
10 #
11 # This program is distributed in the hope that it will be useful,
12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 # GNU Affero General Public License for more details.
15 #
16 # You should have received a copy of the GNU Affero General Public License
17 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
18
19 import sys
20 import os
21 import glob
22 import requests
23 from argparse import ArgumentParser
24 import logging
25
26 from . import _
27 from . import common
28 from . import net
29 from .exception import FDroidException
30
31 options = None
32 config = None
33
34
35 def main():
36
37     global options, config
38
39     # Parse command line...
40     parser = ArgumentParser(usage="%(prog)s [options] [APPID[:VERCODE] [APPID[:VERCODE] ...]]")
41     common.setup_global_opts(parser)
42     parser.add_argument("appid", nargs='*', help=_("applicationId with optional versionCode in the form APPID[:VERCODE]"))
43     options = parser.parse_args()
44
45     config = common.read_config(options)
46
47     tmp_dir = 'tmp'
48     if not os.path.isdir(tmp_dir):
49         logging.info(_("Creating temporary directory"))
50         os.makedirs(tmp_dir)
51
52     unsigned_dir = 'unsigned'
53     if not os.path.isdir(unsigned_dir):
54         logging.error(_("No unsigned directory - nothing to do"))
55         sys.exit(0)
56
57     verified = 0
58     notverified = 0
59
60     vercodes = common.read_pkg_args(options.appid, True)
61
62     for apkfile in sorted(glob.glob(os.path.join(unsigned_dir, '*.apk'))):
63
64         apkfilename = os.path.basename(apkfile)
65         appid, vercode = common.publishednameinfo(apkfile)
66
67         if vercodes and appid not in vercodes:
68             continue
69         if vercodes[appid] and vercode not in vercodes[appid]:
70             continue
71
72         try:
73
74             logging.info("Processing {apkfilename}".format(apkfilename=apkfilename))
75
76             remoteapk = os.path.join(tmp_dir, apkfilename)
77             if os.path.exists(remoteapk):
78                 os.remove(remoteapk)
79             url = 'https://f-droid.org/repo/' + apkfilename
80             logging.info("...retrieving " + url)
81             try:
82                 net.download_file(url, dldir=tmp_dir)
83             except requests.exceptions.HTTPError as e:
84                 try:
85                     net.download_file(url.replace('/repo', '/archive'), dldir=tmp_dir)
86                 except requests.exceptions.HTTPError as e:
87                     raise FDroidException(_('Downloading {url} failed. {error}')
88                                           .format(url=url, error=e))
89
90             compare_result = common.verify_apks(
91                 remoteapk,
92                 os.path.join(unsigned_dir, apkfilename),
93                 tmp_dir)
94             if compare_result:
95                 raise FDroidException(compare_result)
96
97             logging.info("...successfully verified")
98             verified += 1
99
100         except FDroidException as e:
101             logging.info("...NOT verified - {0}".format(e))
102             notverified += 1
103
104     logging.info(_("Finished"))
105     logging.info("{0} successfully verified".format(verified))
106     logging.info("{0} NOT verified".format(notverified))
107
108
109 if __name__ == "__main__":
110     main()