mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-16 04:20:44 +00:00
Merged via /review-pr -> /prepare-pr -> /merge-pr.
Prepared head SHA: 7533b85156
Co-authored-by: gumadeiras <5599352+gumadeiras@users.noreply.github.com>
Co-authored-by: shakkernerd <165377636+shakkernerd@users.noreply.github.com>
Reviewed-by: @shakkernerd
148 lines
5.1 KiB
TypeScript
148 lines
5.1 KiB
TypeScript
import { Command } from "commander";
|
|
import { describe, expect, it, vi } from "vitest";
|
|
import type { ProgramContext } from "./context.js";
|
|
|
|
// Perf: `registerCoreCliByName(...)` dynamically imports registrar modules.
|
|
// Mock the heavy registrars so this suite stays focused on command-registry wiring.
|
|
vi.mock("./register.agent.js", () => ({
|
|
registerAgentCommands: (program: Command) => {
|
|
program.command("agent");
|
|
program.command("agents");
|
|
},
|
|
}));
|
|
|
|
vi.mock("./register.maintenance.js", () => ({
|
|
registerMaintenanceCommands: (program: Command) => {
|
|
program.command("doctor");
|
|
program.command("dashboard");
|
|
program.command("reset");
|
|
program.command("uninstall");
|
|
},
|
|
}));
|
|
|
|
const {
|
|
getCoreCliCommandNames,
|
|
getCoreCliCommandsWithSubcommands,
|
|
registerCoreCliByName,
|
|
registerCoreCliCommands,
|
|
} = await import("./command-registry.js");
|
|
|
|
vi.mock("./register.status-health-sessions.js", () => ({
|
|
registerStatusHealthSessionsCommands: (program: Command) => {
|
|
program.command("status");
|
|
program.command("health");
|
|
program.command("sessions");
|
|
},
|
|
}));
|
|
|
|
const testProgramContext: ProgramContext = {
|
|
programVersion: "0.0.0-test",
|
|
channelOptions: [],
|
|
messageChannelOptions: "",
|
|
agentChannelOptions: "web",
|
|
};
|
|
|
|
describe("command-registry", () => {
|
|
const createProgram = () => new Command();
|
|
const namesOf = (program: Command) => program.commands.map((command) => command.name());
|
|
|
|
const withProcessArgv = async (argv: string[], run: () => Promise<void>) => {
|
|
const prevArgv = process.argv;
|
|
process.argv = argv;
|
|
try {
|
|
await run();
|
|
} finally {
|
|
process.argv = prevArgv;
|
|
}
|
|
};
|
|
|
|
it("includes both agent and agents in core CLI command names", () => {
|
|
const names = getCoreCliCommandNames();
|
|
expect(names).toContain("agent");
|
|
expect(names).toContain("agents");
|
|
});
|
|
|
|
it("returns only commands that support subcommands", () => {
|
|
const names = getCoreCliCommandsWithSubcommands();
|
|
expect(names).toContain("config");
|
|
expect(names).toContain("memory");
|
|
expect(names).toContain("agents");
|
|
expect(names).toContain("browser");
|
|
expect(names).toContain("sessions");
|
|
expect(names).not.toContain("agent");
|
|
expect(names).not.toContain("status");
|
|
expect(names).not.toContain("doctor");
|
|
});
|
|
|
|
it("registerCoreCliByName resolves agents to the agent entry", async () => {
|
|
const program = createProgram();
|
|
const found = await registerCoreCliByName(program, testProgramContext, "agents");
|
|
expect(found).toBe(true);
|
|
const agentsCmd = program.commands.find((c) => c.name() === "agents");
|
|
expect(agentsCmd).toBeDefined();
|
|
// The registrar also installs the singular "agent" command from the same entry.
|
|
const agentCmd = program.commands.find((c) => c.name() === "agent");
|
|
expect(agentCmd).toBeDefined();
|
|
});
|
|
|
|
it("registerCoreCliByName returns false for unknown commands", async () => {
|
|
const program = createProgram();
|
|
const found = await registerCoreCliByName(program, testProgramContext, "nonexistent");
|
|
expect(found).toBe(false);
|
|
});
|
|
|
|
it("registers doctor placeholder for doctor primary command", () => {
|
|
const program = createProgram();
|
|
registerCoreCliCommands(program, testProgramContext, ["node", "openclaw", "doctor"]);
|
|
|
|
expect(namesOf(program)).toEqual(["doctor"]);
|
|
});
|
|
|
|
it("does not narrow to the primary command when help is requested", () => {
|
|
const program = createProgram();
|
|
registerCoreCliCommands(program, testProgramContext, ["node", "openclaw", "doctor", "--help"]);
|
|
|
|
const names = namesOf(program);
|
|
expect(names).toContain("doctor");
|
|
expect(names).toContain("status");
|
|
expect(names.length).toBeGreaterThan(1);
|
|
});
|
|
|
|
it("treats maintenance commands as top-level builtins", async () => {
|
|
const program = createProgram();
|
|
|
|
expect(await registerCoreCliByName(program, testProgramContext, "doctor")).toBe(true);
|
|
|
|
const names = getCoreCliCommandNames();
|
|
expect(names).toContain("doctor");
|
|
expect(names).toContain("dashboard");
|
|
expect(names).toContain("reset");
|
|
expect(names).toContain("uninstall");
|
|
expect(names).not.toContain("maintenance");
|
|
});
|
|
|
|
it("registers grouped core entry placeholders without duplicate command errors", async () => {
|
|
const program = createProgram();
|
|
registerCoreCliCommands(program, testProgramContext, ["node", "openclaw", "vitest"]);
|
|
program.exitOverride();
|
|
await withProcessArgv(["node", "openclaw", "status"], async () => {
|
|
await program.parseAsync(["node", "openclaw", "status"]);
|
|
});
|
|
|
|
const names = namesOf(program);
|
|
expect(names).toContain("status");
|
|
expect(names).toContain("health");
|
|
expect(names).toContain("sessions");
|
|
});
|
|
|
|
it("replaces placeholders when loading a grouped entry by secondary command name", async () => {
|
|
const program = createProgram();
|
|
registerCoreCliCommands(program, testProgramContext, ["node", "openclaw", "doctor"]);
|
|
expect(namesOf(program)).toEqual(["doctor"]);
|
|
|
|
const found = await registerCoreCliByName(program, testProgramContext, "dashboard");
|
|
expect(found).toBe(true);
|
|
expect(namesOf(program)).toEqual(["doctor", "dashboard", "reset", "uninstall"]);
|
|
});
|
|
});
|