mirror of
https://github.com/SideStore/SideStore.git
synced 2026-02-09 06:43:25 +01:00
304 lines
12 KiB
Swift
304 lines
12 KiB
Swift
//
|
|
// VerifyAppOperation.swift
|
|
// AltStore
|
|
//
|
|
// Created by Riley Testut on 5/2/20.
|
|
// Copyright © 2020 Riley Testut. All rights reserved.
|
|
//
|
|
|
|
import Foundation
|
|
import CryptoKit
|
|
|
|
import AltStoreCore
|
|
import AltSign
|
|
import Roxas
|
|
|
|
import RegexBuilder
|
|
|
|
private extension ALTEntitlement
|
|
{
|
|
static var ignoredEntitlements: Set<ALTEntitlement> = [
|
|
.applicationIdentifier,
|
|
.teamIdentifier
|
|
]
|
|
}
|
|
|
|
extension VerifyAppOperation
|
|
{
|
|
enum PermissionReviewMode
|
|
{
|
|
case none
|
|
case all
|
|
case added
|
|
}
|
|
}
|
|
|
|
@objc(VerifyAppOperation)
|
|
final class VerifyAppOperation: ResultOperation<Void>
|
|
{
|
|
let permissionsMode: PermissionReviewMode
|
|
let context: InstallAppOperationContext
|
|
|
|
init(permissionsMode: PermissionReviewMode, context: InstallAppOperationContext)
|
|
{
|
|
self.permissionsMode = permissionsMode
|
|
self.context = context
|
|
|
|
super.init()
|
|
}
|
|
|
|
override func main()
|
|
{
|
|
super.main()
|
|
|
|
do
|
|
{
|
|
if let error = self.context.error
|
|
{
|
|
throw error
|
|
}
|
|
let appName = self.context.app?.name ?? NSLocalizedString("The app", comment: "")
|
|
self.localizedFailure = String(format: NSLocalizedString("%@ could not be installed.", comment: ""), appName)
|
|
|
|
guard let app = self.context.app else {
|
|
throw OperationError.invalidParameters("VerifyAppOperation.main: self.context.app is nil")
|
|
}
|
|
|
|
if !["ny.litritt.ignited", "com.litritt.ignited"].contains(where: { $0 == app.bundleIdentifier }) {
|
|
guard app.bundleIdentifier == self.context.bundleIdentifier else {
|
|
throw VerificationError.mismatchedBundleIdentifiers(sourceBundleID: self.context.bundleIdentifier, app: app)
|
|
}
|
|
}
|
|
|
|
guard ProcessInfo.processInfo.isOperatingSystemAtLeast(app.minimumiOSVersion) else {
|
|
throw VerificationError.iOSVersionNotSupported(app: app, requiredOSVersion: app.minimumiOSVersion)
|
|
}
|
|
|
|
guard let appVersion = self.context.appVersion else {
|
|
return self.finish(.success(()))
|
|
}
|
|
|
|
Task<Void, Never> {
|
|
do
|
|
{
|
|
guard let ipaURL = self.context.ipaURL else { throw OperationError.appNotFound(name: app.name) }
|
|
|
|
|
|
// TODO: @mahee96: appVersion is instantiated source info as AppVersion incoming from source json
|
|
// app is the instantiated ipa downloaded from the specified in the source json in temp dir
|
|
//
|
|
// For alpha and beta/nightly releases, the CFBundleShortVersionString which is the
|
|
// $(MARKETING_VERSION) will be overriden with the commit id before invoking xcode build
|
|
//
|
|
try await self.verifyHash(of: app, at: ipaURL, matches: appVersion)
|
|
try await self.verifyDownloadedVersion(of: app, matches: appVersion)
|
|
try await self.verifyPermissions(of: app, match: appVersion)
|
|
|
|
self.finish(.success(()))
|
|
}
|
|
catch
|
|
{
|
|
self.finish(.failure(error))
|
|
}
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
self.finish(.failure(error))
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension VerifyAppOperation
|
|
{
|
|
func verifyHash(of app: ALTApplication, at ipaURL: URL, @AsyncManaged matches appVersion: AppVersion) async throws
|
|
{
|
|
// Do nothing if source doesn't provide hash.
|
|
guard let expectedHash = await $appVersion.sha256 else { return }
|
|
|
|
let data = try Data(contentsOf: ipaURL)
|
|
let sha256Hash = SHA256.hash(data: data)
|
|
let hashString = sha256Hash.compactMap { String(format: "%02x", $0) }.joined()
|
|
|
|
Logger.sideload.debug("Comparing app hash (\(hashString, privacy: .public)) against expected hash (\(expectedHash, privacy: .public))...")
|
|
|
|
guard hashString == expectedHash else { throw VerificationError.mismatchedHash(hashString, expectedHash: expectedHash, app: app) }
|
|
}
|
|
|
|
func verifyDownloadedVersion(of app: ALTApplication, @AsyncManaged matches appVersion: AppVersion) async throws
|
|
{
|
|
let (version, buildVersion) = await $appVersion.perform { ($0.version, $0.buildVersion) }
|
|
|
|
guard version == app.version else { throw VerificationError.mismatchedVersion(app.version, expectedVersion: version, app: app) }
|
|
|
|
if let buildVersion
|
|
{
|
|
// TODO: @mahee96: requires altsign-marketplace branch release or equivalent
|
|
// guard buildVersion == app.buildVersion else { throw VerificationError.mismatchedBuildVersion(app.buildVersion, expectedVersion: buildVersion, app: app) }
|
|
}
|
|
}
|
|
|
|
func verifyPermissions(of app: ALTApplication, @AsyncManaged match appVersion: AppVersion) async throws
|
|
{
|
|
guard self.permissionsMode != .none else { return }
|
|
guard let storeApp = await $appVersion.app else { throw OperationError.invalidParameters("verifyPermissions requires storeApp to be non-nil") }
|
|
|
|
// Verify source permissions match first.
|
|
let allPermissions = try await self.verifyPermissions(of: app, match: storeApp)
|
|
|
|
guard #available(iOS 15, *) else {
|
|
// Only review downloaded app permissions on iOS 15 and above.
|
|
return
|
|
}
|
|
|
|
switch self.permissionsMode
|
|
{
|
|
case .none: break
|
|
case .all:
|
|
guard let presentingViewController = self.context.presentingViewController else { break } // Don't fail just because we can't show permissions.
|
|
|
|
let allEntitlements = allPermissions.compactMap { $0 as? ALTEntitlement }
|
|
if !allEntitlements.isEmpty
|
|
{
|
|
try await self.review(allEntitlements, for: app, mode: .all, presentingViewController: presentingViewController)
|
|
}
|
|
|
|
case .added:
|
|
let installedAppURL = InstalledApp.fileURL(for: app)
|
|
guard let previousApp = ALTApplication(fileURL: installedAppURL) else { throw OperationError.appNotFound(name: app.name) }
|
|
|
|
var previousEntitlements = Set(previousApp.entitlements.keys)
|
|
for appExtension in previousApp.appExtensions
|
|
{
|
|
previousEntitlements.formUnion(appExtension.entitlements.keys)
|
|
}
|
|
|
|
// Make sure all entitlements already exist in previousApp.
|
|
let addedEntitlements = Array(allPermissions.lazy.compactMap { $0 as? ALTEntitlement }.filter { !previousEntitlements.contains($0) })
|
|
if !addedEntitlements.isEmpty
|
|
{
|
|
// _DO_ throw error if there isn't a presentingViewController.
|
|
guard let presentingViewController = self.context.presentingViewController else { throw VerificationError.addedPermissions(addedEntitlements, appVersion: appVersion) }
|
|
|
|
try await self.review(addedEntitlements, for: app, mode: .added, presentingViewController: presentingViewController)
|
|
}
|
|
}
|
|
}
|
|
|
|
@discardableResult
|
|
func verifyPermissions(of app: ALTApplication, @AsyncManaged match storeApp: StoreApp) async throws -> [any ALTAppPermission]
|
|
{
|
|
// Entitlements
|
|
var allEntitlements = Set(app.entitlements.keys)
|
|
for appExtension in app.appExtensions
|
|
{
|
|
allEntitlements.formUnion(appExtension.entitlements.keys)
|
|
}
|
|
|
|
// Filter out ignored entitlements.
|
|
allEntitlements = allEntitlements.filter { !ALTEntitlement.ignoredEntitlements.contains($0) }
|
|
|
|
if let isDebuggable = app.entitlements[.getTaskAllow] as? Bool, !isDebuggable
|
|
{
|
|
// App has `get-task-allow` entitlement but the value is false, so remove from allEntitlements.
|
|
allEntitlements.remove(.getTaskAllow)
|
|
}
|
|
|
|
// Privacy
|
|
let allPrivacyPermissions = ([app] + app.appExtensions).flatMap { (app) in
|
|
let permissions = app.bundle.infoDictionary?.keys.compactMap { key -> ALTAppPrivacyPermission? in
|
|
if #available(iOS 16, *)
|
|
{
|
|
guard key.wholeMatch(of: Regex.privacyPermission) != nil else { return nil }
|
|
}
|
|
else
|
|
{
|
|
guard key.contains("UsageDescription") else { return nil }
|
|
}
|
|
|
|
let permission = ALTAppPrivacyPermission(rawValue: key)
|
|
return permission
|
|
} ?? []
|
|
|
|
return permissions
|
|
}
|
|
|
|
// Verify permissions.
|
|
let sourcePermissions: Set<AnyHashable> = Set(await $storeApp.perform { $0.permissions.map { AnyHashable($0.permission) } })
|
|
let localPermissions: [any ALTAppPermission] = Array(allEntitlements) + Array(allPrivacyPermissions)
|
|
|
|
// To pass: EVERY permission in localPermissions must also appear in sourcePermissions.
|
|
// If there is a single missing permission, throw error.
|
|
let missingPermissions: [any ALTAppPermission] = localPermissions.filter { permission in
|
|
if sourcePermissions.contains(AnyHashable(permission))
|
|
{
|
|
// `permission` exists in source, so return false.
|
|
return false
|
|
}
|
|
else if permission.type == .privacy
|
|
{
|
|
guard #available(iOS 16, *) else {
|
|
// Assume all privacy permissions _are_ included in source on pre-iOS 16 devices.
|
|
return false
|
|
}
|
|
|
|
// Special-handling for legacy privacy permissions.
|
|
if let match = permission.rawValue.firstMatch(of: Regex.privacyPermission),
|
|
case let legacyPermission = ALTAppPrivacyPermission(rawValue: String(match.1)),
|
|
sourcePermissions.contains(AnyHashable(legacyPermission))
|
|
{
|
|
// The legacy name of this permission exists in the source, so return false.
|
|
return false
|
|
}
|
|
}
|
|
|
|
// Source doesn't contain permission or its legacy name, so assume it is missing.
|
|
return true
|
|
}
|
|
|
|
do
|
|
{
|
|
guard missingPermissions.isEmpty else {
|
|
// There is at least one undeclared permission, so throw error.
|
|
throw VerificationError.undeclaredPermissions(missingPermissions, app: app)
|
|
}
|
|
}
|
|
catch let error as VerificationError where error.code == .undeclaredPermissions
|
|
{
|
|
if let recommendedSources = UserDefaults.shared.recommendedSources, let (sourceID, sourceURL) = await $storeApp.perform({ $0.source.map { ($0.identifier, $0.sourceURL) } })
|
|
{
|
|
let normalizedSourceURL = try? sourceURL.normalized()
|
|
|
|
let isRecommended = recommendedSources.contains { $0.identifier == sourceID || (try? $0.sourceURL?.normalized()) == normalizedSourceURL }
|
|
guard !isRecommended else {
|
|
// Don't enforce permission checking for Recommended Sources for now.
|
|
return localPermissions
|
|
}
|
|
}
|
|
|
|
throw error
|
|
}
|
|
|
|
return localPermissions
|
|
}
|
|
|
|
@MainActor @available(iOS 15, *)
|
|
func review(_ permissions: [ALTEntitlement], for app: AppProtocol, mode: PermissionReviewMode, presentingViewController: UIViewController) async throws
|
|
{
|
|
let reviewPermissionsViewController = ReviewPermissionsViewController(app: app, permissions: permissions, mode: mode)
|
|
let navigationController = UINavigationController(rootViewController: reviewPermissionsViewController)
|
|
|
|
defer {
|
|
navigationController.dismiss(animated: true)
|
|
}
|
|
|
|
try await withCheckedThrowingContinuation { continuation in
|
|
reviewPermissionsViewController.completionHandler = { result in
|
|
continuation.resume(with: result)
|
|
}
|
|
|
|
presentingViewController.present(navigationController, animated: true)
|
|
}
|
|
}
|
|
}
|