Check whether evolving JSON Schemas and OpenAPI 3.1 contracts stay backward-compatible.
jsoncompat compares:
- raw JSON Schema Draft 2020-12 documents;
- OpenAPI 3.1 Schema Objects;
- JSON OpenAPI 3.1 documents with path operations.
If a schema declares $schema, it must use Draft 2020-12 or the OpenAPI 3.1 Schema Object dialect. OpenAPI 3.0-only shortcuts such as nullable are not reinterpreted.
Warning
jsoncompat is alpha software. It is intentionally conservative in places, and it can still miss incompatible changes or report false positives.
The full docs and examples live at jsoncompat.com.
Install the CLI with Cargo:
cargo install jsoncompatPython and JavaScript/WebAssembly packages are documented separately:
Check a serializer-facing schema change:
jsoncompat compat old-schema.json new-schema.json --role serializerCheck both serializer and deserializer compatibility, and ask for fuzzed counterexamples when static analysis finds a problem:
jsoncompat compat old-schema.json new-schema.json --role both --fuzz 1000 --depth 8Check an OpenAPI 3.1 contract:
jsoncompat compat --openapi old-openapi.json new-openapi.jsonGenerate example values accepted by a schema:
jsoncompat generate schema.json --count 5 --prettyCompare schema golden files in CI:
jsoncompat ci old-golden.json new-golden.json --display tableInspect the per-operation request and response schemas generated from an OpenAPI document:
jsoncompat lower-openapi openapi.jsonRun the guided CLI demo:
jsoncompat demo --noninteractivejsoncompat stamp turns a schema into separate writer and reader schemas using
a versioned envelope:
{
"version": 2,
"data": {
"name": "Ada"
}
}Writers emit only the latest schema version, while readers accept a tagged union of historical writer versions. The command stores schema history in a manifest file and appends a new version whenever a change is not compatible in both directions.
jsoncompat stamp --manifest schemas.manifest.json --id user-profile --write-manifest schema.json
jsoncompat stamp --manifest schemas.manifest.json --id user-profile --display writer schema.json > writer.schema.json
jsoncompat stamp --manifest schemas.manifest.json --id user-profile --display reader schema.json > reader.schema.json
jsoncompat codegen --target schema reader.schema.json
jsoncompat codegen --target dataclasses reader.schema.json > reader_models.pyjsoncompat codegen --target dataclasses accepts any JSON Schema document,
canonicalizes it with SchemaDocument::canonical_schema_json(), and emits
frozen, slotted Python dataclasses that import shared construction and
serialization helpers from jsoncompat.codegen.dataclasses. Generated classes
carry the original input schema in __jsoncompat_schema__, cache a
jsoncompat.validator_for(...) validator for runtime checks, and expose:
from_json(...)/from_json_string(...)constructors for schema-checked deserialization;to_json(...)/to_json_string(...)serializers that validate emitted JSON against the attached schema;__jsoncompat_extra__for schema-admitted object properties that are not declared underproperties, includingadditionalPropertiesandpatternProperties;JSONCOMPAT_MISSINGfor omitted optional fields so absent and explicitnullstay distinguishable.
When the schema structure makes it honest, code generation also keeps Python
annotations narrow rather than collapsing to Any, including primitive local
$ref fields rooted under $defs or legacy definitions, plus constrained
tuple-like arrays built from prefixItems.
If the input schema contains x-jsoncompat metadata from jsoncompat stamp,
generated writer envelopes inherit from WriterDataclassModel, which disables
deserialization methods, and generated reader envelopes inherit from
ReaderDataclassModel / ReaderDataclassRootModel, which disable
serialization methods.
Compatibility is directional:
| Role | Question jsoncompat answers |
|---|---|
serializer |
Can old readers still accept every value the new producer may emit? |
deserializer |
Can the new reader still accept every value older producers may have emitted? |
both |
Are both directions safe? |
That is why making a previously required response field optional can be breaking for a serializer, while making a previously optional stored field required can be breaking for a deserializer.
When the inputs are OpenAPI documents, pass --openapi. jsoncompat compares:
- path, query, header, and cookie parameters;
- request bodies and media types;
- response statuses, media types, bodies, and headers;
- removed operations;
- supported local
#/components/...references.
Requests are checked in the deserializer direction. Responses are checked in the serializer direction. --role and --fuzz are raw-JSON-Schema-only flags.
See openapi/README.md for the OpenAPI user guide.
Schema compatibility:
use jsoncompat::{Role, SchemaDocument, check_compat};
use serde_json::json;
let old = SchemaDocument::from_json(&json!({ "type": "string" })).unwrap();
let new = SchemaDocument::from_json(&json!({ "type": ["string", "null"] })).unwrap();
let compatible = check_compat(&old, &new, Role::Deserializer).unwrap();OpenAPI compatibility:
use jsoncompat::{OpenApiDocument, check_openapi_compat};
use serde_json::json;
let old = OpenApiDocument::from_json(&json!({
"openapi": "3.1.0",
"info": { "title": "Pets", "version": "1.0.0" },
"paths": {}
})).unwrap();
let new = old.clone();
let report = check_openapi_compat(&old, &new).unwrap();
assert!(report.is_compatible());The Rust API also exposes structured compatibility errors, OpenAPI issue reports, incompatibility explanations, and schema-guided value generation.
jsoncompat keeps warnings and hard errors separate:
- unsupported-but-valid schema details produce warnings and the modeled comparison continues;
- inputs that would make a verdict unsafe fail before comparison;
- unsupported OpenAPI contract surfaces fail before comparison rather than being silently ignored.
The CLI prints warnings with exact pointers so you can see what was ignored. See developing.md for the detailed support boundaries and the reasoning behind them.
- jsoncompat.com for polished documentation and examples
- openapi/README.md for OpenAPI-specific usage
- developing.md for repository layout, internals, tests, fixtures, benchmarks, and release notes
- docs.rs for the Rust API reference
MIT License. See LICENSE.
