Skip to main content

Jinja2 templating based on environment variables.

Project description

Jinja 2 Command Line Template Renderer

CI

Jinja2 command line renderer that converts environment variables into objects on the context. For example:

DATABASE_MAIN_URI=mysql:3306
DATABASE_MAIN_USERNAME=app
DATABASE_CACHE_URI=redis:6379
DATABASE_CACHE_USERNAME=app

Becomes:

{
    "database": {
        "main": {
            "uri": "mysql:3306",
            "username": "app"
        },
        "cache": {
            "uri": "redis:6379",
            "username": "app"
        }
    }
}

This allows using things like iterations and others for more dynamic templates.

While there are several command line Jinja2 template renderers, they all share a single property that this attempts to solve.

Of those that support environment variables (for Docker, typically), those environment variables are presented to the template verbatum. That is, given this environment:

DATABASE_MAIN_URI=mysql:3306
DATABASE_MAIN_USERNAME=app
DATABASE_CACHE_URI=redis:6379
DATABASE_CACHE_USERNAME=app

A template would have to use them in the following way:

databases: {
    main: {
        uri: {{ DATABASE_MAIN_URI }}
        username: {{ DATABASE_MAIN_USERNAME }}
    cache: {
        uri: {{ DATABASE_CACHE_URI }}
        username {{ DATABASE_CACHE_USERNAME }}

Suppose, though, that you don't necessarily know how many databases your container will have? This construct makes it diffcult in those instances where you have to define N things differently depending on a container deployment, or for base images.

databases: {
    {% for name,definition in databases.items() %}
    {{ name }}: {
        uri: {{ definition.uri }},
        username: {{ definition.username }}
    },
    {% endfor %}

Handling Collisions

Environment variables can sometimes cause interesting problems when building a tree structure. A ValueError will be thrown if a variable is defined twice. Howver, the following is a valid set of environment variables:

AUTH_LDAP=true
AUTH_LDAP_USERNAME=app

In the template context, {{ auth.ldap }} has to be an object as username is a key inside of it. In this case, the value of AUTH_LDAP will be moved down into a special _ key. The two variables would then be:

{{ auth.ldap._ }}=true
{{ auth.ldap.username }}=app

For this to work, almost all _ in environment keys are removed. So, AUTH__LDAP_ is still translated to {{ auth.ldap }}. The only exception is a variable with just underscores. In this case, a '_' is added to the root of the context with that value. Note that all of the following would create this single underscore value:

  • _
  • ___
  • ____________________

If multiple solely-underscore environment variables exist, a ValueError is thrown.

Installation

This can be installed in two ways:

  1. pip3 install j2tmpl
  2. Download the prebuilt binaries.

Note that the prebuilt binaries include the entire Python interpreter so they can be used just as easily as confd.

Usage

This can be used in two ways: processing a single file, or an entire directory.

When a directory path is passed as the template file, j2tmpl will scan the directory for any files with an extension matching template-extensions, an argument that defaults to tmpl,jinja,jinja2,j2,jnj.

Note that it will still output to stdout unless -o is used. If it is, then make sure it's a directory as well. If the target directory doesn't exist, it will be created.

In addition to files matching that pattern, any directory that matches that pattern and also ends with .d will be scanned for template fragments. Template files as well as files in .d directories that end in those extensions will all be concatenated together and rendered into one output file that matches the name of the directory without the template extension and without .d.

For example, given the following directory structure:

foo.conf.jinja.d/
    foo-1.jinja
    foo-2.jinja
bar.conf.jinja.d/
    bar-1.jinja
    bar-2.jinja
foo.conf.jinja
baz.conf.jinja

The output directory will contain the following:

bar.conf
baz.conf
foo.conf

Built-In Filters and extensions

Jinja's do and loopcontrols extensions are enabled by default as are the trim_blocks and lstrip_blocks

Finally, the following additionals filters are avilable:

readfile(str): Read in the contents of the file represented by str. This is particularly useful for container secrets.

boolean(str): Convert the argument into a boolean. A case insensitive comparison to "true", "yes", "on", and "1" will return True. Everything else is false.

b64encode(str): Base 64 encode the value.

b64decode(str): Base 64 decode the value.

Why not confd?

Speaking of confd, why not just use it? While confd is great, it can be a bit too verbose. Having to define a series of config files with all keys enurmated for every single key can be daunting. For certain projects, it probably makes sense, but I would often like to just barf some environment variables into a configuration file with only a tiny amount complexity.

confd is a 5.5mb or so binary and this is still less than 15mb. There are likely ways to make this smaller that I would love to explore.

Project details


Download files

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

Source Distribution

j2tmpl-0.1.1.tar.gz (11.0 kB view details)

Uploaded Source

Built Distribution

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

j2tmpl-0.1.1-py3-none-any.whl (8.6 kB view details)

Uploaded Python 3

File details

Details for the file j2tmpl-0.1.1.tar.gz.

File metadata

  • Download URL: j2tmpl-0.1.1.tar.gz
  • Upload date:
  • Size: 11.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for j2tmpl-0.1.1.tar.gz
Algorithm Hash digest
SHA256 9aa5cfb6ea1dc01bc0cbb1d1df8d42dafde976e0a0c74e08b0fabc34a4cbc5d9
MD5 044a3d6ababa5acd6a0b0df64614ef40
BLAKE2b-256 be387a02244a2ff5251d1c648b2595477fc44da7146e8f4ac0a59bcf7095cf9f

See more details on using hashes here.

Provenance

The following attestation bundles were made for j2tmpl-0.1.1.tar.gz:

Publisher: publish-pypi.yml on ikogan/j2tmpl

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file j2tmpl-0.1.1-py3-none-any.whl.

File metadata

  • Download URL: j2tmpl-0.1.1-py3-none-any.whl
  • Upload date:
  • Size: 8.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for j2tmpl-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 0e2ac59917b2751051b5fa02f73cb8c1ff738c3114715f891cfa8fccdcf50c4b
MD5 61def51ac5142a49d27093d2d3457f25
BLAKE2b-256 b6292c0af2b8417d54498b4f2ae0af6c2484c16278081068caff1bbff0f4119d

See more details on using hashes here.

Provenance

The following attestation bundles were made for j2tmpl-0.1.1-py3-none-any.whl:

Publisher: publish-pypi.yml on ikogan/j2tmpl

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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