mirror of
https://github.com/diegosouzapw/OmniRoute.git
synced 2026-05-23 04:28:06 +00:00
- Use jsonc-parser to update only provider.omniroute in opencode.json,
preserving MCP servers, comments, and other provider entries
- Add /api/cli-tools/keys endpoint returning raw keys for CLI tools UI
(session-auth protected via requireCliToolsAuth)
- Fix OpenCode guide step 3 to use ICU-style {baseUrl} placeholder
instead of broken {{baseUrl}} across all 40+ locales
- Restore valid OpenCode light/dark SVG logos (were broken HTML downloads)
- Add customCliTab translation key to en.json
- Add modelLabels support for human-readable model names in config
- Fix 9 additional locale files (bn, fa, gu, in, mr, sw, ta, te, ur)
that were added after the original PR and still had double-braces
Co-authored-by: JasonLandbridge <JasonLandbridge@users.noreply.github.com>
188 lines
6.8 KiB
TypeScript
188 lines
6.8 KiB
TypeScript
import test from "node:test";
|
|
import assert from "node:assert/strict";
|
|
import fs from "node:fs/promises";
|
|
import path from "node:path";
|
|
|
|
const { CLI_TOOLS } = await import("../../src/shared/constants/cliTools.ts");
|
|
const { resolveOpencodeConfigPath } = await import("../../src/shared/services/cliRuntime.ts");
|
|
const { buildOpenCodeProviderConfig, buildOpenCodeConfigDocument, mergeOpenCodeConfig } =
|
|
await import("../../src/shared/services/opencodeConfig.ts");
|
|
|
|
test("T40: OpenCode card documents config paths and --variant usage", () => {
|
|
const opencode = CLI_TOOLS.opencode;
|
|
assert.ok(opencode, "OpenCode tool card must exist");
|
|
assert.equal(opencode.modelSelectionMode, "multiple");
|
|
assert.equal(opencode.hideComboModels, true);
|
|
assert.equal(opencode.previewConfigMode, "opencode");
|
|
|
|
const notesText = (opencode.notes || [])
|
|
.map((note) => note?.text || "")
|
|
.join(" ")
|
|
.toLowerCase();
|
|
|
|
assert.match(notesText, /\.config\/opencode\/opencode\.json/);
|
|
assert.match(notesText, /%appdata%/);
|
|
assert.match(notesText, /--variant/);
|
|
});
|
|
|
|
test("T40: OpenCode config path resolves per-platform", () => {
|
|
const linuxWithXdg = resolveOpencodeConfigPath(
|
|
"linux",
|
|
{ XDG_CONFIG_HOME: "/tmp/xdg-config-home" },
|
|
"/home/dev"
|
|
);
|
|
assert.equal(linuxWithXdg, path.join("/tmp/xdg-config-home", "opencode", "opencode.json"));
|
|
|
|
const linuxDefault = resolveOpencodeConfigPath("linux", {}, "/home/dev");
|
|
assert.equal(linuxDefault, path.join("/home/dev", ".config", "opencode", "opencode.json"));
|
|
|
|
const windowsPath = resolveOpencodeConfigPath(
|
|
"win32",
|
|
{ APPDATA: "C:\\Users\\dev\\AppData\\Roaming" },
|
|
"C:\\Users\\dev"
|
|
);
|
|
assert.equal(
|
|
windowsPath,
|
|
path.join("C:\\Users\\dev\\AppData\\Roaming", "opencode", "opencode.json")
|
|
);
|
|
});
|
|
|
|
test("T40: OpenCode config generator includes endpoint and selected API key", () => {
|
|
const providerConfig = buildOpenCodeProviderConfig({
|
|
baseUrl: "http://localhost:20128/v1/",
|
|
apiKey: "sk_test_opencode",
|
|
model: "claude-sonnet-4-5-thinking",
|
|
});
|
|
assert.equal(providerConfig.options.baseURL, "http://localhost:20128/v1");
|
|
assert.equal(providerConfig.options.apiKey, "sk_test_opencode");
|
|
assert.ok(providerConfig.models["claude-sonnet-4-5-thinking"]);
|
|
|
|
const mergedConfig = mergeOpenCodeConfig(
|
|
{ provider: { custom: { name: "Custom Provider" } } },
|
|
{
|
|
baseUrl: "http://localhost:20128/v1",
|
|
apiKey: "sk_test_opencode",
|
|
model: "claude-sonnet-4-5-thinking",
|
|
}
|
|
);
|
|
assert.ok(mergedConfig.provider.custom);
|
|
assert.equal(mergedConfig.provider.omniroute.options.baseURL, "http://localhost:20128/v1");
|
|
assert.equal(mergedConfig.provider.omniroute.options.apiKey, "sk_test_opencode");
|
|
});
|
|
|
|
test("T40: OpenCode config document uses current provider schema", () => {
|
|
const configDocument = buildOpenCodeConfigDocument({
|
|
baseUrl: "http://localhost:20128/v1/",
|
|
apiKey: "sk_test_opencode",
|
|
models: ["cc/claude-sonnet-4-20250514", "gg/gemini-2.5-pro"],
|
|
modelLabels: {
|
|
"cc/claude-sonnet-4-20250514": "Claude Sonnet 4.5",
|
|
"gg/gemini-2.5-pro": "Gemini 2.5 Pro",
|
|
},
|
|
});
|
|
|
|
assert.equal(configDocument.$schema, "https://opencode.ai/config.json");
|
|
assert.ok(configDocument.provider.omniroute);
|
|
assert.equal(configDocument.provider.omniroute.npm, "@ai-sdk/openai-compatible");
|
|
assert.equal(configDocument.provider.omniroute.options.baseURL, "http://localhost:20128/v1");
|
|
assert.equal(configDocument.provider.omniroute.options.apiKey, "sk_test_opencode");
|
|
assert.deepEqual(Object.keys(configDocument.provider.omniroute.models), [
|
|
"cc/claude-sonnet-4-20250514",
|
|
"gg/gemini-2.5-pro",
|
|
]);
|
|
assert.equal(
|
|
configDocument.provider.omniroute.models["cc/claude-sonnet-4-20250514"].name,
|
|
"Claude Sonnet 4.5"
|
|
);
|
|
assert.equal(
|
|
configDocument.provider.omniroute.models["gg/gemini-2.5-pro"].name,
|
|
"Gemini 2.5 Pro"
|
|
);
|
|
assert.equal(configDocument.providers, undefined);
|
|
});
|
|
|
|
test("T40: OpenCode explicit multi-model selection overrides fallback defaults", () => {
|
|
const providerConfig = buildOpenCodeProviderConfig({
|
|
baseUrl: "http://localhost:20128/v1/",
|
|
apiKey: "sk_test_opencode",
|
|
models: ["custom/provider-a", "custom/provider-b"],
|
|
modelLabels: {
|
|
"custom/provider-a": "Provider A",
|
|
"custom/provider-b": "Provider B",
|
|
},
|
|
});
|
|
|
|
assert.deepEqual(Object.keys(providerConfig.models), ["custom/provider-a", "custom/provider-b"]);
|
|
assert.equal(providerConfig.models["claude-sonnet-4-5-thinking"], undefined);
|
|
assert.equal(providerConfig.models["custom/provider-a"].name, "Provider A");
|
|
assert.equal(providerConfig.models["custom/provider-b"].name, "Provider B");
|
|
});
|
|
|
|
test("T40: OpenCode merge preserves unrelated config and updates only provider.omniroute", () => {
|
|
const mergedConfig = mergeOpenCodeConfig(
|
|
{
|
|
$schema: "https://opencode.ai/config.json",
|
|
provider: {
|
|
custom: { name: "Custom Provider" },
|
|
omniroute: {
|
|
name: "Old OmniRoute",
|
|
options: { baseURL: "http://old-host/v1", apiKey: "old-key" },
|
|
},
|
|
},
|
|
mcpServers: {
|
|
github: { command: "npx", args: ["-y", "@modelcontextprotocol/server-github"] },
|
|
},
|
|
},
|
|
{
|
|
baseUrl: "http://localhost:20128/v1",
|
|
apiKey: "sk_test_opencode",
|
|
models: ["cx/gpt-5.4"],
|
|
modelLabels: { "cx/gpt-5.4": "GPT-5.4" },
|
|
}
|
|
);
|
|
|
|
assert.deepEqual(mergedConfig.provider.custom, { name: "Custom Provider" });
|
|
assert.deepEqual(mergedConfig.mcpServers, {
|
|
github: { command: "npx", args: ["-y", "@modelcontextprotocol/server-github"] },
|
|
});
|
|
assert.deepEqual(mergedConfig.provider.omniroute.models, {
|
|
"cx/gpt-5.4": { name: "GPT-5.4" },
|
|
});
|
|
});
|
|
|
|
test("T40: OpenCode tool card references theme-aware brand assets", () => {
|
|
const opencode = CLI_TOOLS.opencode;
|
|
assert.equal(opencode.image, undefined);
|
|
assert.equal(opencode.imageLight, "/providers/opencode-light.svg");
|
|
assert.equal(opencode.imageDark, "/providers/opencode-dark.svg");
|
|
});
|
|
|
|
test("T40: OpenCode light/dark provider assets are valid SVG files", async () => {
|
|
const light = await fs.readFile(
|
|
path.join(process.cwd(), "public/providers/opencode-light.svg"),
|
|
"utf-8"
|
|
);
|
|
const dark = await fs.readFile(
|
|
path.join(process.cwd(), "public/providers/opencode-dark.svg"),
|
|
"utf-8"
|
|
);
|
|
|
|
assert.match(light, /^<svg[\s>]/);
|
|
assert.match(dark, /^<svg[\s>]/);
|
|
assert.doesNotMatch(light, /<html/i);
|
|
assert.doesNotMatch(dark, /<html/i);
|
|
});
|
|
|
|
test("T40: Windsurf card documents current official limitations honestly", () => {
|
|
const windsurf = CLI_TOOLS.windsurf;
|
|
assert.ok(windsurf, "Windsurf tool card must exist");
|
|
assert.equal(windsurf.configType, "guide");
|
|
|
|
const notesText = (windsurf.notes || [])
|
|
.map((note) => note?.text || "")
|
|
.join(" ")
|
|
.toLowerCase();
|
|
|
|
assert.match(notesText, /byok/);
|
|
assert.match(notesText, /custom openai-compatible provider/);
|
|
});
|