feat: 添加自定义提醒功能并优化项目配置 - 添加 CustomReminder 模型支持自由文案周期性提醒功能 - 实现自定义提醒的 UI 界面,包括新建、编辑和列表展示 - 集成本地通知服务支持自定义提醒的时间触发 - 更新项目配置文件添加应用显示名称和加密声明 - 修正 iOS 部署目标版本从 26.0 到 17.0 - 修复 FileDownloader 中的线程安全问题 - 优化 ModelManifest 和 Localization 的并发安全性 - 扩展本地化字符串支持多语言提醒相关文本 - 调整项目支持平台范围仅保留 iphoneos 和 iphonesimulator ```
159 lines
5.5 KiB
Swift
159 lines
5.5 KiB
Swift
import Foundation
|
|
|
|
enum DownloadError: Error, LocalizedError {
|
|
case badStatus(Int)
|
|
case sizeMismatch(expected: Int, got: Int)
|
|
|
|
var errorDescription: String? {
|
|
switch self {
|
|
case .badStatus(let code):
|
|
return String(appLoc: "下载失败(HTTP \(code))")
|
|
case .sizeMismatch(let expected, let got):
|
|
return String(appLoc: "文件大小校验失败(预期 \(expected),实际 \(got))")
|
|
}
|
|
}
|
|
}
|
|
|
|
/// 下载单个文件,支持 HTTP Range 断点续传 + 完成后大小校验。
|
|
/// 用 `URLSessionDataDelegate` 把响应体分块写入 `.part`,完成后原子改名为成品。
|
|
///
|
|
/// 注意:文件大小一律用 `FileManager.attributesOfItem` 读取,**不用**
|
|
/// `URL.resourceValues(.fileSizeKey)` —— 后者会把结果缓存在 URL 实例上,
|
|
/// 续传时先读 offset 再读 finalSize 会拿到下载前的陈旧大小,导致误判校验失败。
|
|
///
|
|
/// 一个实例一次处理一个文件(串行)。共享状态用锁保证可见性。
|
|
final class FileDownloader: NSObject, URLSessionDataDelegate, @unchecked Sendable {
|
|
private let configuration: URLSessionConfiguration
|
|
|
|
private let lock = NSLock()
|
|
private var handle: FileHandle?
|
|
private var written: Int = 0
|
|
private var onProgress: ((Int) -> Void)?
|
|
private var responseError: Error?
|
|
private var continuation: CheckedContinuation<Void, Error>?
|
|
|
|
init(configuration: URLSessionConfiguration = .default) {
|
|
self.configuration = configuration
|
|
super.init()
|
|
}
|
|
|
|
/// 不走 URL 资源值缓存的文件大小读取。
|
|
static func fileSize(at url: URL) -> Int {
|
|
guard let attrs = try? FileManager.default.attributesOfItem(atPath: url.path),
|
|
let size = attrs[.size] as? Int else { return 0 }
|
|
return size
|
|
}
|
|
|
|
/// 从 `url` 下载到 `destination`。若存在 `destination.part` 则发 Range 请求续传;
|
|
/// 完成后校验总大小 == `expectedBytes`,通过则原子改名为 `destination`。
|
|
nonisolated func download(
|
|
from url: URL,
|
|
to destination: URL,
|
|
expectedBytes: Int,
|
|
onProgress: (@Sendable (Int) -> Void)? = nil
|
|
) async throws {
|
|
let fm = FileManager.default
|
|
let part = destination.appendingPathExtension("part")
|
|
|
|
// 成品已存在且大小正确 → 跳过
|
|
if Self.fileSize(at: destination) == expectedBytes,
|
|
fm.fileExists(atPath: destination.path) {
|
|
return
|
|
}
|
|
|
|
try fm.createDirectory(
|
|
at: destination.deletingLastPathComponent(), withIntermediateDirectories: true)
|
|
|
|
var offset = 0
|
|
if fm.fileExists(atPath: part.path) {
|
|
offset = Self.fileSize(at: part)
|
|
} else {
|
|
fm.createFile(atPath: part.path, contents: nil)
|
|
}
|
|
|
|
let fileHandle = try FileHandle(forWritingTo: part)
|
|
try fileHandle.seekToEnd()
|
|
|
|
lock.withLock {
|
|
self.handle = fileHandle
|
|
self.written = offset
|
|
self.onProgress = onProgress
|
|
self.responseError = nil
|
|
}
|
|
|
|
var request = URLRequest(url: url)
|
|
if offset > 0 {
|
|
request.setValue("bytes=\(offset)-", forHTTPHeaderField: "Range")
|
|
}
|
|
|
|
let session = URLSession(configuration: configuration, delegate: self, delegateQueue: nil)
|
|
defer { session.finishTasksAndInvalidate() }
|
|
|
|
// 句柄在 didCompleteWithError 内关闭(同一 delegate 队列,串行于 didReceive)。
|
|
try await withCheckedThrowingContinuation { (cont: CheckedContinuation<Void, Error>) in
|
|
lock.lock()
|
|
self.continuation = cont
|
|
lock.unlock()
|
|
session.dataTask(with: request).resume()
|
|
}
|
|
|
|
let finalSize = Self.fileSize(at: part)
|
|
guard finalSize == expectedBytes else {
|
|
try? fm.removeItem(at: part)
|
|
throw DownloadError.sizeMismatch(expected: expectedBytes, got: finalSize)
|
|
}
|
|
|
|
if fm.fileExists(atPath: destination.path) {
|
|
try fm.removeItem(at: destination)
|
|
}
|
|
try fm.moveItem(at: part, to: destination)
|
|
}
|
|
|
|
// MARK: - URLSessionDataDelegate (全部在串行 delegate 队列执行)
|
|
|
|
nonisolated func urlSession(
|
|
_ session: URLSession, dataTask: URLSessionDataTask,
|
|
didReceive response: URLResponse,
|
|
completionHandler: @escaping (URLSession.ResponseDisposition) -> Void
|
|
) {
|
|
if let http = response as? HTTPURLResponse, http.statusCode >= 400 {
|
|
lock.lock(); responseError = DownloadError.badStatus(http.statusCode); lock.unlock()
|
|
completionHandler(.cancel)
|
|
} else {
|
|
completionHandler(.allow)
|
|
}
|
|
}
|
|
|
|
nonisolated func urlSession(
|
|
_ session: URLSession, dataTask: URLSessionDataTask, didReceive data: Data
|
|
) {
|
|
lock.lock()
|
|
try? handle?.write(contentsOf: data)
|
|
written += data.count
|
|
let progress = written
|
|
let callback = onProgress
|
|
lock.unlock()
|
|
callback?(progress)
|
|
}
|
|
|
|
nonisolated func urlSession(
|
|
_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?
|
|
) {
|
|
lock.lock()
|
|
try? handle?.close()
|
|
handle = nil
|
|
let cont = continuation
|
|
continuation = nil
|
|
let respErr = responseError
|
|
lock.unlock()
|
|
|
|
if let respErr {
|
|
cont?.resume(throwing: respErr)
|
|
} else if let error {
|
|
cont?.resume(throwing: error)
|
|
} else {
|
|
cont?.resume()
|
|
}
|
|
}
|
|
}
|