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]) } } @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 = ["/usr/local/bin/clawdis-mac", "/opt/homebrew/bin/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: "'\"'\"'") + "'" } }