* fix(models): update model names to match OpenCode Zen catalog OpenCode Zen recently updated their official model catalog, deprecating several preview and free model variants: DEPRECATED → NEW (Official Zen Names): - gemini-3-pro-preview → gemini-3-pro - gemini-3-flash-preview → gemini-3-flash - grok-code → gpt-5-nano (FREE tier maintained) - glm-4.7-free → big-pickle (FREE tier maintained) - glm-4.6v → glm-4.6 Changes: - Updated 6 source files (model-requirements, delegate-task, think-mode, etc.) - Updated 9 documentation files (installation, configurations, features, etc.) - Updated 14 test files with new model references - Regenerated snapshots to reflect catalog changes - Removed duplicate think-mode entries for preview variants Impact: - FREE tier access preserved via gpt-5-nano and big-pickle - All 55 model-related tests passing - Zero breaking changes - pure string replacement - Aligns codebase with official OpenCode Zen model catalog Verified: - Zero deprecated model names in codebase - All model-related tests pass (55/55) - Snapshots regenerated and validated Affects: 30 files (6 source, 9 docs, 14 tests, 1 snapshot) * fix(multimodal-looker): update fallback chain with glm-4.6v and gpt-5-nano - Change glm-4.6 to glm-4.6v for zai-coding-plan provider - Add opencode/gpt-5-nano as 4th fallback (FREE tier) - Push gpt-5.2 to 5th position Fallback chain now: 1. gemini-3-flash (google, github-copilot, opencode) 2. claude-haiku-4-5 (anthropic, github-copilot, opencode) 3. glm-4.6v (zai-coding-plan) 4. gpt-5-nano (opencode) - FREE 5. gpt-5.2 (openai, github-copilot, opencode) * chore: update bun.lock --------- Co-authored-by: justsisyphus <justsisyphus@users.noreply.github.com>
142 lines
5.8 KiB
TypeScript
142 lines
5.8 KiB
TypeScript
import { describe, it, expect, beforeEach, afterEach, spyOn, mock } from "bun:test"
|
|
|
|
describe("model-resolution check", () => {
|
|
describe("getModelResolutionInfo", () => {
|
|
// #given: Model requirements are defined in model-requirements.ts
|
|
// #when: Getting model resolution info
|
|
// #then: Returns info for all agents and categories with their provider chains
|
|
|
|
it("returns agent requirements with provider chains", async () => {
|
|
const { getModelResolutionInfo } = await import("./model-resolution")
|
|
|
|
const info = getModelResolutionInfo()
|
|
|
|
// #then: Should have agent entries
|
|
const sisyphus = info.agents.find((a) => a.name === "sisyphus")
|
|
expect(sisyphus).toBeDefined()
|
|
expect(sisyphus!.requirement.fallbackChain[0]?.model).toBe("claude-opus-4-5")
|
|
expect(sisyphus!.requirement.fallbackChain[0]?.providers).toContain("anthropic")
|
|
expect(sisyphus!.requirement.fallbackChain[0]?.providers).toContain("github-copilot")
|
|
})
|
|
|
|
it("returns category requirements with provider chains", async () => {
|
|
const { getModelResolutionInfo } = await import("./model-resolution")
|
|
|
|
const info = getModelResolutionInfo()
|
|
|
|
// #then: Should have category entries
|
|
const visual = info.categories.find((c) => c.name === "visual-engineering")
|
|
expect(visual).toBeDefined()
|
|
expect(visual!.requirement.fallbackChain[0]?.model).toBe("gemini-3-pro")
|
|
expect(visual!.requirement.fallbackChain[0]?.providers).toContain("google")
|
|
})
|
|
})
|
|
|
|
describe("getModelResolutionInfoWithOverrides", () => {
|
|
// #given: User has overrides in oh-my-opencode.json
|
|
// #when: Getting resolution info with config
|
|
// #then: Shows user override in Step 1 position
|
|
|
|
it("shows user override for agent when configured", async () => {
|
|
const { getModelResolutionInfoWithOverrides } = await import("./model-resolution")
|
|
|
|
// #given: User has override for oracle agent
|
|
const mockConfig = {
|
|
agents: {
|
|
oracle: { model: "anthropic/claude-opus-4-5" },
|
|
},
|
|
}
|
|
|
|
const info = getModelResolutionInfoWithOverrides(mockConfig)
|
|
|
|
// #then: Oracle should show the override
|
|
const oracle = info.agents.find((a) => a.name === "oracle")
|
|
expect(oracle).toBeDefined()
|
|
expect(oracle!.userOverride).toBe("anthropic/claude-opus-4-5")
|
|
expect(oracle!.effectiveResolution).toBe("User override: anthropic/claude-opus-4-5")
|
|
})
|
|
|
|
it("shows user override for category when configured", async () => {
|
|
const { getModelResolutionInfoWithOverrides } = await import("./model-resolution")
|
|
|
|
// #given: User has override for visual-engineering category
|
|
const mockConfig = {
|
|
categories: {
|
|
"visual-engineering": { model: "openai/gpt-5.2" },
|
|
},
|
|
}
|
|
|
|
const info = getModelResolutionInfoWithOverrides(mockConfig)
|
|
|
|
// #then: visual-engineering should show the override
|
|
const visual = info.categories.find((c) => c.name === "visual-engineering")
|
|
expect(visual).toBeDefined()
|
|
expect(visual!.userOverride).toBe("openai/gpt-5.2")
|
|
expect(visual!.effectiveResolution).toBe("User override: openai/gpt-5.2")
|
|
})
|
|
|
|
it("shows provider fallback when no override exists", async () => {
|
|
const { getModelResolutionInfoWithOverrides } = await import("./model-resolution")
|
|
|
|
// #given: No overrides configured
|
|
const mockConfig = {}
|
|
|
|
const info = getModelResolutionInfoWithOverrides(mockConfig)
|
|
|
|
// #then: Should show provider fallback chain
|
|
const sisyphus = info.agents.find((a) => a.name === "sisyphus")
|
|
expect(sisyphus).toBeDefined()
|
|
expect(sisyphus!.userOverride).toBeUndefined()
|
|
expect(sisyphus!.effectiveResolution).toContain("Provider fallback:")
|
|
expect(sisyphus!.effectiveResolution).toContain("anthropic")
|
|
})
|
|
})
|
|
|
|
describe("checkModelResolution", () => {
|
|
// #given: Doctor check is executed
|
|
// #when: Running the model resolution check
|
|
// #then: Returns pass with details showing resolution flow
|
|
|
|
it("returns pass or warn status with agent and category counts", async () => {
|
|
const { checkModelResolution } = await import("./model-resolution")
|
|
|
|
const result = await checkModelResolution()
|
|
|
|
// #then: Should pass (with cache) or warn (no cache) and show counts
|
|
// In CI without model cache, status is "warn"; locally with cache, status is "pass"
|
|
expect(["pass", "warn"]).toContain(result.status)
|
|
expect(result.message).toMatch(/\d+ agents?, \d+ categories?/)
|
|
})
|
|
|
|
it("includes resolution details in verbose mode details array", async () => {
|
|
const { checkModelResolution } = await import("./model-resolution")
|
|
|
|
const result = await checkModelResolution()
|
|
|
|
// #then: Details should contain agent/category resolution info
|
|
expect(result.details).toBeDefined()
|
|
expect(result.details!.length).toBeGreaterThan(0)
|
|
// Should have Available Models and Configured Models headers
|
|
expect(result.details!.some((d) => d.includes("Available Models"))).toBe(true)
|
|
expect(result.details!.some((d) => d.includes("Configured Models"))).toBe(true)
|
|
expect(result.details!.some((d) => d.includes("Agents:"))).toBe(true)
|
|
expect(result.details!.some((d) => d.includes("Categories:"))).toBe(true)
|
|
// Should have legend
|
|
expect(result.details!.some((d) => d.includes("user override"))).toBe(true)
|
|
})
|
|
})
|
|
|
|
describe("getModelResolutionCheckDefinition", () => {
|
|
it("returns valid check definition", async () => {
|
|
const { getModelResolutionCheckDefinition } = await import("./model-resolution")
|
|
|
|
const def = getModelResolutionCheckDefinition()
|
|
|
|
expect(def.id).toBe("model-resolution")
|
|
expect(def.name).toBe("Model Resolution")
|
|
expect(def.category).toBe("configuration")
|
|
expect(typeof def.check).toBe("function")
|
|
})
|
|
})
|
|
})
|