schema.ts

 1import { Type, Kind, type Static, type TObject, type TProperties } from "@sinclair/typebox";
 2
 3const CustomModelSchema = Type.Object({
 4  provider: Type.String(),
 5  api: Type.String(),
 6  base_url: Type.String(),
 7  id: Type.String(),
 8  name: Type.String(),
 9  reasoning: Type.Boolean(),
10  input: Type.Array(Type.Union([Type.Literal("text"), Type.Literal("image")])),
11  cost: Type.Object({
12    input: Type.Number(),
13    output: Type.Number(),
14    cache_read: Type.Optional(Type.Number()),
15    cache_write: Type.Optional(Type.Number()),
16  }),
17  context_window: Type.Number(),
18  max_tokens: Type.Number(),
19  api_key: Type.Optional(Type.String()),
20  headers: Type.Optional(Type.Record(Type.String(), Type.String())),
21  compat: Type.Optional(
22    Type.Object({
23      supports_store: Type.Optional(Type.Boolean()),
24      supports_developer_role: Type.Optional(Type.Boolean()),
25      supports_reasoning_effort: Type.Optional(Type.Boolean()),
26      supports_usage_in_streaming: Type.Optional(Type.Boolean()),
27      max_tokens_field: Type.Optional(Type.Union([Type.Literal("max_tokens"), Type.Literal("max_completion_tokens")])),
28      requires_tool_result_name: Type.Optional(Type.Boolean()),
29      requires_assistant_after_tool_result: Type.Optional(Type.Boolean()),
30      requires_thinking_as_text: Type.Optional(Type.Boolean()),
31      requires_mistral_tool_ids: Type.Optional(Type.Boolean()),
32      thinking_format: Type.Optional(Type.Union([Type.Literal("openai"), Type.Literal("zai")])),
33    })
34  ),
35});
36
37export const ConfigSchema = Type.Object({
38  defaults: Type.Object({
39    model: Type.String(),
40    cleanup: Type.Boolean(),
41    kagi_session_token: Type.Optional(Type.String()),
42    tabstack_api_key: Type.Optional(Type.String()),
43  }),
44  web: Type.Object({
45    model: Type.Optional(Type.String()),
46    system_prompt_path: Type.Optional(Type.String()),
47    kagi_session_token: Type.Optional(Type.String()),
48    tabstack_api_key: Type.Optional(Type.String()),
49  }),
50  repo: Type.Object({
51    model: Type.Optional(Type.String()),
52    system_prompt_path: Type.Optional(Type.String()),
53    default_depth: Type.Optional(Type.Number({ minimum: 1 })),
54    blob_limit: Type.Optional(Type.String()),
55  }),
56  custom_models: Type.Optional(Type.Record(Type.String(), CustomModelSchema)),
57});
58
59/** Deep-partial version of ConfigSchema for validating TOML override files. */
60export function partialObject<T extends TProperties>(schema: TObject<T>) {
61  const partial: Record<string, unknown> = {};
62  for (const [key, value] of Object.entries(schema.properties)) {
63    const v = value as any;
64    const inner = v[Kind] === 'Object' && v.properties ? partialObject(v) : v;
65    partial[key] = Type.Optional(inner as any);
66  }
67  return Type.Object(partial as any);
68}
69
70export const PartialConfigSchema = Type.Object({
71  defaults: Type.Optional(partialObject(ConfigSchema.properties.defaults)),
72  web: Type.Optional(partialObject(ConfigSchema.properties.web)),
73  repo: Type.Optional(partialObject(ConfigSchema.properties.repo)),
74  custom_models: Type.Optional(Type.Record(Type.String(), CustomModelSchema)),
75});
76
77export type RumiloConfig = Static<typeof ConfigSchema>;
78export type CustomModelConfig = Static<typeof CustomModelSchema>;