3 # install.py - part of the FDroid server tools
4 # Copyright (C) 2013, Ciaran Gultnieks, ciaran@ciarang.com
5 # Copyright (C) 2013-2014 Daniel Martà <mvdan@mvdan.cc>
7 # This program is free software: you can redistribute it and/or modify
8 # it under the terms of the GNU Affero General Public License as published by
9 # the Free Software Foundation, either version 3 of the License, or
10 # (at your option) any later version.
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU Affero General Public License for more details.
17 # You should have received a copy of the GNU Affero General Public License
18 # along with this program. If not, see <http://www.gnu.org/licenses/>.
23 from argparse import ArgumentParser
28 from .common import SdkToolsPopen
29 from .exception import FDroidException
36 p = SdkToolsPopen(['adb', "devices"])
38 raise FDroidException("An error occured when finding devices: %s" % p.output)
39 lines = [l for l in p.output.splitlines() if not l.startswith('* ')]
43 return [l.split()[0] for l in lines]
48 global options, config
50 # Parse command line...
51 parser = ArgumentParser(usage="%(prog)s [options] [APPID[:VERCODE] [APPID[:VERCODE] ...]]")
52 common.setup_global_opts(parser)
53 parser.add_argument("appid", nargs='*', help=_("applicationId with optional versionCode in the form APPID[:VERCODE]"))
54 parser.add_argument("-a", "--all", action="store_true", default=False,
55 help=_("Install all signed applications available"))
56 options = parser.parse_args()
58 if not options.appid and not options.all:
59 parser.error(_("option %s: If you really want to install all the signed apps, use --all") % "all")
61 config = common.read_config(options)
64 if not os.path.isdir(output_dir):
65 logging.info(_("No signed output directory - nothing to do"))
70 vercodes = common.read_pkg_args(options.appid, True)
71 apks = {appid: None for appid in vercodes}
73 # Get the signed apk with the highest vercode
74 for apkfile in sorted(glob.glob(os.path.join(output_dir, '*.apk'))):
77 appid, vercode = common.publishednameinfo(apkfile)
78 except FDroidException:
82 if vercodes[appid] and vercode not in vercodes[appid]:
86 for appid, apk in apks.items():
88 raise FDroidException(_("No signed apk available for %s") % appid)
92 apks = {common.publishednameinfo(apkfile)[0]: apkfile for apkfile in
93 sorted(glob.glob(os.path.join(output_dir, '*.apk')))}
95 for appid, apk in apks.items():
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 '{apkfilename}' on {dev}...").format(apkfilename=apk, dev=dev))
103 p = SdkToolsPopen(['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(_("'{apkfilename}' is already installed on {dev}.")
113 .format(apkfilename=apk, dev=dev))
115 raise FDroidException(_("Failed to install '{apkfilename}' on {dev}: {error}")
116 .format(apkfilename=apk, dev=dev, error=fail))
118 logging.info('\n' + _('Finished'))
121 if __name__ == "__main__":