import ClawdbotProtocol import Foundation import Observation import OSLog @MainActor @Observable final class CronJobsStore { static let shared = CronJobsStore() var jobs: [CronJob] = [] var selectedJobId: String? var runEntries: [CronRunLogEntry] = [] var schedulerEnabled: Bool? var schedulerStorePath: String? var schedulerNextWakeAtMs: Int? var isLoadingJobs = false var isLoadingRuns = false var lastError: String? var statusMessage: String? private let logger = Logger(subsystem: "com.clawdbot", category: "cron.ui") private var refreshTask: Task? private var runsTask: Task? private var eventTask: Task? private var pollTask: Task? private let interval: TimeInterval = 30 private let isPreview: Bool init(isPreview: Bool = ProcessInfo.processInfo.isPreview) { self.isPreview = isPreview } func start() { guard !self.isPreview else { return } guard self.eventTask == nil else { return } self.startGatewaySubscription() self.pollTask = Task.detached { [weak self] in guard let self else { return } await self.refreshJobs() while !Task.isCancelled { try? await Task.sleep(nanoseconds: UInt64(self.interval * 1_000_000_000)) await self.refreshJobs() } } } func stop() { self.refreshTask?.cancel() self.refreshTask = nil self.runsTask?.cancel() self.runsTask = nil self.eventTask?.cancel() self.eventTask = nil self.pollTask?.cancel() self.pollTask = nil } func refreshJobs() async { guard !self.isLoadingJobs else { return } self.isLoadingJobs = true self.lastError = nil self.statusMessage = nil defer { self.isLoadingJobs = false } do { if let status = try? await GatewayConnection.shared.cronStatus() { self.schedulerEnabled = status.enabled self.schedulerStorePath = status.storePath self.schedulerNextWakeAtMs = status.nextWakeAtMs } self.jobs = try await GatewayConnection.shared.cronList(includeDisabled: true) if self.jobs.isEmpty { self.statusMessage = "No cron jobs yet." } } catch { self.logger.error("cron.list failed \(error.localizedDescription, privacy: .public)") self.lastError = error.localizedDescription } } func refreshRuns(jobId: String, limit: Int = 200) async { guard !self.isLoadingRuns else { return } self.isLoadingRuns = true defer { self.isLoadingRuns = false } do { self.runEntries = try await GatewayConnection.shared.cronRuns(jobId: jobId, limit: limit) } catch { self.logger.error("cron.runs failed \(error.localizedDescription, privacy: .public)") self.lastError = error.localizedDescription } } func runJob(id: String, force: Bool = true) async { do { try await GatewayConnection.shared.cronRun(jobId: id, force: force) } catch { self.lastError = error.localizedDescription } } func removeJob(id: String) async { do { try await GatewayConnection.shared.cronRemove(jobId: id) await self.refreshJobs() if self.selectedJobId == id { self.selectedJobId = nil self.runEntries = [] } } catch { self.lastError = error.localizedDescription } } func setJobEnabled(id: String, enabled: Bool) async { do { try await GatewayConnection.shared.cronUpdate( jobId: id, patch: ["enabled": AnyCodable(enabled)]) await self.refreshJobs() } catch { self.lastError = error.localizedDescription } } func upsertJob( id: String?, payload: [String: AnyCodable]) async throws { if let id { try await GatewayConnection.shared.cronUpdate(jobId: id, patch: payload) } else { try await GatewayConnection.shared.cronAdd(payload: payload) } await self.refreshJobs() } // MARK: - Gateway events private func startGatewaySubscription() { self.eventTask?.cancel() self.eventTask = Task { [weak self] in guard let self else { return } let stream = await GatewayConnection.shared.subscribe() for await push in stream { if Task.isCancelled { return } await MainActor.run { [weak self] in self?.handle(push: push) } } } } private func handle(push: GatewayPush) { switch push { case let .event(evt) where evt.event == "cron": guard let payload = evt.payload else { return } if let cronEvt = try? GatewayPayloadDecoding.decode(payload, as: CronEvent.self) { self.handle(cronEvent: cronEvt) } case .seqGap: self.scheduleRefresh() default: break } } private func handle(cronEvent evt: CronEvent) { // Keep UI in sync with the gateway scheduler. self.scheduleRefresh(delayMs: 250) if evt.action == "finished", let selected = self.selectedJobId, selected == evt.jobId { self.scheduleRunsRefresh(jobId: selected, delayMs: 200) } } private func scheduleRefresh(delayMs: Int = 250) { self.refreshTask?.cancel() self.refreshTask = Task { [weak self] in guard let self else { return } try? await Task.sleep(nanoseconds: UInt64(delayMs) * 1_000_000) await self.refreshJobs() } } private func scheduleRunsRefresh(jobId: String, delayMs: Int = 200) { self.runsTask?.cancel() self.runsTask = Task { [weak self] in guard let self else { return } try? await Task.sleep(nanoseconds: UInt64(delayMs) * 1_000_000) await self.refreshRuns(jobId: jobId) } } // MARK: - (no additional RPC helpers) }