Files
SideStore/SideStoreApp/Sources/SideStoreAppKit/Operations/FetchSourceOperation.swift

97 lines
3.5 KiB
Swift
Raw Normal View History

//
// FetchSourceOperation.swift
// AltStore
//
// Created by Riley Testut on 7/30/19.
// Copyright © 2019 Riley Testut. All rights reserved.
//
import CoreData
2023-03-01 00:48:36 -05:00
import Foundation
2023-03-01 00:48:36 -05:00
import SideStoreCore
2023-03-01 14:36:52 -05:00
import RoxasUIKit
@objc(FetchSourceOperation)
2023-03-01 00:48:36 -05:00
final class FetchSourceOperation: ResultOperation<Source> {
let sourceURL: URL
let managedObjectContext: NSManagedObjectContext
2023-03-01 00:48:36 -05:00
private let session: URLSession
2023-03-01 00:48:36 -05:00
2019-09-07 15:37:08 -07:00
private lazy var dateFormatter: ISO8601DateFormatter = {
let dateFormatter = ISO8601DateFormatter()
return dateFormatter
}()
2023-03-01 00:48:36 -05:00
init(sourceURL: URL, managedObjectContext: NSManagedObjectContext = DatabaseManager.shared.persistentContainer.newBackgroundContext()) {
self.sourceURL = sourceURL
self.managedObjectContext = managedObjectContext
2023-03-01 00:48:36 -05:00
let configuration = URLSessionConfiguration.default
configuration.requestCachePolicy = .reloadIgnoringLocalCacheData
configuration.urlCache = nil
2023-03-01 00:48:36 -05:00
session = URLSession(configuration: configuration)
}
2023-03-01 00:48:36 -05:00
override func main() {
super.main()
2023-03-01 00:48:36 -05:00
let dataTask = session.dataTask(with: sourceURL) { data, response, error in
let childContext = DatabaseManager.shared.persistentContainer.newBackgroundContext(withParent: self.managedObjectContext)
childContext.mergePolicy = NSOverwriteMergePolicy
childContext.perform {
2023-03-01 00:48:36 -05:00
do {
let (data, _) = try Result((data, response), error).get()
2023-03-01 00:48:36 -05:00
2023-03-01 14:36:52 -05:00
let decoder = SideStoreCore.JSONDecoder()
2023-03-01 00:48:36 -05:00
decoder.dateDecodingStrategy = .custom { decoder -> Date in
2019-09-07 15:37:08 -07:00
let container = try decoder.singleValueContainer()
let text = try container.decode(String.self)
2023-03-01 00:48:36 -05:00
2019-09-07 15:37:08 -07:00
// Full ISO8601 Format.
self.dateFormatter.formatOptions = [.withFullDate, .withFullTime, .withTimeZone]
2023-03-01 00:48:36 -05:00
if let date = self.dateFormatter.date(from: text) {
2019-09-07 15:37:08 -07:00
return date
}
2023-03-01 00:48:36 -05:00
2019-09-07 15:37:08 -07:00
// Just date portion of ISO8601.
self.dateFormatter.formatOptions = [.withFullDate]
2023-03-01 00:48:36 -05:00
if let date = self.dateFormatter.date(from: text) {
2019-09-07 15:37:08 -07:00
return date
}
2023-03-01 00:48:36 -05:00
2019-09-07 15:37:08 -07:00
throw DecodingError.dataCorruptedError(in: container, debugDescription: "Date is in invalid format.")
2023-03-01 00:48:36 -05:00
}
decoder.managedObjectContext = childContext
decoder.sourceURL = self.sourceURL
2023-03-01 00:48:36 -05:00
let source = try decoder.decode(Source.self, from: data)
let identifier = source.identifier
2023-03-01 00:48:36 -05:00
try childContext.save()
2023-03-01 00:48:36 -05:00
self.managedObjectContext.perform {
2023-03-01 00:48:36 -05:00
if let source = Source.first(satisfying: NSPredicate(format: "%K == %@", #keyPath(Source.identifier), identifier), in: self.managedObjectContext) {
self.finish(.success(source))
2023-03-01 00:48:36 -05:00
} else {
self.finish(.failure(OperationError.noSources))
}
}
2023-03-01 00:48:36 -05:00
} catch {
self.managedObjectContext.perform {
self.finish(.failure(error))
}
}
}
}
2023-03-01 00:48:36 -05:00
progress.addChild(dataTask.progress, withPendingUnitCount: 1)
dataTask.resume()
}
}