mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-10 22:30:44 +00:00
259 lines
11 KiB
TypeScript
259 lines
11 KiB
TypeScript
import type { Command } from "commander";
|
|
import { formatAuthChoiceChoicesForCli } from "../../commands/auth-choice-options.js";
|
|
import type { GatewayDaemonRuntime } from "../../commands/daemon-runtime.js";
|
|
import { CORE_ONBOARD_AUTH_FLAGS } from "../../commands/onboard-core-auth-flags.js";
|
|
import type {
|
|
AuthChoice,
|
|
GatewayAuthChoice,
|
|
GatewayBind,
|
|
NodeManagerChoice,
|
|
ResetScope,
|
|
SecretInputMode,
|
|
TailscaleMode,
|
|
} from "../../commands/onboard-types.js";
|
|
import { setupWizardCommand } from "../../commands/onboard.js";
|
|
import { resolveManifestProviderOnboardAuthFlags } from "../../plugins/provider-auth-choices.js";
|
|
import { defaultRuntime } from "../../runtime.js";
|
|
import { formatDocsLink } from "../../terminal/links.js";
|
|
import { theme } from "../../terminal/theme.js";
|
|
import { runCommandWithRuntime } from "../cli-utils.js";
|
|
|
|
function resolveInstallDaemonFlag(
|
|
command: unknown,
|
|
opts: { installDaemon?: boolean },
|
|
): boolean | undefined {
|
|
if (!command || typeof command !== "object") {
|
|
return undefined;
|
|
}
|
|
const getOptionValueSource =
|
|
"getOptionValueSource" in command ? command.getOptionValueSource : undefined;
|
|
if (typeof getOptionValueSource !== "function") {
|
|
return undefined;
|
|
}
|
|
|
|
// Commander doesn't support option conflicts natively; keep original behavior.
|
|
// If --skip-daemon is explicitly passed, it wins.
|
|
if (getOptionValueSource.call(command, "skipDaemon") === "cli") {
|
|
return false;
|
|
}
|
|
if (getOptionValueSource.call(command, "installDaemon") === "cli") {
|
|
return Boolean(opts.installDaemon);
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
const AUTH_CHOICE_HELP = formatAuthChoiceChoicesForCli({
|
|
includeLegacyAliases: true,
|
|
includeSkip: true,
|
|
});
|
|
|
|
type OnboardAuthFlag = {
|
|
readonly cliOption: string;
|
|
readonly description: string;
|
|
readonly optionKey: string;
|
|
};
|
|
|
|
function extractCliFlags(cliOption: string): string[] {
|
|
return cliOption
|
|
.split(/[ ,|]+/)
|
|
.filter((part) => part.startsWith("-"))
|
|
.map((part) => {
|
|
const equalsIndex = part.indexOf("=");
|
|
return equalsIndex === -1 ? part : part.slice(0, equalsIndex);
|
|
});
|
|
}
|
|
|
|
function resolveOnboardAuthFlags(): OnboardAuthFlag[] {
|
|
const seenCliFlags = new Set<string>();
|
|
const flags: OnboardAuthFlag[] = [];
|
|
for (const flag of [...CORE_ONBOARD_AUTH_FLAGS, ...resolveManifestProviderOnboardAuthFlags()]) {
|
|
const cliFlags = extractCliFlags(flag.cliOption);
|
|
if (cliFlags.some((cliFlag) => seenCliFlags.has(cliFlag))) {
|
|
continue;
|
|
}
|
|
for (const cliFlag of cliFlags) {
|
|
seenCliFlags.add(cliFlag);
|
|
}
|
|
flags.push(flag);
|
|
}
|
|
return flags;
|
|
}
|
|
|
|
const ONBOARD_AUTH_FLAGS = resolveOnboardAuthFlags();
|
|
|
|
function pickOnboardProviderAuthOptionValues(
|
|
opts: Record<string, unknown>,
|
|
): Partial<Record<string, string | undefined>> {
|
|
return Object.fromEntries(
|
|
ONBOARD_AUTH_FLAGS.map((flag) => [flag.optionKey, opts[flag.optionKey] as string | undefined]),
|
|
);
|
|
}
|
|
|
|
export function registerOnboardCommand(program: Command) {
|
|
const command = program
|
|
.command("onboard")
|
|
.description("Interactive onboarding for the gateway, workspace, and skills")
|
|
.addHelpText(
|
|
"after",
|
|
() =>
|
|
`\n${theme.muted("Docs:")} ${formatDocsLink("/cli/onboard", "docs.openclaw.ai/cli/onboard")}\n`,
|
|
)
|
|
.option("--workspace <dir>", "Agent workspace directory (default: ~/.openclaw/workspace)")
|
|
.option(
|
|
"--reset",
|
|
"Reset config + credentials + sessions before running onboard (workspace only with --reset-scope full)",
|
|
)
|
|
.option("--reset-scope <scope>", "Reset scope: config|config+creds+sessions|full")
|
|
.option("--non-interactive", "Run without prompts", false)
|
|
.option("--modern", "Use the Crestodian conversational onboarding preview", false)
|
|
.option(
|
|
"--accept-risk",
|
|
"Acknowledge that agents are powerful and full system access is risky (required for --non-interactive)",
|
|
false,
|
|
)
|
|
.option("--flow <flow>", "Onboard flow: quickstart|advanced|manual|import")
|
|
.option("--mode <mode>", "Onboard mode: local|remote")
|
|
.option("--auth-choice <choice>", `Auth: ${AUTH_CHOICE_HELP}`)
|
|
.option(
|
|
"--token-provider <id>",
|
|
"Token provider id (non-interactive; used with --auth-choice token)",
|
|
)
|
|
.option("--token <token>", "Token value (non-interactive; used with --auth-choice token)")
|
|
.option(
|
|
"--token-profile-id <id>",
|
|
"Auth profile id (non-interactive; default: <provider>:manual)",
|
|
)
|
|
.option("--token-expires-in <duration>", "Optional token expiry duration (e.g. 365d, 12h)")
|
|
.option(
|
|
"--secret-input-mode <mode>",
|
|
"API key persistence mode: plaintext|ref (default: plaintext)",
|
|
)
|
|
.option("--cloudflare-ai-gateway-account-id <id>", "Cloudflare Account ID")
|
|
.option("--cloudflare-ai-gateway-gateway-id <id>", "Cloudflare AI Gateway ID");
|
|
|
|
for (const providerFlag of ONBOARD_AUTH_FLAGS) {
|
|
command.option(providerFlag.cliOption, providerFlag.description);
|
|
}
|
|
|
|
command
|
|
.option("--custom-base-url <url>", "Custom provider base URL")
|
|
.option("--custom-api-key <key>", "Custom provider API key (optional)")
|
|
.option("--custom-model-id <id>", "Custom provider model ID")
|
|
.option("--custom-provider-id <id>", "Custom provider ID (optional; auto-derived by default)")
|
|
.option(
|
|
"--custom-compatibility <mode>",
|
|
"Custom provider API compatibility: openai|anthropic (default: openai)",
|
|
)
|
|
.option("--custom-image-input", "Mark the custom provider model as image-capable")
|
|
.option("--custom-text-input", "Mark the custom provider model as text-only")
|
|
.option("--gateway-port <port>", "Gateway port")
|
|
.option("--gateway-bind <mode>", "Gateway bind: loopback|tailnet|lan|auto|custom")
|
|
.option("--gateway-auth <mode>", "Gateway auth: token|password")
|
|
.option("--gateway-token <token>", "Gateway token (token auth)")
|
|
.option(
|
|
"--gateway-token-ref-env <name>",
|
|
"Gateway token SecretRef env var name (token auth; e.g. OPENCLAW_GATEWAY_TOKEN)",
|
|
)
|
|
.option("--gateway-password <password>", "Gateway password (password auth)")
|
|
.option("--remote-url <url>", "Remote Gateway WebSocket URL")
|
|
.option("--remote-token <token>", "Remote Gateway token (optional)")
|
|
.option("--tailscale <mode>", "Tailscale: off|serve|funnel")
|
|
.option("--tailscale-reset-on-exit", "Reset tailscale serve/funnel on exit")
|
|
.option("--install-daemon", "Install gateway service")
|
|
.option("--no-install-daemon", "Skip gateway service install")
|
|
.option("--skip-daemon", "Skip gateway service install")
|
|
.option("--daemon-runtime <runtime>", "Daemon runtime: node|bun")
|
|
.option("--skip-channels", "Skip channel setup")
|
|
.option("--skip-skills", "Skip skills setup")
|
|
.option("--skip-bootstrap", "Skip creating default agent workspace files")
|
|
.option("--skip-search", "Skip search provider setup")
|
|
.option("--skip-health", "Skip health check")
|
|
.option("--skip-ui", "Skip Control UI/TUI prompts")
|
|
.option("--node-manager <name>", "Node manager for skills: npm|pnpm|bun")
|
|
.option("--import-from <provider>", "Migration provider to run during onboarding")
|
|
.option("--import-source <path>", "Source agent home for --import-from")
|
|
.option("--import-secrets", "Import supported secrets during onboarding migration", false)
|
|
.option("--json", "Output JSON summary", false);
|
|
|
|
command.action(async (opts, commandRuntime) => {
|
|
await runCommandWithRuntime(defaultRuntime, async () => {
|
|
if (opts.modern) {
|
|
const { runCrestodian } = await import("../../crestodian/crestodian.js");
|
|
await runCrestodian({
|
|
message: opts.nonInteractive ? "overview" : undefined,
|
|
yes: false,
|
|
json: Boolean(opts.json),
|
|
interactive: !opts.nonInteractive,
|
|
});
|
|
return;
|
|
}
|
|
const installDaemon = resolveInstallDaemonFlag(commandRuntime, {
|
|
installDaemon: Boolean(opts.installDaemon),
|
|
});
|
|
const gatewayPort =
|
|
typeof opts.gatewayPort === "string" ? Number.parseInt(opts.gatewayPort, 10) : undefined;
|
|
const providerAuthOptionValues = pickOnboardProviderAuthOptionValues(
|
|
opts as Record<string, unknown>,
|
|
);
|
|
await setupWizardCommand(
|
|
{
|
|
workspace: opts.workspace as string | undefined,
|
|
nonInteractive: Boolean(opts.nonInteractive),
|
|
acceptRisk: Boolean(opts.acceptRisk),
|
|
flow: opts.flow as "quickstart" | "advanced" | "manual" | "import" | undefined,
|
|
mode: opts.mode as "local" | "remote" | undefined,
|
|
authChoice: opts.authChoice as AuthChoice | undefined,
|
|
tokenProvider: opts.tokenProvider as string | undefined,
|
|
token: opts.token as string | undefined,
|
|
tokenProfileId: opts.tokenProfileId as string | undefined,
|
|
tokenExpiresIn: opts.tokenExpiresIn as string | undefined,
|
|
secretInputMode: opts.secretInputMode as SecretInputMode | undefined,
|
|
...providerAuthOptionValues,
|
|
cloudflareAiGatewayAccountId: opts.cloudflareAiGatewayAccountId as string | undefined,
|
|
cloudflareAiGatewayGatewayId: opts.cloudflareAiGatewayGatewayId as string | undefined,
|
|
customBaseUrl: opts.customBaseUrl as string | undefined,
|
|
customApiKey: opts.customApiKey as string | undefined,
|
|
customModelId: opts.customModelId as string | undefined,
|
|
customProviderId: opts.customProviderId as string | undefined,
|
|
customCompatibility: opts.customCompatibility as "openai" | "anthropic" | undefined,
|
|
customImageInput:
|
|
opts.customTextInput === true
|
|
? false
|
|
: opts.customImageInput === true
|
|
? true
|
|
: undefined,
|
|
gatewayPort:
|
|
typeof gatewayPort === "number" && Number.isFinite(gatewayPort)
|
|
? gatewayPort
|
|
: undefined,
|
|
gatewayBind: opts.gatewayBind as GatewayBind | undefined,
|
|
gatewayAuth: opts.gatewayAuth as GatewayAuthChoice | undefined,
|
|
gatewayToken: opts.gatewayToken as string | undefined,
|
|
gatewayTokenRefEnv: opts.gatewayTokenRefEnv as string | undefined,
|
|
gatewayPassword: opts.gatewayPassword as string | undefined,
|
|
remoteUrl: opts.remoteUrl as string | undefined,
|
|
remoteToken: opts.remoteToken as string | undefined,
|
|
tailscale: opts.tailscale as TailscaleMode | undefined,
|
|
tailscaleResetOnExit: Boolean(opts.tailscaleResetOnExit),
|
|
reset: Boolean(opts.reset),
|
|
resetScope: opts.resetScope as ResetScope | undefined,
|
|
installDaemon,
|
|
daemonRuntime: opts.daemonRuntime as GatewayDaemonRuntime | undefined,
|
|
skipChannels: Boolean(opts.skipChannels),
|
|
skipSkills: Boolean(opts.skipSkills),
|
|
skipBootstrap: Boolean(opts.skipBootstrap),
|
|
skipSearch: Boolean(opts.skipSearch),
|
|
skipHealth: Boolean(opts.skipHealth),
|
|
skipUi: Boolean(opts.skipUi),
|
|
nodeManager: opts.nodeManager as NodeManagerChoice | undefined,
|
|
importFrom: opts.importFrom as string | undefined,
|
|
importSource: opts.importSource as string | undefined,
|
|
importSecrets: Boolean(opts.importSecrets),
|
|
json: Boolean(opts.json),
|
|
},
|
|
defaultRuntime,
|
|
);
|
|
});
|
|
});
|
|
}
|