Files
openclaw/src/plugins/provider-validation.test.ts
2026-03-28 04:28:54 +00:00

237 lines
6.9 KiB
TypeScript

import { describe, expect, it } from "vitest";
import { normalizeRegisteredProvider } from "./provider-validation.js";
import type { PluginDiagnostic, ProviderPlugin } from "./types.js";
function collectDiagnostics() {
const diagnostics: PluginDiagnostic[] = [];
return {
diagnostics,
pushDiagnostic: (diag: PluginDiagnostic) => {
diagnostics.push(diag);
},
};
}
function makeProvider(overrides: Partial<ProviderPlugin>): ProviderPlugin {
return {
id: "demo",
label: "Demo",
auth: [],
...overrides,
};
}
function expectDiagnosticMessages(
diagnostics: PluginDiagnostic[],
expectedDiagnostics: ReadonlyArray<{ level: PluginDiagnostic["level"]; message: string }>,
) {
expect(diagnostics.map((diag) => ({ level: diag.level, message: diag.message }))).toEqual(
expectedDiagnostics,
);
}
function expectDiagnosticText(diagnostics: PluginDiagnostic[], messages: readonly string[]) {
expect(diagnostics.map((diag) => diag.message)).toEqual([...messages]);
}
function normalizeProviderFixture(provider: ProviderPlugin) {
const { diagnostics, pushDiagnostic } = collectDiagnostics();
const normalizedProvider = normalizeRegisteredProvider({
pluginId: "demo-plugin",
source: "/tmp/demo/index.ts",
provider,
pushDiagnostic,
});
return {
diagnostics,
provider: normalizedProvider,
};
}
function expectNormalizedProviderFixture(params: {
provider: ProviderPlugin;
expectedProvider?: Record<string, unknown>;
expectedDiagnostics?: ReadonlyArray<{ level: PluginDiagnostic["level"]; message: string }>;
expectedDiagnosticText?: readonly string[];
}) {
const result = normalizeProviderFixture(params.provider);
if (params.expectedProvider) {
expect(result.provider).toMatchObject(params.expectedProvider);
}
if (params.expectedDiagnostics) {
expectDiagnosticMessages(result.diagnostics, params.expectedDiagnostics);
}
if (params.expectedDiagnosticText) {
expectDiagnosticText(result.diagnostics, params.expectedDiagnosticText);
}
return result;
}
function expectProviderNormalizationResult(params: {
provider: ProviderPlugin;
expectedProvider?: Record<string, unknown>;
expectedDiagnostics?: ReadonlyArray<{ level: PluginDiagnostic["level"]; message: string }>;
expectedDiagnosticText?: readonly string[];
assert?: (
provider: ReturnType<typeof normalizeRegisteredProvider>,
diagnostics: PluginDiagnostic[],
) => void;
}) {
const { diagnostics, provider } = expectNormalizedProviderFixture(params);
params.assert?.(provider, diagnostics);
}
describe("normalizeRegisteredProvider", () => {
it.each([
{
name: "drops invalid and duplicate auth methods, and clears bad wizard method bindings",
provider: makeProvider({
id: " demo ",
label: " Demo Provider ",
aliases: [" alias-one ", "alias-one", ""],
deprecatedProfileIds: [" demo:legacy ", "demo:legacy", ""],
envVars: [" DEMO_API_KEY ", "DEMO_API_KEY"],
auth: [
{
id: " primary ",
label: " Primary ",
kind: "custom",
wizard: {
choiceId: " demo-primary ",
modelAllowlist: {
allowedKeys: [" demo/model ", "demo/model"],
initialSelections: [" demo/model "],
message: " Demo models ",
},
},
run: async () => ({ profiles: [] }),
},
{
id: "primary",
label: "Duplicate",
kind: "custom",
run: async () => ({ profiles: [] }),
},
{ id: " ", label: "Missing", kind: "custom", run: async () => ({ profiles: [] }) },
],
wizard: {
setup: {
choiceId: " demo-choice ",
methodId: " missing ",
},
modelPicker: {
label: " Demo models ",
methodId: " missing ",
},
},
}),
expectedProvider: {
id: "demo",
label: "Demo Provider",
aliases: ["alias-one"],
deprecatedProfileIds: ["demo:legacy"],
envVars: ["DEMO_API_KEY"],
auth: [
{
id: "primary",
label: "Primary",
wizard: {
choiceId: "demo-primary",
modelAllowlist: {
allowedKeys: ["demo/model"],
initialSelections: ["demo/model"],
message: "Demo models",
},
},
},
],
wizard: {
setup: {
choiceId: "demo-choice",
},
modelPicker: {
label: "Demo models",
},
},
},
expectedDiagnostics: [
{
level: "error",
message: 'provider "demo" auth method duplicated id "primary"',
},
{
level: "error",
message: 'provider "demo" auth method missing id',
},
{
level: "warn",
message:
'provider "demo" setup method "missing" not found; falling back to available methods',
},
{
level: "warn",
message:
'provider "demo" model-picker method "missing" not found; falling back to available methods',
},
],
},
{
name: "drops wizard metadata when a provider has no auth methods",
provider: makeProvider({
wizard: {
setup: {
choiceId: "demo",
},
modelPicker: {
label: "Demo",
},
},
}),
assert: (
provider: ReturnType<typeof normalizeRegisteredProvider>,
diagnostics: PluginDiagnostic[],
) => {
expect(provider?.wizard).toBeUndefined();
expectDiagnosticText(diagnostics, [
'provider "demo" setup metadata ignored because it has no auth methods',
'provider "demo" model-picker metadata ignored because it has no auth methods',
]);
},
},
] as const)(
"$name",
({ provider: inputProvider, expectedProvider, expectedDiagnostics, assert }) => {
expectProviderNormalizationResult({
provider: inputProvider,
...(expectedProvider ? { expectedProvider } : {}),
...(expectedDiagnostics ? { expectedDiagnostics } : {}),
...(assert ? { assert } : {}),
});
},
);
it("prefers catalog when a provider registers both catalog and discovery", () => {
const { provider } = expectNormalizedProviderFixture({
provider: makeProvider({
catalog: {
run: async () => null,
},
discovery: {
run: async () => ({
provider: {
baseUrl: "http://127.0.0.1:8000/v1",
models: [],
},
}),
},
}),
expectedDiagnosticText: [
'provider "demo" registered both catalog and discovery; using catalog',
],
});
expect(provider?.catalog).toBeDefined();
expect(provider?.discovery).toBeUndefined();
});
});