2016-10-21 23:02:19 +00:00
|
|
|
//
|
|
|
|
// Info.swift
|
|
|
|
// mas-cli
|
|
|
|
//
|
|
|
|
// Created by Denis Lebedev on 21/10/2016.
|
|
|
|
// Copyright © 2016 Andrew Naylor. All rights reserved.
|
|
|
|
//
|
|
|
|
|
2018-07-04 20:56:10 +00:00
|
|
|
import Commandant
|
|
|
|
import Result
|
2016-10-21 23:02:19 +00:00
|
|
|
import Foundation
|
|
|
|
|
2018-12-30 06:57:06 +00:00
|
|
|
/// Displays app details. Uses the iTunes Lookup API:
|
|
|
|
/// https://affiliate.itunes.apple.com/resources/documentation/itunes-store-web-service-search-api/#lookup
|
2018-10-14 21:35:48 +00:00
|
|
|
public struct InfoCommand: CommandProtocol {
|
|
|
|
public let verb = "info"
|
|
|
|
public let function = "Display app information from the Mac App Store"
|
2016-10-21 23:02:19 +00:00
|
|
|
|
2018-11-14 04:47:06 +00:00
|
|
|
private let urlSession: URLSession
|
|
|
|
|
|
|
|
public init(urlSession: URLSession = URLSession.shared) {
|
|
|
|
self.urlSession = urlSession
|
|
|
|
}
|
2018-10-14 21:35:48 +00:00
|
|
|
|
|
|
|
public func run(_ options: InfoOptions) -> Result<(), MASError> {
|
2016-10-21 23:02:19 +00:00
|
|
|
guard let infoURLString = infoURLString(options.appId),
|
2018-11-14 04:47:06 +00:00
|
|
|
let searchJson = urlSession.requestSynchronousJSONWithURLString(infoURLString) as? [String: Any] else {
|
2016-10-21 23:02:19 +00:00
|
|
|
return .failure(.searchFailed)
|
|
|
|
}
|
|
|
|
|
|
|
|
guard let resultCount = searchJson[ResultKeys.ResultCount] as? Int, resultCount > 0,
|
|
|
|
let results = searchJson[ResultKeys.Results] as? [[String: Any]],
|
|
|
|
let result = results.first else {
|
|
|
|
print("No results found")
|
|
|
|
return .failure(.noSearchResultsFound)
|
|
|
|
}
|
|
|
|
|
|
|
|
print(AppInfoFormatter.format(appInfo: result))
|
|
|
|
|
|
|
|
return .success(())
|
|
|
|
}
|
|
|
|
|
|
|
|
private func infoURLString(_ appId: String) -> String? {
|
|
|
|
if let urlEncodedAppId = appId.URLEncodedString {
|
|
|
|
return "https://itunes.apple.com/lookup?id=\(urlEncodedAppId)"
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-14 21:35:48 +00:00
|
|
|
public struct InfoOptions: OptionsProtocol {
|
2016-10-21 23:02:19 +00:00
|
|
|
let appId: String
|
|
|
|
|
|
|
|
static func create(_ appId: String) -> InfoOptions {
|
|
|
|
return InfoOptions(appId: appId)
|
|
|
|
}
|
|
|
|
|
2018-10-14 21:35:48 +00:00
|
|
|
public static func evaluate(_ m: CommandMode) -> Result<InfoOptions, CommandantError<MASError>> {
|
2016-10-21 23:02:19 +00:00
|
|
|
return create
|
|
|
|
<*> m <| Argument(usage: "the app id to show info")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private struct AppInfoFormatter {
|
|
|
|
private enum Keys {
|
|
|
|
static let Name = "trackCensoredName"
|
|
|
|
static let Version = "version"
|
|
|
|
static let Price = "formattedPrice"
|
|
|
|
static let Seller = "sellerName"
|
|
|
|
static let VersionReleaseDate = "currentVersionReleaseDate"
|
|
|
|
static let MinimumOS = "minimumOsVersion"
|
|
|
|
static let FileSize = "fileSizeBytes"
|
|
|
|
static let AppStoreUrl = "trackViewUrl"
|
|
|
|
}
|
|
|
|
|
|
|
|
static func format(appInfo: [String: Any]) -> String {
|
|
|
|
let headline = [
|
|
|
|
"\(appInfo.stringOrEmpty(key: Keys.Name))",
|
|
|
|
"\(appInfo.stringOrEmpty(key: Keys.Version))",
|
|
|
|
"[\(appInfo.stringOrEmpty(key: Keys.Price))]",
|
|
|
|
].joined(separator: " ")
|
|
|
|
|
|
|
|
return [
|
|
|
|
headline,
|
|
|
|
"By: \(appInfo.stringOrEmpty(key: Keys.Seller))",
|
|
|
|
"Released: \(humaReadableDate(appInfo.stringOrEmpty(key: Keys.VersionReleaseDate)))",
|
|
|
|
"Minimum OS: \(appInfo.stringOrEmpty(key: Keys.MinimumOS))",
|
|
|
|
"Size: \(humanReadableSize(appInfo.stringOrEmpty(key: Keys.FileSize)))",
|
|
|
|
"From: \(appInfo.stringOrEmpty(key: Keys.AppStoreUrl))",
|
|
|
|
].joined(separator: "\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
private static func humanReadableSize(_ size: String) -> String {
|
|
|
|
let bytesSize = Int64(size) ?? 0
|
|
|
|
return ByteCountFormatter.string(fromByteCount: bytesSize, countStyle: .file)
|
|
|
|
}
|
|
|
|
|
|
|
|
private static func humaReadableDate(_ serverDate: String) -> String {
|
|
|
|
let serverDateFormatter = DateFormatter()
|
|
|
|
serverDateFormatter.locale = Locale(identifier: "en_US_POSIX")
|
|
|
|
serverDateFormatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ssZZZZZ"
|
|
|
|
|
|
|
|
let humanDateFormatter = DateFormatter()
|
|
|
|
humanDateFormatter.timeStyle = .none
|
|
|
|
humanDateFormatter.dateStyle = .medium
|
|
|
|
return serverDateFormatter.date(from: serverDate).flatMap(humanDateFormatter.string(from:)) ?? ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension Dictionary {
|
|
|
|
fileprivate func stringOrEmpty(key: Key) -> String {
|
|
|
|
return self[key] as? String ?? ""
|
|
|
|
}
|
|
|
|
}
|