* refactor: introduce provider plugin registry * refactor: move provider CLI to plugins * docs: add provider plugin implementation notes * refactor: shift provider runtime logic into plugins * refactor: add plugin defaults and summaries * docs: update provider plugin notes * feat(commands): add /commands slash list * Auto-reply: tidy help message * Auto-reply: fix status command lint * Tests: align google shared expectations * Auto-reply: tidy help message * Auto-reply: fix status command lint * refactor: move provider routing into plugins * test: align agent routing expectations * docs: update provider plugin notes * refactor: route replies via provider plugins * docs: note route-reply plugin hooks * refactor: extend provider plugin contract * refactor: derive provider status from plugins * refactor: unify gateway provider control * refactor: use plugin metadata in auto-reply * fix: parenthesize cron target selection * refactor: derive gateway methods from plugins * refactor: generalize provider logout * refactor: route provider logout through plugins * refactor: move WhatsApp web login methods into plugin * refactor: generalize provider log prefixes * refactor: centralize default chat provider * refactor: derive provider lists from registry * refactor: move provider reload noops into plugins * refactor: resolve web login provider via alias * refactor: derive CLI provider options from plugins * refactor: derive prompt provider list from plugins * style: apply biome lint fixes * fix: resolve provider routing edge cases * docs: update provider plugin refactor notes * fix(gateway): harden agent provider routing * refactor: move provider routing into plugins * refactor: move provider CLI to plugins * refactor: derive provider lists from registry * fix: restore slash command parsing * refactor: align provider ids for schema * refactor: unify outbound target resolution * fix: keep outbound labels stable * feat: add msteams to cron surfaces * fix: clean up lint build issues * refactor: localize chat provider alias normalization * refactor: drive gateway provider lists from plugins * docs: update provider plugin notes * style: format message-provider * fix: avoid provider registry init cycles * style: sort message-provider imports * fix: relax provider alias map typing * refactor: move provider routing into plugins * refactor: add plugin pairing/config adapters * refactor: route pairing and provider removal via plugins * refactor: align auto-reply provider typing * test: stabilize telegram media mocks * docs: update provider plugin refactor notes * refactor: pluginize outbound targets * refactor: pluginize provider selection * refactor: generalize text chunk limits * docs: update provider plugin notes * refactor: generalize group session/config * fix: normalize provider id for room detection * fix: avoid provider init in system prompt * style: formatting cleanup * refactor: normalize agent delivery targets * test: update outbound delivery labels * chore: fix lint regressions * refactor: extend provider plugin adapters * refactor: move elevated/block streaming defaults to plugins * refactor: defer outbound send deps to plugins * docs: note plugin-driven streaming/elevated defaults * refactor: centralize webchat provider constant * refactor: add provider setup adapters * refactor: delegate provider add config to plugins * docs: document plugin-driven provider add * refactor: add plugin state/binding metadata * refactor: build agent provider status from plugins * docs: note plugin-driven agent bindings * refactor: centralize internal provider constant usage * fix: normalize WhatsApp targets for groups and E.164 (#631) (thanks @imfing) * refactor: centralize default chat provider * refactor: centralize WhatsApp target normalization * refactor: move provider routing into plugins * refactor: normalize agent delivery targets * chore: fix lint regressions * fix: normalize WhatsApp targets for groups and E.164 (#631) (thanks @imfing) * feat: expand provider plugin adapters * refactor: route auto-reply via provider plugins * fix: align WhatsApp target normalization * fix: normalize WhatsApp targets for groups and E.164 (#631) (thanks @imfing) * refactor: centralize WhatsApp target normalization * feat: add /config chat config updates * docs: add /config get alias * feat(commands): add /commands slash list * refactor: centralize default chat provider * style: apply biome lint fixes * chore: fix lint regressions * fix: clean up whatsapp allowlist typing * style: format config command helpers * refactor: pluginize tool threading context * refactor: normalize session announce targets * docs: note new plugin threading and announce hooks * refactor: pluginize message actions * docs: update provider plugin actions notes * fix: align provider action adapters * refactor: centralize webchat checks * style: format message provider helpers * refactor: move provider onboarding into adapters * docs: note onboarding provider adapters * feat: add msteams onboarding adapter * style: organize onboarding imports * fix: normalize msteams allowFrom types * feat: add plugin text chunk limits * refactor: use plugin chunk limit fallbacks * feat: add provider mention stripping hooks * style: organize provider plugin type imports * refactor: generalize health snapshots * refactor: update macOS health snapshot handling * docs: refresh health snapshot notes * style: format health snapshot updates * refactor: drive security warnings via plugins * docs: note provider security adapter * style: format provider security adapters * refactor: centralize provider account defaults * refactor: type gateway client identity constants * chore: regen gateway protocol swift * fix: degrade health on failed provider probe * refactor: centralize pairing approve hint * docs: add plugin CLI command references * refactor: route auth and tool sends through plugins * docs: expand provider plugin hooks * refactor: document provider docking touchpoints * refactor: normalize internal provider defaults * refactor: streamline outbound delivery wiring * refactor: make provider onboarding plugin-owned * refactor: support provider-owned agent tools * refactor: move telegram draft chunking into telegram module * refactor: infer provider tool sends via extractToolSend * fix: repair plugin onboarding imports * refactor: de-dup outbound target normalization * style: tidy plugin and agent imports * refactor: data-drive provider selection line * fix: satisfy lint after provider plugin rebase * test: deflake gateway-cli coverage * style: format gateway-cli coverage test * refactor(provider-plugins): simplify provider ids * test(pairing-cli): avoid provider-specific ternary * style(macos): swiftformat HealthStore * refactor(sandbox): derive provider tool denylist * fix(sandbox): avoid plugin init in defaults * refactor(provider-plugins): centralize provider aliases * style(test): satisfy biome * refactor(protocol): v3 providers.status maps * refactor(ui): adapt to protocol v3 * refactor(macos): adapt to protocol v3 * test: update providers.status v3 fixtures * refactor(gateway): map provider runtime snapshot * test(gateway): update reload runtime snapshot * refactor(whatsapp): normalize heartbeat provider id * docs(refactor): update provider plugin notes * style: satisfy biome after rebase * fix: describe sandboxed elevated in prompt * feat(gateway): add agent image attachments + live probe * refactor: derive CLI provider options from plugins * fix(gateway): harden agent provider routing * fix(gateway): harden agent provider routing * refactor: align provider ids for schema * fix(protocol): keep agent provider string * fix(gateway): harden agent provider routing * fix(protocol): keep agent provider string * refactor: normalize agent delivery targets * refactor: support provider-owned agent tools * refactor(config): provider-keyed elevated allowFrom * style: satisfy biome * fix(gateway): appease provider narrowing * style: satisfy biome * refactor(reply): move group intro hints into plugin * fix(reply): avoid plugin registry init cycle * refactor(providers): add lightweight provider dock * refactor(gateway): use typed client id in connect * refactor(providers): document docks and avoid init cycles * refactor(providers): make media limit helper generic * fix(providers): break plugin registry import cycles * style: satisfy biome * refactor(status-all): build providers table from plugins * refactor(gateway): delegate web login to provider plugin * refactor(provider): drop web alias * refactor(provider): lazy-load monitors * style: satisfy lint/format * style: format status-all providers table * style: swiftformat gateway discovery model * test: make reload plan plugin-driven * fix: avoid token stringification in status-all * refactor: make provider IDs explicit in status * feat: warn on signal/imessage provider runtime errors * test: cover gateway provider runtime warnings in status * fix: add runtime kind to provider status issues * test: cover health degradation on probe failure * fix: keep routeReply lightweight * style: organize routeReply imports * refactor(web): extract auth-store helpers * refactor(whatsapp): lazy login imports * refactor(outbound): route replies via plugin outbound * docs: update provider plugin notes * style: format provider status issues * fix: make sandbox scope warning wrap-safe * refactor: load outbound adapters from provider plugins * docs: update provider plugin outbound notes * style(macos): fix swiftformat lint * docs: changelog for provider plugins * fix(macos): satisfy swiftformat * fix(macos): open settings via menu action * style: format after rebase * fix(macos): open Settings via menu action --------- Co-authored-by: LK <luke@kyohere.com> Co-authored-by: Luke K (pr-0f3t) <2609441+lc0rp@users.noreply.github.com> Co-authored-by: Xin <xin@imfing.com>
787 lines
30 KiB
Swift
787 lines
30 KiB
Swift
import AppKit
|
|
import ClawdbotDiscovery
|
|
import ClawdbotIPC
|
|
import ClawdbotKit
|
|
import CoreLocation
|
|
import Observation
|
|
import SwiftUI
|
|
|
|
struct GeneralSettings: View {
|
|
@Bindable var state: AppState
|
|
@AppStorage(cameraEnabledKey) private var cameraEnabled: Bool = false
|
|
@AppStorage(locationModeKey) private var locationModeRaw: String = ClawdbotLocationMode.off.rawValue
|
|
@AppStorage(locationPreciseKey) private var locationPreciseEnabled: Bool = true
|
|
private let healthStore = HealthStore.shared
|
|
private let gatewayManager = GatewayProcessManager.shared
|
|
@State private var gatewayDiscovery = GatewayDiscoveryModel(
|
|
localDisplayName: InstanceIdentity.displayName)
|
|
@State private var isInstallingCLI = false
|
|
@State private var cliStatus: String?
|
|
@State private var cliInstalled = false
|
|
@State private var cliInstallLocation: String?
|
|
@State private var gatewayStatus: GatewayEnvironmentStatus = .checking
|
|
@State private var remoteStatus: RemoteStatus = .idle
|
|
@State private var showRemoteAdvanced = false
|
|
private let isPreview = ProcessInfo.processInfo.isPreview
|
|
private var isNixMode: Bool { ProcessInfo.processInfo.isNixMode }
|
|
@State private var lastLocationModeRaw: String = ClawdbotLocationMode.off.rawValue
|
|
|
|
var body: some View {
|
|
ScrollView(.vertical) {
|
|
VStack(alignment: .leading, spacing: 18) {
|
|
if !self.state.onboardingSeen {
|
|
Button {
|
|
DebugActions.restartOnboarding()
|
|
} label: {
|
|
HStack(spacing: 8) {
|
|
Label("Complete onboarding to finish setup", systemImage: "arrow.counterclockwise")
|
|
.font(.callout.weight(.semibold))
|
|
.foregroundStyle(Color.accentColor)
|
|
Spacer(minLength: 0)
|
|
Image(systemName: "chevron.right")
|
|
.font(.caption.weight(.semibold))
|
|
.foregroundStyle(.tertiary)
|
|
}
|
|
.contentShape(Rectangle())
|
|
}
|
|
.buttonStyle(.plain)
|
|
.padding(.bottom, 2)
|
|
}
|
|
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
SettingsToggleRow(
|
|
title: "Clawdbot active",
|
|
subtitle: "Pause to stop the Clawdbot gateway; no messages will be processed.",
|
|
binding: self.activeBinding)
|
|
|
|
self.connectionSection
|
|
|
|
Divider()
|
|
|
|
SettingsToggleRow(
|
|
title: "Launch at login",
|
|
subtitle: "Automatically start Clawdbot after you sign in.",
|
|
binding: self.$state.launchAtLogin)
|
|
|
|
SettingsToggleRow(
|
|
title: "Show Dock icon",
|
|
subtitle: "Keep Clawdbot visible in the Dock instead of menu-bar-only mode.",
|
|
binding: self.$state.showDockIcon)
|
|
|
|
SettingsToggleRow(
|
|
title: "Play menu bar icon animations",
|
|
subtitle: "Enable idle blinks and wiggles on the status icon.",
|
|
binding: self.$state.iconAnimationsEnabled)
|
|
|
|
SettingsToggleRow(
|
|
title: "Allow Canvas",
|
|
subtitle: "Allow the agent to show and control the Canvas panel.",
|
|
binding: self.$state.canvasEnabled)
|
|
|
|
SettingsToggleRow(
|
|
title: "Allow Camera",
|
|
subtitle: "Allow the agent to capture a photo or short video via the built-in camera.",
|
|
binding: self.$cameraEnabled)
|
|
|
|
VStack(alignment: .leading, spacing: 6) {
|
|
Text("Location Access")
|
|
.font(.body)
|
|
|
|
Picker("", selection: self.$locationModeRaw) {
|
|
Text("Off").tag(ClawdbotLocationMode.off.rawValue)
|
|
Text("While Using").tag(ClawdbotLocationMode.whileUsing.rawValue)
|
|
Text("Always").tag(ClawdbotLocationMode.always.rawValue)
|
|
}
|
|
.pickerStyle(.segmented)
|
|
|
|
Toggle("Precise Location", isOn: self.$locationPreciseEnabled)
|
|
.disabled(self.locationMode == .off)
|
|
|
|
Text("Always may require System Settings to approve background location.")
|
|
.font(.footnote)
|
|
.foregroundStyle(.tertiary)
|
|
.fixedSize(horizontal: false, vertical: true)
|
|
}
|
|
|
|
SettingsToggleRow(
|
|
title: "Enable Peekaboo Bridge",
|
|
subtitle: "Allow signed tools (e.g. `peekaboo`) to drive UI automation via PeekabooBridge.",
|
|
binding: self.$state.peekabooBridgeEnabled)
|
|
|
|
SettingsToggleRow(
|
|
title: "Enable debug tools",
|
|
subtitle: "Show the Debug tab with development utilities.",
|
|
binding: self.$state.debugPaneEnabled)
|
|
}
|
|
|
|
Spacer(minLength: 12)
|
|
HStack {
|
|
Spacer()
|
|
Button("Quit Clawdbot") { NSApp.terminate(nil) }
|
|
.buttonStyle(.borderedProminent)
|
|
}
|
|
}
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
.padding(.horizontal, 22)
|
|
.padding(.bottom, 16)
|
|
}
|
|
.onAppear {
|
|
guard !self.isPreview else { return }
|
|
self.refreshCLIStatus()
|
|
self.refreshGatewayStatus()
|
|
self.lastLocationModeRaw = self.locationModeRaw
|
|
}
|
|
.onChange(of: self.state.canvasEnabled) { _, enabled in
|
|
if !enabled {
|
|
CanvasManager.shared.hideAll()
|
|
}
|
|
}
|
|
.onChange(of: self.locationModeRaw) { _, newValue in
|
|
let previous = self.lastLocationModeRaw
|
|
self.lastLocationModeRaw = newValue
|
|
guard let mode = ClawdbotLocationMode(rawValue: newValue) else { return }
|
|
Task {
|
|
let granted = await self.requestLocationAuthorization(mode: mode)
|
|
if !granted {
|
|
await MainActor.run {
|
|
self.locationModeRaw = previous
|
|
self.lastLocationModeRaw = previous
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private var activeBinding: Binding<Bool> {
|
|
Binding(
|
|
get: { !self.state.isPaused },
|
|
set: { self.state.isPaused = !$0 })
|
|
}
|
|
|
|
private var locationMode: ClawdbotLocationMode {
|
|
ClawdbotLocationMode(rawValue: self.locationModeRaw) ?? .off
|
|
}
|
|
|
|
private func requestLocationAuthorization(mode: ClawdbotLocationMode) async -> Bool {
|
|
guard mode != .off else { return true }
|
|
guard CLLocationManager.locationServicesEnabled() else {
|
|
await MainActor.run { LocationPermissionHelper.openSettings() }
|
|
return false
|
|
}
|
|
|
|
let status = CLLocationManager().authorizationStatus
|
|
let requireAlways = mode == .always
|
|
if PermissionManager.isLocationAuthorized(status: status, requireAlways: requireAlways) {
|
|
return true
|
|
}
|
|
let updated = await LocationPermissionRequester.shared.request(always: requireAlways)
|
|
return PermissionManager.isLocationAuthorized(status: updated, requireAlways: requireAlways)
|
|
}
|
|
|
|
private var connectionSection: some View {
|
|
VStack(alignment: .leading, spacing: 10) {
|
|
Text("Clawdbot runs")
|
|
.font(.title3.weight(.semibold))
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
|
|
Picker("", selection: self.$state.connectionMode) {
|
|
Text("Not configured").tag(AppState.ConnectionMode.unconfigured)
|
|
Text("Local (this Mac)").tag(AppState.ConnectionMode.local)
|
|
Text("Remote over SSH").tag(AppState.ConnectionMode.remote)
|
|
}
|
|
.pickerStyle(.segmented)
|
|
.frame(width: 380, alignment: .leading)
|
|
|
|
if self.state.connectionMode == .unconfigured {
|
|
Text("Pick Local or Remote to start the Gateway.")
|
|
.font(.footnote)
|
|
.foregroundStyle(.secondary)
|
|
.fixedSize(horizontal: false, vertical: true)
|
|
}
|
|
|
|
if self.state.connectionMode == .local {
|
|
// In Nix mode, gateway is managed declaratively - no install buttons.
|
|
if !self.isNixMode {
|
|
self.gatewayInstallerCard
|
|
}
|
|
SettingsToggleRow(
|
|
title: "Attach only",
|
|
subtitle: "Use this when the gateway runs externally; the mac app will only attach " +
|
|
"to an already-running gateway and won't start one locally.",
|
|
binding: self.$state.attachExistingGatewayOnly)
|
|
TailscaleIntegrationSection(
|
|
connectionMode: self.state.connectionMode,
|
|
isPaused: self.state.isPaused)
|
|
self.healthRow
|
|
}
|
|
|
|
if self.state.connectionMode == .remote {
|
|
self.remoteCard
|
|
}
|
|
|
|
self.cliInstaller
|
|
}
|
|
}
|
|
|
|
private var remoteCard: some View {
|
|
VStack(alignment: .leading, spacing: 10) {
|
|
HStack(alignment: .center, spacing: 10) {
|
|
Text("SSH")
|
|
.font(.callout.weight(.semibold))
|
|
.frame(width: 48, alignment: .leading)
|
|
TextField("user@host[:22]", text: self.$state.remoteTarget)
|
|
.textFieldStyle(.roundedBorder)
|
|
.frame(maxWidth: .infinity)
|
|
Button {
|
|
Task { await self.testRemote() }
|
|
} label: {
|
|
if self.remoteStatus == .checking {
|
|
ProgressView().controlSize(.small)
|
|
} else {
|
|
Text("Test remote")
|
|
}
|
|
}
|
|
.buttonStyle(.borderedProminent)
|
|
.disabled(self.remoteStatus == .checking || self.state.remoteTarget
|
|
.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty)
|
|
}
|
|
|
|
GatewayDiscoveryInlineList(
|
|
discovery: self.gatewayDiscovery,
|
|
currentTarget: self.state.remoteTarget)
|
|
{ gateway in
|
|
self.applyDiscoveredGateway(gateway)
|
|
}
|
|
.padding(.leading, 58)
|
|
|
|
self.remoteStatusView
|
|
.padding(.leading, 58)
|
|
|
|
DisclosureGroup(isExpanded: self.$showRemoteAdvanced) {
|
|
VStack(alignment: .leading, spacing: 8) {
|
|
LabeledContent("Identity file") {
|
|
TextField("/Users/you/.ssh/id_ed25519", text: self.$state.remoteIdentity)
|
|
.textFieldStyle(.roundedBorder)
|
|
.frame(width: 280)
|
|
}
|
|
LabeledContent("Project root") {
|
|
TextField("/home/you/Projects/clawdbot", text: self.$state.remoteProjectRoot)
|
|
.textFieldStyle(.roundedBorder)
|
|
.frame(width: 280)
|
|
}
|
|
LabeledContent("CLI path") {
|
|
TextField("/Applications/Clawdbot.app/.../clawdbot", text: self.$state.remoteCliPath)
|
|
.textFieldStyle(.roundedBorder)
|
|
.frame(width: 280)
|
|
}
|
|
}
|
|
.padding(.top, 4)
|
|
} label: {
|
|
Text("Advanced")
|
|
.font(.callout.weight(.semibold))
|
|
}
|
|
|
|
// Diagnostics
|
|
VStack(alignment: .leading, spacing: 4) {
|
|
Text("Control channel")
|
|
.font(.caption.weight(.semibold))
|
|
if !self.isControlStatusDuplicate || ControlChannel.shared.lastPingMs != nil {
|
|
let status = self.isControlStatusDuplicate ? nil : self.controlStatusLine
|
|
let ping = ControlChannel.shared.lastPingMs.map { "Ping \(Int($0)) ms" }
|
|
let line = [status, ping].compactMap(\.self).joined(separator: " · ")
|
|
if !line.isEmpty {
|
|
Text(line)
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
}
|
|
if let hb = HeartbeatStore.shared.lastEvent {
|
|
let ageText = age(from: Date(timeIntervalSince1970: hb.ts / 1000))
|
|
Text("Last heartbeat: \(hb.status) · \(ageText)")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
}
|
|
|
|
Text("Tip: enable Tailscale for stable remote access.")
|
|
.font(.footnote)
|
|
.foregroundStyle(.secondary)
|
|
.lineLimit(1)
|
|
}
|
|
.transition(.opacity)
|
|
.onAppear { self.gatewayDiscovery.start() }
|
|
.onDisappear { self.gatewayDiscovery.stop() }
|
|
}
|
|
|
|
private var controlStatusLine: String {
|
|
switch ControlChannel.shared.state {
|
|
case .connected: "Connected"
|
|
case .connecting: "Connecting…"
|
|
case .disconnected: "Disconnected"
|
|
case let .degraded(msg): msg
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
private var remoteStatusView: some View {
|
|
switch self.remoteStatus {
|
|
case .idle:
|
|
EmptyView()
|
|
case .checking:
|
|
Text("Testing…")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
case .ok:
|
|
Label("Ready", systemImage: "checkmark.circle.fill")
|
|
.font(.caption)
|
|
.foregroundStyle(.green)
|
|
case let .failed(message):
|
|
Text(message)
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
.lineLimit(2)
|
|
}
|
|
}
|
|
|
|
private var isControlStatusDuplicate: Bool {
|
|
guard case let .failed(message) = self.remoteStatus else { return false }
|
|
return message == self.controlStatusLine
|
|
}
|
|
|
|
private var cliInstaller: some View {
|
|
VStack(alignment: .leading, spacing: 8) {
|
|
HStack(spacing: 10) {
|
|
Button {
|
|
Task { await self.installCLI() }
|
|
} label: {
|
|
let title = self.cliInstalled ? "Reinstall CLI" : "Install CLI"
|
|
ZStack {
|
|
Text(title)
|
|
.opacity(self.isInstallingCLI ? 0 : 1)
|
|
if self.isInstallingCLI {
|
|
ProgressView()
|
|
.controlSize(.mini)
|
|
}
|
|
}
|
|
.frame(minWidth: 150)
|
|
}
|
|
.disabled(self.isInstallingCLI)
|
|
|
|
if self.isInstallingCLI {
|
|
Text("Working...")
|
|
.font(.callout)
|
|
.foregroundStyle(.secondary)
|
|
} else if self.cliInstalled {
|
|
Label("Installed", systemImage: "checkmark.circle.fill")
|
|
.font(.callout)
|
|
.foregroundStyle(.secondary)
|
|
} else {
|
|
Text("Not installed")
|
|
.font(.callout)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
}
|
|
|
|
if let status = cliStatus {
|
|
Text(status)
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
.lineLimit(2)
|
|
} else if let installLocation = self.cliInstallLocation {
|
|
Text("Found at \(installLocation)")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
.lineLimit(2)
|
|
} else {
|
|
Text("Installs a user-space Node 22+ runtime and the CLI (no Homebrew).")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
.lineLimit(2)
|
|
}
|
|
}
|
|
}
|
|
|
|
private var gatewayInstallerCard: some View {
|
|
VStack(alignment: .leading, spacing: 8) {
|
|
HStack(spacing: 10) {
|
|
Circle()
|
|
.fill(self.gatewayStatusColor)
|
|
.frame(width: 10, height: 10)
|
|
Text(self.gatewayStatus.message)
|
|
.font(.callout)
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
}
|
|
|
|
if let gatewayVersion = self.gatewayStatus.gatewayVersion,
|
|
let required = self.gatewayStatus.requiredGateway,
|
|
gatewayVersion != required
|
|
{
|
|
Text("Installed: \(gatewayVersion) · Required: \(required)")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
} else if let gatewayVersion = self.gatewayStatus.gatewayVersion {
|
|
Text("Gateway \(gatewayVersion) detected")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
|
|
if let node = self.gatewayStatus.nodeVersion {
|
|
Text("Node \(node)")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
|
|
if case let .attachedExisting(details) = self.gatewayManager.status {
|
|
Text(details ?? "Using existing gateway instance")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
|
|
if let failure = self.gatewayManager.lastFailureReason {
|
|
Text("Last failure: \(failure)")
|
|
.font(.caption)
|
|
.foregroundStyle(.red)
|
|
}
|
|
|
|
Button("Recheck") { self.refreshGatewayStatus() }
|
|
.buttonStyle(.bordered)
|
|
|
|
Text("Gateway auto-starts in local mode via launchd (\(gatewayLaunchdLabel)).")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
.lineLimit(2)
|
|
}
|
|
.padding(12)
|
|
.background(Color.gray.opacity(0.08))
|
|
.cornerRadius(10)
|
|
}
|
|
|
|
private func installCLI() async {
|
|
guard !self.isInstallingCLI else { return }
|
|
self.isInstallingCLI = true
|
|
defer { isInstallingCLI = false }
|
|
await CLIInstaller.install { status in
|
|
self.cliStatus = status
|
|
self.refreshCLIStatus()
|
|
}
|
|
}
|
|
|
|
private func refreshCLIStatus() {
|
|
let installLocation = CLIInstaller.installedLocation()
|
|
self.cliInstallLocation = installLocation
|
|
self.cliInstalled = installLocation != nil
|
|
}
|
|
|
|
private func refreshGatewayStatus() {
|
|
Task {
|
|
let status = await Task.detached(priority: .utility) {
|
|
GatewayEnvironment.check()
|
|
}.value
|
|
self.gatewayStatus = status
|
|
}
|
|
}
|
|
|
|
private var gatewayStatusColor: Color {
|
|
switch self.gatewayStatus.kind {
|
|
case .ok: .green
|
|
case .checking: .secondary
|
|
case .missingNode, .missingGateway, .incompatible, .error: .orange
|
|
}
|
|
}
|
|
|
|
private var healthCard: some View {
|
|
let snapshot = self.healthStore.snapshot
|
|
return VStack(alignment: .leading, spacing: 6) {
|
|
HStack(spacing: 8) {
|
|
Circle()
|
|
.fill(self.healthStore.state.tint)
|
|
.frame(width: 10, height: 10)
|
|
Text(self.healthStore.summaryLine)
|
|
.font(.callout.weight(.semibold))
|
|
}
|
|
|
|
if let snap = snapshot {
|
|
let linkId = snap.providerOrder?.first(where: {
|
|
if let summary = snap.providers[$0] { return summary.linked != nil }
|
|
return false
|
|
}) ?? snap.providers.keys.first(where: {
|
|
if let summary = snap.providers[$0] { return summary.linked != nil }
|
|
return false
|
|
})
|
|
let linkLabel =
|
|
linkId.flatMap { snap.providerLabels?[$0] } ??
|
|
linkId?.capitalized ??
|
|
"Link provider"
|
|
let linkAge = linkId.flatMap { snap.providers[$0]?.authAgeMs }
|
|
Text("\(linkLabel) auth age: \(healthAgeString(linkAge))")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
Text("Session store: \(snap.sessions.path) (\(snap.sessions.count) entries)")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
if let recent = snap.sessions.recent.first {
|
|
let lastActivity = recent.updatedAt != nil
|
|
? relativeAge(from: Date(timeIntervalSince1970: (recent.updatedAt ?? 0) / 1000))
|
|
: "unknown"
|
|
Text("Last activity: \(recent.key) \(lastActivity)")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
Text("Last check: \(relativeAge(from: self.healthStore.lastSuccess))")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
} else if let error = self.healthStore.lastError {
|
|
Text(error)
|
|
.font(.caption)
|
|
.foregroundStyle(.red)
|
|
} else {
|
|
Text("Health check pending…")
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
|
|
HStack(spacing: 12) {
|
|
Button {
|
|
Task { await self.healthStore.refresh(onDemand: true) }
|
|
} label: {
|
|
if self.healthStore.isRefreshing {
|
|
ProgressView().controlSize(.small)
|
|
} else {
|
|
Label("Run Health Check", systemImage: "arrow.clockwise")
|
|
}
|
|
}
|
|
.disabled(self.healthStore.isRefreshing)
|
|
|
|
Divider().frame(height: 18)
|
|
|
|
Button {
|
|
self.revealLogs()
|
|
} label: {
|
|
Label("Reveal Logs", systemImage: "doc.text.magnifyingglass")
|
|
}
|
|
}
|
|
}
|
|
.padding(12)
|
|
.background(Color.gray.opacity(0.08))
|
|
.cornerRadius(10)
|
|
}
|
|
}
|
|
|
|
private enum RemoteStatus: Equatable {
|
|
case idle
|
|
case checking
|
|
case ok
|
|
case failed(String)
|
|
}
|
|
|
|
extension GeneralSettings {
|
|
private var healthRow: some View {
|
|
VStack(alignment: .leading, spacing: 6) {
|
|
HStack(spacing: 10) {
|
|
Circle()
|
|
.fill(self.healthStore.state.tint)
|
|
.frame(width: 10, height: 10)
|
|
Text(self.healthStore.summaryLine)
|
|
.font(.callout)
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
}
|
|
|
|
if let detail = self.healthStore.detailLine {
|
|
Text(detail)
|
|
.font(.caption)
|
|
.foregroundStyle(.secondary)
|
|
.fixedSize(horizontal: false, vertical: true)
|
|
}
|
|
|
|
HStack(spacing: 10) {
|
|
Button("Retry now") {
|
|
Task { await HealthStore.shared.refresh(onDemand: true) }
|
|
}
|
|
.disabled(self.healthStore.isRefreshing)
|
|
|
|
Button("Open logs") { self.revealLogs() }
|
|
.buttonStyle(.link)
|
|
.foregroundStyle(.secondary)
|
|
}
|
|
.font(.caption)
|
|
}
|
|
}
|
|
|
|
@MainActor
|
|
func testRemote() async {
|
|
self.remoteStatus = .checking
|
|
let settings = CommandResolver.connectionSettings()
|
|
guard !settings.target.isEmpty else {
|
|
self.remoteStatus = .failed("Set an SSH target first")
|
|
return
|
|
}
|
|
|
|
// Step 1: basic SSH reachability check
|
|
let sshResult = await ShellExecutor.run(
|
|
command: Self.sshCheckCommand(target: settings.target, identity: settings.identity),
|
|
cwd: nil,
|
|
env: nil,
|
|
timeout: 8)
|
|
|
|
guard sshResult.ok else {
|
|
self.remoteStatus = .failed(self.formatSSHFailure(sshResult, target: settings.target))
|
|
return
|
|
}
|
|
|
|
// Step 2: control channel health over tunnel
|
|
let originalMode = AppStateStore.shared.connectionMode
|
|
do {
|
|
try await ControlChannel.shared.configure(mode: .remote(
|
|
target: settings.target,
|
|
identity: settings.identity))
|
|
let data = try await ControlChannel.shared.health(timeout: 10)
|
|
if decodeHealthSnapshot(from: data) != nil {
|
|
self.remoteStatus = .ok
|
|
} else {
|
|
self.remoteStatus = .failed("Control channel returned invalid health JSON")
|
|
}
|
|
} catch {
|
|
self.remoteStatus = .failed(error.localizedDescription)
|
|
}
|
|
|
|
// Restore original mode if we temporarily switched
|
|
switch originalMode {
|
|
case .remote:
|
|
break
|
|
case .local:
|
|
try? await ControlChannel.shared.configure(mode: .local)
|
|
case .unconfigured:
|
|
await ControlChannel.shared.disconnect()
|
|
}
|
|
}
|
|
|
|
private static func sshCheckCommand(target: String, identity: String) -> [String] {
|
|
var args: [String] = [
|
|
"/usr/bin/ssh",
|
|
"-o", "BatchMode=yes",
|
|
"-o", "ConnectTimeout=5",
|
|
"-o", "StrictHostKeyChecking=accept-new",
|
|
"-o", "UpdateHostKeys=yes",
|
|
]
|
|
if !identity.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty {
|
|
args.append(contentsOf: ["-i", identity])
|
|
}
|
|
args.append(target)
|
|
args.append("echo ok")
|
|
return args
|
|
}
|
|
|
|
private func formatSSHFailure(_ response: Response, target: String) -> String {
|
|
let payload = response.payload.flatMap { String(data: $0, encoding: .utf8) }
|
|
let trimmed = payload?
|
|
.trimmingCharacters(in: .whitespacesAndNewlines)
|
|
.split(whereSeparator: \.isNewline)
|
|
.joined(separator: " ")
|
|
if let trimmed,
|
|
trimmed.localizedCaseInsensitiveContains("host key verification failed")
|
|
{
|
|
let host = CommandResolver.parseSSHTarget(target)?.host ?? target
|
|
return "SSH check failed: Host key verification failed. Remove the old key with " +
|
|
"`ssh-keygen -R \(host)` and try again."
|
|
}
|
|
if let trimmed, !trimmed.isEmpty {
|
|
if let message = response.message, message.hasPrefix("exit ") {
|
|
return "SSH check failed: \(trimmed) (\(message))"
|
|
}
|
|
return "SSH check failed: \(trimmed)"
|
|
}
|
|
if let message = response.message {
|
|
return "SSH check failed (\(message))"
|
|
}
|
|
return "SSH check failed"
|
|
}
|
|
|
|
private func revealLogs() {
|
|
let target = LogLocator.bestLogFile()
|
|
|
|
if let target {
|
|
NSWorkspace.shared.selectFile(
|
|
target.path,
|
|
inFileViewerRootedAtPath: target.deletingLastPathComponent().path)
|
|
return
|
|
}
|
|
|
|
let alert = NSAlert()
|
|
alert.messageText = "Log file not found"
|
|
alert.informativeText = """
|
|
Looked for clawdbot logs in /tmp/clawdbot/.
|
|
Run a health check or send a message to generate activity, then try again.
|
|
"""
|
|
alert.alertStyle = .informational
|
|
alert.addButton(withTitle: "OK")
|
|
alert.runModal()
|
|
}
|
|
|
|
private func applyDiscoveredGateway(_ gateway: GatewayDiscoveryModel.DiscoveredGateway) {
|
|
MacNodeModeCoordinator.shared.setPreferredBridgeStableID(gateway.stableID)
|
|
|
|
let host = gateway.tailnetDns ?? gateway.lanHost
|
|
guard let host else { return }
|
|
let user = NSUserName()
|
|
self.state.remoteTarget = GatewayDiscoveryModel.buildSSHTarget(
|
|
user: user,
|
|
host: host,
|
|
port: gateway.sshPort)
|
|
self.state.remoteCliPath = gateway.cliPath ?? ""
|
|
ClawdbotConfigFile.setRemoteGatewayUrl(host: host, port: gateway.gatewayPort)
|
|
}
|
|
}
|
|
|
|
private func healthAgeString(_ ms: Double?) -> String {
|
|
guard let ms else { return "unknown" }
|
|
return msToAge(ms)
|
|
}
|
|
|
|
#if DEBUG
|
|
struct GeneralSettings_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
GeneralSettings(state: .preview)
|
|
.frame(width: SettingsTab.windowWidth, height: SettingsTab.windowHeight)
|
|
.environment(TailscaleService.shared)
|
|
}
|
|
}
|
|
|
|
@MainActor
|
|
extension GeneralSettings {
|
|
static func exerciseForTesting() {
|
|
let state = AppState(preview: true)
|
|
state.connectionMode = .remote
|
|
state.remoteTarget = "user@host:2222"
|
|
state.remoteIdentity = "/tmp/id_ed25519"
|
|
state.remoteProjectRoot = "/tmp/clawdbot"
|
|
state.remoteCliPath = "/tmp/clawdbot"
|
|
|
|
let view = GeneralSettings(state: state)
|
|
view.gatewayStatus = GatewayEnvironmentStatus(
|
|
kind: .ok,
|
|
nodeVersion: "1.0.0",
|
|
gatewayVersion: "1.0.0",
|
|
requiredGateway: nil,
|
|
message: "Gateway ready")
|
|
view.remoteStatus = .failed("SSH failed")
|
|
view.showRemoteAdvanced = true
|
|
view.cliInstalled = true
|
|
view.cliInstallLocation = "/usr/local/bin/clawdbot"
|
|
view.cliStatus = "Installed"
|
|
_ = view.body
|
|
|
|
state.connectionMode = .unconfigured
|
|
_ = view.body
|
|
|
|
state.connectionMode = .local
|
|
view.gatewayStatus = GatewayEnvironmentStatus(
|
|
kind: .error("Gateway offline"),
|
|
nodeVersion: nil,
|
|
gatewayVersion: nil,
|
|
requiredGateway: nil,
|
|
message: "Gateway offline")
|
|
_ = view.body
|
|
}
|
|
}
|
|
#endif
|