mirror of
https://github.com/SideStore/SideStore.git
synced 2026-02-09 06:43:25 +01:00
Manual sideloading is intended to be a fallback for situations where AltStore cannot be used. To emphasize this, we hide the option by default unless the user holds the Option key.
562 lines
22 KiB
Swift
562 lines
22 KiB
Swift
//
|
|
// AppDelegate.swift
|
|
// AltServer
|
|
//
|
|
// Created by Riley Testut on 5/24/19.
|
|
// Copyright © 2019 Riley Testut. All rights reserved.
|
|
//
|
|
|
|
import Cocoa
|
|
import UserNotifications
|
|
|
|
import AltSign
|
|
|
|
import LaunchAtLogin
|
|
|
|
#if STAGING
|
|
private let altstoreAppURL = URL(string: "https://f000.backblazeb2.com/file/altstore-staging/altstore.ipa")!
|
|
#elseif BETA
|
|
private let altstoreAppURL = URL(string: "https://cdn.altstore.io/file/altstore/altstore-beta.ipa")!
|
|
#else
|
|
private let altstoreAppURL = URL(string: "https://cdn.altstore.io/file/altstore/altstore.ipa")!
|
|
#endif
|
|
|
|
extension ALTDevice: MenuDisplayable {}
|
|
|
|
@NSApplicationMain
|
|
class AppDelegate: NSObject, NSApplicationDelegate {
|
|
|
|
private let pluginManager = PluginManager()
|
|
|
|
private var statusItem: NSStatusItem?
|
|
|
|
private var connectedDevices = [ALTDevice]()
|
|
|
|
private weak var authenticationAlert: NSAlert?
|
|
|
|
@IBOutlet private var appMenu: NSMenu!
|
|
@IBOutlet private var connectedDevicesMenu: NSMenu!
|
|
@IBOutlet private var sideloadIPAConnectedDevicesMenu: NSMenu!
|
|
@IBOutlet private var enableJITMenu: NSMenu!
|
|
|
|
@IBOutlet private var launchAtLoginMenuItem: NSMenuItem!
|
|
@IBOutlet private var installMailPluginMenuItem: NSMenuItem!
|
|
@IBOutlet private var installAltStoreMenuItem: NSMenuItem!
|
|
@IBOutlet private var sideloadAppMenuItem: NSMenuItem!
|
|
|
|
private weak var authenticationAppleIDTextField: NSTextField?
|
|
private weak var authenticationPasswordTextField: NSSecureTextField?
|
|
|
|
private var connectedDevicesMenuController: MenuController<ALTDevice>!
|
|
private var sideloadIPAConnectedDevicesMenuController: MenuController<ALTDevice>!
|
|
private var enableJITMenuController: MenuController<ALTDevice>!
|
|
|
|
private var _jitAppListMenuControllers = [AnyObject]()
|
|
|
|
func applicationDidFinishLaunching(_ aNotification: Notification)
|
|
{
|
|
UserDefaults.standard.registerDefaults()
|
|
|
|
UNUserNotificationCenter.current().delegate = self
|
|
|
|
ServerConnectionManager.shared.start()
|
|
ALTDeviceManager.shared.start()
|
|
|
|
let item = NSStatusBar.system.statusItem(withLength: -1)
|
|
item.menu = self.appMenu
|
|
item.button?.image = NSImage(named: "MenuBarIcon")
|
|
self.statusItem = item
|
|
|
|
self.appMenu.delegate = self
|
|
|
|
self.sideloadAppMenuItem.keyEquivalentModifierMask = .option
|
|
self.sideloadAppMenuItem.isAlternate = true
|
|
|
|
let placeholder = NSLocalizedString("No Connected Devices", comment: "")
|
|
|
|
self.connectedDevicesMenuController = MenuController<ALTDevice>(menu: self.connectedDevicesMenu, items: [])
|
|
self.connectedDevicesMenuController.placeholder = placeholder
|
|
self.connectedDevicesMenuController.action = { [weak self] device in
|
|
self?.installAltStore(to: device)
|
|
}
|
|
|
|
self.sideloadIPAConnectedDevicesMenuController = MenuController<ALTDevice>(menu: self.sideloadIPAConnectedDevicesMenu, items: [])
|
|
self.sideloadIPAConnectedDevicesMenuController.placeholder = placeholder
|
|
self.sideloadIPAConnectedDevicesMenuController.action = { [weak self] device in
|
|
self?.sideloadIPA(to: device)
|
|
}
|
|
|
|
self.enableJITMenuController = MenuController<ALTDevice>(menu: self.enableJITMenu, items: [])
|
|
self.enableJITMenuController.placeholder = placeholder
|
|
|
|
UNUserNotificationCenter.current().requestAuthorization(options: [.alert]) { (success, error) in
|
|
guard success else { return }
|
|
|
|
if !UserDefaults.standard.didPresentInitialNotification
|
|
{
|
|
let content = UNMutableNotificationContent()
|
|
content.title = NSLocalizedString("AltServer Running", comment: "")
|
|
content.body = NSLocalizedString("AltServer runs in the background as a menu bar app listening for AltStore.", comment: "")
|
|
|
|
let request = UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil)
|
|
UNUserNotificationCenter.current().add(request)
|
|
|
|
UserDefaults.standard.didPresentInitialNotification = true
|
|
}
|
|
}
|
|
|
|
if self.pluginManager.isUpdateAvailable
|
|
{
|
|
self.installMailPlugin()
|
|
}
|
|
}
|
|
|
|
func applicationWillTerminate(_ aNotification: Notification)
|
|
{
|
|
// Insert code here to tear down your application
|
|
}
|
|
}
|
|
|
|
private extension AppDelegate
|
|
{
|
|
@objc func installAltStore(to device: ALTDevice)
|
|
{
|
|
self.installApplication(at: altstoreAppURL, to: device)
|
|
}
|
|
|
|
@objc func sideloadIPA(to device: ALTDevice)
|
|
{
|
|
let openPanel = NSOpenPanel()
|
|
openPanel.canChooseDirectories = false
|
|
openPanel.allowsMultipleSelection = false
|
|
openPanel.allowedFileTypes = ["ipa"]
|
|
openPanel.begin { (response) in
|
|
guard let fileURL = openPanel.url, response == .OK else { return }
|
|
self.installApplication(at: fileURL, to: device)
|
|
}
|
|
}
|
|
|
|
func enableJIT(for app: InstalledApp, on device: ALTDevice)
|
|
{
|
|
func finish(_ result: Result<Void, Error>)
|
|
{
|
|
DispatchQueue.main.async {
|
|
switch result
|
|
{
|
|
case .failure(let error):
|
|
self.showErrorAlert(error: error, localizedFailure: String(format: NSLocalizedString("JIT compilation could not be enabled for %@.", comment: ""), app.name))
|
|
|
|
case .success:
|
|
let alert = NSAlert()
|
|
alert.messageText = String(format: NSLocalizedString("Successfully enabled JIT for %@.", comment: ""), app.name)
|
|
alert.informativeText = String(format: NSLocalizedString("JIT will remain enabled until you quit the app. You can now disconnect %@ from your computer.", comment: ""), device.name)
|
|
alert.runModal()
|
|
}
|
|
}
|
|
}
|
|
|
|
ALTDeviceManager.shared.prepare(device) { (result) in
|
|
switch result
|
|
{
|
|
case .failure(let error as NSError): return finish(.failure(error))
|
|
case .success:
|
|
ALTDeviceManager.shared.startDebugConnection(to: device) { (connection, error) in
|
|
guard let connection = connection else {
|
|
return finish(.failure(error! as NSError))
|
|
}
|
|
|
|
connection.enableUnsignedCodeExecutionForProcess(withName: app.executableName) { (success, error) in
|
|
guard success else {
|
|
return finish(.failure(error!))
|
|
}
|
|
|
|
finish(.success(()))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func installApplication(at url: URL, to device: ALTDevice)
|
|
{
|
|
let alert = NSAlert()
|
|
alert.messageText = NSLocalizedString("Please enter your Apple ID and password.", comment: "")
|
|
alert.informativeText = NSLocalizedString("Your Apple ID and password are not saved and are only sent to Apple for authentication.", comment: "")
|
|
|
|
let textFieldSize = NSSize(width: 300, height: 22)
|
|
|
|
let appleIDTextField = NSTextField(frame: NSRect(x: 0, y: 0, width: textFieldSize.width, height: textFieldSize.height))
|
|
appleIDTextField.delegate = self
|
|
appleIDTextField.translatesAutoresizingMaskIntoConstraints = false
|
|
appleIDTextField.placeholderString = NSLocalizedString("Apple ID", comment: "")
|
|
alert.window.initialFirstResponder = appleIDTextField
|
|
self.authenticationAppleIDTextField = appleIDTextField
|
|
|
|
let passwordTextField = NSSecureTextField(frame: NSRect(x: 0, y: 0, width: textFieldSize.width, height: textFieldSize.height))
|
|
passwordTextField.delegate = self
|
|
passwordTextField.translatesAutoresizingMaskIntoConstraints = false
|
|
passwordTextField.placeholderString = NSLocalizedString("Password", comment: "")
|
|
self.authenticationPasswordTextField = passwordTextField
|
|
|
|
appleIDTextField.nextKeyView = passwordTextField
|
|
|
|
let stackView = NSStackView(frame: NSRect(x: 0, y: 0, width: textFieldSize.width, height: textFieldSize.height * 2))
|
|
stackView.orientation = .vertical
|
|
stackView.distribution = .equalSpacing
|
|
stackView.spacing = 0
|
|
stackView.addArrangedSubview(appleIDTextField)
|
|
stackView.addArrangedSubview(passwordTextField)
|
|
alert.accessoryView = stackView
|
|
|
|
alert.addButton(withTitle: NSLocalizedString("Install", comment: ""))
|
|
alert.addButton(withTitle: NSLocalizedString("Cancel", comment: ""))
|
|
|
|
self.authenticationAlert = alert
|
|
self.validate()
|
|
|
|
NSRunningApplication.current.activate(options: .activateIgnoringOtherApps)
|
|
|
|
let response = alert.runModal()
|
|
guard response == .alertFirstButtonReturn else { return }
|
|
|
|
let username = appleIDTextField.stringValue
|
|
let password = passwordTextField.stringValue
|
|
|
|
func install()
|
|
{
|
|
ALTDeviceManager.shared.installApplication(at: url, to: device, appleID: username, password: password) { (result) in
|
|
switch result
|
|
{
|
|
case .success(let application):
|
|
let content = UNMutableNotificationContent()
|
|
content.title = NSLocalizedString("Installation Succeeded", comment: "")
|
|
content.body = String(format: NSLocalizedString("%@ was successfully installed on %@.", comment: ""), application.name, device.name)
|
|
|
|
let request = UNNotificationRequest(identifier: UUID().uuidString, content: content, trigger: nil)
|
|
UNUserNotificationCenter.current().add(request)
|
|
|
|
case .failure(InstallError.cancelled), .failure(ALTAppleAPIError.requiresTwoFactorAuthentication):
|
|
// Ignore
|
|
break
|
|
|
|
case .failure(let error):
|
|
self.showErrorAlert(error: error, localizedFailure: String(format: NSLocalizedString("Could not install app to %@.", comment: ""), device.name))
|
|
}
|
|
}
|
|
}
|
|
|
|
if !self.pluginManager.isMailPluginInstalled || self.pluginManager.isUpdateAvailable
|
|
{
|
|
AnisetteDataManager.shared.isXPCAvailable { (isAvailable) in
|
|
if isAvailable
|
|
{
|
|
// XPC service is available, so we don't need to install/update Mail plug-in.
|
|
// Users can still manually do so from the AltServer menu.
|
|
install()
|
|
}
|
|
else
|
|
{
|
|
DispatchQueue.main.async {
|
|
self.installMailPlugin { (result) in
|
|
switch result
|
|
{
|
|
case .failure: break
|
|
case .success: install()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
install()
|
|
}
|
|
}
|
|
|
|
func showErrorAlert(error: Error, localizedFailure: String)
|
|
{
|
|
let nsError = error as NSError
|
|
|
|
let alert = NSAlert()
|
|
alert.alertStyle = .critical
|
|
alert.messageText = localizedFailure
|
|
|
|
var messageComponents = [String]()
|
|
|
|
let separator: String
|
|
switch error
|
|
{
|
|
case ALTServerError.maximumFreeAppLimitReached: separator = "\n\n"
|
|
default: separator = " "
|
|
}
|
|
|
|
if let errorFailure = nsError.localizedFailure
|
|
{
|
|
if let failureReason = nsError.localizedFailureReason
|
|
{
|
|
if nsError.localizedDescription.starts(with: errorFailure)
|
|
{
|
|
alert.messageText = errorFailure
|
|
messageComponents.append(failureReason)
|
|
}
|
|
else
|
|
{
|
|
alert.messageText = errorFailure
|
|
messageComponents.append(nsError.localizedDescription)
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// No failure reason given.
|
|
|
|
if nsError.localizedDescription.starts(with: errorFailure)
|
|
{
|
|
// No need to duplicate errorFailure in both title and message.
|
|
alert.messageText = localizedFailure
|
|
messageComponents.append(nsError.localizedDescription)
|
|
}
|
|
else
|
|
{
|
|
alert.messageText = errorFailure
|
|
messageComponents.append(nsError.localizedDescription)
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
alert.messageText = localizedFailure
|
|
messageComponents.append(nsError.localizedDescription)
|
|
}
|
|
|
|
if let recoverySuggestion = nsError.localizedRecoverySuggestion
|
|
{
|
|
messageComponents.append(recoverySuggestion)
|
|
}
|
|
|
|
let informativeText = messageComponents.joined(separator: separator)
|
|
alert.informativeText = informativeText
|
|
|
|
NSRunningApplication.current.activate(options: .activateIgnoringOtherApps)
|
|
|
|
alert.runModal()
|
|
}
|
|
|
|
@objc func toggleLaunchAtLogin(_ item: NSMenuItem)
|
|
{
|
|
LaunchAtLogin.isEnabled.toggle()
|
|
}
|
|
|
|
@objc func handleInstallMailPluginMenuItem(_ item: NSMenuItem)
|
|
{
|
|
if !self.pluginManager.isMailPluginInstalled || self.pluginManager.isUpdateAvailable
|
|
{
|
|
self.installMailPlugin()
|
|
}
|
|
else
|
|
{
|
|
self.uninstallMailPlugin()
|
|
}
|
|
}
|
|
|
|
private func installMailPlugin(completion: ((Result<Void, Error>) -> Void)? = nil)
|
|
{
|
|
self.pluginManager.installMailPlugin { (result) in
|
|
DispatchQueue.main.async {
|
|
switch result
|
|
{
|
|
case .failure(PluginError.cancelled): break
|
|
case .failure(let error):
|
|
let alert = NSAlert()
|
|
alert.messageText = NSLocalizedString("Failed to Install Mail Plug-in", comment: "")
|
|
alert.informativeText = error.localizedDescription
|
|
alert.runModal()
|
|
|
|
case .success:
|
|
let alert = NSAlert()
|
|
alert.messageText = NSLocalizedString("Mail Plug-in Installed", comment: "")
|
|
alert.informativeText = NSLocalizedString("Please restart Mail and enable AltPlugin in Mail's Preferences. Mail must be running when installing or refreshing apps with AltServer.", comment: "")
|
|
alert.runModal()
|
|
}
|
|
|
|
completion?(result)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func uninstallMailPlugin()
|
|
{
|
|
self.pluginManager.uninstallMailPlugin { (result) in
|
|
DispatchQueue.main.async {
|
|
switch result
|
|
{
|
|
case .failure(PluginError.cancelled): break
|
|
case .failure(let error):
|
|
let alert = NSAlert()
|
|
alert.messageText = NSLocalizedString("Failed to Uninstall Mail Plug-in", comment: "")
|
|
alert.informativeText = error.localizedDescription
|
|
alert.runModal()
|
|
|
|
case .success:
|
|
let alert = NSAlert()
|
|
alert.messageText = NSLocalizedString("Mail Plug-in Uninstalled", comment: "")
|
|
alert.informativeText = NSLocalizedString("Please restart Mail for changes to take effect. You will not be able to use AltServer until the plug-in is reinstalled.", comment: "")
|
|
alert.runModal()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extension AppDelegate: NSMenuDelegate
|
|
{
|
|
func menuWillOpen(_ menu: NSMenu)
|
|
{
|
|
guard menu == self.appMenu else { return }
|
|
|
|
// Clear any cached _jitAppListMenuControllers.
|
|
self._jitAppListMenuControllers.removeAll()
|
|
|
|
self.connectedDevices = ALTDeviceManager.shared.availableDevices
|
|
|
|
self.connectedDevicesMenuController.items = self.connectedDevices
|
|
self.sideloadIPAConnectedDevicesMenuController.items = self.connectedDevices
|
|
self.enableJITMenuController.items = self.connectedDevices
|
|
|
|
self.launchAtLoginMenuItem.target = self
|
|
self.launchAtLoginMenuItem.action = #selector(AppDelegate.toggleLaunchAtLogin(_:))
|
|
self.launchAtLoginMenuItem.state = LaunchAtLogin.isEnabled ? .on : .off
|
|
|
|
if self.pluginManager.isUpdateAvailable
|
|
{
|
|
self.installMailPluginMenuItem.title = NSLocalizedString("Update Mail Plug-in", comment: "")
|
|
}
|
|
else if self.pluginManager.isMailPluginInstalled
|
|
{
|
|
self.installMailPluginMenuItem.title = NSLocalizedString("Uninstall Mail Plug-in", comment: "")
|
|
}
|
|
else
|
|
{
|
|
self.installMailPluginMenuItem.title = NSLocalizedString("Install Mail Plug-in", comment: "")
|
|
}
|
|
self.installMailPluginMenuItem.target = self
|
|
self.installMailPluginMenuItem.action = #selector(AppDelegate.handleInstallMailPluginMenuItem(_:))
|
|
|
|
// Need to re-set this every time menu appears so we can refresh device app list.
|
|
self.enableJITMenuController.submenuHandler = { [weak self] device in
|
|
let submenu = NSMenu(title: NSLocalizedString("Sideloaded Apps", comment: ""))
|
|
|
|
guard let `self` = self else { return submenu }
|
|
|
|
let submenuController = MenuController<InstalledApp>(menu: submenu, items: [])
|
|
submenuController.placeholder = NSLocalizedString("Loading...", comment: "")
|
|
submenuController.action = { [weak self] (appInfo) in
|
|
self?.enableJIT(for: appInfo, on: device)
|
|
}
|
|
|
|
// Keep strong reference
|
|
self._jitAppListMenuControllers.append(submenuController)
|
|
|
|
ALTDeviceManager.shared.fetchInstalledApps(on: device) { (installedApps, error) in
|
|
DispatchQueue.main.async {
|
|
guard let installedApps = installedApps else {
|
|
print("Failed to fetch installed apps from \(device).", error!)
|
|
submenuController.placeholder = error?.localizedDescription
|
|
return
|
|
}
|
|
|
|
print("Fetched \(installedApps.count) apps for \(device).")
|
|
|
|
let sortedApps = installedApps.sorted { (app1, app2) in
|
|
if app1.name == app2.name
|
|
{
|
|
return app1.bundleIdentifier < app2.bundleIdentifier
|
|
}
|
|
else
|
|
{
|
|
return app1.name < app2.name
|
|
}
|
|
}
|
|
|
|
submenuController.items = sortedApps
|
|
|
|
if submenuController.items.isEmpty
|
|
{
|
|
submenuController.placeholder = NSLocalizedString("No Sideloaded Apps", comment: "")
|
|
}
|
|
}
|
|
}
|
|
|
|
return submenu
|
|
}
|
|
}
|
|
|
|
func menuDidClose(_ menu: NSMenu)
|
|
{
|
|
// Clearing _jitAppListMenuControllers now prevents action handler from being called.
|
|
// self._jitAppListMenuControllers = []
|
|
}
|
|
|
|
func menu(_ menu: NSMenu, willHighlight item: NSMenuItem?)
|
|
{
|
|
guard menu == self.appMenu else { return }
|
|
|
|
// The submenu won't update correctly if the user holds/releases
|
|
// the Option key while the submenu is visible.
|
|
// Workaround: temporarily set submenu to nil to dismiss it,
|
|
// which will then cause the correct submenu to appear.
|
|
|
|
let previousItem: NSMenuItem
|
|
switch item
|
|
{
|
|
case self.sideloadAppMenuItem: previousItem = self.installAltStoreMenuItem
|
|
case self.installAltStoreMenuItem: previousItem = self.sideloadAppMenuItem
|
|
default: return
|
|
}
|
|
|
|
let submenu = previousItem.submenu
|
|
previousItem.submenu = nil
|
|
previousItem.submenu = submenu
|
|
}
|
|
}
|
|
|
|
extension AppDelegate: NSTextFieldDelegate
|
|
{
|
|
func controlTextDidChange(_ obj: Notification)
|
|
{
|
|
self.validate()
|
|
}
|
|
|
|
func controlTextDidEndEditing(_ obj: Notification)
|
|
{
|
|
self.validate()
|
|
}
|
|
|
|
private func validate()
|
|
{
|
|
guard
|
|
let appleID = self.authenticationAppleIDTextField?.stringValue.trimmingCharacters(in: .whitespacesAndNewlines),
|
|
let password = self.authenticationPasswordTextField?.stringValue.trimmingCharacters(in: .whitespacesAndNewlines)
|
|
else { return }
|
|
|
|
if appleID.isEmpty || password.isEmpty
|
|
{
|
|
self.authenticationAlert?.buttons.first?.isEnabled = false
|
|
}
|
|
else
|
|
{
|
|
self.authenticationAlert?.buttons.first?.isEnabled = true
|
|
}
|
|
|
|
self.authenticationAlert?.layout()
|
|
}
|
|
}
|
|
|
|
extension AppDelegate: UNUserNotificationCenterDelegate
|
|
{
|
|
func userNotificationCenter(_ center: UNUserNotificationCenter, willPresent notification: UNNotification, withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void)
|
|
{
|
|
completionHandler([.alert, .sound, .badge])
|
|
}
|
|
}
|