GET/PATCH /api/providers/config, subset POST /refresh, and
GET /api/providers/:id/diagnostic (JSON { diagnostic }, §6.4). PATCH order
is validate→save→reload→clear; a malformed body or invalid merged config
returns 422 without writing, and a save failure returns 500 without
reloading (no file/registry divergence). Web client + types extended.
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
101 lines
3.7 KiB
TypeScript
101 lines
3.7 KiB
TypeScript
/**
|
|
* v2.3 provider config file (`/data/coder-providers.json`) — schema + loader.
|
|
*
|
|
* Layers config-backed overrides/custom-ACP entries over the hardcoded built-ins
|
|
* (see provider-config-registry.ts). Loading NEVER throws at startup (design.md
|
|
* §2.1): a missing file, invalid JSON, or schema mismatch all fall back to
|
|
* `{ providers: {} }` (built-ins only, all enabled).
|
|
*/
|
|
import { readFileSync, writeFileSync } from 'node:fs';
|
|
import { z } from 'zod';
|
|
|
|
// Schemas verbatim from design.md §2.2.
|
|
export const ProviderOverrideSchema = z.object({
|
|
extends: z.enum(['acp']).optional(), // v2.3: only 'acp' for custom; built-ins omit extends
|
|
label: z.string().min(1).optional(),
|
|
description: z.string().optional(),
|
|
command: z.array(z.string().min(1)).min(1).optional(), // [binary, ...args]
|
|
env: z.record(z.string()).optional(),
|
|
enabled: z.boolean().optional(), // default true
|
|
order: z.number().int().optional(), // UI sort key
|
|
models: z.array(z.object({ id: z.string(), label: z.string() })).optional(),
|
|
additionalModels: z.array(z.object({ id: z.string(), label: z.string() })).optional(),
|
|
});
|
|
|
|
export const CoderProvidersFileSchema = z.object({
|
|
providers: z.record(ProviderOverrideSchema).default({}),
|
|
});
|
|
|
|
export type ProviderOverride = z.infer<typeof ProviderOverrideSchema>;
|
|
export type CoderProvidersFile = z.infer<typeof CoderProvidersFileSchema>;
|
|
|
|
/**
|
|
* PATCH body schema (design.md §6.2). A partial providers map where each value
|
|
* is either a full override object (REPLACES that id's override) or `null`
|
|
* (DELETES the override → revert to the built-in default). Ids absent from the
|
|
* patch are left untouched. The route validates the body against this first
|
|
* (malformed → 422) so a bad shape can never reach the merge/save step.
|
|
*/
|
|
export const ProviderConfigPatchSchema = z.object({
|
|
providers: z.record(ProviderOverrideSchema.nullable()).default({}),
|
|
});
|
|
|
|
export type ProviderConfigPatch = z.infer<typeof ProviderConfigPatchSchema>;
|
|
|
|
/**
|
|
* Shallow per-id merge (design.md §6.2 / Paseo `patchConfig`). Each key in
|
|
* `patch.providers` REPLACES that id's override object wholesale (NOT a deep
|
|
* field merge); a `null` value DELETES the override. Returns a new object —
|
|
* never mutates `current`. The result is a plain CoderProvidersFile (no nulls),
|
|
* which the route re-validates against CoderProvidersFileSchema before save.
|
|
*/
|
|
export function mergeProviderConfigPatch(
|
|
current: CoderProvidersFile,
|
|
patch: ProviderConfigPatch,
|
|
): CoderProvidersFile {
|
|
const providers: Record<string, ProviderOverride> = { ...current.providers };
|
|
for (const [id, override] of Object.entries(patch.providers)) {
|
|
if (override === null) {
|
|
delete providers[id];
|
|
} else {
|
|
providers[id] = override;
|
|
}
|
|
}
|
|
return { providers };
|
|
}
|
|
|
|
/** Read + parse + validate. Falls back to built-ins-only on any failure; never throws. */
|
|
export function load(path: string): CoderProvidersFile {
|
|
let raw: string;
|
|
try {
|
|
raw = readFileSync(path, 'utf8');
|
|
} catch {
|
|
// Missing file → built-ins only. Expected, not an error.
|
|
return { providers: {} };
|
|
}
|
|
|
|
let json: unknown;
|
|
try {
|
|
json = JSON.parse(raw);
|
|
} catch (err) {
|
|
console.error(`provider-config: invalid JSON in ${path} — using built-ins only`, err);
|
|
return { providers: {} };
|
|
}
|
|
|
|
const parsed = CoderProvidersFileSchema.safeParse(json);
|
|
if (!parsed.success) {
|
|
console.error(
|
|
`provider-config: schema validation failed for ${path} — using built-ins only`,
|
|
parsed.error.flatten(),
|
|
);
|
|
return { providers: {} };
|
|
}
|
|
|
|
return parsed.data;
|
|
}
|
|
|
|
/** Write the config back to disk (used by the Phase 4 PATCH route). */
|
|
export function save(path: string, config: CoderProvidersFile): void {
|
|
writeFileSync(path, `${JSON.stringify(config, null, 2)}\n`, 'utf8');
|
|
}
|