Files
openclaw/extensions/deepinfra/surface-model-catalogs.test.ts
Georgi Atsev 9e7c2b356b fix(deepinfra): load all DeepInfra models when user wants to browse t… (#84549)
* fix(deepinfra): load all DeepInfra models when user wants to browse them during onboarding

* docs(deepinfra): align TTS default

* fix(deepinfra): refresh video fallbacks

* fix(deepinfra): share credential-aware catalog discovery

* test(deepinfra): narrow catalog regression types

* test(deepinfra): keep catalog narrowing across callback

* fix(deepinfra): preserve default model in live catalog

* fix(deepinfra): align default model pricing

* fix(deepinfra): keep pixverse as video default

* docs(deepinfra): match video fallback default

* fix(deepinfra): honor config api keys for live catalog

* test(e2e): wait for watchdog stdio close

* test(media): align live harness provider expectation

* fix(deepinfra): always augment custom catalogs

* test(e2e): resolve watchdog commands before spawning

---------

Co-authored-by: Peter Steinberger <steipete@gmail.com>
2026-05-27 10:43:37 +01:00

288 lines
8.9 KiB
TypeScript

import { beforeEach, describe, expect, it, vi } from "vitest";
import { resetDeepInfraModelCacheForTest } from "./provider-models.js";
import {
listDeepInfraImageGenCatalog,
listDeepInfraVideoGenCatalog,
resolveDeepInfraVideoModelCapabilities,
} from "./surface-model-catalogs.js";
beforeEach(() => {
resetDeepInfraModelCacheForTest();
});
function makeCtx(overrides: Partial<Parameters<typeof listDeepInfraImageGenCatalog>[0]> = {}) {
return {
config: {},
env: { ...process.env },
resolveProviderApiKey: (_id?: string) => ({
apiKey: undefined,
discoveryApiKey: undefined,
}),
resolveProviderAuth: () => ({
apiKey: undefined,
mode: "none" as const,
source: "none" as const,
}),
...overrides,
} as Parameters<typeof listDeepInfraImageGenCatalog>[0];
}
function withKeyCtx(): Parameters<typeof listDeepInfraImageGenCatalog>[0] {
return makeCtx({
resolveProviderApiKey: () => ({
apiKey: "sk-test",
discoveryApiKey: "sk-test",
}),
});
}
const surfaceEntry = (id: string, surfaceTag: string, extra: Record<string, unknown> = {}) => ({
id,
object: "model" as const,
owned_by: "deepinfra",
metadata: {
description: id,
tags: [surfaceTag],
pricing: {},
...extra,
},
});
async function withLiveFetch(
mockFetch: ReturnType<typeof vi.fn>,
run: () => Promise<void>,
) {
const env = { ...process.env };
delete process.env.NODE_ENV;
delete process.env.VITEST;
process.env.DEEPINFRA_API_KEY = "sk-test";
vi.stubGlobal("fetch", mockFetch);
try {
await run();
} finally {
if (env.NODE_ENV !== undefined) {
process.env.NODE_ENV = env.NODE_ENV;
} else {
delete process.env.NODE_ENV;
}
if (env.VITEST !== undefined) {
process.env.VITEST = env.VITEST;
} else {
delete process.env.VITEST;
}
if (env.DEEPINFRA_API_KEY !== undefined) {
process.env.DEEPINFRA_API_KEY = env.DEEPINFRA_API_KEY;
} else {
delete process.env.DEEPINFRA_API_KEY;
}
vi.unstubAllGlobals();
}
}
describe("listDeepInfraImageGenCatalog", () => {
it("returns null when no discoveryApiKey is configured", async () => {
const result = await listDeepInfraImageGenCatalog(makeCtx());
expect(result).toBeNull();
});
it("returns null when live discovery succeeds but the response has zero image-gen entries", async () => {
const mockFetch = vi.fn().mockResolvedValue({
ok: true,
json: () =>
Promise.resolve({
data: [
surfaceEntry("anthropic/claude-sonnet-4-6", "chat", {
context_length: 200000,
max_tokens: 8192,
pricing: { input_tokens: 3, output_tokens: 15 },
}),
],
}),
});
await withLiveFetch(mockFetch, async () => {
const result = await listDeepInfraImageGenCatalog(withKeyCtx());
expect(result).toBeNull();
});
});
it("returns null under VITEST even with a key (static fallback owns offline)", async () => {
// The default VITEST env path makes discoverDeepInfraSurfaces emit the
// manifest fallback (live=false), and the catalog provider rejects
// non-live results so it cannot serve stale offline data as "live".
const result = await listDeepInfraImageGenCatalog(withKeyCtx());
expect(result).toBeNull();
});
it("projects discovered image-gen entries when a key is configured and discovery is live", async () => {
const mockFetch = vi.fn().mockResolvedValue({
ok: true,
json: () =>
Promise.resolve({
data: [
surfaceEntry("black-forest-labs/FLUX-2-pro", "image-gen", {
pricing: { per_image_unit: 0.08 },
default_width: 1024,
default_height: 1024,
default_iterations: 28,
}),
surfaceEntry("ByteDance/Seedream-4", "image-gen", {
pricing: { per_image_unit: 0.03 },
}),
surfaceEntry("anthropic/claude-sonnet-4-6", "chat", {
context_length: 200000,
max_tokens: 8192,
pricing: { input_tokens: 3, output_tokens: 15 },
}),
],
}),
});
await withLiveFetch(mockFetch, async () => {
const result = await listDeepInfraImageGenCatalog(withKeyCtx());
expect(result).not.toBeNull();
expect(result?.map((e) => e.model)).toEqual([
"black-forest-labs/FLUX-2-pro",
"ByteDance/Seedream-4",
]);
for (const entry of result ?? []) {
expect(entry.kind).toBe("image_generation");
expect(entry.provider).toBe("deepinfra");
expect(entry.source).toBe("live");
}
});
});
});
describe("listDeepInfraVideoGenCatalog", () => {
it("returns null when no discoveryApiKey is configured", async () => {
const result = await listDeepInfraVideoGenCatalog(makeCtx());
expect(result).toBeNull();
});
it("returns null when live discovery succeeds but the response has zero video-gen entries", async () => {
// Current production state: TTS/STT/T2V models lack the OPENAI tag the
// backend filter requires, so a key-authenticated discovery still
// produces zero video-gen entries. We must return null so the registered
// provider's static fallback list is consulted instead of an empty
// "live" answer.
const mockFetch = vi.fn().mockResolvedValue({
ok: true,
json: () =>
Promise.resolve({
data: [
surfaceEntry("anthropic/claude-sonnet-4-6", "chat", {
context_length: 200000,
max_tokens: 8192,
pricing: { input_tokens: 3, output_tokens: 15 },
}),
surfaceEntry("black-forest-labs/FLUX-2-pro", "image-gen", {
pricing: { per_image_unit: 0.08 },
}),
],
}),
});
await withLiveFetch(mockFetch, async () => {
const result = await listDeepInfraVideoGenCatalog(withKeyCtx());
expect(result).toBeNull();
});
});
it("projects discovered video-gen entries with capability shape", async () => {
const mockFetch = vi.fn().mockResolvedValue({
ok: true,
json: () =>
Promise.resolve({
data: [
surfaceEntry("Wan-AI/Wan2.6-T2V", "video-gen", {
pricing: { output_seconds: 0.05 },
}),
surfaceEntry("ByteDance/Seedance-2.0", "video-gen", {
pricing: { output_seconds: 0.08 },
}),
],
}),
});
await withLiveFetch(mockFetch, async () => {
const result = await listDeepInfraVideoGenCatalog(withKeyCtx());
expect(result).not.toBeNull();
expect(result?.map((e) => e.model)).toEqual([
"Wan-AI/Wan2.6-T2V",
"ByteDance/Seedance-2.0",
]);
const first = result?.[0];
expect(first?.kind).toBe("video_generation");
expect(first?.capabilities?.generate?.supportsAspectRatio).toBe(true);
expect(first?.capabilities?.generate?.supportedDurationSeconds).toEqual([5, 8]);
});
});
});
describe("resolveDeepInfraVideoModelCapabilities", () => {
it("returns capabilities for a discovered video-gen model", async () => {
const mockFetch = vi.fn().mockResolvedValue({
ok: true,
json: () =>
Promise.resolve({
data: [
surfaceEntry("Wan-AI/Wan2.6-T2V", "video-gen", {
pricing: { output_seconds: 0.05 },
}),
],
}),
});
await withLiveFetch(mockFetch, async () => {
const caps = await resolveDeepInfraVideoModelCapabilities({
model: "Wan-AI/Wan2.6-T2V",
} as Parameters<typeof resolveDeepInfraVideoModelCapabilities>[0]);
expect(caps).toBeDefined();
expect(caps?.generate?.supportsAspectRatio).toBe(true);
});
});
it("strips the deepinfra/ prefix when matching", async () => {
const mockFetch = vi.fn().mockResolvedValue({
ok: true,
json: () =>
Promise.resolve({
data: [
surfaceEntry("Wan-AI/Wan2.6-T2V", "video-gen", {
pricing: { output_seconds: 0.05 },
}),
],
}),
});
await withLiveFetch(mockFetch, async () => {
const caps = await resolveDeepInfraVideoModelCapabilities({
model: "deepinfra/Wan-AI/Wan2.6-T2V",
} as Parameters<typeof resolveDeepInfraVideoModelCapabilities>[0]);
expect(caps).toBeDefined();
});
});
it("returns undefined for an unknown model", async () => {
const mockFetch = vi.fn().mockResolvedValue({
ok: true,
json: () =>
Promise.resolve({
data: [
surfaceEntry("Wan-AI/Wan2.6-T2V", "video-gen", {
pricing: { output_seconds: 0.05 },
}),
],
}),
});
await withLiveFetch(mockFetch, async () => {
const caps = await resolveDeepInfraVideoModelCapabilities({
model: "ByteDance/Seedance-2.0",
} as Parameters<typeof resolveDeepInfraVideoModelCapabilities>[0]);
expect(caps).toBeUndefined();
});
});
});