Settings Schema
The settings schema is the structural contract between guest TOML configs, the Rust backend, and the TypeScript frontend. Pydantic models in Python are the single source of truth. JSON Schema is generated from them. Three languages — Python, Rust, TypeScript — must parse settings identically.
Key files:
| File | Role |
|---|---|
src/capsem/builder/schema.py | Pydantic models (canonical schema) |
config/settings-schema.json | Generated JSON Schema |
config/defaults.json | Generated defaults from guest TOML configs |
crates/capsem-core/tests/settings_spec.rs | Rust conformance tests |
frontend/src/lib/__tests__/settings_spec.test.ts | TypeScript conformance tests |
tests/test_settings_spec.py | Python schema + conformance tests |
tests/settings_spec/golden.json | Golden fixture (shared by all three) |
Two-Node-Type Design
Section titled “Two-Node-Type Design”The settings tree has exactly two node types, discriminated by the kind field:
graph TD
ROOT["SettingsRoot"]
ROOT --> G1["GroupNode\nkind=group"]
ROOT --> G2["GroupNode\nkind=group"]
G1 --> S1["SettingNode\nkind=setting\nsetting_type=bool"]
G1 --> S2["SettingNode\nkind=setting\nsetting_type=text"]
G1 --> S3["SettingNode\nkind=setting\nsetting_type=action"]
G2 --> G3["GroupNode\nkind=group"]
G3 --> S4["SettingNode\nkind=setting\nsetting_type=mcp_tool"]
GroupNode (kind="group"): container with children.
| Field | Type | Required | Description |
|---|---|---|---|
key | string | yes | Dot-separated path (e.g. ai.anthropic) |
name | string | yes | Display name |
description | string | no | Help text |
enabled_by | string | no | Key of a bool setting that gates this group |
enabled | bool | no | Effective enabled state (default true) |
collapsed | bool | yes | Whether the UI renders this group collapsed |
children | SettingsNode[] | yes | Nested groups and settings |
SettingNode (kind="setting"): everything else — regular settings, actions, and MCP tools. The setting_type field determines which subfields are relevant.
| Field | Type | Required | Description |
|---|---|---|---|
key | string | yes | Dot-separated path |
name | string | yes | Display name |
description | string | yes | Help text |
setting_type | SettingType | yes | Data type (see enum table below) |
default_value | any | no | Default from guest config |
effective_value | any | no | Resolved value (corp > user > default) |
source | PolicySource | no | Where effective value came from |
modified | string | no | ISO timestamp of last user change |
corp_locked | bool | no | Whether corp.toml overrides this |
enabled_by | string | no | Key of a bool setting that gates this |
enabled | bool | no | Effective enabled state |
collapsed | bool | no | UI collapse state |
metadata | SettingMetadata | no | Extra fields (defaults to empty) |
history | HistoryEntry[] | no | Audit trail of value changes |
Actions (check_update, preset_select, rerun_wizard) and MCP tools are SettingNode variants. They use setting_type="action" or setting_type="mcp_tool" with the relevant metadata fields. Consumers check setting_type, not kind.
SettingType Enum
Section titled “SettingType Enum”13 values. The first 11 are data types with stored values. The last two are structural variants.
| Value | Category | Description |
|---|---|---|
text | value | Free-form string |
number | value | Integer with optional min/max |
url | value | URL string |
email | value | Email address |
apikey | value | API key (masked input, prefix hint) |
bool | value | Boolean toggle |
file | value | { path, content } object |
kv_map | value | { key: value } dictionary |
string_list | value | Array of strings |
int_list | value | Array of integers |
float_list | value | Array of floats |
action | structural | UI button/widget, no stored value |
mcp_tool | structural | MCP tool definition |
Metadata Fields
Section titled “Metadata Fields”All metadata lives in a single SettingMetadata object. Most fields are optional with sensible defaults. Fields are grouped by purpose.
Common fields
Section titled “Common fields”| Field | Type | Default | Description |
|---|---|---|---|
domains | string[] | [] | Domain patterns for network policy |
choices | string[] | [] | Valid options (drives select widget) |
min | int | null | Minimum value (number types) |
max | int | null | Maximum value (number types) |
rules | dict | {} | HTTP method permissions per rule |
env_vars | string[] | [] | Environment variables injected into guest |
collapsed | bool | false | Default collapse state |
format | string | null | Value format hint (e.g. domain_list) |
docs_url | string | null | Link to external documentation |
prefix | string | null | Expected value prefix (e.g. sk-ant-) |
filetype | string | null | File syntax type (e.g. json) |
widget | Widget | null | Override default UI widget |
side_effect | SideEffect | null | Frontend action on value change |
hidden | bool | false | Exclude from UI, keep for policy |
builtin | bool | false | Non-removable (system setting) |
mask | bool | false | Mask display value |
validator | string | null | Regex pattern for validation |
Action-specific
Section titled “Action-specific”| Field | Type | Default | Description |
|---|---|---|---|
action | ActionKind | null | Action identifier (check_update, preset_select, rerun_wizard) |
MCP tool-specific
Section titled “MCP tool-specific”| Field | Type | Default | Description |
|---|---|---|---|
origin | McpToolOrigin | null | Where the tool runs (builtin, remote, in_vm) |
MCP server-specific (legacy)
Section titled “MCP server-specific (legacy)”| Field | Type | Default | Description |
|---|---|---|---|
transport | McpTransport | null | Protocol (stdio, sse) |
command | string | null | Executable path (stdio transport) |
url | string | null | Server URL (sse transport) |
args | string[] | [] | Command arguments |
env | dict | {} | Environment variables for the server process |
headers | dict | {} | HTTP headers (sse transport) |
JSON Schema Generation
Section titled “JSON Schema Generation”The schema generation pipeline runs from Pydantic models to two output files:
flowchart LR
PM["schema.py\nPydantic models"] --> MSJ["model_json_schema()"]
MSJ --> SCH["config/settings-schema.json"]
GC["guest/config/*.toml"] --> GD["generate_defaults_json()"]
GD --> DEF["config/defaults.json"]
just schema regenerates both files:
just schema# Runs: uv run python scripts/generate_schema.py# Outputs:# config/settings-schema.json (JSON Schema from Pydantic)# config/defaults.json (defaults from guest TOML configs)The JSON Schema is derived from SettingsRoot.model_json_schema(). It contains $defs for all model types (GroupNode, SettingNode, SettingMetadata, enums) and a properties.settings array at the root.
Cross-Language Conformance
Section titled “Cross-Language Conformance”A golden fixture at tests/settings_spec/golden.json is the contract. Three test suites parse the same fixture and verify identical structure:
flowchart TD
GOLDEN["tests/settings_spec/golden.json\n(shared fixture)"]
EXPECTED["tests/settings_spec/expected.json\n(expected counts + fields)"]
GOLDEN --> PY["Python\ntests/test_settings_spec.py\n73 tests"]
GOLDEN --> RS["Rust\ncrates/capsem-core/tests/settings_spec.rs\n12 tests"]
GOLDEN --> TS["TypeScript\nfrontend/.../settings_spec.test.ts\n14 tests"]
EXPECTED --> PY
EXPECTED --> RS
EXPECTED --> TS
PY --> V["All three agree on:\n- total setting count\n- per-type counts\n- group count\n- setting fields\n- roundtrip serialization"]
RS --> V
TS --> V
99 tests total (73 Python, 12 Rust, 14 TypeScript). Every test suite checks:
| Assertion | Verified by |
|---|---|
| Golden fixture parses | All three |
| Total setting count matches expected.json | All three |
| Per-type counts match expected.json | All three |
| Group count matches expected.json | All three |
| Setting key, name, type, enabled_by match | All three |
| Roundtrip serialize/deserialize | Python, Rust |
| All 13 setting types present | All three |
Action settings have metadata.action | All three |
MCP tool settings have metadata.origin | All three |
File settings have { path, content } | All three |
| Hidden/builtin settings exist | All three |
enabled_by references a valid bool | Python, TypeScript |
Any schema change requires updating the golden fixture, expected.json, and all three test suites. just test runs all of them.
Data Flow
Section titled “Data Flow”Two parallel paths connect guest TOML configs to the running application:
flowchart TD
subgraph "Schema Path (dev time)"
PM["schema.py\nPydantic models"] --> JSG["model_json_schema()"]
JSG --> SCHEMA["config/settings-schema.json"]
SCHEMA --> TESTS["Conformance tests\n(Python + Rust + TypeScript)"]
end
subgraph "Data Path (build time)"
TOML["guest/config/*.toml\n(ai, mcp, security, vm)"] --> GEN["generate_defaults_json()"]
GEN --> DEF["config/defaults.json"]
DEF --> RUST["Rust include_str!()\nregistry.rs"]
RUST --> BOOT["Boot-time config\ninjection"]
end
subgraph "Golden Fixture Path (test time)"
GOLDEN2["tests/settings_spec/golden.json"] --> PY2["Python tests"]
GOLDEN2 --> RS2["Rust tests"]
GOLDEN2 --> TS2["TypeScript tests"]
end
The data path: guest TOML configs are processed by generate_defaults_json() into config/defaults.json. Rust embeds this file at compile time via include_str!() in registry.rs. At boot, the registry resolves settings (corp > user > defaults) and injects the result into the VM.
The schema path: Pydantic models generate JSON Schema for documentation and validation. The conformance tests ensure all three languages agree on parsing.
Design Decision: Two Node Types
Section titled “Design Decision: Two Node Types”The original schema had four node types:
| Old type | Discriminant |
|---|---|
| Group | kind="group" |
| Leaf | kind="leaf" |
| Action | kind="action" |
| McpServer | kind="mcp_server" |
This was simplified to two:
| New type | Discriminant | Covers |
|---|---|---|
| GroupNode | kind="group" | Containers with children |
| SettingNode | kind="setting" | Regular settings, actions, MCP tools |
The four-type design forced consumers to match on kind with four arms, even though actions and MCP servers share nearly all fields with regular settings. The two-type design uses setting_type as the discriminant for behavior:
- Regular settings:
setting_typein{text, number, bool, ...}— value fields populated - Actions:
setting_type="action"—metadata.actionspecifies the action kind - MCP tools:
setting_type="mcp_tool"—metadata.originspecifies where the tool runs
Consumers match on kind (two arms: group vs. setting), then check setting_type when they need type-specific behavior. MCP servers are GroupNodes containing server config settings and MCP tool SettingNodes as children. Tool categories (snapshots, network) are nested sub-groups within the server GroupNode.
The Rust conformance tests use local test-only structs with the two-node schema. The live app’s SettingsNode in capsem-core still uses the old four-variant enum for backward compatibility — migration is tracked separately.