mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-12 09:41:11 +00:00
262 lines
8.9 KiB
TypeScript
262 lines
8.9 KiB
TypeScript
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
|
|
type ChutesModelsModule = typeof import("./models.js");
|
|
|
|
let chutesModels: ChutesModelsModule;
|
|
|
|
async function withLiveChutesDiscovery<T>(
|
|
fetchMock: ReturnType<typeof vi.fn>,
|
|
run: () => Promise<T>,
|
|
options?: { now?: string },
|
|
): Promise<T> {
|
|
const oldNodeEnv = process.env.NODE_ENV;
|
|
const oldVitest = process.env.VITEST;
|
|
delete process.env.NODE_ENV;
|
|
delete process.env.VITEST;
|
|
if (options?.now) {
|
|
vi.useFakeTimers();
|
|
vi.setSystemTime(new Date(options.now));
|
|
}
|
|
vi.stubGlobal("fetch", fetchMock);
|
|
|
|
try {
|
|
return await run();
|
|
} finally {
|
|
process.env.NODE_ENV = oldNodeEnv;
|
|
process.env.VITEST = oldVitest;
|
|
vi.unstubAllGlobals();
|
|
if (options?.now) {
|
|
vi.useRealTimers();
|
|
}
|
|
}
|
|
}
|
|
|
|
function createAuthEchoFetchMock() {
|
|
return vi.fn().mockImplementation((_url, init?: { headers?: Record<string, string> }) => {
|
|
const auth = init?.headers?.Authorization ?? "";
|
|
return Promise.resolve({
|
|
ok: true,
|
|
json: async () => ({
|
|
data: [{ id: auth ? `${auth}-model` : "public-model" }],
|
|
}),
|
|
});
|
|
});
|
|
}
|
|
|
|
describe("chutes-models", () => {
|
|
beforeEach(async () => {
|
|
vi.resetModules();
|
|
chutesModels = await import("./models.js");
|
|
});
|
|
|
|
it("buildChutesModelDefinition returns config with required fields", () => {
|
|
const entry = chutesModels.CHUTES_MODEL_CATALOG[0];
|
|
const def = chutesModels.buildChutesModelDefinition(entry);
|
|
expect(def.id).toBe(entry.id);
|
|
expect(def.name).toBe(entry.name);
|
|
expect(def.reasoning).toBe(entry.reasoning);
|
|
expect(def.input).toEqual(entry.input);
|
|
expect(def.cost).toEqual(entry.cost);
|
|
expect(def.contextWindow).toBe(entry.contextWindow);
|
|
expect(def.maxTokens).toBe(entry.maxTokens);
|
|
expect(def.compat?.supportsUsageInStreaming).toBe(false);
|
|
});
|
|
|
|
it("discoverChutesModels returns static catalog when accessToken is empty", async () => {
|
|
const models = await chutesModels.discoverChutesModels("");
|
|
expect(models).toHaveLength(chutesModels.CHUTES_MODEL_CATALOG.length);
|
|
expect(models.map((m) => m.id)).toEqual(chutesModels.CHUTES_MODEL_CATALOG.map((m) => m.id));
|
|
});
|
|
|
|
it("discoverChutesModels returns static catalog in test env by default", async () => {
|
|
const models = await chutesModels.discoverChutesModels("test-token");
|
|
expect(models).toHaveLength(chutesModels.CHUTES_MODEL_CATALOG.length);
|
|
expect(models[0]?.id).toBe("Qwen/Qwen3-32B");
|
|
});
|
|
|
|
it("discoverChutesModels correctly maps API response when not in test env", async () => {
|
|
const mockFetch = vi.fn().mockResolvedValue({
|
|
ok: true,
|
|
json: async () => ({
|
|
data: [
|
|
{ id: "zai-org/GLM-4.7-TEE" },
|
|
{
|
|
id: "new-provider/new-model-r1",
|
|
supported_features: ["reasoning"],
|
|
input_modalities: ["text", "image"],
|
|
context_length: 200000,
|
|
max_output_length: 16384,
|
|
pricing: { prompt: 0.1, completion: 0.2 },
|
|
},
|
|
{ id: "new-provider/simple-model" },
|
|
],
|
|
}),
|
|
});
|
|
await withLiveChutesDiscovery(mockFetch, async () => {
|
|
const models = await chutesModels.discoverChutesModels("test-token-real-fetch");
|
|
expect(models.length).toBeGreaterThan(0);
|
|
if (models.length === 3) {
|
|
expect(models[0]?.id).toBe("zai-org/GLM-4.7-TEE");
|
|
expect(models[1]?.reasoning).toBe(true);
|
|
expect(models[1]?.compat?.supportsUsageInStreaming).toBe(false);
|
|
}
|
|
});
|
|
});
|
|
|
|
it("discoverChutesModels retries without auth on 401", async () => {
|
|
const mockFetch = vi.fn().mockImplementation((url, init) => {
|
|
if (init?.headers?.Authorization === "Bearer test-token-error") {
|
|
return Promise.resolve({
|
|
ok: false,
|
|
status: 401,
|
|
});
|
|
}
|
|
return Promise.resolve({
|
|
ok: true,
|
|
json: async () => ({
|
|
data: [
|
|
{
|
|
id: "Qwen/Qwen3-32B",
|
|
name: "Qwen/Qwen3-32B",
|
|
supported_features: ["reasoning"],
|
|
input_modalities: ["text"],
|
|
context_length: 40960,
|
|
max_output_length: 40960,
|
|
pricing: { prompt: 0.08, completion: 0.24 },
|
|
},
|
|
{
|
|
id: "unsloth/Mistral-Nemo-Instruct-2407",
|
|
name: "unsloth/Mistral-Nemo-Instruct-2407",
|
|
input_modalities: ["text"],
|
|
context_length: 131072,
|
|
max_output_length: 131072,
|
|
pricing: { prompt: 0.02, completion: 0.04 },
|
|
},
|
|
{
|
|
id: "deepseek-ai/DeepSeek-V3-0324-TEE",
|
|
name: "deepseek-ai/DeepSeek-V3-0324-TEE",
|
|
supported_features: ["reasoning"],
|
|
input_modalities: ["text"],
|
|
context_length: 131072,
|
|
max_output_length: 65536,
|
|
pricing: { prompt: 0.28, completion: 0.42 },
|
|
},
|
|
],
|
|
}),
|
|
});
|
|
});
|
|
await withLiveChutesDiscovery(mockFetch, async () => {
|
|
const models = await chutesModels.discoverChutesModels("test-token-error");
|
|
expect(models.length).toBeGreaterThan(0);
|
|
expect(mockFetch).toHaveBeenCalled();
|
|
});
|
|
});
|
|
|
|
it("caches fallback static catalog for non-OK responses", async () => {
|
|
const mockFetch = vi.fn().mockResolvedValue({
|
|
ok: false,
|
|
status: 503,
|
|
});
|
|
|
|
await withLiveChutesDiscovery(mockFetch, async () => {
|
|
const first = await chutesModels.discoverChutesModels("chutes-fallback-token");
|
|
const second = await chutesModels.discoverChutesModels("chutes-fallback-token");
|
|
expect(first.map((m) => m.id)).toEqual(chutesModels.CHUTES_MODEL_CATALOG.map((m) => m.id));
|
|
expect(second.map((m) => m.id)).toEqual(chutesModels.CHUTES_MODEL_CATALOG.map((m) => m.id));
|
|
expect(mockFetch).toHaveBeenCalledTimes(1);
|
|
});
|
|
});
|
|
|
|
it("scopes discovery cache by access token", async () => {
|
|
const mockFetch = vi
|
|
.fn()
|
|
.mockImplementation((_url, init?: { headers?: Record<string, string> }) => {
|
|
const auth = init?.headers?.Authorization;
|
|
if (auth === "Bearer chutes-token-a") {
|
|
return Promise.resolve({
|
|
ok: true,
|
|
json: async () => ({
|
|
data: [{ id: "private/model-a" }],
|
|
}),
|
|
});
|
|
}
|
|
if (auth === "Bearer chutes-token-b") {
|
|
return Promise.resolve({
|
|
ok: true,
|
|
json: async () => ({
|
|
data: [{ id: "private/model-b" }],
|
|
}),
|
|
});
|
|
}
|
|
return Promise.resolve({
|
|
ok: true,
|
|
json: async () => ({
|
|
data: [{ id: "public/model" }],
|
|
}),
|
|
});
|
|
});
|
|
await withLiveChutesDiscovery(mockFetch, async () => {
|
|
const modelsA = await chutesModels.discoverChutesModels("chutes-token-a");
|
|
const modelsB = await chutesModels.discoverChutesModels("chutes-token-b");
|
|
const modelsASecond = await chutesModels.discoverChutesModels("chutes-token-a");
|
|
expect(modelsA[0]?.id).toBe("private/model-a");
|
|
expect(modelsB[0]?.id).toBe("private/model-b");
|
|
expect(modelsASecond[0]?.id).toBe("private/model-a");
|
|
expect(mockFetch).toHaveBeenCalledTimes(2);
|
|
});
|
|
});
|
|
|
|
it("evicts oldest token entries when cache reaches max size", async () => {
|
|
const mockFetch = createAuthEchoFetchMock();
|
|
|
|
await withLiveChutesDiscovery(mockFetch, async () => {
|
|
for (let i = 0; i < 150; i += 1) {
|
|
await chutesModels.discoverChutesModels(`cache-token-${i}`);
|
|
}
|
|
|
|
await chutesModels.discoverChutesModels("cache-token-0");
|
|
expect(mockFetch).toHaveBeenCalledTimes(151);
|
|
});
|
|
});
|
|
|
|
it("prunes expired token cache entries during subsequent discovery", async () => {
|
|
const mockFetch = createAuthEchoFetchMock();
|
|
|
|
await withLiveChutesDiscovery(
|
|
mockFetch,
|
|
async () => {
|
|
await chutesModels.discoverChutesModels("token-a");
|
|
vi.advanceTimersByTime(5 * 60 * 1000 + 1);
|
|
await chutesModels.discoverChutesModels("token-b");
|
|
await chutesModels.discoverChutesModels("token-a");
|
|
expect(mockFetch).toHaveBeenCalledTimes(3);
|
|
},
|
|
{ now: "2026-03-01T00:00:00.000Z" },
|
|
);
|
|
});
|
|
|
|
it("does not cache 401 fallback under the failed token key", async () => {
|
|
const mockFetch = vi
|
|
.fn()
|
|
.mockImplementation((_url, init?: { headers?: Record<string, string> }) => {
|
|
if (init?.headers?.Authorization === "Bearer failed-token") {
|
|
return Promise.resolve({
|
|
ok: false,
|
|
status: 401,
|
|
});
|
|
}
|
|
return Promise.resolve({
|
|
ok: true,
|
|
json: async () => ({
|
|
data: [{ id: "public/model" }],
|
|
}),
|
|
});
|
|
});
|
|
await withLiveChutesDiscovery(mockFetch, async () => {
|
|
await chutesModels.discoverChutesModels("failed-token");
|
|
await chutesModels.discoverChutesModels("failed-token");
|
|
expect(mockFetch).toHaveBeenCalledTimes(4);
|
|
});
|
|
});
|
|
});
|