Jinja 2 Command Line Template Renderer
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:
pip3 install j2tmpl
- 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.