2019-06-17 14:49:23 -07:00
|
|
|
//
|
2019-07-30 17:00:04 -07:00
|
|
|
// FetchSourceOperation.swift
|
2019-06-17 14:49:23 -07:00
|
|
|
// AltStore
|
|
|
|
|
//
|
2019-07-30 17:00:04 -07:00
|
|
|
// Created by Riley Testut on 7/30/19.
|
2019-06-17 14:49:23 -07:00
|
|
|
// Copyright © 2019 Riley Testut. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
import Foundation
|
2020-03-24 13:27:44 -07:00
|
|
|
import CoreData
|
|
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
import AltStoreCore
|
2019-06-17 14:49:23 -07:00
|
|
|
import Roxas
|
|
|
|
|
|
2022-11-22 13:02:19 -06:00
|
|
|
extension SourceError
|
|
|
|
|
{
|
|
|
|
|
enum Code: Int, ALTErrorCode
|
|
|
|
|
{
|
|
|
|
|
typealias Error = SourceError
|
|
|
|
|
|
|
|
|
|
case unsupported
|
|
|
|
|
case duplicateBundleID
|
2022-11-23 19:14:20 -06:00
|
|
|
case duplicateVersion
|
2022-11-22 13:02:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static func unsupported(_ source: Source) -> SourceError { SourceError(code: .unsupported, source: source) }
|
2022-11-23 19:14:20 -06:00
|
|
|
static func duplicateBundleID(_ bundleID: String, source: Source) -> SourceError { SourceError(code: .duplicateBundleID, source: source, bundleID: bundleID) }
|
|
|
|
|
static func duplicateVersion(_ version: String, for app: StoreApp, source: Source) -> SourceError { SourceError(code: .duplicateVersion, source: source, app: app, version: version) }
|
2022-11-22 13:02:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct SourceError: ALTLocalizedError
|
|
|
|
|
{
|
2022-11-23 19:14:20 -06:00
|
|
|
let code: Code
|
2022-11-22 13:02:19 -06:00
|
|
|
var errorTitle: String?
|
|
|
|
|
var errorFailure: String?
|
|
|
|
|
|
|
|
|
|
@Managed var source: Source
|
2022-11-23 19:14:20 -06:00
|
|
|
@Managed var app: StoreApp?
|
|
|
|
|
var bundleID: String?
|
|
|
|
|
var version: String?
|
2022-11-22 13:02:19 -06:00
|
|
|
|
|
|
|
|
var errorFailureReason: String {
|
|
|
|
|
switch self.code
|
|
|
|
|
{
|
|
|
|
|
case .unsupported: return String(format: NSLocalizedString("The source “%@” is not supported by this version of AltStore.", comment: ""), self.$source.name)
|
|
|
|
|
case .duplicateBundleID:
|
2022-11-23 19:14:20 -06:00
|
|
|
let bundleIDFragment = self.bundleID.map { String(format: NSLocalizedString("the bundle identifier %@", comment: ""), $0) } ?? NSLocalizedString("the same bundle identifier", comment: "")
|
2022-11-22 13:02:19 -06:00
|
|
|
let failureReason = String(format: NSLocalizedString("The source “%@” contains multiple apps with %@.", comment: ""), self.$source.name, bundleIDFragment)
|
|
|
|
|
return failureReason
|
2022-11-23 19:14:20 -06:00
|
|
|
|
|
|
|
|
case .duplicateVersion:
|
|
|
|
|
var versionFragment = NSLocalizedString("duplicate versions", comment: "")
|
|
|
|
|
if let version
|
|
|
|
|
{
|
|
|
|
|
versionFragment += " (\(version))"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let appFragment: String
|
|
|
|
|
if let name = self.$app.name, let bundleID = self.$app.bundleIdentifier
|
|
|
|
|
{
|
|
|
|
|
appFragment = name + " (\(bundleID))"
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
appFragment = NSLocalizedString("one or more apps", comment: "")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let failureReason = String(format: NSLocalizedString("The source “%@” contains %@ for %@.", comment: ""), self.$source.name, versionFragment, appFragment)
|
|
|
|
|
return failureReason
|
2022-11-22 13:02:19 -06:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-30 17:00:04 -07:00
|
|
|
@objc(FetchSourceOperation)
|
2023-01-04 09:52:12 -05:00
|
|
|
final class FetchSourceOperation: ResultOperation<Source>
|
2019-06-17 14:49:23 -07:00
|
|
|
{
|
2019-07-30 17:00:04 -07:00
|
|
|
let sourceURL: URL
|
2020-03-24 13:27:44 -07:00
|
|
|
let managedObjectContext: NSManagedObjectContext
|
2019-07-30 17:00:04 -07:00
|
|
|
|
2019-09-19 11:27:38 -07:00
|
|
|
private let session: URLSession
|
2019-06-17 14:49:23 -07:00
|
|
|
|
2019-09-07 15:37:08 -07:00
|
|
|
private lazy var dateFormatter: ISO8601DateFormatter = {
|
|
|
|
|
let dateFormatter = ISO8601DateFormatter()
|
2019-06-17 14:49:23 -07:00
|
|
|
return dateFormatter
|
|
|
|
|
}()
|
|
|
|
|
|
2020-03-24 13:27:44 -07:00
|
|
|
init(sourceURL: URL, managedObjectContext: NSManagedObjectContext = DatabaseManager.shared.persistentContainer.newBackgroundContext())
|
2019-07-30 17:00:04 -07:00
|
|
|
{
|
|
|
|
|
self.sourceURL = sourceURL
|
2020-03-24 13:27:44 -07:00
|
|
|
self.managedObjectContext = managedObjectContext
|
2019-09-19 11:27:38 -07:00
|
|
|
|
|
|
|
|
let configuration = URLSessionConfiguration.default
|
|
|
|
|
configuration.requestCachePolicy = .reloadIgnoringLocalCacheData
|
|
|
|
|
configuration.urlCache = nil
|
|
|
|
|
|
|
|
|
|
self.session = URLSession(configuration: configuration)
|
2019-07-30 17:00:04 -07:00
|
|
|
}
|
|
|
|
|
|
2019-06-17 14:49:23 -07:00
|
|
|
override func main()
|
|
|
|
|
{
|
|
|
|
|
super.main()
|
|
|
|
|
|
2019-07-30 17:00:04 -07:00
|
|
|
let dataTask = self.session.dataTask(with: self.sourceURL) { (data, response, error) in
|
2020-08-27 16:23:50 -07:00
|
|
|
|
|
|
|
|
let childContext = DatabaseManager.shared.persistentContainer.newBackgroundContext(withParent: self.managedObjectContext)
|
|
|
|
|
childContext.mergePolicy = NSOverwriteMergePolicy
|
|
|
|
|
childContext.perform {
|
2019-06-17 14:49:23 -07:00
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
let (data, _) = try Result((data, response), error).get()
|
|
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
let decoder = AltStoreCore.JSONDecoder()
|
2019-09-07 15:37:08 -07:00
|
|
|
decoder.dateDecodingStrategy = .custom({ (decoder) -> Date in
|
|
|
|
|
let container = try decoder.singleValueContainer()
|
|
|
|
|
let text = try container.decode(String.self)
|
|
|
|
|
|
|
|
|
|
// Full ISO8601 Format.
|
|
|
|
|
self.dateFormatter.formatOptions = [.withFullDate, .withFullTime, .withTimeZone]
|
|
|
|
|
if let date = self.dateFormatter.date(from: text)
|
|
|
|
|
{
|
|
|
|
|
return date
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Just date portion of ISO8601.
|
|
|
|
|
self.dateFormatter.formatOptions = [.withFullDate]
|
|
|
|
|
if let date = self.dateFormatter.date(from: text)
|
|
|
|
|
{
|
|
|
|
|
return date
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw DecodingError.dataCorruptedError(in: container, debugDescription: "Date is in invalid format.")
|
|
|
|
|
})
|
|
|
|
|
|
2020-08-27 16:23:50 -07:00
|
|
|
decoder.managedObjectContext = childContext
|
2020-03-24 13:27:44 -07:00
|
|
|
decoder.sourceURL = self.sourceURL
|
2019-06-17 14:49:23 -07:00
|
|
|
|
2019-07-30 17:00:04 -07:00
|
|
|
let source = try decoder.decode(Source.self, from: data)
|
2020-08-27 16:23:50 -07:00
|
|
|
let identifier = source.identifier
|
2019-11-04 13:38:54 -08:00
|
|
|
|
2022-11-22 13:02:19 -06:00
|
|
|
try self.verify(source)
|
|
|
|
|
|
2020-08-27 16:23:50 -07:00
|
|
|
try childContext.save()
|
|
|
|
|
|
|
|
|
|
self.managedObjectContext.perform {
|
|
|
|
|
if let source = Source.first(satisfying: NSPredicate(format: "%K == %@", #keyPath(Source.identifier), identifier), in: self.managedObjectContext)
|
|
|
|
|
{
|
|
|
|
|
self.finish(.success(source))
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
self.finish(.failure(OperationError.noSources))
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-06-17 14:49:23 -07:00
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
2020-08-27 16:23:50 -07:00
|
|
|
self.managedObjectContext.perform {
|
|
|
|
|
self.finish(.failure(error))
|
|
|
|
|
}
|
2019-06-17 14:49:23 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
self.progress.addChild(dataTask.progress, withPendingUnitCount: 1)
|
|
|
|
|
|
|
|
|
|
dataTask.resume()
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-11-22 13:02:19 -06:00
|
|
|
|
|
|
|
|
private extension FetchSourceOperation
|
|
|
|
|
{
|
|
|
|
|
func verify(_ source: Source) throws
|
|
|
|
|
{
|
|
|
|
|
#if !BETA
|
|
|
|
|
if let trustedSourceIDs = UserDefaults.shared.trustedSourceIDs
|
|
|
|
|
{
|
|
|
|
|
guard trustedSourceIDs.contains(source.identifier) || source.identifier == Source.altStoreIdentifier else { throw SourceError(code: .unsupported, source: source) }
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
var bundleIDs = Set<String>()
|
|
|
|
|
for app in source.apps
|
|
|
|
|
{
|
|
|
|
|
guard !bundleIDs.contains(app.bundleIdentifier) else { throw SourceError.duplicateBundleID(app.bundleIdentifier, source: source) }
|
|
|
|
|
bundleIDs.insert(app.bundleIdentifier)
|
2022-11-23 19:14:20 -06:00
|
|
|
|
|
|
|
|
var versions = Set<String>()
|
|
|
|
|
for version in app.versions
|
|
|
|
|
{
|
|
|
|
|
guard !versions.contains(version.version) else { throw SourceError.duplicateVersion(version.version, for: app, source: source) }
|
|
|
|
|
versions.insert(version.version)
|
|
|
|
|
}
|
2022-11-22 13:02:19 -06:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|