mas/Sources/MasKit/Commands/Upgrade.swift

108 lines
3.6 KiB
Swift
Raw Normal View History

//
// Upgrade.swift
// mas-cli
//
// Created by Andrew Naylor on 30/12/2015.
// Copyright © 2015 Andrew Naylor. All rights reserved.
//
2018-07-04 20:56:10 +00:00
import Commandant
2021-04-26 23:44:17 +00:00
import Foundation
import PromiseKit
2021-05-09 20:25:20 +00:00
import enum Swift.Result
2018-07-04 20:56:10 +00:00
/// Command which upgrades apps with new versions available in the Mac App Store.
public struct UpgradeCommand: CommandProtocol {
public typealias Options = UpgradeOptions
public let verb = "upgrade"
public let function = "Upgrade outdated apps from the Mac App Store"
private let appLibrary: AppLibrary
private let storeSearch: StoreSearch
2020-03-29 03:15:07 +00:00
/// Public initializer.
public init() {
self.init(appLibrary: MasAppLibrary())
}
/// Internal initializer.
/// - Parameter appLibrary: AppLibrary manager.
/// - Parameter storeSearch: StoreSearch manager.
init(appLibrary: AppLibrary = MasAppLibrary(), storeSearch: StoreSearch = MasStoreSearch()) {
self.appLibrary = appLibrary
self.storeSearch = storeSearch
}
2019-01-12 01:28:03 +00:00
/// Runs the command.
2021-03-22 05:46:17 +00:00
public func run(_ options: Options) -> Result<Void, MASError> {
let apps: [(installedApp: SoftwareProduct, storeApp: SearchResult)]
do {
2021-04-21 23:15:26 +00:00
apps = try findOutdatedApps(options)
} catch {
// Bubble up MASErrors
2021-03-22 22:13:48 +00:00
return .failure(error as? MASError ?? .searchFailed)
}
guard apps.count > 0 else {
printWarning("Nothing found to upgrade")
return .success(())
}
print("Upgrading \(apps.count) outdated application\(apps.count > 1 ? "s" : ""):")
print(apps.map { "\($0.installedApp.appName) (\($0.installedApp.bundleVersion)) -> (\($0.storeApp.version))" }.joined(separator: "\n"))
let appIds = apps.map(\.installedApp.itemIdentifier.uint64Value)
do {
try downloadAll(appIds).wait()
} catch {
return .failure(error as? MASError ?? .downloadFailed(error: error as NSError))
}
return .success(())
}
2021-04-21 23:15:26 +00:00
private func findOutdatedApps(_ options: Options) throws -> [(SoftwareProduct, SearchResult)] {
let apps: [SoftwareProduct] = options.apps.isEmpty
? appLibrary.installedApps
:
options.apps.compactMap {
if let appId = UInt64($0) {
// if argument a UInt64, lookup app by id using argument
return appLibrary.installedApp(forId: appId)
} else {
// if argument not a UInt64, lookup app by name using argument
return appLibrary.installedApp(named: $0)
}
2021-04-21 23:15:26 +00:00
}
let promises = apps.map { installedApp in
2021-04-21 23:15:26 +00:00
// only upgrade apps whose local version differs from the store version
firstly {
storeSearch.lookup(app: installedApp.itemIdentifier.intValue)
}.map { result -> (SoftwareProduct, SearchResult)? in
guard let storeApp = result, installedApp.isOutdatedWhenComparedTo(storeApp) else {
return nil
}
2021-04-21 23:15:26 +00:00
return (installedApp, storeApp)
}
}
return try when(fulfilled: promises).wait().compactMap { $0 }
2021-04-21 23:15:26 +00:00
}
2015-12-30 21:20:25 +00:00
}
2016-09-14 21:55:04 +00:00
public struct UpgradeOptions: OptionsProtocol {
let apps: [String]
2019-01-12 01:06:02 +00:00
static func create(_ apps: [String]) -> UpgradeOptions {
2021-03-22 05:46:17 +00:00
UpgradeOptions(apps: apps)
2016-09-14 21:55:04 +00:00
}
2019-01-12 01:06:02 +00:00
public static func evaluate(_ mode: CommandMode) -> Result<UpgradeOptions, CommandantError<MASError>> {
2021-03-22 05:46:17 +00:00
create
2019-01-12 01:06:02 +00:00
<*> mode <| Argument(defaultValue: [], usage: "app(s) to upgrade")
2016-09-14 21:55:04 +00:00
}
2016-09-17 12:58:38 +00:00
}