F-Droid desktop client
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

263 lines
7.0 KiB

// Copyright (c) 2015, Daniel Martí <mvdan@mvdan.cc>
// See LICENSE for licensing information
package main
import (
"encoding/csv"
"fmt"
"io"
"os"
"strconv"
"strings"
"mvdan.cc/fdroidcl/adb"
"mvdan.cc/fdroidcl/fdroid"
)
var cmdInstall = &Command{
UsageLine: "install [<appid...>]",
Short: "Install or upgrade apps",
Long: `
Install or upgrade apps. When given no arguments, it reads a comma-separated
list of apps to install from standard input, like:
packageName,versionCode,versionName
foo.bar,120,1.2.0
`[1:],
}
var (
installUpdates = cmdInstall.Fset.Bool("u", false, "Upgrade all installed apps")
installDryRun = cmdInstall.Fset.Bool("n", false, "Only print the operations that would be done")
installUpdatesExclude = cmdInstall.Fset.String("e", "", "Exclude apps from upgrading (comma-separated list)")
installSkipError = cmdInstall.Fset.Bool("s", false, "Skip to the next application if a download or install error occurs")
installUser = cmdInstall.Fset.String("user", "", `Install/upgrade for specified user <USER_ID|current|all>
default: installs app for the current user; upgrades apps of all users and installs the new version only for the users of the old version
USER_ID: installs app for USER_ID; upgrades only apps of USER_ID and installs the new version only for USER_ID
current: installs app for the current user; upgrades only apps of the current user and installs the new version only for the current user
all: installs app for all users; upgrades apps of all users and installs the new version for all users`)
)
func init() {
cmdInstall.Run = runInstall
}
func runInstall(args []string) error {
if *installUpdates && len(args) > 0 {
return fmt.Errorf("-u can only be used without arguments")
}
if *installUpdatesExclude != "" && !*installUpdates {
return fmt.Errorf("-e can only be used for upgrading (i.e. -u)")
}
device, err := oneDevice()
if err != nil {
return err
}
inst, err := device.Installed()
if err != nil {
return err
}
if *installUser != "" && *installUser != "all" && *installUser != "current" {
n, err := strconv.Atoi(*installUser)
if err != nil {
return fmt.Errorf("-user has to be <USER_ID|current|all>")
}
if n < 0 {
return fmt.Errorf("-user cannot have a negative number as USER_ID")
}
allUids := adb.AllUserIds(inst)
if _, exists := allUids[n]; !exists {
return fmt.Errorf("user %d does not exist", n)
}
}
if *installUser == "current" || (*installUser == "" && !*installUpdates) {
uid, err := device.CurrentUserId()
if err != nil {
return err
}
*installUser = strconv.Itoa(uid)
}
if *installUpdates {
apps, err := loadIndexes()
if err != nil {
return err
}
var filterUser *int
if *installUser == "all" || *installUser == "" {
filterUser = nil
} else {
n, err := strconv.Atoi(*installUser)
if err != nil {
return err
}
filterUser = &n
}
apps = filterAppsUpdates(apps, inst, device, filterUser)
if *installUpdatesExclude != "" {
excludeApps := strings.Split(*installUpdatesExclude, ",")
installApps := make([]fdroid.App, 0)
for _, app := range apps {
shouldExclude := false
for _, exclude := range excludeApps {
if app.PackageName == exclude {
shouldExclude = true
break
}
}
if shouldExclude {
continue
}
installApps = append(installApps, app)
}
apps = installApps
}
if len(apps) == 0 {
fmt.Fprintln(os.Stderr, "All apps up to date.")
}
return downloadAndDo(apps, inst, device)
}
if len(args) == 0 {
// The CSV input is as follows:
//
// packageName,versionCode,versionName
// foo.bar,120,1.2.0
// ...
r := csv.NewReader(os.Stdin)
r.FieldsPerRecord = 3
r.Read()
for {
record, err := r.Read()
if err == io.EOF {
break
}
if err != nil {
return fmt.Errorf("error parsing CSV: %v", err)
}
// convert "foo.bar,120" into "foo.bar:120" for findApps
args = append(args, record[0]+":"+record[1])
}
}
apps, err := findApps(args)
if err != nil {
return err
}
var toInstall []fdroid.App
for _, app := range apps {
p, e := inst[app.PackageName]
if !e {
// installing an app from scratch
toInstall = append(toInstall, app)
continue
}
suggested := app.SuggestedApk(device)
if suggested == nil {
return fmt.Errorf("no suitable APKs found for %s", app.PackageName)
}
if p.VersCode >= suggested.VersCode {
if !(*installUser == "all" && len(p.NotInstalledForUsers) > 0) { // ensure that it can't install for other user
okSkip := *installUser == "all"
if !okSkip {
n, err := strconv.Atoi(*installUser)
if err != nil {
return err
}
isInstalledForUser := false
for _, uid := range p.InstalledForUsers {
if uid == n {
isInstalledForUser = true
break
}
}
if isInstalledForUser {
okSkip = true
}
}
if okSkip {
fmt.Printf("%s is up to date\n", app.PackageName)
// app is already up to date
continue
}
}
}
// upgrading an existing app
toInstall = append(toInstall, app)
}
return downloadAndDo(toInstall, inst, device)
}
func downloadAndDo(apps []fdroid.App, installed map[string]adb.Package, device *adb.Device) error {
type downloaded struct {
apk *fdroid.Apk
app fdroid.App
path string
}
toInstall := make([]downloaded, 0)
for _, app := range apps {
apk := app.SuggestedApk(device)
if apk == nil {
return fmt.Errorf("no suitable APKs found for %s", app.PackageName)
}
if *installDryRun {
fmt.Printf("install %s:%d\n", app.PackageName, apk.VersCode)
continue
}
path, err := downloadApk(apk)
if err != nil {
if *installSkipError {
fmt.Printf("Downloading %s failed, skipping...\n", app.PackageName)
continue
}
return err
}
toInstall = append(toInstall, downloaded{apk: apk, app: app, path: path})
}
if *installDryRun {
return nil
}
for _, t := range toInstall {
var installedPkg *adb.Package = nil
if p, e := installed[t.app.PackageName]; e {
installedPkg = &p
}
if err := installApk(device, t.apk, installedPkg, t.path); err != nil {
if *installSkipError {
fmt.Printf("Installing %s failed, skipping...\n", t.apk.AppID)
continue
}
return err
}
}
return nil
}
func installApk(device *adb.Device, apk *fdroid.Apk, devicePkg *adb.Package, path string) error {
fmt.Printf("Installing %s\n", apk.AppID)
userId := "all"
if *installUser != "all" {
if *installUpdates && *installUser == "" {
if devicePkg == nil {
return fmt.Errorf("failed to get device package although it should be installed (please report this error)")
}
if len((*devicePkg).InstalledForUsers) > 0 {
userId = strconv.Itoa((*devicePkg).InstalledForUsers[0])
}
} else {
userId = *installUser
}
}
if userId == "all" {
if err := device.Install(path); err != nil {
return fmt.Errorf("could not install %s: %v", apk.AppID, err)
}
} else {
if err := device.InstallUser(path, userId); err != nil {
return fmt.Errorf("could not install %s: %v", apk.AppID, err)
}
}
return nil
}