mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-16 04:20:44 +00:00
* CLI argv: add strict root help invocation guard * Entry: add root help fast-path bootstrap bypass * CLI context: lazily resolve channel options * CLI context tests: cover lazy channel option resolution * CLI argv tests: cover root help invocation detection * Changelog: note additional startup path optimizations * Changelog: split startup follow-up into #30975 entry * CLI channel options: load precomputed startup metadata * CLI channel options tests: cover precomputed metadata path * Build: generate CLI startup metadata during build * Build script: invoke CLI startup metadata generator * CLI routes: preload plugins for routed health * CLI routes tests: assert health plugin preload * CLI: add experimental bundled entry and snapshot helper * Tools: compare CLI startup entries in benchmark script * Docs: add startup tuning notes for Pi and VM hosts * CLI: drop bundled entry runtime toggle * Build: remove bundled and snapshot scripts * Tools: remove bundled-entry benchmark shortcut * Docs: remove bundled startup bench examples * Docs: remove Pi bundled entry mention * Docs: remove VM bundled entry mention * Changelog: remove bundled startup follow-up claims * Build: remove snapshot helper script * Build: remove CLI bundle tsdown config * Doctor: add low-power startup optimization hints * Doctor: run startup optimization hint checks * Doctor tests: cover startup optimization host targeting * Doctor tests: mock startup optimization note export * CLI argv: require strict root-only help fast path * CLI argv tests: cover mixed root-help invocations * CLI channel options: merge metadata with runtime catalog * CLI channel options tests: assert dynamic catalog merge * Changelog: align #30975 startup follow-up scope * Docs tests: remove secondary-entry startup bench note * Docs Pi: add systemd recovery reference link * Docs VPS: add systemd recovery reference link
99 lines
3.3 KiB
TypeScript
99 lines
3.3 KiB
TypeScript
import { afterEach, describe, expect, it, vi } from "vitest";
|
|
|
|
const readFileSyncMock = vi.hoisted(() => vi.fn());
|
|
const listCatalogMock = vi.hoisted(() => vi.fn());
|
|
const listPluginsMock = vi.hoisted(() => vi.fn());
|
|
const ensurePluginRegistryLoadedMock = vi.hoisted(() => vi.fn());
|
|
|
|
vi.mock("node:fs", async () => {
|
|
const actual = await vi.importActual<typeof import("node:fs")>("node:fs");
|
|
const base = ("default" in actual ? actual.default : actual) as Record<string, unknown>;
|
|
return {
|
|
...actual,
|
|
default: {
|
|
...base,
|
|
readFileSync: readFileSyncMock,
|
|
},
|
|
readFileSync: readFileSyncMock,
|
|
};
|
|
});
|
|
|
|
vi.mock("../channels/registry.js", () => ({
|
|
CHAT_CHANNEL_ORDER: ["telegram", "discord"],
|
|
}));
|
|
|
|
vi.mock("../channels/plugins/catalog.js", () => ({
|
|
listChannelPluginCatalogEntries: listCatalogMock,
|
|
}));
|
|
|
|
vi.mock("../channels/plugins/index.js", () => ({
|
|
listChannelPlugins: listPluginsMock,
|
|
}));
|
|
|
|
vi.mock("./plugin-registry.js", () => ({
|
|
ensurePluginRegistryLoaded: ensurePluginRegistryLoadedMock,
|
|
}));
|
|
|
|
async function loadModule() {
|
|
return await import("./channel-options.js");
|
|
}
|
|
|
|
describe("resolveCliChannelOptions", () => {
|
|
afterEach(() => {
|
|
delete process.env.OPENCLAW_EAGER_CHANNEL_OPTIONS;
|
|
vi.resetModules();
|
|
vi.clearAllMocks();
|
|
});
|
|
|
|
it("uses precomputed startup metadata when available", async () => {
|
|
readFileSyncMock.mockReturnValue(
|
|
JSON.stringify({ channelOptions: ["cached", "telegram", "cached"] }),
|
|
);
|
|
listCatalogMock.mockReturnValue([{ id: "catalog-only" }]);
|
|
|
|
const mod = await loadModule();
|
|
expect(mod.resolveCliChannelOptions()).toEqual(["cached", "telegram", "catalog-only"]);
|
|
expect(listCatalogMock).toHaveBeenCalledOnce();
|
|
});
|
|
|
|
it("falls back to dynamic catalog resolution when metadata is missing", async () => {
|
|
readFileSyncMock.mockImplementation(() => {
|
|
throw new Error("ENOENT");
|
|
});
|
|
listCatalogMock.mockReturnValue([{ id: "feishu" }, { id: "telegram" }]);
|
|
|
|
const mod = await loadModule();
|
|
expect(mod.resolveCliChannelOptions()).toEqual(["telegram", "discord", "feishu"]);
|
|
expect(listCatalogMock).toHaveBeenCalledOnce();
|
|
});
|
|
|
|
it("respects eager mode and includes loaded plugin ids", async () => {
|
|
process.env.OPENCLAW_EAGER_CHANNEL_OPTIONS = "1";
|
|
readFileSyncMock.mockReturnValue(JSON.stringify({ channelOptions: ["cached"] }));
|
|
listCatalogMock.mockReturnValue([{ id: "zalo" }]);
|
|
listPluginsMock.mockReturnValue([{ id: "custom-a" }, { id: "custom-b" }]);
|
|
|
|
const mod = await loadModule();
|
|
expect(mod.resolveCliChannelOptions()).toEqual([
|
|
"telegram",
|
|
"discord",
|
|
"zalo",
|
|
"custom-a",
|
|
"custom-b",
|
|
]);
|
|
expect(ensurePluginRegistryLoadedMock).toHaveBeenCalledOnce();
|
|
expect(listPluginsMock).toHaveBeenCalledOnce();
|
|
});
|
|
|
|
it("keeps dynamic catalog resolution when external catalog env is set", async () => {
|
|
process.env.OPENCLAW_PLUGIN_CATALOG_PATHS = "/tmp/plugins-catalog.json";
|
|
readFileSyncMock.mockReturnValue(JSON.stringify({ channelOptions: ["cached", "telegram"] }));
|
|
listCatalogMock.mockReturnValue([{ id: "custom-catalog" }]);
|
|
|
|
const mod = await loadModule();
|
|
expect(mod.resolveCliChannelOptions()).toEqual(["cached", "telegram", "custom-catalog"]);
|
|
expect(listCatalogMock).toHaveBeenCalledOnce();
|
|
delete process.env.OPENCLAW_PLUGIN_CATALOG_PATHS;
|
|
});
|
|
});
|