mirror of
https://github.com/openclaw/openclaw.git
synced 2026-02-19 18:39:20 -05:00
Merged via /review-pr -> /prepare-pr -> /merge-pr.
Prepared head SHA: f60cd10f6d
Co-authored-by: ngutman <1540134+ngutman@users.noreply.github.com>
Co-authored-by: ngutman <1540134+ngutman@users.noreply.github.com>
Reviewed-by: @ngutman
203 lines
7.0 KiB
Swift
203 lines
7.0 KiB
Swift
import OpenClawKit
|
|
import CoreLocation
|
|
import Foundation
|
|
|
|
@MainActor
|
|
final class LocationService: NSObject, CLLocationManagerDelegate {
|
|
enum Error: Swift.Error {
|
|
case timeout
|
|
case unavailable
|
|
}
|
|
|
|
private let manager = CLLocationManager()
|
|
private var authContinuation: CheckedContinuation<CLAuthorizationStatus, Never>?
|
|
private var locationContinuation: CheckedContinuation<CLLocation, Swift.Error>?
|
|
private var updatesContinuation: AsyncStream<CLLocation>.Continuation?
|
|
private var isStreaming = false
|
|
private var significantLocationCallback: (@Sendable (CLLocation) -> Void)?
|
|
private var isMonitoringSignificantChanges = false
|
|
|
|
override init() {
|
|
super.init()
|
|
self.manager.delegate = self
|
|
self.manager.desiredAccuracy = kCLLocationAccuracyBest
|
|
}
|
|
|
|
func authorizationStatus() -> CLAuthorizationStatus {
|
|
self.manager.authorizationStatus
|
|
}
|
|
|
|
func accuracyAuthorization() -> CLAccuracyAuthorization {
|
|
if #available(iOS 14.0, *) {
|
|
return self.manager.accuracyAuthorization
|
|
}
|
|
return .fullAccuracy
|
|
}
|
|
|
|
func ensureAuthorization(mode: OpenClawLocationMode) async -> CLAuthorizationStatus {
|
|
guard CLLocationManager.locationServicesEnabled() else { return .denied }
|
|
|
|
let status = self.manager.authorizationStatus
|
|
if status == .notDetermined {
|
|
self.manager.requestWhenInUseAuthorization()
|
|
let updated = await self.awaitAuthorizationChange()
|
|
if mode != .always { return updated }
|
|
}
|
|
|
|
if mode == .always {
|
|
let current = self.manager.authorizationStatus
|
|
if current == .authorizedWhenInUse {
|
|
self.manager.requestAlwaysAuthorization()
|
|
return await self.awaitAuthorizationChange()
|
|
}
|
|
return current
|
|
}
|
|
|
|
return self.manager.authorizationStatus
|
|
}
|
|
|
|
func currentLocation(
|
|
params: OpenClawLocationGetParams,
|
|
desiredAccuracy: OpenClawLocationAccuracy,
|
|
maxAgeMs: Int?,
|
|
timeoutMs: Int?) async throws -> CLLocation
|
|
{
|
|
let now = Date()
|
|
if let maxAgeMs,
|
|
let cached = self.manager.location,
|
|
now.timeIntervalSince(cached.timestamp) * 1000 <= Double(maxAgeMs)
|
|
{
|
|
return cached
|
|
}
|
|
|
|
self.manager.desiredAccuracy = Self.accuracyValue(desiredAccuracy)
|
|
let timeout = max(0, timeoutMs ?? 10000)
|
|
return try await self.withTimeout(timeoutMs: timeout) {
|
|
try await self.requestLocation()
|
|
}
|
|
}
|
|
|
|
private func requestLocation() async throws -> CLLocation {
|
|
try await withCheckedThrowingContinuation { cont in
|
|
self.locationContinuation = cont
|
|
self.manager.requestLocation()
|
|
}
|
|
}
|
|
|
|
private func awaitAuthorizationChange() async -> CLAuthorizationStatus {
|
|
await withCheckedContinuation { cont in
|
|
self.authContinuation = cont
|
|
}
|
|
}
|
|
|
|
private func withTimeout<T: Sendable>(
|
|
timeoutMs: Int,
|
|
operation: @escaping @Sendable () async throws -> T) async throws -> T
|
|
{
|
|
try await AsyncTimeout.withTimeoutMs(timeoutMs: timeoutMs, onTimeout: { Error.timeout }, operation: operation)
|
|
}
|
|
|
|
private static func accuracyValue(_ accuracy: OpenClawLocationAccuracy) -> CLLocationAccuracy {
|
|
switch accuracy {
|
|
case .coarse:
|
|
kCLLocationAccuracyKilometer
|
|
case .balanced:
|
|
kCLLocationAccuracyHundredMeters
|
|
case .precise:
|
|
kCLLocationAccuracyBest
|
|
}
|
|
}
|
|
|
|
func startLocationUpdates(
|
|
desiredAccuracy: OpenClawLocationAccuracy,
|
|
significantChangesOnly: Bool) -> AsyncStream<CLLocation>
|
|
{
|
|
self.stopLocationUpdates()
|
|
|
|
self.manager.desiredAccuracy = Self.accuracyValue(desiredAccuracy)
|
|
self.manager.pausesLocationUpdatesAutomatically = true
|
|
self.manager.allowsBackgroundLocationUpdates = true
|
|
|
|
self.isStreaming = true
|
|
if significantChangesOnly {
|
|
self.manager.startMonitoringSignificantLocationChanges()
|
|
} else {
|
|
self.manager.startUpdatingLocation()
|
|
}
|
|
|
|
return AsyncStream(bufferingPolicy: .bufferingNewest(1)) { continuation in
|
|
self.updatesContinuation = continuation
|
|
continuation.onTermination = { @Sendable _ in
|
|
Task { @MainActor in
|
|
self.stopLocationUpdates()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func stopLocationUpdates() {
|
|
guard self.isStreaming else { return }
|
|
self.isStreaming = false
|
|
self.manager.stopUpdatingLocation()
|
|
self.manager.stopMonitoringSignificantLocationChanges()
|
|
self.updatesContinuation?.finish()
|
|
self.updatesContinuation = nil
|
|
}
|
|
|
|
func startMonitoringSignificantLocationChanges(onUpdate: @escaping @Sendable (CLLocation) -> Void) {
|
|
self.significantLocationCallback = onUpdate
|
|
guard !self.isMonitoringSignificantChanges else { return }
|
|
self.isMonitoringSignificantChanges = true
|
|
self.manager.startMonitoringSignificantLocationChanges()
|
|
}
|
|
|
|
func stopMonitoringSignificantLocationChanges() {
|
|
guard self.isMonitoringSignificantChanges else { return }
|
|
self.isMonitoringSignificantChanges = false
|
|
self.significantLocationCallback = nil
|
|
self.manager.stopMonitoringSignificantLocationChanges()
|
|
}
|
|
|
|
nonisolated func locationManagerDidChangeAuthorization(_ manager: CLLocationManager) {
|
|
let status = manager.authorizationStatus
|
|
Task { @MainActor in
|
|
if let cont = self.authContinuation {
|
|
self.authContinuation = nil
|
|
cont.resume(returning: status)
|
|
}
|
|
}
|
|
}
|
|
|
|
nonisolated func locationManager(_ manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
|
|
let locs = locations
|
|
Task { @MainActor in
|
|
// Resolve the one-shot continuation first (if any).
|
|
if let cont = self.locationContinuation {
|
|
self.locationContinuation = nil
|
|
if let latest = locs.last {
|
|
cont.resume(returning: latest)
|
|
} else {
|
|
cont.resume(throwing: Error.unavailable)
|
|
}
|
|
// Don't return — also forward to significant-change callback below
|
|
// so both consumers receive updates when both are active.
|
|
}
|
|
if let callback = self.significantLocationCallback, let latest = locs.last {
|
|
callback(latest)
|
|
}
|
|
if let latest = locs.last, let updates = self.updatesContinuation {
|
|
updates.yield(latest)
|
|
}
|
|
}
|
|
}
|
|
|
|
nonisolated func locationManager(_ manager: CLLocationManager, didFailWithError error: Swift.Error) {
|
|
let err = error
|
|
Task { @MainActor in
|
|
guard let cont = self.locationContinuation else { return }
|
|
self.locationContinuation = nil
|
|
cont.resume(throwing: err)
|
|
}
|
|
}
|
|
}
|