import AppKit import Foundation enum LaunchdManager { private static func runLaunchctl(_ args: [String]) { let process = Process() process.launchPath = "/bin/launchctl" process.arguments = args try? process.run() } static func startClawdis() { let userTarget = "gui/\(getuid())/\(launchdLabel)" self.runLaunchctl(["kickstart", "-k", userTarget]) } static func stopClawdis() { let userTarget = "gui/\(getuid())/\(launchdLabel)" self.runLaunchctl(["stop", userTarget]) } } enum LaunchAgentManager { private static var plistURL: URL { FileManager.default.homeDirectoryForCurrentUser .appendingPathComponent("Library/LaunchAgents/com.steipete.clawdis.plist") } static func status() -> Bool { guard FileManager.default.fileExists(atPath: self.plistURL.path) else { return false } let result = self.runLaunchctl(["print", "gui/\(getuid())/\(launchdLabel)"]) return result == 0 } static func set(enabled: Bool, bundlePath: String) { if enabled { self.writePlist(bundlePath: bundlePath) _ = self.runLaunchctl(["bootout", "gui/\(getuid())/\(launchdLabel)"]) _ = self.runLaunchctl(["bootstrap", "gui/\(getuid())", self.plistURL.path]) _ = self.runLaunchctl(["kickstart", "-k", "gui/\(getuid())/\(launchdLabel)"]) } else { _ = self.runLaunchctl(["bootout", "gui/\(getuid())/\(launchdLabel)"]) try? FileManager.default.removeItem(at: self.plistURL) } } private static func writePlist(bundlePath: String) { let plist = """ Label com.steipete.clawdis ProgramArguments \(bundlePath)/Contents/MacOS/Clawdis WorkingDirectory \(FileManager.default.homeDirectoryForCurrentUser.path) RunAtLoad MachServices com.steipete.clawdis.xpc StandardOutPath /tmp/clawdis.log StandardErrorPath /tmp/clawdis.log """ try? plist.write(to: self.plistURL, atomically: true, encoding: .utf8) } @discardableResult private static func runLaunchctl(_ args: [String]) -> Int32 { let process = Process() process.launchPath = "/bin/launchctl" process.arguments = args process.standardOutput = Pipe() process.standardError = Pipe() try? process.run() process.waitUntilExit() return process.terminationStatus } } @MainActor enum CLIInstaller { static func install(statusHandler: @escaping @Sendable (String) async -> Void) async { let helper = Bundle.main.bundleURL.appendingPathComponent("Contents/MacOS/ClawdisCLI") guard FileManager.default.isExecutableFile(atPath: helper.path) else { await statusHandler("Helper missing in bundle; rebuild via scripts/package-mac-app.sh") return } let targets = cliHelperSearchPaths.map { "\($0)/clawdis-mac" } let result = await self.privilegedSymlink(source: helper.path, targets: targets) await statusHandler(result) } private static func privilegedSymlink(source: String, targets: [String]) async -> String { let escapedSource = self.shellEscape(source) let targetList = targets.map(self.shellEscape).joined(separator: " ") let cmds = [ "mkdir -p /usr/local/bin /opt/homebrew/bin", targets.map { "ln -sf \(escapedSource) \($0)" }.joined(separator: "; ") ].joined(separator: "; ") let script = """ do shell script "\(cmds)" with administrator privileges """ let proc = Process() proc.executableURL = URL(fileURLWithPath: "/usr/bin/osascript") proc.arguments = ["-e", script] let pipe = Pipe() proc.standardOutput = pipe proc.standardError = pipe do { try proc.run() proc.waitUntilExit() let data = pipe.fileHandleForReading.readDataToEndOfFile() let output = String(data: data, encoding: .utf8)?.trimmingCharacters(in: .whitespacesAndNewlines) ?? "" if proc.terminationStatus == 0 { return output.isEmpty ? "CLI helper linked into \(targetList)" : output } if output.lowercased().contains("user canceled") { return "Install canceled" } return "Failed to install CLI helper: \(output)" } catch { return "Failed to run installer: \(error.localizedDescription)" } } private static func shellEscape(_ path: String) -> String { "'" + path.replacingOccurrences(of: "'", with: "'\"'\"'") + "'" } }