macsync/@packages/ireminders/Sources/IReminderSync/SyncManager.swift

169 lines
5.4 KiB
Swift

import AppKit
import EventKit
import Foundation
import LilithLogging
import MacSyncShared
private let log = AppLogger.logger(for: "IReminder.Sync")
// MARK: - Stats
public struct IReminderSyncStats: Equatable, Sendable {
public var calendarCount: Int = 0
public var reminderCount: Int = 0
public init() {}
}
// MARK: - Sync Error
public enum IReminderSyncError: Equatable, Sendable {
case none
case reminderAccessRequired
case backendUnreachable
case connectionFailed(String)
public var message: String {
switch self {
case .none: return ""
case .reminderAccessRequired: return "Reminders access required"
case .backendUnreachable: return "Cannot connect to backend server"
case .connectionFailed(let m): return m
}
}
public var isConnectionError: Bool {
switch self {
case .backendUnreachable, .connectionFailed: return true
default: return false
}
}
}
// MARK: - SyncManager
@MainActor
public final class SyncManager: BaseSyncManager<IReminderSyncStats, IReminderSyncError> {
public static let shared = SyncManager()
public let reader = ReminderReader.shared
private let apiClient = APIClient.shared
private let reminderBatchSize = 200
private lazy var sendQueueClient: SendQueueClient<ReminderSendTransport> = {
let transport = ReminderSendTransport(apiClient: apiClient)
let sender = ReminderSender(eventStore: reader.eventStore)
return SendQueueClient(label: "ireminders", transport: transport, interval: 60) { item in
await sender.apply(item)
}
}()
private init() {
super.init(
initialStats: IReminderSyncStats(),
noError: .none,
persistenceKey: "ireminders",
timerInterval: 300
)
}
// MARK: - Authorization hooks
public override func isAuthorized() async -> Bool { reader.isAuthorized }
public override func requestAuthorization() async -> Bool {
await reader.requestAuthorization()
}
public override func onAuthorizationDenied() {
log.warning("Reminders access denied")
syncError = .reminderAccessRequired
}
// MARK: - Send-queue lifecycle
public override func didStartSync() {
sendQueueClient.start()
}
public override func willStopSync() {
sendQueueClient.stop()
}
public func openReminderAccessSettings() {
if let url = URL(string: "x-apple.systempreferences:com.apple.preference.security?Privacy_Reminders") {
NSWorkspace.shared.open(url)
}
}
// MARK: - Sync cycle
public override func performSync() async {
log.info("performSync starting")
let df = ISO8601DateFormatter()
// Phase 1: Reminder lists (calendars). Server stats currently only
// surfaces totalReminders; reminder-list payloads are computed for
// local stats display.
currentOperation = "Fetching reminder lists…"
let calendars = reader.fetchCalendars()
log.info("Fetched \(calendars.count) reminder lists")
stats.calendarCount = calendars.count
// Phase 2: Reminders
currentOperation = "Fetching reminders…"
let reminders = await reader.fetchReminders(since: lastSync)
log.info("Fetched \(reminders.count) reminders")
if !reminders.isEmpty {
let batches = stride(from: 0, to: reminders.count, by: reminderBatchSize).map {
Array(reminders[$0..<min($0 + reminderBatchSize, reminders.count)])
}
for (idx, batch) in batches.enumerated() {
currentOperation = "Syncing reminders… (\(idx + 1)/\(batches.count))"
let payloads = batch.map { r in
SyncReminderPayload(
reminderIdentifier: r.reminderIdentifier,
calendarIdentifier: r.calendarIdentifier,
title: r.title,
notes: r.notes,
dueDate: r.dueDate.map { df.string(from: $0) },
priority: r.priority,
isCompleted: r.isCompleted,
modifiedAt: r.lastModifiedDate.map { df.string(from: $0) }
)
}
do {
let synced = try await apiClient.syncReminders(payloads)
log.info("Batch \(idx + 1)/\(batches.count) synced=\(synced)")
if syncError.isConnectionError { syncError = .none }
} catch {
log.warning("Reminder batch \(idx + 1) failed: \(error.localizedDescription)")
setConnectionError(error)
}
}
}
await updateStats()
currentOperation = "Sync complete"
log.info("performSync complete lists=\(calendars.count) reminders=\(reminders.count)")
}
private func updateStats() async {
do {
let response = try await apiClient.getStats()
stats.reminderCount = response.totalReminders
log.info("Stats: reminders=\(response.totalReminders)")
} catch {
log.warning("updateStats failed: \(error.localizedDescription)")
}
}
private func setConnectionError(_ error: Error) {
if SyncConnectionErrorHeuristic.isConnectionError(error) {
syncError = .backendUnreachable
}
}
}