mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 20:10:42 +00:00
* feat(nvidia): add NVIDIA provider with onboarding flow Add the NVIDIA build.nvidia.com API as a bundled provider. Default model is nvidia/nvidia/nemotron-3-super-120b-a12b: first segment is the provider id, remaining "nvidia/nemotron-3-super-120b-a12b" is the literal upstream model id (which happens to start with "nvidia/" because NVIDIA is also the model maker). Supporting core change: introduce a provider capability flag nativeIdsIncludeProviderPrefix so providers whose native catalog ids intentionally include their provider prefix (OpenRouter) opt into self-prefix dedupe in modelKey, without hardcoding provider names in core. Providers whose ids merely happen to start with their own name (NVIDIA) leave the flag unset and get the full <provider>/<model-id> concatenation. - extensions/nvidia/*: new plugin, catalog, onboarding, tests, docs - extensions/openrouter/index.ts: declare nativeIdsIncludeProviderPrefix - src/plugins/types.ts: add field to ProviderPlugin - src/plugins/registry.ts: populate self-prefix set on registration - src/agents/provider-self-prefix.ts: sync accessor used by modelKey - src/agents/model-ref-shared.ts: modelKey consults the flag - test updates for affected surfaces Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * refactor(model-picker): simplify literal-prefix display to label-only * fix(model-picker): pass workspaceDir/env to allowlist literal-prefix resolution * chore: untrack generated baseline JSON artifacts (gitignored) * fix(nvidia): show literal model ref in picker and onboarding notes * fix(nvidia): show hint whenever display label differs from stored config * fix(nvidia): drop redundant hint from Keep current label * fix(nvidia): restore literal double-prefix display labels * fix(picker): handle literal-prefix fast path * fix(picker): show literal keep label * fix(docs): update nvidia provider docs * fix(nvidia): update test helper imports * fix(changelog): add nvidia provider entry --------- Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
158 lines
4.8 KiB
TypeScript
158 lines
4.8 KiB
TypeScript
import fs from "node:fs";
|
|
import {
|
|
registerSingleProviderPlugin,
|
|
resolveProviderPluginChoice,
|
|
} from "openclaw/plugin-sdk/plugin-test-runtime";
|
|
import { describe, expect, it } from "vitest";
|
|
import plugin from "./index.js";
|
|
|
|
type NvidiaManifest = {
|
|
providerAuthChoices?: Array<{ choiceId?: string; method?: string; provider?: string }>;
|
|
};
|
|
|
|
function readManifest(): NvidiaManifest {
|
|
return JSON.parse(
|
|
fs.readFileSync(new URL("./openclaw.plugin.json", import.meta.url), "utf8"),
|
|
) as NvidiaManifest;
|
|
}
|
|
|
|
async function registerNvidiaProvider() {
|
|
return registerSingleProviderPlugin(plugin);
|
|
}
|
|
|
|
describe("nvidia provider hooks", () => {
|
|
it("registers the nvidia provider with correct metadata", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
expect(provider.id).toBe("nvidia");
|
|
expect(provider.label).toBe("NVIDIA");
|
|
expect(provider.docsPath).toBe("/providers/nvidia");
|
|
expect(provider.envVars).toEqual(["NVIDIA_API_KEY"]);
|
|
});
|
|
|
|
it("registers API-key auth choice metadata", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
expect(provider.auth?.map((method) => method.id)).toEqual(["api-key"]);
|
|
|
|
const choice = resolveProviderPluginChoice({
|
|
providers: [provider],
|
|
choice: "nvidia-api-key",
|
|
});
|
|
expect(choice?.provider.id).toBe("nvidia");
|
|
expect(choice?.method.id).toBe("api-key");
|
|
expect(readManifest().providerAuthChoices).toEqual(
|
|
expect.arrayContaining([
|
|
expect.objectContaining({
|
|
provider: "nvidia",
|
|
method: "api-key",
|
|
choiceId: "nvidia-api-key",
|
|
}),
|
|
]),
|
|
);
|
|
});
|
|
|
|
it("keeps nvidia auth setup metadata aligned", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
expect(
|
|
provider.auth.map((method) => ({
|
|
id: method.id,
|
|
label: method.label,
|
|
hint: method.hint,
|
|
choiceId: method.wizard?.choiceId,
|
|
groupId: method.wizard?.groupId,
|
|
groupLabel: method.wizard?.groupLabel,
|
|
groupHint: method.wizard?.groupHint,
|
|
})),
|
|
).toEqual([
|
|
{
|
|
id: "api-key",
|
|
label: "NVIDIA API key",
|
|
hint: "Direct API key",
|
|
choiceId: "nvidia-api-key",
|
|
groupId: "nvidia",
|
|
groupLabel: "NVIDIA",
|
|
groupHint: "Direct API key",
|
|
},
|
|
]);
|
|
});
|
|
|
|
it("keeps nvidia wizard setup metadata aligned", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
expect(provider.wizard?.setup).toMatchObject({
|
|
choiceId: "nvidia-api-key",
|
|
choiceLabel: "NVIDIA API key",
|
|
groupId: "nvidia",
|
|
groupLabel: "NVIDIA",
|
|
groupHint: "Direct API key",
|
|
methodId: "api-key",
|
|
});
|
|
});
|
|
|
|
it("keeps nvidia model picker metadata aligned", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
expect(provider.wizard?.modelPicker).toMatchObject({
|
|
label: "NVIDIA (custom)",
|
|
hint: "Use NVIDIA-hosted open models",
|
|
methodId: "api-key",
|
|
});
|
|
});
|
|
|
|
it("does not override replay policy for standard openai-compatible transport", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
// NVIDIA uses standard OpenAI-compatible API without custom replay logic
|
|
expect(provider.buildReplayPolicy).toBeUndefined();
|
|
});
|
|
|
|
it("does not override stream wrapper for standard models", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
// NVIDIA uses standard streaming without custom wrappers
|
|
expect(provider.wrapStreamFn).toBeUndefined();
|
|
});
|
|
|
|
it("surfaces the bundled NVIDIA models via augmentModelCatalog", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
const entries = await provider.augmentModelCatalog?.({
|
|
env: process.env,
|
|
entries: [],
|
|
});
|
|
|
|
expect(entries?.map((entry) => entry.id)).toEqual([
|
|
"nvidia/nemotron-3-super-120b-a12b",
|
|
"moonshotai/kimi-k2.5",
|
|
"minimaxai/minimax-m2.5",
|
|
"z-ai/glm5",
|
|
]);
|
|
expect(entries?.every((entry) => entry.provider === "nvidia")).toBe(true);
|
|
});
|
|
|
|
it("opts into literal provider-prefix preservation", async () => {
|
|
const provider = await registerNvidiaProvider();
|
|
|
|
// NVIDIA's ids like nvidia/nemotron-... sit alongside moonshotai/...,
|
|
// minimaxai/..., z-ai/... in the same catalog, so the leading nvidia/
|
|
// is a vendor namespace rather than a redundant provider prefix. The
|
|
// flag keeps the canonical ref as nvidia/nvidia/nemotron-... instead
|
|
// of letting the default string-based dedupe collapse it.
|
|
expect(provider.preserveLiteralProviderPrefix).toBe(true);
|
|
});
|
|
|
|
it("registers nvidia provider through the plugin api", () => {
|
|
const registeredProviders: string[] = [];
|
|
|
|
plugin.register({
|
|
registerProvider(provider: { id: string }) {
|
|
registeredProviders.push(provider.id);
|
|
},
|
|
} as any);
|
|
|
|
expect(registeredProviders).toContain("nvidia");
|
|
});
|
|
});
|