mirror of
https://github.com/glittercowboy/get-shit-done
synced 2026-04-25 17:25:23 +02:00
The Codex and OpenCode install paths read `model_overrides` only from `~/.gsd/defaults.json` (global). A per-project override set in `.planning/config.json` — the reporter's exact setup for `gsd-codebase-mapper` — was silently dropped, so the child agent inherited the runtime's default model regardless of `model_overrides`. Neither runtime has an inline `model` parameter on its spawn API (Codex `spawn_agent(agent_type, message)`, OpenCode `task(description, prompt, subagent_type, task_id, command)`), so the per-agent model must reach the child via the static config GSD writes at install time. That config was being populated from the wrong source. Fix: add `readGsdEffectiveModelOverrides(targetDir)` which merges `~/.gsd/defaults.json` with per-project `.planning/config.json`, with per-project keys winning on conflict. Both install sites now call it and walk up from the install root to locate `.planning/` — matching the precedence `readGsdRuntimeProfileResolver` already uses for #2517. Also update the Codex Task()->spawn_agent mapping block so it no longer says "omit" without context: it now documents that per-agent overrides are embedded in the agent TOML and notes the restriction that Codex only permits `spawn_agent` when the user explicitly requested sub-agents (do the work inline otherwise). Regression tests (`tests/bug-2256-model-overrides-transport.test.cjs`) cover: global-only, project-only, project-wins-on-conflict, walking up from a nested `targetDir`, Codex TOML `model =` emission, and OpenCode frontmatter `model:` emission. Closes #2256 Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
154 lines
5.7 KiB
JavaScript
154 lines
5.7 KiB
JavaScript
/**
|
|
* Regression tests for issue #2256 — per-agent model_overrides transport
|
|
* for Codex and OpenCode runtimes.
|
|
*
|
|
* The bug: model_overrides set in per-project `.planning/config.json` were
|
|
* never read by the Codex / OpenCode install paths, which only probed
|
|
* `~/.gsd/defaults.json`. As a result, the configured per-agent model was
|
|
* dropped and child agents inherited the runtime's default model.
|
|
*
|
|
* These tests lock in the fix: per-project overrides must be honored, and
|
|
* per-project keys must win over global when both are present.
|
|
*/
|
|
|
|
process.env.GSD_TEST_MODE = '1';
|
|
|
|
const { test, describe, beforeEach, afterEach } = require('node:test');
|
|
const assert = require('node:assert/strict');
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
const os = require('os');
|
|
|
|
const {
|
|
readGsdEffectiveModelOverrides,
|
|
generateCodexAgentToml,
|
|
convertClaudeToOpencodeFrontmatter,
|
|
getCodexSkillAdapterHeader,
|
|
} = require('../bin/install.js');
|
|
|
|
function makeTmp(prefix) {
|
|
return fs.mkdtempSync(path.join(os.tmpdir(), `gsd-2256-${prefix}-`));
|
|
}
|
|
|
|
function writeJson(p, obj) {
|
|
fs.mkdirSync(path.dirname(p), { recursive: true });
|
|
fs.writeFileSync(p, JSON.stringify(obj, null, 2));
|
|
}
|
|
|
|
function rmr(p) {
|
|
try { fs.rmSync(p, { recursive: true, force: true }); } catch { /* noop */ }
|
|
}
|
|
|
|
describe('bug #2256 — readGsdEffectiveModelOverrides', () => {
|
|
let projectDir;
|
|
let homeDir;
|
|
let origHome;
|
|
|
|
beforeEach(() => {
|
|
projectDir = makeTmp('proj');
|
|
homeDir = makeTmp('home');
|
|
origHome = process.env.HOME;
|
|
process.env.HOME = homeDir;
|
|
});
|
|
|
|
afterEach(() => {
|
|
if (origHome === undefined) delete process.env.HOME;
|
|
else process.env.HOME = origHome;
|
|
rmr(projectDir);
|
|
rmr(homeDir);
|
|
});
|
|
|
|
test('returns null when neither source defines model_overrides', () => {
|
|
const result = readGsdEffectiveModelOverrides(projectDir);
|
|
assert.strictEqual(result, null);
|
|
});
|
|
|
|
test('reads overrides from ~/.gsd/defaults.json (global only)', () => {
|
|
writeJson(path.join(homeDir, '.gsd', 'defaults.json'), {
|
|
model_overrides: { 'gsd-codebase-mapper': 'gpt-5-mini' },
|
|
});
|
|
const result = readGsdEffectiveModelOverrides(projectDir);
|
|
assert.deepStrictEqual(result, { 'gsd-codebase-mapper': 'gpt-5-mini' });
|
|
});
|
|
|
|
test('reads overrides from per-project .planning/config.json', () => {
|
|
writeJson(path.join(projectDir, '.planning', 'config.json'), {
|
|
model_overrides: { 'gsd-codebase-mapper': 'claude-haiku-4-5' },
|
|
});
|
|
const result = readGsdEffectiveModelOverrides(projectDir);
|
|
assert.deepStrictEqual(result, { 'gsd-codebase-mapper': 'claude-haiku-4-5' });
|
|
});
|
|
|
|
test('per-project overrides win over global on conflict', () => {
|
|
writeJson(path.join(homeDir, '.gsd', 'defaults.json'), {
|
|
model_overrides: { 'gsd-codebase-mapper': 'global-model', 'gsd-planner': 'opus' },
|
|
});
|
|
writeJson(path.join(projectDir, '.planning', 'config.json'), {
|
|
model_overrides: { 'gsd-codebase-mapper': 'project-model' },
|
|
});
|
|
const result = readGsdEffectiveModelOverrides(projectDir);
|
|
// Per-project wins on conflict; non-conflicting global keys are preserved.
|
|
assert.deepStrictEqual(result, {
|
|
'gsd-codebase-mapper': 'project-model',
|
|
'gsd-planner': 'opus',
|
|
});
|
|
});
|
|
|
|
test('walks up from nested targetDir to find .planning/', () => {
|
|
writeJson(path.join(projectDir, '.planning', 'config.json'), {
|
|
model_overrides: { 'gsd-planner': 'project-opus' },
|
|
});
|
|
const nested = path.join(projectDir, '.codex');
|
|
fs.mkdirSync(nested, { recursive: true });
|
|
const result = readGsdEffectiveModelOverrides(nested);
|
|
assert.deepStrictEqual(result, { 'gsd-planner': 'project-opus' });
|
|
});
|
|
});
|
|
|
|
describe('bug #2256 — Codex adapter embeds per-project override', () => {
|
|
const agentContent = `---\nname: gsd-codebase-mapper\ndescription: Maps codebase\n---\n\nbody\n`;
|
|
|
|
test('generateCodexAgentToml embeds model when override provided', () => {
|
|
const toml = generateCodexAgentToml(
|
|
'gsd-codebase-mapper',
|
|
agentContent,
|
|
{ 'gsd-codebase-mapper': 'gpt-5-mini' },
|
|
);
|
|
assert.match(toml, /^model = "gpt-5-mini"$/m);
|
|
});
|
|
|
|
test('generateCodexAgentToml omits model when no override', () => {
|
|
const toml = generateCodexAgentToml('gsd-codebase-mapper', agentContent, null);
|
|
assert.doesNotMatch(toml, /^model\s*=/m);
|
|
});
|
|
});
|
|
|
|
describe('bug #2256 — OpenCode adapter embeds per-project override', () => {
|
|
test('convertClaudeToOpencodeFrontmatter embeds model on agent frontmatter', () => {
|
|
const input = `---\nname: gsd-codebase-mapper\ndescription: Maps codebase\n---\n\nbody\n`;
|
|
const out = convertClaudeToOpencodeFrontmatter(input, {
|
|
isAgent: true,
|
|
modelOverride: 'claude-haiku-4-5',
|
|
});
|
|
assert.match(out, /^model: claude-haiku-4-5$/m);
|
|
assert.match(out, /^mode: subagent$/m);
|
|
});
|
|
|
|
test('convertClaudeToOpencodeFrontmatter omits model when override absent', () => {
|
|
const input = `---\nname: gsd-codebase-mapper\ndescription: Maps codebase\n---\n\nbody\n`;
|
|
const out = convertClaudeToOpencodeFrontmatter(input, { isAgent: true, modelOverride: null });
|
|
assert.doesNotMatch(out, /^model:/m);
|
|
});
|
|
});
|
|
|
|
describe('bug #2256 — Codex skill adapter header documents transport', () => {
|
|
test('Task(model=...) line no longer says "omit" without explanation', () => {
|
|
const header = getCodexSkillAdapterHeader('gsd-plan-phase');
|
|
// Header must mention that per-agent model_overrides are embedded in agent
|
|
// TOML so spawn_agent picks them up automatically — the old text said
|
|
// "Codex uses per-role config, not inline model selection" which left
|
|
// users thinking their model_overrides were silently ignored.
|
|
assert.match(header, /model_overrides/);
|
|
});
|
|
});
|