Files
openclaw/src/wizard/setup.plugin-config.test.ts
Josh Lehman 2d846e1f1a fix: coerce integer plugin config input (#63346)
* Wizard: coerce integer plugin config input

Regeneration-Prompt: |
  Fix the interactive plugin-config wizard so JSON Schema fields declared as type "integer" are coerced from text input the same way type "number" already is. Keep the change narrow in src/wizard/setup.plugin-config.ts rather than refactoring the broader prompt flow. Add a focused regression test in src/wizard/setup.plugin-config.test.ts that exercises setupPluginConfig with an integer-typed schema field, verifies the text response "3" is stored as numeric 3, and run only the relevant wizard test slice before committing.

* Wizard: type select mock in setup plugin config test

Regeneration-Prompt: |
  Fix the CI type failure on PR #63346 in src/wizard/setup.plugin-config.test.ts with the smallest possible change. The new integer-coercion test needs its mocked prompter to satisfy the generic WizardPrompter select signature, matching the surrounding test style without changing production code or test behavior. After the one-line test fix, rerun pnpm tsgo --pretty false and pnpm test src/wizard/setup.plugin-config.test.ts on branch aristotle-3f605963-fix-config-integer-coercion.

* Wizard: coerce integer plugin config input

* Changelog: remove stray conflict marker
2026-04-09 13:57:06 -07:00

386 lines
10 KiB
TypeScript

import { describe, expect, it, vi } from "vitest";
import type { OpenClawConfig } from "../config/config.js";
import type { PluginConfigUiHint } from "../plugins/types.js";
import type { WizardPrompter } from "./prompts.js";
import {
discoverConfigurablePlugins,
discoverUnconfiguredPlugins,
setupPluginConfig,
} from "./setup.plugin-config.js";
const loadPluginManifestRegistry = vi.fn();
vi.mock("../plugins/manifest-registry.js", () => ({
loadPluginManifestRegistry,
}));
function makeManifestPlugin(
id: string,
uiHints?: Record<string, PluginConfigUiHint>,
configSchema?: Record<string, unknown>,
) {
return {
id,
name: id,
configUiHints: uiHints,
configSchema,
enabled: true,
enabledByDefault: true,
};
}
describe("discoverConfigurablePlugins", () => {
it("returns plugins with non-advanced uiHints", () => {
const plugins = [
makeManifestPlugin("openshell", {
mode: { label: "Mode", help: "Sandbox mode" },
gateway: { label: "Gateway", help: "Gateway name" },
gpu: { label: "GPU", advanced: true },
}),
];
const result = discoverConfigurablePlugins({ manifestPlugins: plugins });
expect(result).toHaveLength(1);
expect(result[0]).toBeDefined();
expect(result[0].id).toBe("openshell");
expect(Object.keys(result[0].uiHints)).toEqual(["mode", "gateway"]);
// Advanced field excluded
expect(result[0].uiHints.gpu).toBeUndefined();
});
it("excludes plugins with no uiHints", () => {
const plugins = [makeManifestPlugin("bare-plugin")];
const result = discoverConfigurablePlugins({ manifestPlugins: plugins });
expect(result).toHaveLength(0);
});
it("excludes sensitive fields from promptable hints", () => {
const plugins = [
makeManifestPlugin("secret-plugin", {
endpoint: { label: "Endpoint" },
apiKey: { label: "API Key", sensitive: true },
}),
];
const result = discoverConfigurablePlugins({ manifestPlugins: plugins });
expect(result).toHaveLength(1);
// sensitive fields are still included in uiHints for discovery —
// they are skipped at prompt time, not at discovery time
expect(result[0].uiHints.endpoint).toBeDefined();
expect(result[0].uiHints.apiKey).toBeDefined();
});
it("excludes plugins where all fields are advanced", () => {
const plugins = [
makeManifestPlugin("all-advanced", {
gpu: { label: "GPU", advanced: true },
timeout: { label: "Timeout", advanced: true },
}),
];
const result = discoverConfigurablePlugins({ manifestPlugins: plugins });
expect(result).toHaveLength(0);
});
it("sorts results alphabetically by name", () => {
const plugins = [
makeManifestPlugin("zeta", { a: { label: "A" } }),
makeManifestPlugin("alpha", { b: { label: "B" } }),
];
const result = discoverConfigurablePlugins({ manifestPlugins: plugins });
expect(result.map((p) => p.id)).toEqual(["alpha", "zeta"]);
});
});
describe("discoverUnconfiguredPlugins", () => {
it("returns plugins with at least one unconfigured field", () => {
const plugins = [
makeManifestPlugin("openshell", {
mode: { label: "Mode" },
gateway: { label: "Gateway" },
}),
];
const config: OpenClawConfig = {
plugins: {
entries: {
openshell: {
config: { mode: "mirror" },
},
},
},
};
const result = discoverUnconfiguredPlugins({
manifestPlugins: plugins,
config,
});
// gateway is unconfigured
expect(result).toHaveLength(1);
expect(result[0]).toBeDefined();
expect(result[0].id).toBe("openshell");
});
it("excludes plugins where all fields are configured", () => {
const plugins = [
makeManifestPlugin("openshell", {
mode: { label: "Mode" },
gateway: { label: "Gateway" },
}),
];
const config: OpenClawConfig = {
plugins: {
entries: {
openshell: {
config: { mode: "mirror", gateway: "my-gw" },
},
},
},
};
const result = discoverUnconfiguredPlugins({
manifestPlugins: plugins,
config,
});
expect(result).toHaveLength(0);
});
it("treats empty string as unconfigured", () => {
const plugins = [
makeManifestPlugin("test-plugin", {
endpoint: { label: "Endpoint" },
}),
];
const config: OpenClawConfig = {
plugins: {
entries: {
"test-plugin": {
config: { endpoint: "" },
},
},
},
};
const result = discoverUnconfiguredPlugins({
manifestPlugins: plugins,
config,
});
expect(result).toHaveLength(1);
});
it("returns empty when no plugins have uiHints", () => {
const plugins = [makeManifestPlugin("bare")];
const result = discoverUnconfiguredPlugins({
manifestPlugins: plugins,
config: {},
});
expect(result).toHaveLength(0);
});
it("treats dotted uiHint paths as configured when nested config exists", () => {
const plugins = [
makeManifestPlugin(
"brave",
{
"webSearch.mode": { label: "Brave Search Mode" },
},
{
type: "object",
properties: {
webSearch: {
type: "object",
properties: {
mode: {
type: "string",
enum: ["web", "llm-context"],
},
},
},
},
},
),
];
const config: OpenClawConfig = {
plugins: {
entries: {
brave: {
config: {
webSearch: {
mode: "llm-context",
},
},
},
},
},
};
const result = discoverUnconfiguredPlugins({
manifestPlugins: plugins,
config,
});
expect(result).toHaveLength(0);
});
});
describe("setupPluginConfig", () => {
it("allows skipping plugin setup from the multiselect prompt", async () => {
loadPluginManifestRegistry.mockReturnValue({
plugins: [
{
...makeManifestPlugin("device-pairing", {
enabled: { label: "Enable pairing" },
}),
enabledByDefault: true,
},
],
});
const note = vi.fn(async () => {});
const select = vi.fn(async () => {
throw new Error("select should not run when plugin setup is skipped");
});
const text = vi.fn(async () => {
throw new Error("text should not run when plugin setup is skipped");
});
const confirm = vi.fn(async () => {
throw new Error("confirm should not run when plugin setup is skipped");
});
const result = await setupPluginConfig({
config: {
plugins: {
entries: {
"device-pairing": {
enabled: true,
},
},
},
},
prompter: {
intro: vi.fn(async () => {}),
outro: vi.fn(async () => {}),
note,
select: select as unknown as WizardPrompter["select"],
multiselect: vi.fn(async () => ["__skip__"]) as unknown as WizardPrompter["multiselect"],
text,
confirm,
progress: vi.fn(() => ({ update: vi.fn(), stop: vi.fn() })),
},
});
expect(result).toEqual({
plugins: {
entries: {
"device-pairing": {
enabled: true,
},
},
},
});
expect(note).not.toHaveBeenCalled();
});
it("writes dotted uiHint values into nested plugin config", async () => {
loadPluginManifestRegistry.mockReturnValue({
plugins: [
{
...makeManifestPlugin(
"brave",
{
"webSearch.mode": { label: "Brave Search Mode" },
},
{
type: "object",
additionalProperties: false,
properties: {
webSearch: {
type: "object",
additionalProperties: false,
properties: {
mode: {
type: "string",
enum: ["web", "llm-context"],
},
},
},
},
},
),
enabledByDefault: true,
},
],
});
const result = await setupPluginConfig({
config: {
plugins: {
entries: {
brave: {
enabled: true,
},
},
},
},
prompter: {
intro: vi.fn(async () => {}),
outro: vi.fn(async () => {}),
note: vi.fn(async () => {}),
select: vi.fn(async () => "llm-context") as unknown as WizardPrompter["select"],
multiselect: vi.fn(async () => ["brave"]) as unknown as WizardPrompter["multiselect"],
text: vi.fn(async () => ""),
confirm: vi.fn(async () => true),
progress: vi.fn(() => ({ update: vi.fn(), stop: vi.fn() })),
},
});
expect(result.plugins?.entries?.brave?.config).toEqual({
webSearch: {
mode: "llm-context",
},
});
expect(result.plugins?.entries?.brave?.config?.["webSearch.mode"]).toBeUndefined();
});
it("coerces integer schema fields from text input", async () => {
loadPluginManifestRegistry.mockReturnValue({
plugins: [
makeManifestPlugin(
"retry-plugin",
{
retries: { label: "Retries" },
},
{
type: "object",
additionalProperties: false,
properties: {
retries: {
type: "integer",
},
},
},
),
],
});
const result = await setupPluginConfig({
config: {
plugins: {
entries: {
"retry-plugin": {
enabled: true,
},
},
},
},
prompter: {
intro: vi.fn(async () => {}),
outro: vi.fn(async () => {}),
note: vi.fn(async () => {}),
select: vi.fn(async () => "") as unknown as WizardPrompter["select"],
multiselect: vi.fn(async () => [
"retry-plugin",
]) as unknown as WizardPrompter["multiselect"],
text: vi.fn(async () => "3") as unknown as WizardPrompter["text"],
confirm: vi.fn(async () => true),
progress: vi.fn(() => ({ update: vi.fn(), stop: vi.fn() })),
},
});
expect(result.plugins?.entries?.["retry-plugin"]?.config).toEqual({
retries: 3,
});
});
});