Skip to main content

More JSON Configuration! JSON configuration files with `$ref` and template overlays

Project description

More JSON Configuration!

PyPI Latest Release Build Status Coverage Status

A JSON template format intended for configuration files.

See changes

Overview

This module reads JSON files and expands references found within. It is much like the IETF's JSON Reference specification, but with the following differences:

  1. This module uses the dot (.) as a path separator in the URL fragment. For example, an absolute reference looks like {"$ref": "#message.type.name"}, and a relative reference looks like {"$ref": "#..type.name"}. This syntax better matches that used by Javascript.
  2. The properties found in a $ref object are not ignored. Rather, they are to override the referenced object properties. This allows you to reference a default document, and replace the particular properties as needed. more below
  3. You can reference
    • http URLs
    • files
    • environment variables
    • keyring values
    • AWS SSM parameters
    • AWS S3 files

Quick guide

Load your configuration file:

from mo_json_config import get

config = get("my_config.json")

Schemes

This module can load configuration from a number of sources, and you can access them via URI scheme. Beyond the common file and https schemes, there are

Environment Variables

Use the env scheme for accessing environment variables:

{
    "host": "mail.example.com",
    "username": "ekyle",
    "password": {"$ref": "env://MAIL_PASSWORD"}
}

Keystore Values

The keyring library can be used with the keyring scheme:

{
    "host": "mail.example.com",
    "username": "ekyle",
    "password": {"$ref": "keyring://ekyle@mail.example.com"}
}

The host is in <username>@<server_name> format; invoking keyring.get_password(server_name, username). You may also set the username as a parameter:

{
    "host": "mail.example.com",
    "username": "ekyle",
    "password": {"$ref": "keyring://mail.example.com?username=ekyle"}
}

Be sure to pip install keyring to use keyring

AWS SSM

The ssm scheme can be used to read from the AWS parameter store. Here is an example that will read all parameters that start with "/configuration" and adds them to the global configuration object:

from mo_json_config import get, configuration

configuration |= get("ssm:///configuration")

String Template Expansion

Before going into the minutia of expanding $ref objects, let's look at simpler string template expansion: Any string that contains a reference (of the form {scheme://...}) will have that reference replaced with the string it points to.

Consider an example where we want to name a number of components with a common application name. You may define them by using.

{
    "database_name": "{env://APP_NAME}-database"
    "queue_name": "{env://APP_NAME}-queue"
}

If we assume

export APP_NAME=my-app-name

then the above JSON will expand to

{
    "database_name": "my-app-name-database"
    "queue_name": "my-app-name-queue"
}

These references can be used in the $ref object, as well, providing another level of indirection to the configuration file.

Using references in config files

The $ref property is special. Its value is interpreted as a URL pointing to more JSON

Absolute Internal Reference

The simplest form of URL is an absolute reference to a node in the same document:

{
    "message": "Hello world",
    "repeat": {"$ref": "#message"}
}

The reference must start with #, and the object with the $ref is replaced with the value it points to:

{
    "message": "Hello world",
    "repeat": "Hello world"
}

Relative Internal References

References that start with dot (.) are relative, with each additional dot referring to successive parents. In this case the .. refers to the ref-object's parent, and expands just like the previous example:

{
    "message": "Hello world",
    "repeat": {"$ref": "#..message"}
}

File References

Configuration is often stored on the local file system. You can in-line the JSON found in a file by using the file:// scheme:

It is good practice to store sensitive data in a secure place...

{# LOCATED IN C:\users\kyle\password.json
    "host": "database.example.com",
    "username": "kyle",
    "password": "pass123"
}

...and then refer to it in your configuration file:

{
    "host": "example.com",
    "port": "8080",
    "$ref": "file:///C:/users/kyle/password.json"
}

which will be expanded at run-time to:

{
    "host": "example.com",
    "port": "8080",
    "username": "kyle",
    "password": "pass123"
}

