mirror of
https://github.com/immich-app/immich.git
synced 2025-11-02 10:37:11 -05:00
* fix: local sync task never runs on iOS * chore: rename ios register method * refactor from using dart callback to dart entrypoint + more logs * check if file exists before hashing * reschedule local sync task * chore: rename background worker logger * refactor: move file exists check inside repo --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
162 lines
5.9 KiB
Swift
162 lines
5.9 KiB
Swift
import BackgroundTasks
|
|
|
|
class BackgroundWorkerApiImpl: BackgroundWorkerFgHostApi {
|
|
func enableSyncWorker() throws {
|
|
BackgroundWorkerApiImpl.scheduleLocalSync()
|
|
print("BackgroundUploadImpl:enableSyncWorker Local Sync worker scheduled")
|
|
}
|
|
|
|
func enableUploadWorker() throws {
|
|
BackgroundWorkerApiImpl.updateUploadEnabled(true)
|
|
|
|
BackgroundWorkerApiImpl.scheduleRefreshUpload()
|
|
BackgroundWorkerApiImpl.scheduleProcessingUpload()
|
|
print("BackgroundUploadImpl:enableUploadWorker Scheduled background upload tasks")
|
|
}
|
|
|
|
func disableUploadWorker() throws {
|
|
BackgroundWorkerApiImpl.updateUploadEnabled(false)
|
|
BackgroundWorkerApiImpl.cancelUploadTasks()
|
|
print("BackgroundUploadImpl:disableUploadWorker Disabled background upload tasks")
|
|
}
|
|
|
|
public static let backgroundUploadEnabledKey = "immich:background:backup:enabled"
|
|
|
|
private static let localSyncTaskID = "app.alextran.immich.background.localSync"
|
|
private static let refreshUploadTaskID = "app.alextran.immich.background.refreshUpload"
|
|
private static let processingUploadTaskID = "app.alextran.immich.background.processingUpload"
|
|
|
|
private static func updateUploadEnabled(_ isEnabled: Bool) {
|
|
return UserDefaults.standard.set(isEnabled, forKey: BackgroundWorkerApiImpl.backgroundUploadEnabledKey)
|
|
}
|
|
|
|
private static func cancelUploadTasks() {
|
|
BackgroundWorkerApiImpl.updateUploadEnabled(false)
|
|
BGTaskScheduler.shared.cancel(taskRequestWithIdentifier: refreshUploadTaskID);
|
|
BGTaskScheduler.shared.cancel(taskRequestWithIdentifier: processingUploadTaskID);
|
|
}
|
|
|
|
public static func registerBackgroundWorkers() {
|
|
BGTaskScheduler.shared.register(
|
|
forTaskWithIdentifier: processingUploadTaskID, using: nil) { task in
|
|
if task is BGProcessingTask {
|
|
handleBackgroundProcessing(task: task as! BGProcessingTask)
|
|
}
|
|
}
|
|
|
|
BGTaskScheduler.shared.register(
|
|
forTaskWithIdentifier: refreshUploadTaskID, using: nil) { task in
|
|
if task is BGAppRefreshTask {
|
|
handleBackgroundRefresh(task: task as! BGAppRefreshTask, taskType: .refreshUpload)
|
|
}
|
|
}
|
|
|
|
BGTaskScheduler.shared.register(
|
|
forTaskWithIdentifier: localSyncTaskID, using: nil) { task in
|
|
if task is BGAppRefreshTask {
|
|
handleBackgroundRefresh(task: task as! BGAppRefreshTask, taskType: .localSync)
|
|
}
|
|
}
|
|
}
|
|
|
|
private static func scheduleLocalSync() {
|
|
let backgroundRefresh = BGAppRefreshTaskRequest(identifier: localSyncTaskID)
|
|
backgroundRefresh.earliestBeginDate = Date(timeIntervalSinceNow: 5 * 60) // 5 mins
|
|
|
|
do {
|
|
try BGTaskScheduler.shared.submit(backgroundRefresh)
|
|
} catch {
|
|
print("Could not schedule the local sync task \(error.localizedDescription)")
|
|
}
|
|
}
|
|
|
|
private static func scheduleRefreshUpload() {
|
|
let backgroundRefresh = BGAppRefreshTaskRequest(identifier: refreshUploadTaskID)
|
|
backgroundRefresh.earliestBeginDate = Date(timeIntervalSinceNow: 5 * 60) // 5 mins
|
|
|
|
do {
|
|
try BGTaskScheduler.shared.submit(backgroundRefresh)
|
|
} catch {
|
|
print("Could not schedule the refresh upload task \(error.localizedDescription)")
|
|
}
|
|
}
|
|
|
|
private static func scheduleProcessingUpload() {
|
|
let backgroundProcessing = BGProcessingTaskRequest(identifier: processingUploadTaskID)
|
|
|
|
backgroundProcessing.requiresNetworkConnectivity = true
|
|
backgroundProcessing.earliestBeginDate = Date(timeIntervalSinceNow: 15 * 60) // 15 mins
|
|
|
|
do {
|
|
try BGTaskScheduler.shared.submit(backgroundProcessing)
|
|
} catch {
|
|
print("Could not schedule the processing upload task \(error.localizedDescription)")
|
|
}
|
|
}
|
|
|
|
private static func handleBackgroundRefresh(task: BGAppRefreshTask, taskType: BackgroundTaskType) {
|
|
let maxSeconds: Int?
|
|
|
|
switch taskType {
|
|
case .localSync:
|
|
maxSeconds = 15
|
|
scheduleLocalSync()
|
|
case .refreshUpload:
|
|
maxSeconds = 20
|
|
scheduleRefreshUpload()
|
|
case .processingUpload:
|
|
print("Unexpected background refresh task encountered")
|
|
return;
|
|
}
|
|
|
|
// Restrict the refresh task to run only for a maximum of (maxSeconds) seconds
|
|
runBackgroundWorker(task: task, taskType: taskType, maxSeconds: maxSeconds)
|
|
}
|
|
|
|
private static func handleBackgroundProcessing(task: BGProcessingTask) {
|
|
scheduleProcessingUpload()
|
|
// There are no restrictions for processing tasks. Although, the OS could signal expiration at any time
|
|
runBackgroundWorker(task: task, taskType: .processingUpload, maxSeconds: nil)
|
|
}
|
|
|
|
/**
|
|
* Executes the background worker within the context of a background task.
|
|
* This method creates a BackgroundWorker, sets up task expiration handling,
|
|
* and manages the synchronization between the background task and the Flutter engine.
|
|
*
|
|
* - Parameters:
|
|
* - task: The iOS background task that provides the execution context
|
|
* - taskType: The type of background operation to perform (refresh or processing)
|
|
* - maxSeconds: Optional timeout for the operation in seconds
|
|
*/
|
|
private static func runBackgroundWorker(task: BGTask, taskType: BackgroundTaskType, maxSeconds: Int?) {
|
|
let semaphore = DispatchSemaphore(value: 0)
|
|
var isSuccess = true
|
|
|
|
let backgroundWorker = BackgroundWorker(taskType: taskType, maxSeconds: maxSeconds) { success in
|
|
isSuccess = success
|
|
semaphore.signal()
|
|
}
|
|
|
|
task.expirationHandler = {
|
|
DispatchQueue.main.async {
|
|
backgroundWorker.close()
|
|
}
|
|
isSuccess = false
|
|
|
|
// Schedule a timer to signal the semaphore after 2 seconds
|
|
Timer.scheduledTimer(withTimeInterval: 2, repeats: false) { _ in
|
|
semaphore.signal()
|
|
}
|
|
}
|
|
|
|
DispatchQueue.main.async {
|
|
backgroundWorker.run()
|
|
}
|
|
|
|
semaphore.wait()
|
|
task.setTaskCompleted(success: isSuccess)
|
|
print("Background task completed with success: \(isSuccess)")
|
|
}
|
|
}
|