mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-18 10:24:45 +00:00
* fix(cli): fast-path bare channels help * fix(cli): normalize channels add argv gating * fix(cli): restore channel add completion flags
193 lines
6.6 KiB
TypeScript
193 lines
6.6 KiB
TypeScript
import type { OpenClawConfig } from "../config/types.openclaw.js";
|
|
import {
|
|
resolveManifestCommandAliasOwnerInRegistry,
|
|
type PluginManifestCommandAliasRecord,
|
|
type PluginManifestCommandAliasRegistry,
|
|
} from "../plugins/manifest-command-aliases.js";
|
|
import {
|
|
normalizeLowercaseStringOrEmpty,
|
|
normalizeOptionalLowercaseString,
|
|
} from "../shared/string-coerce.js";
|
|
import { resolveCliArgvInvocation } from "./argv-invocation.js";
|
|
import {
|
|
resolveCliCommandPathPolicy,
|
|
resolveCliNetworkProxyPolicy,
|
|
} from "./command-path-policy.js";
|
|
import { isReservedNonPluginCommandRoot } from "./command-registration-policy.js";
|
|
|
|
const ROOT_HELP_ALIASES = new Set(["tools"]);
|
|
|
|
export function rewriteUpdateFlagArgv(argv: string[]): string[] {
|
|
const index = argv.indexOf("--update");
|
|
if (index === -1) {
|
|
return argv;
|
|
}
|
|
|
|
const next = [...argv];
|
|
next.splice(index, 1, "update");
|
|
return next;
|
|
}
|
|
|
|
export function shouldEnsureCliPath(argv: string[]): boolean {
|
|
const invocation = resolveCliArgvInvocation(argv);
|
|
if (invocation.hasHelpOrVersion || shouldStartCrestodianForBareRoot(argv)) {
|
|
return false;
|
|
}
|
|
return resolveCliCommandPathPolicy(invocation.commandPath).ensureCliPath;
|
|
}
|
|
|
|
export function shouldUseRootHelpFastPath(
|
|
argv: string[],
|
|
env: NodeJS.ProcessEnv = process.env,
|
|
): boolean {
|
|
const invocation = resolveCliArgvInvocation(argv);
|
|
return (
|
|
env.OPENCLAW_DISABLE_CLI_STARTUP_HELP_FAST_PATH !== "1" &&
|
|
(invocation.isRootHelpInvocation ||
|
|
(invocation.commandPath.length === 1 &&
|
|
ROOT_HELP_ALIASES.has(invocation.commandPath[0] ?? "") &&
|
|
invocation.hasHelpOrVersion) ||
|
|
(invocation.commandPath.length === 1 &&
|
|
invocation.commandPath[0] === "help" &&
|
|
invocation.hasHelpOrVersion))
|
|
);
|
|
}
|
|
|
|
export function shouldUseBrowserHelpFastPath(
|
|
argv: string[],
|
|
env: NodeJS.ProcessEnv = process.env,
|
|
): boolean {
|
|
if (env.OPENCLAW_DISABLE_CLI_STARTUP_HELP_FAST_PATH === "1") {
|
|
return false;
|
|
}
|
|
const invocation = resolveCliArgvInvocation(argv);
|
|
return (
|
|
invocation.commandPath.length === 1 &&
|
|
invocation.commandPath[0] === "browser" &&
|
|
invocation.hasHelpOrVersion
|
|
);
|
|
}
|
|
|
|
export function shouldStartCrestodianForBareRoot(argv: string[]): boolean {
|
|
const invocation = resolveCliArgvInvocation(argv);
|
|
return invocation.commandPath.length === 0 && !invocation.hasHelpOrVersion;
|
|
}
|
|
|
|
export function shouldStartCrestodianForModernOnboard(argv: string[]): boolean {
|
|
const invocation = resolveCliArgvInvocation(argv);
|
|
return (
|
|
invocation.commandPath[0] === "onboard" &&
|
|
argv.includes("--modern") &&
|
|
!invocation.hasHelpOrVersion
|
|
);
|
|
}
|
|
|
|
export function shouldStartProxyForCli(argv: string[]): boolean {
|
|
const policyArgv = rewriteUpdateFlagArgv(argv);
|
|
const invocation = resolveCliArgvInvocation(policyArgv);
|
|
const [primary] = invocation.commandPath;
|
|
if (invocation.hasHelpOrVersion || !primary) {
|
|
return false;
|
|
}
|
|
if (invocation.commandPath.length === 1 && primary === "channels") {
|
|
return false;
|
|
}
|
|
return resolveCliNetworkProxyPolicy(policyArgv) === "default";
|
|
}
|
|
|
|
export function resolveMissingPluginCommandMessage(
|
|
pluginId: string,
|
|
config?: OpenClawConfig,
|
|
options?: {
|
|
registry?: PluginManifestCommandAliasRegistry;
|
|
resolveCommandAliasOwner?: (params: {
|
|
command: string | undefined;
|
|
config?: OpenClawConfig;
|
|
registry?: PluginManifestCommandAliasRegistry;
|
|
}) => PluginManifestCommandAliasRecord | undefined;
|
|
},
|
|
): string | null {
|
|
const normalizedPluginId = normalizeLowercaseStringOrEmpty(pluginId);
|
|
if (!normalizedPluginId) {
|
|
return null;
|
|
}
|
|
const allow =
|
|
Array.isArray(config?.plugins?.allow) && config.plugins.allow.length > 0
|
|
? config.plugins.allow
|
|
.filter((entry): entry is string => typeof entry === "string")
|
|
.map((entry) => normalizeOptionalLowercaseString(entry))
|
|
.filter(Boolean)
|
|
: [];
|
|
const commandAlias = options?.registry
|
|
? resolveManifestCommandAliasOwnerInRegistry({
|
|
command: normalizedPluginId,
|
|
registry: options.registry,
|
|
})
|
|
: options?.resolveCommandAliasOwner?.({
|
|
command: normalizedPluginId,
|
|
config,
|
|
...(options?.registry ? { registry: options.registry } : {}),
|
|
});
|
|
const parentPluginId = commandAlias?.pluginId;
|
|
if (parentPluginId) {
|
|
if (allow.length > 0 && !allow.includes(parentPluginId)) {
|
|
return (
|
|
`"${normalizedPluginId}" is not a plugin; it is a command provided by the ` +
|
|
`"${parentPluginId}" plugin. Add "${parentPluginId}" to \`plugins.allow\` ` +
|
|
`instead of "${normalizedPluginId}".`
|
|
);
|
|
}
|
|
if (config?.plugins?.entries?.[parentPluginId]?.enabled === false) {
|
|
return (
|
|
`The \`openclaw ${normalizedPluginId}\` command is unavailable because ` +
|
|
`\`plugins.entries.${parentPluginId}.enabled=false\`. Re-enable that entry if you want ` +
|
|
"the bundled plugin command surface."
|
|
);
|
|
}
|
|
if (
|
|
commandAlias.kind !== "runtime-slash" &&
|
|
commandAlias.enabledByDefault !== true &&
|
|
config?.plugins?.entries?.[parentPluginId]?.enabled !== true
|
|
) {
|
|
return (
|
|
`The \`openclaw ${normalizedPluginId}\` command is provided by the ` +
|
|
`"${parentPluginId}" plugin, but that bundled plugin is disabled by default. Run ` +
|
|
`\`openclaw plugins enable ${parentPluginId}\` to enable that CLI surface.`
|
|
);
|
|
}
|
|
if (commandAlias.kind === "runtime-slash") {
|
|
const cliHint = commandAlias.cliCommand
|
|
? `Use \`openclaw ${commandAlias.cliCommand}\` for related CLI operations, or `
|
|
: "Use ";
|
|
return (
|
|
`"${normalizedPluginId}" is a runtime slash command (/${normalizedPluginId}), not a CLI command. ` +
|
|
`It is provided by the "${parentPluginId}" plugin. ` +
|
|
`${cliHint}\`/${normalizedPluginId}\` in a chat session.`
|
|
);
|
|
}
|
|
}
|
|
|
|
if (isReservedNonPluginCommandRoot(normalizedPluginId)) {
|
|
return null;
|
|
}
|
|
|
|
if (allow.length > 0 && !allow.includes(normalizedPluginId)) {
|
|
if (parentPluginId && allow.includes(parentPluginId)) {
|
|
return null;
|
|
}
|
|
return (
|
|
`The \`openclaw ${normalizedPluginId}\` command is unavailable because ` +
|
|
`\`plugins.allow\` excludes "${normalizedPluginId}". Add "${normalizedPluginId}" to ` +
|
|
`\`plugins.allow\` if you want that bundled plugin CLI surface.`
|
|
);
|
|
}
|
|
if (config?.plugins?.entries?.[normalizedPluginId]?.enabled === false) {
|
|
return (
|
|
`The \`openclaw ${normalizedPluginId}\` command is unavailable because ` +
|
|
`\`plugins.entries.${normalizedPluginId}.enabled=false\`. Re-enable that entry if you want ` +
|
|
"the bundled plugin CLI surface."
|
|
);
|
|
}
|
|
return null;
|
|
}
|