Files
get-shit-done/sdk/src/query/config-query.ts
Tom Boucher 726003563a fix(#2544): config-get exits 1 on missing key (was 10, UNIX convention is 1)
Change ErrorClassification for 'Key not found' in configGet from Validation
(exit 10) to Execution (exit 1), matching git config --get. Callers using
`gsd-sdk query config-get k || fallback` need a non-zero exit to trigger the
fallback branch; 10 worked technically but was semantically wrong and noisy.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-22 09:53:46 -04:00

160 lines
6.9 KiB
TypeScript

/**
* Config-get and resolve-model query handlers.
*
* Ported from get-shit-done/bin/lib/config.cjs and commands.cjs.
* Provides raw config.json traversal and model profile resolution.
*
* @example
* ```typescript
* import { configGet, resolveModel } from './config-query.js';
*
* const result = await configGet(['workflow.auto_advance'], '/project');
* // { data: true }
*
* const model = await resolveModel(['gsd-planner'], '/project');
* // { data: { model: 'opus', profile: 'balanced' } }
* ```
*/
import { readFile } from 'node:fs/promises';
import { GSDError, ErrorClassification } from '../errors.js';
import { loadConfig } from '../config.js';
import { planningPaths } from './helpers.js';
import type { QueryHandler } from './utils.js';
// ─── MODEL_PROFILES ─────────────────────────────────────────────────────────
/**
* Mapping of GSD agent type to model alias for each profile tier.
*
* Ported from get-shit-done/bin/lib/model-profiles.cjs.
*/
export const MODEL_PROFILES: Record<string, Record<string, string>> = {
'gsd-planner': { quality: 'opus', balanced: 'opus', budget: 'sonnet', adaptive: 'opus' },
'gsd-roadmapper': { quality: 'opus', balanced: 'sonnet', budget: 'sonnet', adaptive: 'sonnet' },
'gsd-executor': { quality: 'opus', balanced: 'sonnet', budget: 'sonnet', adaptive: 'sonnet' },
'gsd-phase-researcher': { quality: 'opus', balanced: 'sonnet', budget: 'haiku', adaptive: 'sonnet' },
'gsd-project-researcher': { quality: 'opus', balanced: 'sonnet', budget: 'haiku', adaptive: 'sonnet' },
'gsd-research-synthesizer': { quality: 'sonnet', balanced: 'sonnet', budget: 'haiku', adaptive: 'haiku' },
'gsd-debugger': { quality: 'opus', balanced: 'sonnet', budget: 'sonnet', adaptive: 'opus' },
'gsd-codebase-mapper': { quality: 'sonnet', balanced: 'haiku', budget: 'haiku', adaptive: 'haiku' },
'gsd-verifier': { quality: 'sonnet', balanced: 'sonnet', budget: 'haiku', adaptive: 'sonnet' },
'gsd-plan-checker': { quality: 'sonnet', balanced: 'sonnet', budget: 'haiku', adaptive: 'haiku' },
'gsd-integration-checker': { quality: 'sonnet', balanced: 'sonnet', budget: 'haiku', adaptive: 'haiku' },
'gsd-nyquist-auditor': { quality: 'sonnet', balanced: 'sonnet', budget: 'haiku', adaptive: 'haiku' },
'gsd-ui-researcher': { quality: 'opus', balanced: 'sonnet', budget: 'haiku', adaptive: 'sonnet' },
'gsd-ui-checker': { quality: 'sonnet', balanced: 'sonnet', budget: 'haiku', adaptive: 'haiku' },
'gsd-ui-auditor': { quality: 'sonnet', balanced: 'sonnet', budget: 'haiku', adaptive: 'haiku' },
'gsd-doc-writer': { quality: 'opus', balanced: 'sonnet', budget: 'haiku', adaptive: 'sonnet' },
'gsd-doc-verifier': { quality: 'sonnet', balanced: 'sonnet', budget: 'haiku', adaptive: 'haiku' },
};
/** Valid model profile names. */
export const VALID_PROFILES: string[] = Object.keys(MODEL_PROFILES['gsd-planner']);
// ─── configGet ──────────────────────────────────────────────────────────────
/**
* Query handler for config-get command.
*
* Reads raw .planning/config.json and traverses dot-notation key paths.
* Does NOT merge with defaults (matches gsd-tools.cjs behavior).
*
* @param args - args[0] is the dot-notation key path (e.g., 'workflow.auto_advance')
* @param projectDir - Project root directory
* @returns QueryResult with the config value at the given path
* @throws GSDError with Validation classification if key missing or not found
*/
export const configGet: QueryHandler = async (args, projectDir) => {
const keyPath = args[0];
if (!keyPath) {
throw new GSDError('Usage: config-get <key.path>', ErrorClassification.Validation);
}
const paths = planningPaths(projectDir);
let raw: string;
try {
raw = await readFile(paths.config, 'utf-8');
} catch {
throw new GSDError(`No config.json found at ${paths.config}`, ErrorClassification.Validation);
}
let config: Record<string, unknown>;
try {
config = JSON.parse(raw) as Record<string, unknown>;
} catch {
throw new GSDError(`Malformed config.json at ${paths.config}`, ErrorClassification.Validation);
}
const keys = keyPath.split('.');
let current: unknown = config;
for (const key of keys) {
if (current === undefined || current === null || typeof current !== 'object') {
throw new GSDError(`Key not found: ${keyPath}`, ErrorClassification.Execution);
}
current = (current as Record<string, unknown>)[key];
}
if (current === undefined) {
throw new GSDError(`Key not found: ${keyPath}`, ErrorClassification.Execution);
}
return { data: current };
};
// ─── resolveModel ───────────────────────────────────────────────────────────
/**
* Query handler for resolve-model command.
*
* Resolves the model alias for a given agent type based on the current profile.
* Uses loadConfig (with defaults) and MODEL_PROFILES for lookup.
*
* @param args - args[0] is the agent type (e.g., 'gsd-planner')
* @param projectDir - Project root directory
* @returns QueryResult with { model, profile } or { model, profile, unknown_agent: true }
* @throws GSDError with Validation classification if agent type not provided
*/
export const resolveModel: QueryHandler = async (args, projectDir) => {
const agentType = args[0];
if (!agentType) {
throw new GSDError('agent-type required', ErrorClassification.Validation);
}
const config = await loadConfig(projectDir);
const profile = String(config.model_profile || 'balanced').toLowerCase();
// Check per-agent override first
const overrides = (config as Record<string, unknown>).model_overrides as Record<string, string> | undefined;
const override = overrides?.[agentType];
if (override) {
const agentModels = MODEL_PROFILES[agentType];
const result = agentModels
? { model: override, profile }
: { model: override, profile, unknown_agent: true };
return { data: result };
}
// resolve_model_ids: "omit" -- return empty string
const resolveModelIds = (config as Record<string, unknown>).resolve_model_ids;
if (resolveModelIds === 'omit') {
const agentModels = MODEL_PROFILES[agentType];
const result = agentModels
? { model: '', profile }
: { model: '', profile, unknown_agent: true };
return { data: result };
}
// Fall back to profile lookup
const agentModels = MODEL_PROFILES[agentType];
if (!agentModels) {
return { data: { model: 'sonnet', profile, unknown_agent: true } };
}
if (profile === 'inherit') {
return { data: { model: 'inherit', profile } };
}
const alias = agentModels[profile] || agentModels['balanced'] || 'sonnet';
return { data: { model: alias, profile } };
};