deep-linking-sample/Apps/Wikipedia/WMF Framework/CacheItemMigrationPolicy.swift
Javier Cicchelli 9bcdaa697b [Setup] Basic project structure (#1)
This PR contains all the work related to setting up this project as required to implement the [Assignment](https://repo.rock-n-code.com/rock-n-code/deep-linking-assignment/wiki/Assignment) on top, as intended.

To summarise this work:
- [x] created a new **Xcode** project;
- [x] cloned the `Wikipedia` app and inserted it into the **Xcode** project;
- [x] created the `Locations` app and also, its `Libraries` package;
- [x] created the `Shared` package to share dependencies between the apps;
- [x] added a `Makefile` file and implemented some **environment** and **help** commands.

Co-authored-by: Javier Cicchelli <javier@rock-n-code.com>
Reviewed-on: rock-n-code/deep-linking-assignment#1
2023-04-08 18:37:13 +00:00

63 lines
2.9 KiB
Swift

import Foundation
enum CacheItemMigrationPolicyError: Error {
case unrecognizedSourceAttributeTypes
}
class CacheItemMigrationPolicy: NSEntityMigrationPolicy {
private let fetcher = ImageFetcher()
override func createDestinationInstances(forSource sInstance: NSManagedObject, in mapping: NSEntityMapping, manager: NSMigrationManager) throws {
if sInstance.entity.name == "CacheItem" {
guard let key = sInstance.primitiveValue(forKey: "key") as? String,
let variant = sInstance.primitiveValue(forKey: "variant") as? Int64,
let date = sInstance.primitiveValue(forKey: "date") as? Date else {
throw CacheItemMigrationPolicyError.unrecognizedSourceAttributeTypes
}
let destinationItem = NSEntityDescription.insertNewObject(forEntityName: "CacheItem", into: manager.destinationContext)
destinationItem.setValue(key, forKey: "key")
let newVariant = String(variant)
destinationItem.setValue(newVariant, forKey: "variant")
destinationItem.setValue(date, forKey: "date")
var isDownloaded = false
autoreleasepool { () -> Void in
guard
let fileName = fetcher.uniqueFileNameForItemKey(key, variant: newVariant),
let headerFileName = fetcher.uniqueHeaderFileNameForItemKey(key, variant: newVariant) else {
return
}
let fileURL = CacheFileWriterHelper.fileURL(for: fileName)
let filePath = fileURL.path
// artifically create and save image response header
var headers: [String: String] = [:]
headers["Content-Type"] = FileManager.default.getValueForExtendedFileAttributeNamed(WMFExtendedFileAttributeNameMIMEType, forFileAtPath: filePath)
let values = try? fileURL.resourceValues(forKeys: [URLResourceKey.fileSizeKey])
if let fileSize = values?.fileSize {
headers["Content-Length"] = String(fileSize)
}
guard !headers.isEmpty else {
return
}
CacheFileWriterHelper.saveResponseHeader(headerFields: headers, toNewFileName: headerFileName) { (result) in
switch result {
case .success, .exists:
isDownloaded = true
case .failure:
break
}
}
}
destinationItem.setValue(isDownloaded, forKey: "isDownloaded")
destinationItem.setValue(nil, forKey: "url")
manager.associate(sourceInstance: sInstance, withDestinationInstance: destinationItem, for: mapping)
}
}
}