mirror of https://github.com/openclaw/openclaw.git
382 lines
14 KiB
TypeScript
382 lines
14 KiB
TypeScript
import type { Api, Model } from "@mariozechner/pi-ai";
|
|
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
|
|
const providerRuntimeMocks = vi.hoisted(() => ({
|
|
resolveProviderModernModelRef: vi.fn(),
|
|
}));
|
|
|
|
vi.mock("../plugins/provider-runtime.js", () => ({
|
|
resolveProviderModernModelRef: providerRuntimeMocks.resolveProviderModernModelRef,
|
|
}));
|
|
|
|
import { isModernModelRef } from "./live-model-filter.js";
|
|
import { normalizeModelCompat } from "./model-compat.js";
|
|
|
|
const baseModel = (): Model<Api> =>
|
|
({
|
|
id: "glm-4.7",
|
|
name: "GLM-4.7",
|
|
api: "openai-completions",
|
|
provider: "zai",
|
|
baseUrl: "https://api.z.ai/api/coding/paas/v4",
|
|
reasoning: true,
|
|
input: ["text"],
|
|
cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
|
|
contextWindow: 8192,
|
|
maxTokens: 1024,
|
|
}) as Model<Api>;
|
|
|
|
function supportsDeveloperRole(model: Model<Api>): boolean | undefined {
|
|
return (model.compat as { supportsDeveloperRole?: boolean } | undefined)?.supportsDeveloperRole;
|
|
}
|
|
|
|
function supportsUsageInStreaming(model: Model<Api>): boolean | undefined {
|
|
return (model.compat as { supportsUsageInStreaming?: boolean } | undefined)
|
|
?.supportsUsageInStreaming;
|
|
}
|
|
|
|
function supportsStrictMode(model: Model<Api>): boolean | undefined {
|
|
return (model.compat as { supportsStrictMode?: boolean } | undefined)?.supportsStrictMode;
|
|
}
|
|
|
|
function expectSupportsDeveloperRoleForcedOff(overrides?: Partial<Model<Api>>): void {
|
|
const model = { ...baseModel(), ...overrides };
|
|
delete (model as { compat?: unknown }).compat;
|
|
const normalized = normalizeModelCompat(model as Model<Api>);
|
|
expect(supportsDeveloperRole(normalized)).toBe(false);
|
|
}
|
|
|
|
function expectSupportsUsageInStreamingForcedOff(overrides?: Partial<Model<Api>>): void {
|
|
const model = { ...baseModel(), ...overrides };
|
|
delete (model as { compat?: unknown }).compat;
|
|
const normalized = normalizeModelCompat(model as Model<Api>);
|
|
expect(supportsUsageInStreaming(normalized)).toBe(false);
|
|
}
|
|
|
|
function expectSupportsStrictModeForcedOff(overrides?: Partial<Model<Api>>): void {
|
|
const model = { ...baseModel(), ...overrides };
|
|
delete (model as { compat?: unknown }).compat;
|
|
const normalized = normalizeModelCompat(model as Model<Api>);
|
|
expect(supportsStrictMode(normalized)).toBe(false);
|
|
}
|
|
|
|
beforeEach(() => {
|
|
providerRuntimeMocks.resolveProviderModernModelRef.mockReset();
|
|
providerRuntimeMocks.resolveProviderModernModelRef.mockReturnValue(undefined);
|
|
});
|
|
|
|
describe("normalizeModelCompat — Anthropic baseUrl", () => {
|
|
const anthropicBase = (): Model<Api> =>
|
|
({
|
|
id: "claude-opus-4-6",
|
|
name: "claude-opus-4-6",
|
|
api: "anthropic-messages",
|
|
provider: "anthropic",
|
|
reasoning: true,
|
|
input: ["text"],
|
|
cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
|
|
contextWindow: 200_000,
|
|
maxTokens: 8_192,
|
|
}) as Model<Api>;
|
|
|
|
it("strips /v1 suffix from anthropic-messages baseUrl", () => {
|
|
const model = { ...anthropicBase(), baseUrl: "https://api.anthropic.com/v1" };
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized.baseUrl).toBe("https://api.anthropic.com");
|
|
});
|
|
|
|
it("strips trailing /v1/ (with slash) from anthropic-messages baseUrl", () => {
|
|
const model = { ...anthropicBase(), baseUrl: "https://api.anthropic.com/v1/" };
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized.baseUrl).toBe("https://api.anthropic.com");
|
|
});
|
|
|
|
it("leaves anthropic-messages baseUrl without /v1 unchanged", () => {
|
|
const model = { ...anthropicBase(), baseUrl: "https://api.anthropic.com" };
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized.baseUrl).toBe("https://api.anthropic.com");
|
|
});
|
|
|
|
it("leaves baseUrl undefined unchanged for anthropic-messages", () => {
|
|
const model = anthropicBase();
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized.baseUrl).toBeUndefined();
|
|
});
|
|
|
|
it("does not strip /v1 from non-anthropic-messages models", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "openai",
|
|
api: "openai-responses" as Api,
|
|
baseUrl: "https://api.openai.com/v1",
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized.baseUrl).toBe("https://api.openai.com/v1");
|
|
});
|
|
|
|
it("strips /v1 from custom Anthropic proxy baseUrl", () => {
|
|
const model = {
|
|
...anthropicBase(),
|
|
baseUrl: "https://my-proxy.example.com/anthropic/v1",
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized.baseUrl).toBe("https://my-proxy.example.com/anthropic");
|
|
});
|
|
});
|
|
|
|
describe("normalizeModelCompat", () => {
|
|
it("forces supportsDeveloperRole off for z.ai models", () => {
|
|
expectSupportsDeveloperRoleForcedOff();
|
|
});
|
|
|
|
it("forces supportsDeveloperRole off for moonshot models", () => {
|
|
expectSupportsDeveloperRoleForcedOff({
|
|
provider: "moonshot",
|
|
baseUrl: "https://api.moonshot.ai/v1",
|
|
});
|
|
});
|
|
|
|
it("forces supportsDeveloperRole off for custom moonshot-compatible endpoints", () => {
|
|
expectSupportsDeveloperRoleForcedOff({
|
|
provider: "custom-kimi",
|
|
baseUrl: "https://api.moonshot.cn/v1",
|
|
});
|
|
});
|
|
|
|
it("forces supportsDeveloperRole off for DashScope provider ids", () => {
|
|
expectSupportsDeveloperRoleForcedOff({
|
|
provider: "dashscope",
|
|
baseUrl: "https://dashscope.aliyuncs.com/compatible-mode/v1",
|
|
});
|
|
});
|
|
|
|
it("forces supportsDeveloperRole off for DashScope-compatible endpoints", () => {
|
|
expectSupportsDeveloperRoleForcedOff({
|
|
provider: "custom-qwen",
|
|
baseUrl: "https://dashscope-intl.aliyuncs.com/compatible-mode/v1",
|
|
});
|
|
});
|
|
|
|
it("leaves native api.openai.com model untouched", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "openai",
|
|
baseUrl: "https://api.openai.com/v1",
|
|
};
|
|
delete (model as { compat?: unknown }).compat;
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized.compat).toBeUndefined();
|
|
});
|
|
|
|
it("forces supportsDeveloperRole off for Azure OpenAI (Chat Completions, not Responses API)", () => {
|
|
expectSupportsDeveloperRoleForcedOff({
|
|
provider: "azure-openai",
|
|
baseUrl: "https://my-deployment.openai.azure.com/openai",
|
|
});
|
|
});
|
|
it("forces supportsDeveloperRole off for generic custom openai-completions provider", () => {
|
|
expectSupportsDeveloperRoleForcedOff({
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://cpa.example.com/v1",
|
|
});
|
|
});
|
|
|
|
it("forces supportsUsageInStreaming off for generic custom openai-completions provider", () => {
|
|
expectSupportsUsageInStreamingForcedOff({
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://cpa.example.com/v1",
|
|
});
|
|
});
|
|
|
|
it("forces supportsStrictMode off for z.ai models", () => {
|
|
expectSupportsStrictModeForcedOff();
|
|
});
|
|
|
|
it("forces supportsStrictMode off for custom openai-completions provider", () => {
|
|
expectSupportsStrictModeForcedOff({
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://cpa.example.com/v1",
|
|
});
|
|
});
|
|
|
|
it("forces supportsDeveloperRole off for Qwen proxy via openai-completions", () => {
|
|
expectSupportsDeveloperRoleForcedOff({
|
|
provider: "qwen-proxy",
|
|
baseUrl: "https://qwen-api.example.org/compatible-mode/v1",
|
|
});
|
|
});
|
|
|
|
it("leaves openai-completions model with empty baseUrl untouched", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "openai",
|
|
};
|
|
delete (model as { baseUrl?: unknown }).baseUrl;
|
|
delete (model as { compat?: unknown }).compat;
|
|
const normalized = normalizeModelCompat(model as Model<Api>);
|
|
expect(normalized.compat).toBeUndefined();
|
|
});
|
|
|
|
it("forces supportsDeveloperRole off for malformed baseUrl values", () => {
|
|
expectSupportsDeveloperRoleForcedOff({
|
|
provider: "custom-cpa",
|
|
baseUrl: "://api.openai.com malformed",
|
|
});
|
|
});
|
|
|
|
it("respects explicit supportsDeveloperRole true on non-native endpoints", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://proxy.example.com/v1",
|
|
compat: { supportsDeveloperRole: true },
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(supportsDeveloperRole(normalized)).toBe(true);
|
|
});
|
|
|
|
it("respects explicit supportsUsageInStreaming true on non-native endpoints", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://proxy.example.com/v1",
|
|
compat: { supportsUsageInStreaming: true },
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(supportsUsageInStreaming(normalized)).toBe(true);
|
|
});
|
|
|
|
it("preserves explicit supportsUsageInStreaming false on non-native endpoints", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://proxy.example.com/v1",
|
|
compat: { supportsUsageInStreaming: false },
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(supportsUsageInStreaming(normalized)).toBe(false);
|
|
});
|
|
|
|
it("still forces flags off when not explicitly set by user", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://proxy.example.com/v1",
|
|
};
|
|
delete (model as { compat?: unknown }).compat;
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(supportsDeveloperRole(normalized)).toBe(false);
|
|
expect(supportsUsageInStreaming(normalized)).toBe(false);
|
|
expect(supportsStrictMode(normalized)).toBe(false);
|
|
});
|
|
|
|
it("respects explicit supportsStrictMode true on non-native endpoints", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://proxy.example.com/v1",
|
|
compat: { supportsStrictMode: true },
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(supportsStrictMode(normalized)).toBe(true);
|
|
});
|
|
|
|
it("does not mutate caller model when forcing supportsDeveloperRole off", () => {
|
|
const model = {
|
|
...baseModel(),
|
|
provider: "custom-cpa",
|
|
baseUrl: "https://proxy.example.com/v1",
|
|
};
|
|
delete (model as { compat?: unknown }).compat;
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized).not.toBe(model);
|
|
expect(supportsDeveloperRole(model)).toBeUndefined();
|
|
expect(supportsUsageInStreaming(model)).toBeUndefined();
|
|
expect(supportsStrictMode(model)).toBeUndefined();
|
|
expect(supportsDeveloperRole(normalized)).toBe(false);
|
|
expect(supportsUsageInStreaming(normalized)).toBe(false);
|
|
expect(supportsStrictMode(normalized)).toBe(false);
|
|
});
|
|
|
|
it("does not override explicit compat false", () => {
|
|
const model = baseModel();
|
|
model.compat = {
|
|
supportsDeveloperRole: false,
|
|
supportsUsageInStreaming: false,
|
|
supportsStrictMode: false,
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(supportsDeveloperRole(normalized)).toBe(false);
|
|
expect(supportsUsageInStreaming(normalized)).toBe(false);
|
|
expect(supportsStrictMode(normalized)).toBe(false);
|
|
});
|
|
|
|
it("leaves fully explicit non-native compat untouched", () => {
|
|
const model = baseModel();
|
|
model.baseUrl = "https://proxy.example.com/v1";
|
|
model.compat = {
|
|
supportsDeveloperRole: false,
|
|
supportsUsageInStreaming: true,
|
|
supportsStrictMode: true,
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(normalized).toBe(model);
|
|
});
|
|
|
|
it("preserves explicit usage compat when developer role is explicitly enabled", () => {
|
|
const model = baseModel();
|
|
model.baseUrl = "https://proxy.example.com/v1";
|
|
model.compat = {
|
|
supportsDeveloperRole: true,
|
|
supportsUsageInStreaming: true,
|
|
supportsStrictMode: true,
|
|
};
|
|
const normalized = normalizeModelCompat(model);
|
|
expect(supportsDeveloperRole(normalized)).toBe(true);
|
|
expect(supportsUsageInStreaming(normalized)).toBe(true);
|
|
expect(supportsStrictMode(normalized)).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe("isModernModelRef", () => {
|
|
it("uses provider runtime hooks before fallback heuristics", () => {
|
|
providerRuntimeMocks.resolveProviderModernModelRef.mockReturnValue(false);
|
|
|
|
expect(isModernModelRef({ provider: "openrouter", id: "claude-opus-4-6" })).toBe(false);
|
|
});
|
|
|
|
it("includes plugin-advertised modern models", () => {
|
|
providerRuntimeMocks.resolveProviderModernModelRef.mockImplementation(({ provider, context }) =>
|
|
provider === "openai" &&
|
|
["gpt-5.4", "gpt-5.4-pro", "gpt-5.4-mini", "gpt-5.4-nano"].includes(context.modelId)
|
|
? true
|
|
: provider === "openai-codex" && context.modelId === "gpt-5.4"
|
|
? true
|
|
: provider === "opencode" && ["claude-opus-4-6", "gemini-3-pro"].includes(context.modelId)
|
|
? true
|
|
: provider === "opencode-go"
|
|
? true
|
|
: undefined,
|
|
);
|
|
|
|
expect(isModernModelRef({ provider: "openai", id: "gpt-5.4" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "openai", id: "gpt-5.4-pro" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "openai", id: "gpt-5.4-mini" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "openai", id: "gpt-5.4-nano" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "openai-codex", id: "gpt-5.4" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "opencode", id: "claude-opus-4-6" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "opencode", id: "gemini-3-pro" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "opencode-go", id: "kimi-k2.5" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "opencode-go", id: "glm-5" })).toBe(true);
|
|
expect(isModernModelRef({ provider: "opencode-go", id: "minimax-m2.5" })).toBe(true);
|
|
});
|
|
|
|
it("excludes provider-declined modern models", () => {
|
|
providerRuntimeMocks.resolveProviderModernModelRef.mockImplementation(({ provider, context }) =>
|
|
provider === "opencode" && context.modelId === "minimax-m2.5" ? false : undefined,
|
|
);
|
|
|
|
expect(isModernModelRef({ provider: "opencode", id: "minimax-m2.5" })).toBe(false);
|
|
});
|
|
});
|