Object-oriented JSONSchema
- Traverse schema like paths
- Access schema on demand with separate dereferencing accessor layer
pip install jsonschema-pathAlternatively you can download the code and install from the repository:
pip install -e git+https://github.com/p1c2u/jsonschema-path.git#egg=jsonschema_path>>> from jsonschema_path import SchemaPath
>>> d = {
... "properties": {
... "info": {
... "$ref": "#/$defs/Info",
... },
... },
... "$defs": {
... "Info": {
... "properties": {
... "title": {
... "$ref": "http://example.com",
... },
... "version": {
... "type": "string",
... "default": "1.0",
... },
... },
... },
... },
... }
>>> path = SchemaPath.from_dict(d)
>>> # Stat keys
>>> "properties" in path
True
>>> # Concatenate paths with /
>>> info_path = path / "properties" / "info"
>>> # Stat keys with implicit dereferencing
>>> "properties" in info_path
True
>>> # Concatenate paths with implicit dereferencing
>>> version_path = info_path / "properties" / "version"
>>> # Open content with implicit dereferencing
>>> with version_path.open() as contents:
... print(contents)
{'type': 'string', 'default': '1.0'}Two SchemaPath instances are equal if they have the same parts
and point to the same SchemaAccessor. SchemaAccessor identity
is per-resource-handle: same wrapped dict (by reference), same
base_uri, and same internal resolver instance. In practice:
Paths derived from the same accessor compare equal as expected:
>>> accessor = SchemaAccessor.from_schema(d) >>> SchemaPath(accessor) / "properties" == SchemaPath(accessor) / "properties" True
Paths from separate
from_dictorfrom_schemacalls do not compare equal even with identical arguments, because each call builds its own accessor:>>> SchemaPath.from_dict(d) == SchemaPath.from_dict(d) False
SchemaAccessoris hashable, so accessors and paths can be used as set members and dict keys.
This is also why the "build one accessor, reuse it" pattern below matters: it is both a caching optimisation and the contract you need for path equality to behave the way you expect.
The resolved-path cache is intended for repeated path lookups and may significantly improve
read_value/membership hot paths. When the underlying referencing
registry grows (e.g. an external $ref pulls in a new document),
cached entries are rebound to the new registry on read instead of
being discarded. This relies on registries growing monotonically —
resources are added, never replaced. Handlers that return drifting
content for the same URI violate that assumption; disable caching with
resolved_cache_maxsize=0 if you need to defend against it.
This cache is enabled by default
(resolved_cache_maxsize=128). You can disable it when creating paths or
accessors, for example:
>>> path = SchemaPath.from_dict(d, resolved_cache_maxsize=0)Build one SchemaAccessor per schema document and reuse it for
every SchemaPath you derive from that document. Treat the accessor
as the long-lived handle to the resource; treat paths as cheap views over
it.
>>> from jsonschema_path import SchemaAccessor, SchemaPath
>>> # Construct the accessor once, with caching enabled.
>>> accessor = SchemaAccessor.from_schema(d, resolved_cache_maxsize=128)
>>> # Derive as many paths as you like from the same accessor.
>>> root = SchemaPath(accessor)
>>> info = root / "properties" / "info"
>>> version = info / "properties" / "version"
>>> # Every path over `accessor` shares the same resolved-ref cache.
>>> with version.open() as contents:
... ...Benchmarks mirror the lightweight (dependency-free) JSON output format used in pathable.
Run locally with Poetry:
poetry run python -m tests.benchmarks.bench_parse --output reports/bench-parse.json
poetry run python -m tests.benchmarks.bench_lookup --output reports/bench-lookup.jsonFor a quick smoke run:
poetry run python -m tests.benchmarks.bench_parse --output reports/bench-parse.quick.json --quick
poetry run python -m tests.benchmarks.bench_lookup --output reports/bench-lookup.quick.json --quickYou can also control repeats/warmup and resolved cache maxsize via env vars:
export JSONSCHEMA_PATH_BENCH_REPEATS=5
export JSONSCHEMA_PATH_BENCH_WARMUP=1
export JSONSCHEMA_PATH_BENCH_RESOLVED_CACHE_MAXSIZE=64Compare two results:
poetry run python -m tests.benchmarks.compare_results \
--baseline reports/bench-lookup-master.json \
--candidate reports/bench-lookup.json \
--tolerance 0.20- openapi-core
- Python library that adds client-side and server-side support for the OpenAPI.
- openapi-spec-validator
- Python library that validates OpenAPI Specs against the OpenAPI 2.0 (aka Swagger) and OpenAPI 3.0 specification
- openapi-schema-validator
- Python library that validates schema against the OpenAPI Schema Specification v3.0.
Copyright (c) 2017-2025, Artur Maciag, All rights reserved. Apache-2.0