169 lines
6.8 KiB
TypeScript
169 lines
6.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-6")
|
|
expect(sisyphus!.requirement.fallbackChain[0]?.providers).toContain("anthropic")
|
|
})
|
|
|
|
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.1-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-6" },
|
|
},
|
|
}
|
|
|
|
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-6")
|
|
expect(oracle!.effectiveResolution).toBe("User override: anthropic/claude-opus-4-6")
|
|
})
|
|
|
|
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")
|
|
})
|
|
|
|
it("captures user variant for agent when configured", async () => {
|
|
const { getModelResolutionInfoWithOverrides } = await import("./model-resolution")
|
|
|
|
//#given User has model with variant override for oracle agent
|
|
const mockConfig = {
|
|
agents: {
|
|
oracle: { model: "openai/gpt-5.2", variant: "xhigh" },
|
|
},
|
|
}
|
|
|
|
//#when getting resolution info with config
|
|
const info = getModelResolutionInfoWithOverrides(mockConfig)
|
|
|
|
//#then Oracle should have userVariant set
|
|
const oracle = info.agents.find((a) => a.name === "oracle")
|
|
expect(oracle).toBeDefined()
|
|
expect(oracle!.userOverride).toBe("openai/gpt-5.2")
|
|
expect(oracle!.userVariant).toBe("xhigh")
|
|
})
|
|
|
|
it("captures user variant for category when configured", async () => {
|
|
const { getModelResolutionInfoWithOverrides } = await import("./model-resolution")
|
|
|
|
//#given User has model with variant override for visual-engineering category
|
|
const mockConfig = {
|
|
categories: {
|
|
"visual-engineering": { model: "google/gemini-3-flash-preview", variant: "high" },
|
|
},
|
|
}
|
|
|
|
//#when getting resolution info with config
|
|
const info = getModelResolutionInfoWithOverrides(mockConfig)
|
|
|
|
//#then visual-engineering should have userVariant set
|
|
const visual = info.categories.find((c) => c.name === "visual-engineering")
|
|
expect(visual).toBeDefined()
|
|
expect(visual!.userOverride).toBe("google/gemini-3-flash-preview")
|
|
expect(visual!.userVariant).toBe("high")
|
|
})
|
|
})
|
|
|
|
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)
|
|
})
|
|
})
|
|
|
|
})
|