Configuration Templating Language
Project description
Configuration Templating Language
Simple to learn but yet powerful language for templating your configuration files. It is a 'slang' of the web2py's templating language, written from scratch and optimized for textual non-html data.
Features
-
Simple to learn - a person who has some idea of the Python syntax could dive into conftl for 15 min.
-
Powerful - Python code in templating.
-
Command line tool for rendering.
-
Different methods for trigerring rendering from Python code.
-
Suitable for system administration, devops and similar roles.
-
Performance - minimal code base optimized for performance.
-
Platform independent - tested under Linux and Windows, should work on other Unix platforms as well, Python 2.7 compatible, Python 3.x compatible.
Getting Started
Install conftl using pip:
pip install conftl
Alternatively download the source code:
git clone https://github.com/ttt-fifo/conftl
cd conftl
python setup.py install
Hello world from the command line:
$ render -c "name='John Smith'"
Hello, {{=name}}
Hello, John Smith
NOTE: Write Hello, {{=name}}
on stdin, followed by Enter, Ctr+D
Hello world from the Python REPL:
>>>
>>> from conftl import render
>>>
>>> render(content='Hello, {{=name}}', context=dict(name='John Smith'))
'Hello, John Smith'
>>>
Prerequisites
Linux or other Unix distribution or Windows.
Python 2.7 or Python 3.x
Please place an issue in case the current implementation is not working with your platform and I will try to help.
Python Modules: future
Templating Kickstart
- Clear text from the template is printed to the output as is
lorem ipsum dolor sim amet
text clear lorem ipsum
will go exactly the same into the output
lorem ipsum dolor sim amet
text clear lorem ipsum
- Python code in template should be written using tags
{{ ...code... }}
For example if you would like to assign a value 3
to i
, you can do it using the following syntax:
{{i = 3}}
You could also write multiline Python code as well - like the following example:
{{
import sys
def one():
return 1
i = 3
}}
- Printing a variable value to the output is done by tagging it and placing = sign in front of the variable like this
{{=myvar}}
For example if i
has the value of 3
and you put in template:
{{=i}}
you will receive in the output:
3
- Combining a Python code block with clear text and variable outputs - you should not indent the code block as you normally do with Python, but you should determine it with
{{pass}}
special keyword instead.
Whenever you write a code block into the original Python interpreter you indent the code. Lets take the following example of original Python code block:
for i in range(0, 2):
print('X', i)
The equivalent of the above code would be:
{{for i in range(0, 2):}}
X {{=i}}
{{pass}}
-
You are able to pass values to template variables from outside of the template - there are multiple methods to give 'context' to the template, e.g. assigning variable values outside of the template. Look the follow up sections.
-
Advanced templating topics could be found at TEMPLATING.md
Examples
Take a look at the examples folder from the project repository.
Command Line Tool for Rendering (render)
- The render command line tool works as follows:
render -i templatename.tmpl -o filename.conf
will take the template from file templatename.tmpl
and write the output to filename.conf
WARNING: filename.conf will be overwriten!!!
In case input template is not given by -i, you would be expected to place template code on stdin.
NOTE: For Linux and other Unix systems write template code and finish it with Ctr + D
NOTE: For Windows finish template code with with Ctr + Z and then hit ENTER
In case the output filename (-o) is not given, the output will be written to stdout.
- Giving context variables on the command line
You want to give i
value of 4
and use it in your template. Use -c flag:
render -i templatename.tmpl -o filename.conf -c i=4
For assigning values to multiple variables, just repeat -c flag multiple times:
render -i templatename.tmpl -o filename.conf -c i=4 -c j=8 -c x=2
For assigning complex variable datatypes, wrap assignment in double quote like this:
render -i templatename.tmpl -o filename.conf -c "mydict={'a': 1, 'b': 'string'}"
- Context from json file
The json file format should be similar to:
{"myvar": 4,
"otherthing": [1, 3, 5],
"stringsomething": "hello world"
}
You can invoke render by giving the -j option like this:
render -i mytemplate.tmpl -j mycontext.ctx
NOTE: the command line variables have precedence over json file, e.g. if you assign i=2 in json file and i=3 on command line, the final value of i will be i=3.
- Environment in context for convenience
For convenience the ENV dictionary is automatically included in the context and it contains the OS environment variables. The following example prints them on the screen:
render
{{for e in ENV:}}
{{=e}} : {{=ENV[e]}}
{{pass}}
..................................
... environment will come here ...
..................................
NOTE: the ENV is included automatically in context only with the command line tool, rendering from Python (the next section) does not have ENV automatically in context.
How to use ENV in templates?
Many devops / sysadmin systems pass data to their underlying scripts via environment variables. As an example the following shell commands:
$ export SYSTEM=production
$
$ render
{{if ENV['SYSTEM'] == 'production':}}
Listen 80
{{elif ENV['SYSTEM'] == 'ci_cd':}}
Listen 8080
{{elif ENV['SYSTEM'] == 'devel':}}
Listen 8081
{{else:}}
{{raise RuntimeError('wrong SYSTEM')}}
{{pass}}
(Ctr+D at the end)
will give you the output based on the SYSTEM environment variable:
Listen 80
- Command line tool help
See render -h
Rendering Template from Python
There are three interfaces for rendering a template from Python: the function render(...)
, the class Render
and the decorator @template(...)
. Please see the explanation below:
- render(...) function
Consider the following example:
>>>
>>> from conftl import render
>>> render(content='{{=i}}', context=dict(i=8))
'8'
>>>
As you can see, you can give the context= value, which is a dict, containing your variable data.
The signature of the function follows:
render(infile=None,
outfile=None,
context=None,
content=None,
delimiters=None)
You can use the function by giving infile= as argument (this is the template file). If not given, you should give the content= value - this would be a string with the template content.
Output file could be given by outfile= argument. If given, the output will be written to this file. On outfile= absence, the output is returned as string.
In case you need to use other delimiters than the default {{ }}
, you can change the delimiters like this:
>>>
>>> render(content='[[=i]]', context=dict(i=7), delimiters='[[ ]]')
'7'
>>>
- template decorator
Define a function which returns the context as a dict. Decorate your function with template decorator:
from conftl import template
# Define your function, which should output a dict
# with the template context and decorate it with
# template decorator
@template(infile='mytemplate.tmpl', outfile='myconf.conf')
def template_myconf(*arg, **kwarg):
# ...here your complex computations...
i = ....
j = ....
x = '.....'
return dict(i=i, j=j, templ_var=x)
if __name__ == '__main__':
# Here invoke your function and it should create
# the needed myconf.conf
template_myconf(... some args...)
The possible arguments for template decorator are
@template(infile=None,
outfile=None,
content=None,
delimiters=None)
You must give eihter infile= or content= as input. You can omit outfile= and in this case the decorated function will return the output as a string. Changing delimiters= is also possible. The function, decorated with template(...) must return dict, otherwise exception is raised.
This type of context computation is well know by the web2py users, because this is the layout of the web2py controller.
- Render object
An object from Render class could be used in a long running processes. Load the object in memory once and use it multiple times for templating multiple files:
from conftl import Render
rndr = Render()
# ... use it multiple times like this
rndr.instream = open('filename.tmpl', 'r')
rndr.outstream = open('otherfile.conf', 'w')
rndr.context = dict(i=..., j=..., somevar='...')
rndr()
rndr.instream.close()
rndr.outstream.close()
# ....
The instream
and outstream
should be file handles or StringIO objects.
Known Limitations
-
Arbitrary Python code is possible to be executed by the current templating language. I would advice against giving opportunity to the end-users to write template code, unless you know what you are doing. Multiple attack vectors could be used by a malicious end-user who has the possibility to execute arbitrary Python code.
-
In case you want to template a HTML output, you would be better off using the web2py's templating language (called yatl). Yatl has XML escaping switched on by default and also multiple HTML helper functions.
Contributing
Testing implementation on different platforms.
Do not hesitate to fork me on github.
Place issue if you spot issues with this code.
Versioning
See the tags on this repository.
Authors
Todor Todorov - ttt-fifo
License
BSD + other copyright credits
See LICENSE for details.
Acknowledgments
Thanks to Massimo Di Pierro and the web2py team for the inspiration.
See Also
Differences between conftl and yatl from the document differences_yatl.txt
web2py templating language yatl
Another implementation of the same templating language may be found at the weppy project.
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
Built Distribution
Hashes for conftl-0.5.3-py2.py3-none-any.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | becc375209beed78b9d53399f97c36c18cb723320ac1a1f80294e68bfd2c9f93 |
|
MD5 | 4878c22b57fe7dad4dd945e65f7cfdf8 |
|
BLAKE2b-256 | 741e420908576de28f97f9c9b3db2d322ad0c4543d83c55282aefb4adca5d1b0 |