UNPKG

@gguf/claw

Version:

Multi-channel AI gateway with extensible messaging integrations

454 lines (451 loc) 19.3 kB
import { nt as pathExists, ot as resolveUserPath, v as restoreTerminalState } from "./entry.js"; import "./auth-profiles-DFa1zzNy.js"; import { r as resolveCliName, t as formatCliCommand } from "./command-format-D3syQOZg.js"; import "./exec-CBKBIMpA.js"; import { h as DEFAULT_BOOTSTRAP_FILENAME } from "./agent-scope-RzK9Zcks.js"; import "./github-copilot-token-DuFIqfeC.js"; import "./model-DY9t1aT6.js"; import "./pi-model-discovery-Do3xMEtM.js"; import "./frontmatter-D-YR-Ghi.js"; import "./skills-DJmGZazd.js"; import "./manifest-registry-DS2iK5AZ.js"; import "./config-B2kL1ciP.js"; import "./client-CDjZdZtI.js"; import "./call-DXhJGwEy.js"; import "./message-channel-CVHJDItx.js"; import "./pairing-token-Byh6drgn.js"; import "./subagent-registry-C7Edpn23.js"; import "./sessions-BD5dyLxb.js"; import "./tokens-D5RzuaYP.js"; import "./normalize-Db7Xtx2v.js"; import "./accounts-BpzwDfBB.js"; import "./bindings-KqaGKS1E.js"; import "./logging-CFvkxgcX.js"; import "./send-BFiRlO6V.js"; import "./plugins-RqhjLCb6.js"; import "./send-CWDtU8Gi.js"; import "./with-timeout-nVWy7PWz.js"; import "./deliver-CeVE1Jh-.js"; import "./diagnostic-774btyou.js"; import "./diagnostic-session-state-HO94DMou.js"; import "./accounts-B3D4iWUP.js"; import "./send-DCCBUsNn.js"; import "./image-ops-lDlFpoR2.js"; import "./pi-embedded-helpers-NVEtaJwl.js"; import "./sandbox-CO-R8v6J.js"; import "./common-DcSh1hZE.js"; import "./chrome-52ZF7gmE.js"; import "./tailscale-BzRVNhMW.js"; import "./auth-Dq2pFnjj.js"; import "./server-context-C1f7cijA.js"; import "./routes-BJmh1Ify.js"; import "./redact-C2s6sr73.js"; import "./errors-CFEPAPWc.js"; import "./fs-safe-3YwsxSr5.js"; import "./paths-CXKNzNjU.js"; import "./ssrf-Cv2DiHsm.js"; import "./store-e6fIrP17.js"; import "./ports-59j-bA53.js"; import "./trash-DVLJ0k6q.js"; import "./dock-BZuwgj1O.js"; import "./accounts-Jg3_1Y-r.js"; import "./paths-CXpciDEv.js"; import "./thinking-CJPPUYWd.js"; import "./models-config-5zJ39mUb.js"; import "./reply-prefix-DJOqzjBt.js"; import "./memory-cli-C3uas9sI.js"; import "./manager-D3ZmOwqt.js"; import "./gemini-auth-CdgAkz2K.js"; import "./sqlite-1-TCahE6.js"; import "./retry-BcjnIuo-.js"; import "./chunk-Z2NYDchs.js"; import "./markdown-tables-GCHx-nGT.js"; import "./fetch-guard-B5BAqVyD.js"; import "./local-roots-SkDT1Wv8.js"; import "./ir-DAwVi0a7.js"; import "./render-e7fENCYH.js"; import "./commands-registry-CRmMPJQ9.js"; import "./image-QGUQCdGh.js"; import "./tool-display-Cs1uRaRV.js"; import "./runner-D81xREgc.js"; import "./model-catalog-DLoDxnxL.js"; import "./session-utils-C3Oi9cXA.js"; import "./skill-commands-D2rmj6w8.js"; import "./workspace-dirs-BXftTDSV.js"; import "./pairing-store-CruPwgBw.js"; import "./fetch-vg2oFVIH.js"; import "./exec-approvals-BZA6z4HM.js"; import "./nodes-screen-x7YKy8Ay.js"; import "./session-cost-usage-CpxG9Mup.js"; import "./pi-tools.policy-DeRwGLEO.js"; import "./control-service-DGleRjGJ.js"; import "./stagger-CArN2YVJ.js"; import "./channel-selection-DoXlsBhe.js"; import "./send-C3RIrBTr.js"; import "./outbound-attachment-CdFWGHgJ.js"; import "./delivery-queue-DKdWlR61.js"; import "./send-0Nz3O9Jc.js"; import "./resolve-route-BONxzeyg.js"; import "./channel-activity-BYGpAtHP.js"; import "./tables-BrqD0SUa.js"; import "./proxy-DL3MD6-P.js"; import "./links-CW8Bx7rK.js"; import "./cli-utils-CCaEbxAz.js"; import "./help-format-B0pWGnZs.js"; import "./progress-BAHiAaDW.js"; import "./replies-CuUAAFF2.js"; import { b as waitForGatewayReachable, f as openUrl, g as resolveControlUiLinks, i as detectBrowserOpenSupport, m as probeGatewayReachable, o as formatControlUiSshHint } from "./onboard-helpers-CEx2tGVB.js"; import "./prompt-style-DwCXob2h.js"; import "./pairing-labels-DtxjElSq.js"; import "./note-D3Xn5qjj.js"; import "./register.subclis-CPEP_g__.js"; import "./command-registry-D81FNA9E.js"; import "./program-context--isQKnLs.js"; import { r as installCompletion } from "./completion-cli-D8CbjU3K.js"; import { a as gatewayInstallErrorHint, i as buildGatewayInstallPlan, n as GATEWAY_DAEMON_RUNTIME_OPTIONS, t as DEFAULT_GATEWAY_DAEMON_RUNTIME } from "./daemon-runtime-DL5x9CWn.js"; import "./runtime-guard-DcwOrtgH.js"; import { r as isSystemdUserServiceAvailable } from "./systemd-BeKZd5oD.js"; import { t as resolveGatewayService } from "./service-CIkh5YiN.js"; import { r as healthCommand } from "./health-75545GOF.js"; import { t as ensureControlUiAssetsBuilt } from "./control-ui-assets-CwSEyfND.js"; import { t as formatHealthCheckFailure } from "./health-format-DkjSgkDx.js"; import { r as ensureCompletionCacheExists, t as checkShellCompletionStatus } from "./doctor-completion-CkzolxKG.js"; import { t as runTui } from "./tui-CrNYDXCj.js"; import os from "node:os"; import path from "node:path"; import fs from "node:fs/promises"; //#region src/wizard/onboarding.completion.ts async function resolveProfileHint(shell) { const home = process.env.HOME || os.homedir(); if (shell === "zsh") return "~/.zshrc"; if (shell === "bash") return await pathExists(path.join(home, ".bashrc")) ? "~/.bashrc" : "~/.bash_profile"; if (shell === "fish") return "~/.config/fish/config.fish"; return "$PROFILE"; } function formatReloadHint(shell, profileHint) { if (shell === "powershell") return "Restart your shell (or reload your PowerShell profile)."; return `Restart your shell or run: source ${profileHint}`; } async function setupOnboardingShellCompletion(params) { const deps = { resolveCliName, checkShellCompletionStatus, ensureCompletionCacheExists, installCompletion, ...params.deps }; const cliName = deps.resolveCliName(); const completionStatus = await deps.checkShellCompletionStatus(cliName); if (completionStatus.usesSlowPattern) { if (await deps.ensureCompletionCacheExists(cliName)) await deps.installCompletion(completionStatus.shell, true, cliName); return; } if (completionStatus.profileInstalled && !completionStatus.cacheExists) { await deps.ensureCompletionCacheExists(cliName); return; } if (!completionStatus.profileInstalled) { if (!(params.flow === "quickstart" ? true : await params.prompter.confirm({ message: `Enable ${completionStatus.shell} shell completion for ${cliName}?`, initialValue: true }))) return; if (!await deps.ensureCompletionCacheExists(cliName)) { await params.prompter.note(`Failed to generate completion cache. Run \`${cliName} completion --install\` later.`, "Shell completion"); return; } await deps.installCompletion(completionStatus.shell, true, cliName); const profileHint = await resolveProfileHint(completionStatus.shell); await params.prompter.note(`Shell completion installed. ${formatReloadHint(completionStatus.shell, profileHint)}`, "Shell completion"); } } //#endregion //#region src/wizard/onboarding.finalize.ts async function finalizeOnboardingWizard(options) { const { flow, opts, baseConfig, nextConfig, settings, prompter, runtime } = options; const withWizardProgress = async (label, options, work) => { const progress = prompter.progress(label); try { return await work(progress); } finally { progress.stop(options.doneMessage); } }; const systemdAvailable = process.platform === "linux" ? await isSystemdUserServiceAvailable() : true; if (process.platform === "linux" && !systemdAvailable) await prompter.note("Systemd user services are unavailable. Skipping lingering checks and service install.", "Systemd"); if (process.platform === "linux" && systemdAvailable) { const { ensureSystemdUserLingerInteractive } = await import("./systemd-linger-CgYzp_8H.js").then((n) => n.r); await ensureSystemdUserLingerInteractive({ runtime, prompter: { confirm: prompter.confirm, note: prompter.note }, reason: "Linux installs use a systemd user service by default. Without lingering, systemd stops the user session on logout/idle and kills the Gateway.", requireConfirm: false }); } const explicitInstallDaemon = typeof opts.installDaemon === "boolean" ? opts.installDaemon : void 0; let installDaemon; if (explicitInstallDaemon !== void 0) installDaemon = explicitInstallDaemon; else if (process.platform === "linux" && !systemdAvailable) installDaemon = false; else if (flow === "quickstart") installDaemon = true; else installDaemon = await prompter.confirm({ message: "Install Gateway service (recommended)", initialValue: true }); if (process.platform === "linux" && !systemdAvailable && installDaemon) { await prompter.note("Systemd user services are unavailable; skipping service install. Use your container supervisor or `docker compose up -d`.", "Gateway service"); installDaemon = false; } if (installDaemon) { const daemonRuntime = flow === "quickstart" ? DEFAULT_GATEWAY_DAEMON_RUNTIME : await prompter.select({ message: "Gateway service runtime", options: GATEWAY_DAEMON_RUNTIME_OPTIONS, initialValue: opts.daemonRuntime ?? DEFAULT_GATEWAY_DAEMON_RUNTIME }); if (flow === "quickstart") await prompter.note("QuickStart uses Node for the Gateway service (stable + supported).", "Gateway service runtime"); const service = resolveGatewayService(); const loaded = await service.isLoaded({ env: process.env }); if (loaded) { const action = await prompter.select({ message: "Gateway service already installed", options: [ { value: "restart", label: "Restart" }, { value: "reinstall", label: "Reinstall" }, { value: "skip", label: "Skip" } ] }); if (action === "restart") await withWizardProgress("Gateway service", { doneMessage: "Gateway service restarted." }, async (progress) => { progress.update("Restarting Gateway service…"); await service.restart({ env: process.env, stdout: process.stdout }); }); else if (action === "reinstall") await withWizardProgress("Gateway service", { doneMessage: "Gateway service uninstalled." }, async (progress) => { progress.update("Uninstalling Gateway service…"); await service.uninstall({ env: process.env, stdout: process.stdout }); }); } if (!loaded || loaded && !await service.isLoaded({ env: process.env })) { const progress = prompter.progress("Gateway service"); let installError = null; try { progress.update("Preparing Gateway service…"); const { programArguments, workingDirectory, environment } = await buildGatewayInstallPlan({ env: process.env, port: settings.port, token: settings.gatewayToken, runtime: daemonRuntime, warn: (message, title) => prompter.note(message, title), config: nextConfig }); progress.update("Installing Gateway service…"); await service.install({ env: process.env, stdout: process.stdout, programArguments, workingDirectory, environment }); } catch (err) { installError = err instanceof Error ? err.message : String(err); } finally { progress.stop(installError ? "Gateway service install failed." : "Gateway service installed."); } if (installError) { await prompter.note(`Gateway service install failed: ${installError}`, "Gateway"); await prompter.note(gatewayInstallErrorHint(), "Gateway"); } } } if (!opts.skipHealth) { await waitForGatewayReachable({ url: resolveControlUiLinks({ bind: nextConfig.gateway?.bind ?? "loopback", port: settings.port, customBindHost: nextConfig.gateway?.customBindHost, basePath: void 0 }).wsUrl, token: settings.gatewayToken, deadlineMs: 15e3 }); try { await healthCommand({ json: false, timeoutMs: 1e4 }, runtime); } catch (err) { runtime.error(formatHealthCheckFailure(err)); await prompter.note([ "Docs:", "https://docs.openclaw.ai/gateway/health", "https://docs.openclaw.ai/gateway/troubleshooting" ].join("\n"), "Health check help"); } } const controlUiEnabled = nextConfig.gateway?.controlUi?.enabled ?? baseConfig.gateway?.controlUi?.enabled ?? true; if (!opts.skipUi && controlUiEnabled) { const controlUiAssets = await ensureControlUiAssetsBuilt(runtime); if (!controlUiAssets.ok && controlUiAssets.message) runtime.error(controlUiAssets.message); } await prompter.note([ "Add nodes for extra features:", "- macOS app (system + notifications)", "- iOS app (camera/canvas)", "- Android app (camera/canvas)" ].join("\n"), "Optional apps"); const controlUiBasePath = nextConfig.gateway?.controlUi?.basePath ?? baseConfig.gateway?.controlUi?.basePath; const links = resolveControlUiLinks({ bind: settings.bind, port: settings.port, customBindHost: settings.customBindHost, basePath: controlUiBasePath }); const authedUrl = settings.authMode === "token" && settings.gatewayToken ? `${links.httpUrl}#token=${encodeURIComponent(settings.gatewayToken)}` : links.httpUrl; const gatewayProbe = await probeGatewayReachable({ url: links.wsUrl, token: settings.authMode === "token" ? settings.gatewayToken : void 0, password: settings.authMode === "password" ? nextConfig.gateway?.auth?.password : "" }); const gatewayStatusLine = gatewayProbe.ok ? "Gateway: reachable" : `Gateway: not detected${gatewayProbe.detail ? ` (${gatewayProbe.detail})` : ""}`; const bootstrapPath = path.join(resolveUserPath(options.workspaceDir), DEFAULT_BOOTSTRAP_FILENAME); const hasBootstrap = await fs.access(bootstrapPath).then(() => true).catch(() => false); await prompter.note([ `Web UI: ${links.httpUrl}`, settings.authMode === "token" && settings.gatewayToken ? `Web UI (with token): ${authedUrl}` : void 0, `Gateway WS: ${links.wsUrl}`, gatewayStatusLine, "Docs: https://docs.openclaw.ai/web/control-ui" ].filter(Boolean).join("\n"), "Control UI"); let controlUiOpened = false; let controlUiOpenHint; let hatchChoice = null; let launchedTui = false; if (!opts.skipUi && gatewayProbe.ok) { if (hasBootstrap) await prompter.note([ "This is the defining action that makes your agent you.", "Please take your time.", "The more you tell it, the better the experience will be.", "We will send: \"Wake up, my friend!\"" ].join("\n"), "Start TUI (best option!)"); await prompter.note([ "Gateway token: shared auth for the Gateway + Control UI.", "Stored in: ~/.openclaw/openclaw.json (gateway.auth.token) or OPENCLAW_GATEWAY_TOKEN.", `View token: ${formatCliCommand("openclaw config get gateway.auth.token")}`, `Generate token: ${formatCliCommand("openclaw doctor --generate-gateway-token")}`, "Web UI stores a copy in this browser's localStorage (openclaw.control.settings.v1).", `Open the dashboard anytime: ${formatCliCommand("openclaw dashboard --no-open")}`, "If prompted: paste the token into Control UI settings (or use the tokenized dashboard URL)." ].join("\n"), "Token"); hatchChoice = await prompter.select({ message: "How do you want to hatch your bot?", options: [ { value: "tui", label: "Hatch in TUI (recommended)" }, { value: "web", label: "Open the Web UI" }, { value: "later", label: "Do this later" } ], initialValue: "tui" }); if (hatchChoice === "tui") { restoreTerminalState("pre-onboarding tui", { resumeStdinIfPaused: true }); await runTui({ url: links.wsUrl, token: settings.authMode === "token" ? settings.gatewayToken : void 0, password: settings.authMode === "password" ? nextConfig.gateway?.auth?.password : "", deliver: false, message: hasBootstrap ? "Wake up, my friend!" : void 0 }); launchedTui = true; } else if (hatchChoice === "web") { if ((await detectBrowserOpenSupport()).ok) { controlUiOpened = await openUrl(authedUrl); if (!controlUiOpened) controlUiOpenHint = formatControlUiSshHint({ port: settings.port, basePath: controlUiBasePath, token: settings.authMode === "token" ? settings.gatewayToken : void 0 }); } else controlUiOpenHint = formatControlUiSshHint({ port: settings.port, basePath: controlUiBasePath, token: settings.authMode === "token" ? settings.gatewayToken : void 0 }); await prompter.note([ `Dashboard link (with token): ${authedUrl}`, controlUiOpened ? "Opened in your browser. Keep that tab to control OpenClaw." : "Copy/paste this URL in a browser on this machine to control OpenClaw.", controlUiOpenHint ].filter(Boolean).join("\n"), "Dashboard ready"); } else await prompter.note(`When you're ready: ${formatCliCommand("openclaw dashboard --no-open")}`, "Later"); } else if (opts.skipUi) await prompter.note("Skipping Control UI/TUI prompts.", "Control UI"); await prompter.note(["Back up your agent workspace.", "Docs: https://docs.openclaw.ai/concepts/agent-workspace"].join("\n"), "Workspace backup"); await prompter.note("Running agents on your computer is risky — harden your setup: https://docs.openclaw.ai/security", "Security"); await setupOnboardingShellCompletion({ flow, prompter }); if (!opts.skipUi && settings.authMode === "token" && Boolean(settings.gatewayToken) && hatchChoice === null) { if ((await detectBrowserOpenSupport()).ok) { controlUiOpened = await openUrl(authedUrl); if (!controlUiOpened) controlUiOpenHint = formatControlUiSshHint({ port: settings.port, basePath: controlUiBasePath, token: settings.gatewayToken }); } else controlUiOpenHint = formatControlUiSshHint({ port: settings.port, basePath: controlUiBasePath, token: settings.gatewayToken }); await prompter.note([ `Dashboard link (with token): ${authedUrl}`, controlUiOpened ? "Opened in your browser. Keep that tab to control OpenClaw." : "Copy/paste this URL in a browser on this machine to control OpenClaw.", controlUiOpenHint ].filter(Boolean).join("\n"), "Dashboard ready"); } const webSearchKey = (nextConfig.tools?.web?.search?.apiKey ?? "").trim(); const webSearchEnv = (process.env.BRAVE_API_KEY ?? "").trim(); const hasWebSearchKey = Boolean(webSearchKey || webSearchEnv); await prompter.note(hasWebSearchKey ? [ "Web search is enabled, so your agent can look things up online when needed.", "", webSearchKey ? "API key: stored in config (tools.web.search.apiKey)." : "API key: provided via BRAVE_API_KEY env var (Gateway environment).", "Docs: https://docs.openclaw.ai/tools/web" ].join("\n") : [ "If you want your agent to be able to search the web, you’ll need an API key.", "", "OpenClaw uses Brave Search for the `web_search` tool. Without a Brave Search API key, web search won’t work.", "", "Set it up interactively:", `- Run: ${formatCliCommand("openclaw configure --section web")}`, "- Enable web_search and paste your Brave Search API key", "", "Alternative: set BRAVE_API_KEY in the Gateway environment (no config changes).", "Docs: https://docs.openclaw.ai/tools/web" ].join("\n"), "Web search (optional)"); await prompter.note("What now: https://openclaw.ai/showcase (\"What People Are Building\").", "What now"); await prompter.outro(controlUiOpened ? "Onboarding complete. Dashboard opened; keep that tab to control OpenClaw." : "Onboarding complete. Use the dashboard link above to control OpenClaw."); return { launchedTui }; } //#endregion export { finalizeOnboardingWizard };