Please notice the triple slash (///) is referring to an absolute file reference.

References To Objects

Ref-objects that point to other objects (dicts) are not replaced completely, but rather are merged with the target; with the ref-object properties taking precedence. This is seen in the example above: The "host" property is not overwritten by the target's.

Relative File Reference

Here is the same, using a relative file reference; which is relative to the file that contains this JSON

{#LOCATED IN C:\users\kyle\dev-debug.json
    "host": "example.com",
    "port": "8080",
    "$ref": "file://password.json"
}

Home Directory Reference

You may also use the tilde (~) to refer to the current user's home directory. Here is the same again, but this example can be anywhere in the file system.

{
    "host": "example.com",
    "port": "8080",
    "$ref": "file://~/password.json"
}

HTTP Reference

Configuration can be stored remotely, especially in the case of larger configurations which are too unwieldy to inline:

{
    "schema":{"$ref": "https://example.com/sources/my_db.json"}
}

Scheme-Relative Reference

You are also able to leave the scheme off, so that whole constellations of configuration files can refer to each other no matter if they are on the local file system, or remote:

{# LOCATED AT SOMEWHERE AT http://example.com
    "schema":{"$ref": "///sources/my_db.json"}
}

And, of course, relative references are also allowed:

{# LOCATED AT http://example.com/sources/dev-debug.json
    "schema":{"$ref": "//sources/my_db.json"}
}

Fragment Reference

Some remote configuration files are quite large...

{# LOCATED IN C:\users\kyle\password.json
    "database":{
        "username": "kyle",
        "password": "pass123"
    },
    "email":{
        "username": "ekyle",
        "password": "pass123"
    }
}

... and you only need one fragment. For this use the hash (#) followed by the dot-delimited path into the document:

{
    "host": "mail.example.com",
    "username": "ekyle",
    "password": {"$ref": "//~/password.json#email.password"}
}

Parameters Reference

You can reference the variables found in $ref URL by using the param scheme. For example, the following JSON document demands that it be provided with a password parameter:

{ # LOCATED AT http://example.com/machine_config.json
    "host": "mail.example.com",
    "username": "ekyle",
    "password": {"$ref": "param:///password"}
}

The param scheme only accepts dot-delimited paths.

The above parametric JSON can be expanded with a $ref

{"config": {
	"$ref": "http://example.com/machine_config.json?password=pass123"
}}

expands to

{"config": {
    "host": "mail.example.com",
    "username": "ekyle",
    "password": "pass123"
}}

URL parameters and $ref properties can conflict. Let's consider

{"config": {
	"$ref": "http://example.com/machine_config.json?password=pass123",
	"password": "123456"
}}

the URL paramters are used to expand the given document, then the $ref properties override the contents of the document:

{"config": {
    "host": "mail.example.com",
    "username": "ekyle",
    "password": "123456"
}}

Comments

JSON parsing is performed using Hjson, as such there are numerous flexibilities in the syntax. The most important is comments:

End-of-line Comments are allowed, using either # or // prefix:

    "key1": "value1",  //Comment 1
    "key1": "value1",  # Comment 1

Multiline comments are also allowed, using either Python's triple-quotes (""" ... """) or Javascript's block quotes /*...*/

{
    "key1": /* Comment 1 */ "value1",
}
    "key1": """Comment 1""" "value1",

Version Changes, Features

Version 5

June 2025 - removed general string replacment using moustaches {{param_name}}. Only {param://param_name} syntax is allowed. This is more consistent with string replacement in general.

Project details


Release history Release notifications | RSS feed

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

mo_json_config-5.703.26061.tar.gz (21.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

mo_json_config-5.703.26061-py3-none-any.whl (19.4 kB view details)

Uploaded Python 3

File details

Details for the file mo_json_config-5.703.26061.tar.gz.

File metadata

  • Download URL: mo_json_config-5.703.26061.tar.gz
  • Upload date:
  • Size: 21.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.1

File hashes

Hashes for mo_json_config-5.703.26061.tar.gz
Algorithm Hash digest
SHA256 e52f4a3dc994871a46f34ff798eab1ca72a42f1032ee7507c507a5156c102b74
MD5 2a5fe72706c31a4e1ff6aec97ce1f1bb
BLAKE2b-256 3d09f0cad6cd32143e6be6812a38716dd6d38fa996019548dd002bdce4a9f704

See more details on using hashes here.

File details

Details for the file mo_json_config-5.703.26061-py3-none-any.whl.

File metadata

File hashes

Hashes for mo_json_config-5.703.26061-py3-none-any.whl
Algorithm Hash digest
SHA256 8e7d92255365f4171b12e9e4d26c5f40f5710041726c2138c5638d0439ae44cc
MD5 09e2d3fdf37cf00933e2a82d4c3e01f2
BLAKE2b-256 72d87a36f8ac06416d370021918e499b89c75106d94cc9fad6a66582925da8ae

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page