JSON contract registry and validator for structured LLM and agent outputs.
JSON contract registry and validator for structured LLM and agent outputs.
JSON Contracts · v0.1.2
by Json-contracts
JSON Contracts MCP Server
JSON Contracts is a local MCP contract server for natural-language-to-JSON workflows. It gives agents Git-controlled JSON contracts, rules, examples, and JSON Schema validation tools so any model can reliably convert natural-language requests into schema-valid JSON.
The MCP server does not call an LLM provider.
The MCP server does not need API keys.
The MCP server does not use BAML, LangChain, Markdown, or a DSL.
The MCP server does not use MCP sampling.
The MCP server does not generate JSON by itself.
The optional Studio is a separate package/repo for live demos and local contract testing. The MCP stdio server itself does not call LLM providers.
Your user provides natural language.
Your agent chooses the model.
Your agent performs the natural-language-to-JSON conversion.
json-contracts provides the contract and validates the result.
Correct mental model
Not this:
json-contracts = generator
This:
json-contracts = schema contract registry + validator
Like TypeScript:
developer writes code
TypeScript validates it
With json-contracts:
agent writes JSON
json-contracts validates it
Architecture
User
↓
Agent using whatever model the user picked
↓
json-contracts MCP server
- lists available JSON contracts
- returns schemas, rules, examples, and instructions
- validates agent-produced JSON
- returns repair contracts when validation fails
↓
Agent generates or repairs JSON itself
↓
App consumes valid JSON
The MCP server never performs natural-language-to-JSON conversion itself. It only provides contracts, validation, and repair guidance.
Install
Install globally:
npm install -g json-contracts
json-contracts --help
Or run without installing:
npx -y json-contracts@latest
The npm package is json-contracts; the installed CLI binary is still json-contracts.
30-second setup
Create a starter contract folder and validate it:
mkdir my-json-contracts
cd my-json-contracts
npx -y json-contracts@latest init
npx -y json-contracts@latest validate
Then add the MCP config below to your agent host. Point JSON_CONTRACTS_DIR at the json-contracts folder that init created.
Validate contracts in CI without starting MCP:
npx -y json-contracts@latest validate --contracts ./json-contracts
npx -y json-contracts@latest lint --strict --contracts ./json-contracts
By default, the server starts as a local stdio MCP server and loads contracts from:
./json-contracts
MCP config
{
"mcpServers": {
"json-contracts": {
"command": "npx",
"args": ["-y", "json-contracts@latest"],
"env": {
"JSON_CONTRACTS_DIR": "./json-contracts"
}
}
}
}
Adding a new behavior only requires adding a new .json file to the contracts folder. No MCP config change is required.
MCP host config snippets
Most MCP hosts use the same stdio shape. Adapt paths for your machine and point JSON_CONTRACTS_DIR at your app-owned contracts folder.
Claude Desktop
{
"mcpServers": {
"json-contracts": {
"command": "npx",
"args": ["-y", "json-contracts@latest"],
"env": {
"JSON_CONTRACTS_DIR": "/absolute/path/to/json-contracts"
}
}
}
}
Cursor / Windsurf / VS Code MCP-compatible hosts
{
"mcpServers": {
"json-contracts": {
"command": "npx",
"args": ["-y", "json-contracts@latest"],
"env": {
"JSON_CONTRACTS_DIR": "./json-contracts"
}
}
}
}
Continue or generic stdio MCP clients
{
"name": "json-contracts",
"command": "npx",
"args": ["-y", "json-contracts@latest"],
"env": {
"JSON_CONTRACTS_DIR": "./json-contracts"
}
}
If npx startup is too slow or your host requires explicit executables, install globally and use:
{
"command": "json-contracts",
"args": []
}
Adopt in your agent/app
Use json-contracts as a local contract/validation tool beside the model your app already uses.
1. Add the MCP server
If your agent host supports MCP, add the server config above and point JSON_CONTRACTS_DIR at the contracts folder your app owns.
If your app has its own agent runtime, connect to the same MCP stdio server and call the tools directly. The important part is the flow, not the host.
2. Write one contract per JSON behavior
Create files such as:
json-contracts/support-ticket.json
json-contracts/real-estate-lead.json
json-contracts/chart-generation.json
Each file contains:
schemafor the final JSON shaperulesfor app-specific mapping behaviorexamplesfor model guidance- optional
operationsfor create/edit behavior
3. Pass app/system variables as context
Do not hide runtime variables in the user prompt. Pass them as context:
{
"contract": "real-estate-lead",
"input": "I'm pre-approved for a 4 bedroom house in Durham NC up to 900k.",
"context": {
"current_datetime": "2026-05-03T00:00:00Z",
"email": "JohnnyAppleseed@gmail.com",
"source": "website-lead-form"
}
}
json-contracts passes context through unchanged. Contracts decide how to use it through rules/examples. The final JSON still must match the schema.
4. Give your agent this tool policy
Use this as the system/developer instruction for your agent:
When converting natural language into app JSON, use json-contracts.
Create flow:
1. Call get_json_contract with contract, input, and context.
2. Generate JSON using the returned schema, rules, examples, input, and context.
3. Call validate_json.
4. If invalid, call get_repair_contract, repair the JSON, and validate again.
5. Return only validated JSON to the app.
Edit flow:
1. Call get_edit_contract with contract, currentJson, input, and context.
2. Return the complete updated object, not a patch.
3. Call validate_json.
4. Repair and validate again if needed.
Never skip validation. Never add fields that are not allowed by the schema. Do not copy context fields into output unless the schema allows them and the contract rules say to use them.
5. Runtime loop in your app
At request time, your app should do:
user input + app context
-> get_json_contract or get_edit_contract
-> your chosen model generates JSON
-> validate_json
-> if invalid: get_repair_contract -> model repairs -> validate_json
-> app consumes valid JSON
The MCP server does not call the model and does not mutate output. Your app/agent remains in control of model choice, provider keys, context, and business defaults.
Contract files
Each contract is one .json file in json-contracts/.
The contract name is derived from the filename:
json-contracts/support-ticket.json -> support-ticket
There is no manifest file and no manually maintained resources file. Git handles versioning.
Contract shape
{
"description": "Convert natural language into a support ticket object.",
"rules": [
"If the user says urgent, severity must be critical.",
"Summary must be under 80 characters.",
"Category must be authentication, billing, bug, feature_request, or other."
],
"operations": {
"create": {
"enabled": true
},
"edit": {
"enabled": true,
"return": "full_object",
"rules": [
"Start from currentJson.",
"Apply only the user's requested change.",
"Preserve all unspecified fields exactly.",
"Return the complete updated JSON object."
]
}
},
"schema": {
"type": "object",
"additionalProperties": false,
"properties": {
"summary": {
"type": "string",
"maxLength": 80
},
"severity": {
"type": "string",
"enum": ["low", "medium", "high", "critical"]
},
"category": {
"type": "string",
"enum": ["authentication", "billing", "bug", "feature_request", "other"]
}
},
"required": ["summary", "severity", "category"]
},
"examples": [
{
"input": "Urgent, users cannot log in after SSO update.",
"output": {
"summary": "Users cannot log in after SSO update",
"severity": "critical",
"category": "authentication"
}
}
]
}
Rules:
schemais required and must be valid JSON Schema. Omit$schemafor the default 2020-12 validator, or set$schemato draft-07 or 2020-12 explicitly.descriptionis optional but recommended.rulesis optional and defaults to[].examplesis optional and defaults to[]. Exampleoutputvalues must validate againstschema.operationsis optional and defaults to enabledcreateandeditoperations. Operation metadata belongs at the top level, not inside the JSON Schema.nameis optional, but the filename is the source of truth.- A
versionfield is rejected; use Git for versioning. - Contract files are plain JSON.
Included example contracts
The default json-contracts/ folder includes examples from different app categories where teams commonly rebuild the same AI glue code:
| Contract | Industry/app pattern | Converts natural language into |
|---|---|---|
support-ticket |
SaaS support | Triage-ready support tickets. |
create-filter |
Internal tools/API builders | API filter objects. |
chart-generation |
BI/analytics tools | Dashboard chart generation specs. |
patient-intake |
Healthcare intake | Triage and appointment-routing objects. |
ecommerce-return |
Ecommerce support | Return, refund, exchange, and warranty requests. |
real-estate-lead |
Real estate CRM | Buyer, renter, seller, and lease lead profiles. |
legal-client-intake |
Legal tech intake | Matter routing and conflict-check data. |
expense-report |
Finance/expense apps | Reimbursement and expense line items. |
Each one uses the same MCP tools: read the contract, let the model create or edit JSON, validate it, and repair if needed. New apps should not need a new bespoke prompt framework just to get reliable JSON.
MCP resources
Every loaded contract is exposed dynamically as:
json-contract://{contractName}
Examples:
json-contract://support-ticket
json-contract://create-filter
json-contract://chart-generation
resources/list returns all loaded contracts. resources/read returns the normalized full contract JSON.
Stable MCP tools
list_contracts
Input:
{}
Output:
{
"contracts": [
{
"name": "support-ticket",
"description": "Convert natural language into a support ticket object.",
"contractHash": "sha256:...",
"schemaHash": "sha256:..."
}
]
}
read_contract
Input:
{
"contract": "support-ticket"
}
Returns the selected contract's description, rules, operations, schema, examples, contractHash, and schemaHash.
Hashes are deterministic SHA-256 values over the normalized contract/schema payload. They are for logging, cache keys, and audit trails; contract files still use Git for versioning.
get_json_contract
Input:
{
"contract": "support-ticket",
"input": "Urgent, users cannot log in after SSO update.",
"context": {}
}
Output:
{
"contract": "support-ticket",
"contractHash": "sha256:...",
"schemaHash": "sha256:...",
"operation": "create",
"instructions": [
"Convert the input into JSON.",
"Return JSON only.",
"Do not return markdown.",
"Do not include commentary.",
"Do not include extra keys.",
"Match the schema exactly.",
"Use enum values exactly.",
"Follow all rules.",
"Use examples as guidance."
],
"description": "Convert natural language into a support ticket object.",
"rules": [
"If the user says urgent, severity must be critical.",
"Summary must be under 80 characters.",
"Category must be authentication, billing, bug, feature_request, or other."
],
"operationRules": [],
"schema": {},
"examples": [],
"operationExamples": [],
"input": "Urgent, users cannot log in after SSO update.",
"context": {}
}
The agent/model uses this contract to produce JSON. The MCP server does not produce it.
get_edit_contract
Input:
{
"contract": "create-filter",
"currentJson": {
"status": "open",
"limit": 50
},
"input": "we want the last 20 closed tickets",
"context": {}
}
Output:
{
"contract": "create-filter",
"contractHash": "sha256:...",
"schemaHash": "sha256:...",
"operation": "edit",
"instructions": [
"Start from currentJson.",
"Apply only the user's requested change.",
"Preserve all unspecified fields exactly.",
"Return the complete updated JSON object, not a patch.",
"Return JSON only."
],
"description": "Convert natural language into a structured API filter object.",
"rules": [
"Only include fields that are explicitly requested or clearly implied."
],
"operationRules": [
"Preserve all unspecified fields exactly."
],
"schema": {},
"examples": [],
"operationExamples": [],
"currentJson": {
"status": "open",
"limit": 50
},
"input": "we want the last 20 closed tickets",
"context": {}
}
The agent/model uses this edit contract to return the complete updated JSON object. The MCP server first validates currentJson against the selected contract, and the agent should validate the edited object with validate_json afterward.
Context and system variables
context is an intentional pass-through object for app/system variables that should help the model interpret the user's input.
Examples:
{
"contract": "real-estate-lead",
"input": "I'm pre-approved for a 4 bedroom house in Durham NC up to 900k. We need a pool and a large backyard for dogs. We are looking to move this summer.",
"context": {
"current_datetime": "2026-05-03T00:00:00Z",
"email": "JohnnyAppleseed@gmail.com"
}
}
json-contracts does not interpret, normalize, or validate context against a separate schema. It returns the object unchanged in the contract payload so the model can use it with the contract rules and output schema.
Important behavior:
- Put external variables in
context, not by appending hidden text toinput. - The final JSON must still match the contract schema.
- Context fields should not appear in the final JSON unless the contract schema allows them.
- Contracts can define how to use context through
rules, examples, and schema shape. - Relative values such as "this summer" or "last week" should be resolved by the model using whatever date/time/location context the app provides.
validate_json
Input:
{
"contract": "support-ticket",
"json": {
"summary": "Users cannot log in after SSO update",
"severity": "critical",
"category": "authentication"
}
}
Success:
{
"valid": true,
"contract": "support-ticket",
"json": {
"summary": "Users cannot log in after SSO update",
"severity": "critical",
"category": "authentication"
},
"errors": []
}
Failure:
{
"valid": false,
"contract": "support-ticket",
"errors": [
{
"path": "/severity",
"message": "must be equal to one of the allowed values",
"keyword": "enum"
}
]
}
valid is never true unless Ajv validates the JSON against the contract schema.
get_repair_contract
Input:
{
"contract": "support-ticket",
"invalidJson": {
"summary": "Users cannot log in",
"severity": "urgent"
},
"validationErrors": []
}
Output:
{
"contract": "support-ticket",
"contractHash": "sha256:...",
"schemaHash": "sha256:...",
"instructions": [
"Repair the JSON so it validates against the schema.",
"Return JSON only.",
"Do not return markdown.",
"Do not include commentary.",
"Do not include extra keys.",
"Preserve valid fields where possible."
],
"schema": {},
"rules": [],
"examples": [],
"invalidJson": {},
"validationErrors": []
}
The agent/model uses this repair contract to produce corrected JSON. The MCP server does not repair by calling a model. When validation errors are available, instructions also includes deterministic field-specific repair hints such as adding missing required fields, removing extra fields, or choosing allowed enum values.
status
Input:
{}
Output:
{
"server": "json-contracts",
"version": "0.1.0",
"contractsDir": "/absolute/path/to/json-contracts",
"loaded": 3,
"contracts": [
{
"name": "support-ticket",
"description": "Convert natural language into a support ticket object.",
"contractHash": "sha256:...",
"schemaHash": "sha256:..."
}
],
"watchContracts": true,
"allowInvalidContracts": false
}
Use this to debug host configuration, loaded contracts, and the exact contract/schema hashes in use.
reload_contracts
Input:
{}
Output:
{
"loaded": 3,
"contracts": ["support-ticket", "create-filter", "chart-generation"]
}
Optional MCP prompts
The server also exposes reusable prompt helpers for MCP hosts that support prompts:
json_contract_promptedit_contract_promptrepair_contract_prompt
These prompts only render contract text for the agent/model. They do not call a model and they do not generate JSON inside the MCP server.
Correct flow
User:
Create a support ticket: urgent, users cannot log in after SSO update.
Agent calls:
{
"tool": "get_json_contract",
"arguments": {
"contract": "support-ticket",
"input": "Urgent, users cannot log in after SSO update."
}
}
MCP returns schema, rules, examples, instructions, and input.
Agent/model produces:
{
"summary": "Users cannot log in after SSO update",
"severity": "critical",
"category": "authentication"
}
Agent calls:
{
"tool": "validate_json",
"arguments": {
"contract": "support-ticket",
"json": {
"summary": "Users cannot log in after SSO update",
"severity": "critical",
"category": "authentication"
}
}
}
MCP returns:
{
"valid": true,
"contract": "support-ticket",
"json": {
"summary": "Users cannot log in after SSO update",
"severity": "critical",
"category": "authentication"
},
"errors": []
}
If invalid, the agent calls get_repair_contract, uses its own model to repair, and calls validate_json again.
Edit flow
For existing JSON, the agent calls get_edit_contract with the current JSON plus a natural-language change request:
{
"tool": "get_edit_contract",
"arguments": {
"contract": "create-filter",
"currentJson": {
"status": "open",
"limit": 50
},
"input": "we want the last 20 closed tickets"
}
}
The model returns the complete edited object:
{
"status": "closed",
"limit": 20
}
Then the agent validates that final edited object with validate_json.
Contract validation and linting CLI
Use the CLI in CI or pre-commit checks without starting an MCP host:
json-contracts validate --contracts ./json-contracts
validate loads every .json contract, checks the contract shape, validates the JSON Schema, and validates example outputs against the schema. It exits nonzero if any contract is invalid.
For advisory checks:
json-contracts lint --contracts ./json-contracts
json-contracts lint --strict --contracts ./json-contracts
lint includes the same validation checks, then prints generic schema-quality warnings such as empty schemas, missing examples, object schemas without additionalProperties:false, or broad additionalProperties:true. --strict exits nonzero when warnings are found.
Both commands support machine-readable output:
json-contracts validate --json --contracts ./json-contracts
json-contracts lint --json --strict --contracts ./json-contracts
Minimal app integration example
A tiny Node MCP client is included at examples/node-client. It shows the app-side loop:
get_json_contract -> your model -> validate_json -> optional get_repair_contract -> your model -> validate_json
It intentionally does not call an LLM provider. Replace its placeholder getModelJson() with your own model call.
Studio
The local web Studio is intentionally not bundled with this MCP package. It lives in a separate repo/package so the MCP server stays small, local, and focused.
Use the Studio when you want a browser UI for testing contracts, provider-backed demos, manual repair loops, or drafting new contract files.
For a quick demo with starter contracts:
mkdir my-json-contracts
cd my-json-contracts
npx -y json-contracts@latest init
npx -y json-contracts-studio@latest
Repository:
https://github.com/json-contracts/json-contracts-studio
Pi local integration
This repo also includes a project-local Pi extension at .pi/extensions/json-contracts-mcp.ts. It starts the local MCP stdio server and exposes the server tools to Pi as jc_* tools for manual testing.
From this repo on Windows PowerShell:
npm run build
pi
Then try:
/jc-status
See docs/pi-integration.md for the full setup and test prompts.
Documentation site and content strategy
The docs/ folder can be published directly with GitHub Pages from the /docs branch folder setting.
docs/index.mdis the docs landing page.docs/github-pages.mdexplains how to enable GitHub Pages.docs/content-marketing-plan.mdcontains the short-form video/content plan for positioningjson-contractsaround structured output, MCP validation, and production-safe JSON flows.
Git-controlled behavior
The json-contracts/ folder is the product surface.
- Add behavior by adding a new
.jsonfile. - Edit behavior by editing an existing
.jsonfile. - Review behavior through Git pull requests.
- Roll back behavior through Git.
No provider keys, SDKs, sampling, or MCP config changes are required when contracts change.
Environment variables
| Variable | Default | Description |
|---|---|---|
JSON_CONTRACTS_DIR |
./json-contracts |
Folder containing contract .json files. |
MCP_TRANSPORT |
stdio |
Transport. v1 implements stdio. |
PORT |
3000 |
Reserved for future HTTP transport. |
API_AUTH_TOKEN |
unset | Reserved for future HTTP Bearer auth. |
DEBUG |
false |
Enables debug logging to stderr. |
WATCH_CONTRACTS |
true |
Watches local contracts and reloads on changes. |
ALLOW_INVALID_CONTRACTS |
false |
If true, invalid contracts are skipped with warnings. |
Studio-only LLM provider variables are documented in the separate json-contracts-studio repo. The MCP stdio server does not use them.
In stdio mode, logs are written to stderr only. The server never writes logs to stdout.
Security notes
The MCP server in json-contracts:
- never calls an LLM provider
- never uses provider API keys
- never performs MCP sampling
- never executes anything from contract files
- never uses
eval - never imports or executes code from contract files
- never logs full user input unless
DEBUG=true - validates MCP tool inputs with Zod
- enforces contract file, schema, and examples limits
- rejects unsafe contract names and resource URIs
- prevents path traversal
The optional separate Studio can call LLM providers only when you configure a key in its local UI or .env file; those provider calls are not part of the MCP stdio server.
Licensing and trademarks
Project owner: Harry Giunta.
Licensing model:
- Runtime code, docs, examples, tests, and project infrastructure are licensed under Apache-2.0. See
LICENSEandNOTICE. - Official starter contracts in
json-contracts/are licensed under Apache-2.0 OR MIT, at your option. Seejson-contracts/LICENSE.md. - Third-party or marketplace contract packs may use creator-selected licenses. They should include their own license file or license metadata and must not imply official status unless approved.
The project name json-contracts is subject to the trademark policy in TRADEMARKS.md. Copyright licenses do not grant trademark rights.
See CONTRIBUTING.md for contribution licensing and project guidelines.
Development
npm install
npm test
npm run build
npm run dev
Run the published package as an MCP stdio server:
npx -y json-contracts@latest
Or from this repository:
npm run dev
Docker
Docker support is optional. A sample Dockerfile is included for packaging the stdio server with its default contracts.