mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-06 05:51:15 +08:00
143 lines
3.9 KiB
TypeScript
143 lines
3.9 KiB
TypeScript
// Vercel Ai Gateway tests cover provider catalog plugin behavior.
|
|
import { afterEach, describe, expect, it, vi } from "vitest";
|
|
|
|
const { fetchWithSsrFGuardMock } = vi.hoisted(() => ({
|
|
fetchWithSsrFGuardMock: vi.fn(),
|
|
}));
|
|
|
|
vi.mock("openclaw/plugin-sdk/ssrf-runtime", () => ({
|
|
fetchWithSsrFGuard: fetchWithSsrFGuardMock,
|
|
}));
|
|
|
|
import {
|
|
discoverVercelAiGatewayModels,
|
|
getStaticVercelAiGatewayModelCatalog,
|
|
VERCEL_AI_GATEWAY_BASE_URL,
|
|
VERCEL_AI_GATEWAY_DEFAULT_CONTEXT_WINDOW,
|
|
VERCEL_AI_GATEWAY_DEFAULT_MAX_TOKENS,
|
|
} from "./api.js";
|
|
import {
|
|
buildStaticVercelAiGatewayProvider,
|
|
buildVercelAiGatewayProvider,
|
|
} from "./provider-catalog.js";
|
|
|
|
const STATIC_MODEL_IDS = [
|
|
"anthropic/claude-opus-4.6",
|
|
"openai/gpt-5.4",
|
|
"openai/gpt-5.4-pro",
|
|
"moonshotai/kimi-k2.6",
|
|
];
|
|
|
|
function restoreEnvVar(name: "NODE_ENV" | "VITEST", value: string | undefined): void {
|
|
if (value === undefined) {
|
|
delete process.env[name];
|
|
} else {
|
|
process.env[name] = value;
|
|
}
|
|
}
|
|
|
|
async function withLiveDiscovery<T>(run: () => Promise<T>): Promise<T> {
|
|
const oldNodeEnv = process.env.NODE_ENV;
|
|
const oldVitest = process.env.VITEST;
|
|
delete process.env.NODE_ENV;
|
|
delete process.env.VITEST;
|
|
try {
|
|
return await run();
|
|
} finally {
|
|
restoreEnvVar("NODE_ENV", oldNodeEnv);
|
|
restoreEnvVar("VITEST", oldVitest);
|
|
}
|
|
}
|
|
|
|
afterEach(() => {
|
|
fetchWithSsrFGuardMock.mockReset();
|
|
});
|
|
|
|
describe("vercel ai gateway provider catalog", () => {
|
|
it("builds the bundled Vercel AI Gateway defaults", async () => {
|
|
const provider = await buildVercelAiGatewayProvider();
|
|
|
|
expect(provider).toStrictEqual({
|
|
baseUrl: VERCEL_AI_GATEWAY_BASE_URL,
|
|
api: "anthropic-messages",
|
|
models: getStaticVercelAiGatewayModelCatalog(),
|
|
});
|
|
});
|
|
|
|
it("exposes the static fallback model catalog", () => {
|
|
expect(getStaticVercelAiGatewayModelCatalog().map((model) => model.id)).toStrictEqual(
|
|
STATIC_MODEL_IDS,
|
|
);
|
|
});
|
|
|
|
it("builds an offline static provider catalog", () => {
|
|
expect(buildStaticVercelAiGatewayProvider()).toStrictEqual({
|
|
baseUrl: VERCEL_AI_GATEWAY_BASE_URL,
|
|
api: "anthropic-messages",
|
|
models: getStaticVercelAiGatewayModelCatalog(),
|
|
});
|
|
});
|
|
|
|
it("falls back to the static catalog for malformed successful model list payloads", async () => {
|
|
for (const payload of [[], { data: {} }, { data: [null] }]) {
|
|
fetchWithSsrFGuardMock.mockResolvedValueOnce({
|
|
response: {
|
|
ok: true,
|
|
status: 200,
|
|
json: async () => payload,
|
|
},
|
|
release: async () => {},
|
|
});
|
|
|
|
await withLiveDiscovery(async () => {
|
|
expect(await discoverVercelAiGatewayModels()).toStrictEqual(
|
|
getStaticVercelAiGatewayModelCatalog(),
|
|
);
|
|
});
|
|
}
|
|
});
|
|
|
|
it("falls back from malformed live token metadata", async () => {
|
|
fetchWithSsrFGuardMock.mockResolvedValueOnce({
|
|
response: {
|
|
ok: true,
|
|
status: 200,
|
|
json: async () => ({
|
|
data: [
|
|
{
|
|
id: "anthropic/claude-opus-4.6",
|
|
name: "Claude Opus 4.6",
|
|
context_window: -1,
|
|
max_tokens: 128_000.5,
|
|
tags: ["vision", "reasoning"],
|
|
},
|
|
{
|
|
id: "custom/provider-model",
|
|
name: "Custom model",
|
|
context_window: Number.POSITIVE_INFINITY,
|
|
max_tokens: 0,
|
|
tags: ["reasoning"],
|
|
},
|
|
],
|
|
}),
|
|
},
|
|
release: async () => {},
|
|
});
|
|
|
|
await withLiveDiscovery(async () => {
|
|
const models = await discoverVercelAiGatewayModels();
|
|
|
|
expect(models[0]).toMatchObject({
|
|
id: "anthropic/claude-opus-4.6",
|
|
contextWindow: 1_000_000,
|
|
maxTokens: 128_000,
|
|
});
|
|
expect(models[1]).toMatchObject({
|
|
id: "custom/provider-model",
|
|
contextWindow: VERCEL_AI_GATEWAY_DEFAULT_CONTEXT_WINDOW,
|
|
maxTokens: VERCEL_AI_GATEWAY_DEFAULT_MAX_TOKENS,
|
|
});
|
|
});
|
|
});
|
|
});
|