mirror of
https://github.com/X1a0He/Adobe-Downloader.git
synced 2025-11-25 03:14:57 +08:00
1. AboutView: Subdivided some Views 2. AboutView: Added the display of system version number and specific chip model 3. AppCardView: Added the display of the minimum system version and module display 4. AppCardView: Optimized the display of the number of available versions and the number of dependent components 5. VersionPickerView: Added the product icon display in the Header part 6. VersionPickerView: Optimized the download architecture prompt 7. VersionPickerView: Optimized the display of product version, provided the display of productVersion and buildGuid 8. VersionPickerView: Optimized the display of dependent components, added the display of total number of dependencies and buildGuid 9. VersionPickerView: In DEBUG mode, you can check whether the dependent component hits the correct version, the selected version, and the reason for not hitting, and added the selected version icon 10. VersionPickerView: Adjusted the version display of dependent components, and no longer displayed baseVersion by default 11. VersionPickerView: Added optional module display and information
199 lines
7.4 KiB
Swift
199 lines
7.4 KiB
Swift
//
|
|
// Adobe Downloader
|
|
//
|
|
// Created by X1a0He on 2024/10/30.
|
|
//
|
|
/*
|
|
Adobe Exit Code
|
|
107: 架构不一致或安装文件被损坏
|
|
103: 权限问题
|
|
182: 表示创建的包不包含要安装的包
|
|
133: 磁盘空间不足
|
|
*/
|
|
import Foundation
|
|
|
|
actor InstallManager {
|
|
enum InstallError: Error, LocalizedError {
|
|
case setupNotFound
|
|
case installationFailed(String)
|
|
case cancelled
|
|
case permissionDenied
|
|
|
|
var errorDescription: String? {
|
|
switch self {
|
|
case .setupNotFound: return String(localized: "找不到安装程序")
|
|
case .installationFailed(let message): return message
|
|
case .cancelled: return String(localized: "安装已取消")
|
|
case .permissionDenied: return String(localized: "权限被拒绝")
|
|
}
|
|
}
|
|
}
|
|
|
|
private var installationProcess: Process?
|
|
private var progressHandler: ((Double, String) -> Void)?
|
|
private let setupPath = "/Library/Application Support/Adobe/Adobe Desktop Common/HDBox/Setup"
|
|
|
|
actor InstallationState {
|
|
var isCompleted = false
|
|
var error: Error?
|
|
var hasExitCode0 = false
|
|
var lastOutputTime = Date()
|
|
|
|
func markCompleted() {
|
|
isCompleted = true
|
|
}
|
|
|
|
func setError(_ error: Error) {
|
|
if !isCompleted {
|
|
self.error = error
|
|
isCompleted = true
|
|
}
|
|
}
|
|
|
|
func setExitCode0() {
|
|
hasExitCode0 = true
|
|
}
|
|
|
|
func updateLastOutputTime() {
|
|
lastOutputTime = Date()
|
|
}
|
|
|
|
func getTimeSinceLastOutput() -> TimeInterval {
|
|
return Date().timeIntervalSince(lastOutputTime)
|
|
}
|
|
|
|
var shouldContinue: Bool {
|
|
!isCompleted
|
|
}
|
|
|
|
var hasReceivedExitCode0: Bool {
|
|
hasExitCode0
|
|
}
|
|
}
|
|
|
|
private func executeInstallation(
|
|
at appPath: URL,
|
|
progressHandler: @escaping (Double, String) -> Void
|
|
) async throws {
|
|
guard FileManager.default.fileExists(atPath: setupPath) else {
|
|
throw InstallError.setupNotFound
|
|
}
|
|
|
|
let driverPath = appPath.appendingPathComponent("driver.xml").path
|
|
guard FileManager.default.fileExists(atPath: driverPath) else {
|
|
throw InstallError.installationFailed("找不到 driver.xml 文件")
|
|
}
|
|
|
|
let attributes = try? FileManager.default.attributesOfItem(atPath: driverPath)
|
|
if let permissions = attributes?[.posixPermissions] as? NSNumber {
|
|
if permissions.int16Value & 0o444 == 0 {
|
|
throw InstallError.installationFailed("driver.xml 文件没有读取权限")
|
|
}
|
|
}
|
|
|
|
let installCommand = "sudo \"\(setupPath)\" --install=1 --driverXML=\"\(driverPath)\""
|
|
|
|
await MainActor.run {
|
|
progressHandler(0.0, String(localized: "正在准备安装..."))
|
|
}
|
|
|
|
try await withCheckedThrowingContinuation { (continuation: CheckedContinuation<Void, Error>) in
|
|
Task.detached {
|
|
do {
|
|
try await PrivilegedHelperManager.shared.executeInstallation(installCommand) { output in
|
|
Task { @MainActor in
|
|
if let range = output.range(of: "Exit Code:\\s*(-?[0-9]+)", options: .regularExpression),
|
|
let codeStr = output[range].split(separator: ":").last?.trimmingCharacters(in: .whitespaces),
|
|
let exitCode = Int(codeStr) {
|
|
|
|
if exitCode == 0 {
|
|
progressHandler(1.0, String(localized: "安装完成"))
|
|
PrivilegedHelperManager.shared.executeCommand("pkill -f Setup") { _ in }
|
|
continuation.resume()
|
|
return
|
|
} else {
|
|
let errorMessage: String
|
|
switch exitCode {
|
|
case 107:
|
|
errorMessage = String(localized: "架构或版本不一致 (退出代码: \(exitCode))")
|
|
case 103:
|
|
errorMessage = String(localized: "权限问题 (退出代码: \(exitCode))")
|
|
case 182:
|
|
errorMessage = String(localized: "安装文件不完整或损坏 (退出代码: \(exitCode))")
|
|
case -1:
|
|
errorMessage = String(localized: "Setup 组件未被处理 (退出代码: \(exitCode))")
|
|
default:
|
|
errorMessage = String(localized: "(退出代码: \(exitCode))")
|
|
}
|
|
progressHandler(0.0, errorMessage)
|
|
continuation.resume(throwing: InstallError.installationFailed(errorMessage))
|
|
return
|
|
}
|
|
}
|
|
|
|
if let progress = await self.parseProgress(from: output) {
|
|
progressHandler(progress, String(localized: "正在安装..."))
|
|
}
|
|
}
|
|
}
|
|
} catch {
|
|
continuation.resume(throwing: error)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private func parseProgress(from output: String) -> Double? {
|
|
if let range = output.range(of: "Exit Code:\\s*(-?[0-9]+)", options: .regularExpression),
|
|
let codeStr = output[range].split(separator: ":").last?.trimmingCharacters(in: .whitespaces),
|
|
let exitCode = Int(codeStr) {
|
|
if exitCode == 0 {
|
|
return 1.0
|
|
}
|
|
}
|
|
|
|
if output.range(of: "Progress:\\s*[0-9]+/[0-9]+", options: .regularExpression) != nil {
|
|
return nil
|
|
}
|
|
|
|
if let range = output.range(of: "Progress:\\s*([0-9]{1,3})%", options: .regularExpression),
|
|
let progressStr = output[range].split(separator: ":").last?.trimmingCharacters(in: .whitespaces),
|
|
let progressValue = Double(progressStr.replacingOccurrences(of: "%", with: "")) {
|
|
return progressValue / 100.0
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func install(
|
|
at appPath: URL,
|
|
progressHandler: @escaping (Double, String) -> Void
|
|
) async throws {
|
|
try await executeInstallation(
|
|
at: appPath,
|
|
progressHandler: progressHandler
|
|
)
|
|
}
|
|
|
|
func cancel() {
|
|
PrivilegedHelperManager.shared.executeCommand("pkill -f Setup") { _ in }
|
|
}
|
|
|
|
func getInstallCommand(for driverPath: String) -> String {
|
|
return "sudo \"\(setupPath)\" --install=1 --driverXML=\"\(driverPath)\""
|
|
}
|
|
|
|
func retry(
|
|
at appPath: URL,
|
|
progressHandler: @escaping (Double, String) -> Void
|
|
) async throws {
|
|
cancel()
|
|
try await Task.sleep(nanoseconds: 1_000_000_000)
|
|
|
|
try await executeInstallation(
|
|
at: appPath,
|
|
progressHandler: progressHandler
|
|
)
|
|
}
|
|
}
|
|
|