2 # -*- coding: utf-8 -*-
4 # install.py - part of the FDroid server tools
5 # Copyright (C) 2013, Ciaran Gultnieks, ciaran@ciarang.com
6 # Copyright (C) 2013-2014 Daniel Martà <mvdan@mvdan.cc>
8 # This program is free software: you can redistribute it and/or modify
9 # it under the terms of the GNU Affero General Public License as published by
10 # the Free Software Foundation, either version 3 of the License, or
11 # (at your option) any later version.
13 # This program is distributed in the hope that it will be useful,
14 # but WITHOUT ANY WARRANTY; without even the implied warranty of
15 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 # GNU Affero General Public License for more details.
18 # You should have received a copy of the GNU Affero General Public License
19 # along with this program. If not, see <http://www.gnu.org/licenses/>.
24 from optparse import OptionParser, OptionError
28 from common import FDroidPopen, FDroidException
35 p = FDroidPopen([config['adb'], "devices"])
37 raise FDroidException("An error occured when finding devices: %s" % p.output)
38 lines = p.output.splitlines()
39 if lines[0].startswith('* daemon not running'):
44 return [l.split()[0] for l in lines]
49 global options, config
51 # Parse command line...
52 parser = OptionParser(usage="Usage: %prog [options] [APPID[:VERCODE] [APPID[:VERCODE] ...]]")
53 parser.add_option("-v", "--verbose", action="store_true", default=False,
54 help="Spew out even more information than normal")
55 parser.add_option("-q", "--quiet", action="store_true", default=False,
56 help="Restrict output to warnings and errors")
57 parser.add_option("-a", "--all", action="store_true", default=False,
58 help="Install all signed applications available")
59 (options, args) = parser.parse_args()
61 if not args and not options.all:
62 raise OptionError("If you really want to install all the signed apps, use --all", "all")
64 config = common.read_config(options)
67 if not os.path.isdir(output_dir):
68 logging.info("No signed output directory - nothing to do")
73 vercodes = common.read_pkg_args(args, True)
74 apks = {appid: None for appid in vercodes}
76 # Get the signed apk with the highest vercode
77 for apkfile in sorted(glob.glob(os.path.join(output_dir, '*.apk'))):
79 appid, vercode = common.apknameinfo(apkfile)
82 if vercodes[appid] and vercode not in vercodes[appid]:
86 for appid, apk in apks.iteritems():
88 raise FDroidException("No signed apk available for %s" % appid)
92 apks = {common.apknameinfo(apkfile)[0]: apkfile for apkfile in
93 sorted(glob.glob(os.path.join(output_dir, '*.apk')))}
95 for appid, apk in apks.iteritems():
96 # Get device list each time to avoid device not found errors
99 raise FDroidException("No attached devices found")
100 logging.info("Installing %s..." % apk)
102 logging.info("Installing %s on %s..." % (apk, dev))
103 p = FDroidPopen([config['adb'], "-s", dev, "install", apk])
105 for line in p.output.splitlines():
106 if line.startswith("Failure"):
111 if fail == "INSTALL_FAILED_ALREADY_EXISTS":
112 logging.warn("%s is already installed on %s." % (apk, dev))
114 raise FDroidException("Failed to install %s on %s: %s" % (
117 logging.info("\nFinished")
119 if __name__ == "__main__":