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.

559 lines
22 KiB

// Copyright © 2015 Abhishek Banthia
import Cocoa
import CoreLoggerKit
class VersionUpdateHandler: NSObject {
enum VersionUpdateHandlerPriority: Comparable {
case defaultPri
case low
case medium
case high
}
static let kSecondsInDay: Double = 86400.0
static let kMacAppStoreRefreshDelay: Double = 5.0
static let kMacRequestTimeout: Double = 60.0
static let kVersionCheckLastVersionKey = "VersionCheckLastVersionKey"
static let kVersionIgnoreVersionKey = "VersionCheckIgnoreVersionKey"
5 years ago
static let kMacAppStoreIDKey = "VersionCheckAppStoreIDKey"
static let kVersionLastCheckedKey = "VersionLastCheckedKey"
static let kVersionLastRemindedKey = "VersionLastRemindedKey"
static let kVersionMacAppStoreBundleID = "com.apple.AppStore";
static let kVersionMacAppStoreAppID = 1056643111
5 years ago
private var appStoreCountry: String!
private var applicationVersion: String!
private var applicationBundleID: String = Bundle.main.bundleIdentifier ?? "N/A"
4 years ago
private var updatePriority = VersionUpdateHandlerPriority.defaultPri
public var useAllAvailableLanguages: Bool = true
private var onlyPromptIfMainWindowIsAvailable: Bool = true
private var checkAtLaunch: Bool = true
private var checkPeriod: Double = 0.0
private var remindPeriod: Double = 1.0
private var verboseLogging: Bool = true
5 years ago
private var checkingForNewVersion: Bool = false
private var remoteVersionsDict: [String: Any] = [:]
private var downloadError: Error?
private var dataTask: URLSessionDataTask? = .none
private var visibleLocalAlert: NSAlert?
private var visibleRemoteAlert: NSAlert?
private var remoteRepeater: Repeater?
private var localRepeater: Repeater?
private var showOnFirstLaunch: Bool = false
public var previewMode: Bool = false
private var versionDetails: String?
10 months ago
private let store: DataStore
10 months ago
init(with dataStore: DataStore) {
// Setup App Store Country
10 months ago
store = dataStore
appStoreCountry = Locale.current.regionCode
if appStoreCountry == "150" {
appStoreCountry = "eu"
4 years ago
} else if appStoreCountry.replacingOccurrences(of: "[A-Za-z]{2}", with: "", options: .regularExpression, range: appStoreCountry.startIndex ..< appStoreCountry.endIndex).isEmpty == false {
appStoreCountry = "us"
}
// Setup App Version
var appVersion = Bundle.main.object(forInfoDictionaryKey: "CFBundleShortVersionString") as? String
if appVersion == nil {
appVersion = Bundle.main.object(forInfoDictionaryKey: kCFBundleVersionKey as String) as? String
}
applicationVersion = appVersion ?? "N/A"
// Bundle Identifier
self.applicationBundleID = Bundle.main.bundleIdentifier ?? "com.abhishek.Clocker"
//default settings
self.updatePriority = .defaultPri;
self.useAllAvailableLanguages = true;
self.onlyPromptIfMainWindowIsAvailable = true;
self.checkAtLaunch = true;
self.checkPeriod = 0.0;
self.remindPeriod = 1.0;
self.verboseLogging = true;
super.init()
applicationLaunched()
}
5 years ago
private func inThisVersionTitle() -> String {
return "New in this version"
}
5 years ago
private func updateAvailableTitle() -> String {
return "New version available"
}
private func versionLabelFormat() -> String {
return "Version %@"
}
private func okayButtonLabel() -> String {
return "OK"
}
private func ignoreButtonLabel() -> String {
return "Ignore"
}
private func downloadButtonLabel() -> String {
return "Download"
}
private func remindButtonLabel() -> String {
return "Remind Me Later"
}
private func updatedURL() -> URL {
// Last resort
return URL(string: "macappstore://itunes.apple.com/us/app/clocker/id1056643111")!
5 years ago
}
@objc private func setLastChecked(_ date: Date) {
5 years ago
UserDefaults.standard.set(date, forKey: VersionUpdateHandler.kVersionLastCheckedKey)
}
private func lastChecked() -> Date? {
10 months ago
return store.retrieve(key: VersionUpdateHandler.kVersionLastCheckedKey) as? Date
5 years ago
}
private func setLastReminded(_ date: Date?) {
UserDefaults.standard.set(date, forKey: VersionUpdateHandler.kVersionLastRemindedKey)
}
private func lastReminded() -> Date? {
10 months ago
return store.retrieve(key: VersionUpdateHandler.kVersionLastRemindedKey) as? Date
5 years ago
}
private func ignoredVersion() -> String? {
10 months ago
return store.retrieve(key: VersionUpdateHandler.kVersionIgnoreVersionKey) as? String
5 years ago
}
private func setIgnoredVersion(_ version: String) {
UserDefaults.standard.set(version, forKey: VersionUpdateHandler.kVersionIgnoreVersionKey)
}
private func setViewedVersionDetails(_ viewed: Bool) {
UserDefaults.standard.set(viewed ? applicationVersion : nil, forKey: VersionUpdateHandler.kVersionCheckLastVersionKey)
}
private func viewedVersionDetails() -> Bool {
10 months ago
let lastVersionKey = store.retrieve(key: VersionUpdateHandler.kVersionCheckLastVersionKey) as? String ?? ""
5 years ago
return lastVersionKey == applicationVersion
}
private func lastVersion() -> String {
10 months ago
return store.retrieve(key: VersionUpdateHandler.kVersionCheckLastVersionKey) as? String ?? ""
}
5 years ago
private func setLastVersion(_ version: String) {
UserDefaults.standard.set(version, forKey: VersionUpdateHandler.kVersionCheckLastVersionKey)
}
private func localVersionsDict() -> [String: Any] {
return [String: Any]()
}
private func versionDetails(_ version: String, _ dict: [String: Any]) -> String? {
if let versionData = dict[version] as? String {
return versionData
} else if let versionDataArray = dict[version] as? NSArray {
return versionDataArray.componentsJoined(by: "\n")
}
return nil
}
5 years ago
private func versionDetails(since lastVersion: String, in dict: [String: Any]) -> String? {
var lastVersionCopy = lastVersion
if previewMode {
lastVersionCopy = "0"
}
var newVersionFound = false
3 years ago
var details = ""
5 years ago
let versions = dict.keys.sorted()
for version in versions {
if version.compareVersion(lastVersionCopy) == .orderedDescending {
newVersionFound = true
}
details.append(versionDetails(version, dict) ?? "")
details.append("\n")
}
if newVersionFound {
return details.trimmingCharacters(in: CharacterSet.newlines)
}
return nil
}
5 years ago
private func shouldCheckForNewVersion() -> Bool {
if (!self.previewMode) {
if let lastRemindedDate = lastReminded() {
// Reminder takes priority over check period
if Date().timeIntervalSince(lastRemindedDate) < Double(remindPeriod * Self.kSecondsInDay) {
if verboseLogging {
Logger.info("iVersion did not check for a new version because the user last asked to be reminded less than \(self.remindPeriod) days ago")
}
return false
}
} else if let lastCheckedDate = lastChecked(), Date().timeIntervalSince(lastCheckedDate) < Double(self.checkPeriod * Self.kSecondsInDay) {
if (self.verboseLogging) {
Logger.info("iVersion did not check for a new version because the last check was less than \(self.checkPeriod) days ago")
}
return false
}
} else if (self.verboseLogging) {
Logger.info("iVersion debug mode is enabled - make sure you disable this for release")
}
// perform the check
5 years ago
return true
}
private func checkForNewVersionInBackground() {
var newerVersionAvailable = false
var osVersionSupported = false
var latestVersion: String? = nil
var versions: [String:String]? = nil
var itunesServiceURL = "http://itunes.apple.com/\(self.appStoreCountry ?? "us")/lookup"
itunesServiceURL = itunesServiceURL.appendingFormat("?bundleId=%@", self.applicationBundleID)
if (verboseLogging) {
Logger.info("iVersion is checking \(itunesServiceURL) for a new app version...")
}
dataTask = NetworkManager.task(with: itunesServiceURL) { [weak self] response, error in
guard let self, let data = response else {return }
if (error != nil || response == nil) {
Logger.info("Response is nil or error is non-nil")
}
let json = try? JSONSerialization.jsonObject(with: data, options: .mutableContainers)
if let unwrapped = json as? [String: Any],
let results = unwrapped["results"] as? Array<Any>,
let firstResult = results.first as? [String: Any],
let bundleID = firstResult["bundleId"] as? String {
if (bundleID == self.applicationBundleID) {
guard let minimumSupportedOSVersion = firstResult["minimumOsVersion"] as? String else { return }
let version = ProcessInfo.processInfo.operatingSystemVersion
let systemVersion = "\(version.majorVersion).\(version.minorVersion).\(version.patchVersion)"
osVersionSupported = systemVersion.compareVersion(minimumSupportedOSVersion) != ComparisonResult.orderedAscending
if (!osVersionSupported) {
Logger.info("Current OS version is not supported")
}
// get version details
let releaseNotes = firstResult["releaseNotes"]
latestVersion = firstResult["version"] as? String
if let version = latestVersion, osVersionSupported {
versions = [version : (releaseNotes as? String) ?? ""]
}
newerVersionAvailable = latestVersion?.compareVersion(self.applicationVersion) == .orderedDescending
if (verboseLogging) {
if (newerVersionAvailable) {
Logger.info("iVersion found a new version \(latestVersion ?? "N/A") of the app on iTunes. Current version is \(self.applicationVersion ?? "nil")")
} else {
Logger.info("iVersion did not find a new version of the app on iTunes. Current version is \(self.applicationVersion ?? "nil") and the latest version is \(latestVersion ?? "nil")")
}
}
} else {
if (verboseLogging) {
Logger.info("iVersion found that the application bundle ID \(self.applicationBundleID) does not match the bundle ID of the app found on iTunes \(bundleID) with the specified App Store ID")
}
}
} else {
Logger.info("Server returned an error while fetching version info")
}
//TODO: Set download error
Logger.info("Versions downloaded \(versions ?? [:])")
performSelector(onMainThread: #selector(setRemoteVersionsDict(_:)),
with: versions,
waitUntilDone: true)
performSelector(onMainThread: #selector(setLastChecked(_:)),
with: Date(),
waitUntilDone: true)
performSelector(onMainThread: #selector(Self.downloadVersionsData),
with: nil,
waitUntilDone: true)
}
dataTask?.resume()
}
@objc private func setRemoteVersionsDict(_ dict: [String: Any]?) {
if let unwrappedDict = dict {
Logger.info("Setting Remote Versions Dict to \(unwrappedDict)")
remoteVersionsDict = unwrappedDict
}
}
private func checkForNewVersion() {
if (!self.checkingForNewVersion) {
self.checkingForNewVersion = true
DispatchQueue.global(qos: .userInitiated).async {
self.checkForNewVersionInBackground()
}
}
}
5 years ago
private func applicationLaunched() {
if checkAtLaunch {
checkIfNewVersion()
if (shouldCheckForNewVersion()) {
checkForNewVersion()
}
5 years ago
} else if verboseLogging {
Logger.info("iVersion will not check for updatess because checkAtLaunch option is disabled")
5 years ago
}
}
private func versionDetailsString() -> String {
if versionDetails == nil {
if viewedVersionDetails() {
versionDetails = versionDetails(applicationVersion, localVersionsDict())
}
} else {
5 years ago
versionDetails = versionDetails(since: lastVersion(), in: localVersionsDict())
}
return versionDetails!
}
5 years ago
private func mostRecentVersionInDict(_ dict: [String: Any]) -> String {
// return [dictionary.allKeys sortedArrayUsingSelector:@selector(compareVersion:)].lastObject;
// TODO: Fix this sorting
return dict.keys.sorted().last ?? ""
}
private func showAlertWithTitle(_ title: String,
_ details: String,
_ defaultButton: String,
_ ignoreButton: String?,
_ remindButton: String?) -> NSAlert {
Logger.info("Showing alert")
5 years ago
let floatMax = CGFloat.greatestFiniteMagnitude
let alert = NSAlert()
alert.messageText = title
alert.informativeText = inThisVersionTitle()
alert.addButton(withTitle: defaultButton)
let scrollView = NSScrollView(frame: NSRect(x: 0.0,
y: 0.0,
width: 380.0,
height: 15.0))
let contentSize = scrollView.contentSize
scrollView.borderType = .bezelBorder
scrollView.hasVerticalScroller = true
scrollView.hasHorizontalScroller = false
scrollView.autoresizingMask = [.width, .height]
let textView = NSTextView(frame: NSRect(x: 0.0,
y: 0.0,
width: contentSize.width,
height: contentSize.height))
3 years ago
textView.minSize = NSSize(width: 0.0, height: contentSize.height)
textView.maxSize = NSSize(width: floatMax, height: floatMax)
5 years ago
textView.isVerticallyResizable = true
textView.isHorizontallyResizable = false
textView.isEditable = false
textView.autoresizingMask = .width
3 years ago
textView.textContainer?.containerSize = NSSize(width: contentSize.width, height: floatMax)
5 years ago
textView.textContainer?.widthTracksTextView = true
textView.string = details
scrollView.documentView = textView
textView.sizeToFit()
let height = min(200.0, scrollView.documentView?.frame.size.height ?? 200.0) + 3.0
3 years ago
scrollView.frame = NSRect(x: 0.0, y: 0.0, width: scrollView.frame.size.width, height: height)
5 years ago
alert.accessoryView = scrollView
if let ignoreButtonTitle = ignoreButton {
alert.addButton(withTitle: ignoreButtonTitle)
5 years ago
}
if let remindButtonTitle = remindButton {
alert.addButton(withTitle: remindButtonTitle)
5 years ago
let modalResponse = alert.runModal()
if modalResponse == .alertFirstButtonReturn {
// Right most button
didDismissAlert(alert, 0)
} else if modalResponse == .alertSecondButtonReturn {
didDismissAlert(alert, 1)
} else {
didDismissAlert(alert, 2)
}
}
return alert
}
private func showIgnoreButton() -> Bool {
return ignoreButtonLabel().isEmpty == false && updatePriority < VersionUpdateHandlerPriority.medium
5 years ago
}
private func showRemindButtton() -> Bool {
return remindButtonLabel().isEmpty == false && updatePriority < VersionUpdateHandlerPriority.high
5 years ago
}
private func didDismissAlert(_ alert: NSAlert, _ buttonIndex: Int) {
5 years ago
// Get Button Indice
let downloadButtonIndex = 0
let ignoreButtonIndex = showIgnoreButton() ? 1 : 0
let remindButtonIndex = showRemindButtton() ? ignoreButtonIndex + 1 : 0
let latestVersion = mostRecentVersionInDict(self.remoteVersionsDict)
if (self.visibleLocalAlert == alert) {
setViewedVersionDetails(true)
visibleLocalAlert = nil
return
}
if (buttonIndex == downloadButtonIndex) {
setLastReminded(nil)
showAppPageInAppStore()
} else if (buttonIndex == ignoreButtonIndex) {
// ignore this version
setIgnoredVersion(latestVersion)
setLastReminded(nil)
} else if (buttonIndex == remindButtonIndex) {
setLastReminded(Date())
}
self.visibleRemoteAlert = nil
5 years ago
}
private func showAppPageInAppStore() {
if (self.verboseLogging) {
Logger.info("iVersion will open App Store using the following URL \(updatedURL())")
}
NSWorkspace.shared.open(updatedURL())
}
5 years ago
@objc private func downloadVersionsData() {
if onlyPromptIfMainWindowIsAvailable, NSApplication.shared.mainWindow == nil {
Logger.info("Main window not available in downloadVersionsData")
remoteRepeater = Repeater(interval: .seconds(5), mode: .once) { _ in
5 years ago
OperationQueue.main.addOperation { [weak self] in
guard let self = self else {
return
}
self.downloadVersionsData()
}
}
remoteRepeater?.start()
return
5 years ago
}
if checkingForNewVersion {
checkingForNewVersion = false
4 years ago
if remoteVersionsDict.isEmpty {
5 years ago
if downloadError != nil {
Logger.info("Update Check Failed because of \(downloadError!.localizedDescription)")
5 years ago
} else {
Logger.info("Version Update Check because an unknown error occurred")
5 years ago
}
return
5 years ago
}
}
let details = versionDetails(since: applicationVersion, in: remoteVersionsDict)
let mostRecentVersion = mostRecentVersionInDict(remoteVersionsDict)
if details != nil {
Logger.info("About to show visible remote alert")
5 years ago
// Check if ignored
let showDetails = ignoredVersion() != mostRecentVersion || previewMode
// show details
if showDetails && self.visibleRemoteAlert == nil {
var title = updateAvailableTitle()
title = title.appending(" (\(mostRecentVersion))")
self.visibleRemoteAlert = showAlertWithTitle(title,
details ?? "N/A",
self.downloadButtonLabel(),
showIgnoreButton() ? self.ignoreButtonLabel() : nil,
showRemindButtton() ? self.remindButtonLabel() : nil)
}
remoteRepeater = nil
5 years ago
}
}
private func checkIfNewVersion() {
if onlyPromptIfMainWindowIsAvailable, NSApplication.shared.mainWindow == nil {
Logger.info("Main window not available in checkIfNewVersion")
localRepeater = Repeater(interval: .seconds(5), mode: .once) { _ in
OperationQueue.main.addOperation { [weak self] in
guard let self = self else {
return
}
self.checkIfNewVersion()
}
}
localRepeater?.start()
return
}
let lastVersionString = lastVersion()
4 years ago
if lastVersionString.isEmpty == false || showOnFirstLaunch || previewMode {
if applicationVersion.compareVersion(lastVersionString) == ComparisonResult.orderedDescending || previewMode {
// Clear Reminder
setLastReminded(nil)
if (self.versionDetails != nil && visibleLocalAlert == nil && visibleRemoteAlert == nil) {
Logger.info("Visible Local Alert about to be display")
visibleLocalAlert = showAlertWithTitle(inThisVersionTitle(), self.versionDetailsString(), okayButtonLabel(), nil, nil)
} else {
Logger.info("Skipping to show local alert because version details is \(self.versionDetails ?? "nil")")
}
}
} else {
//record this as last viewed release
Logger.info("Set Viewed Version Details")
setViewedVersionDetails(true)
}
localRepeater = nil
}
}
extension String {
func compareVersion(_ version: String) -> ComparisonResult {
return compare(version,
options: CompareOptions.numeric,
range: nil,
locale: nil)
}
func compareVersionDescending(_ version: String) -> ComparisonResult {
let comparsionResult = (0 - compareVersion(version).rawValue)
switch comparsionResult {
case -1:
return ComparisonResult.orderedAscending
case 0:
return ComparisonResult.orderedSame
case 1:
return ComparisonResult.orderedDescending
default:
assertionFailure("Invalid Comparison Result")
return .orderedSame
}
}
}