This vignette describes the template syntax supported by jinjar, following the structure of the Jinja Template Designer Documentation. It is designed to act as a reference when writing templates.
The jinjar R package is powered by the inja C++ library. The syntax is very similar to that of the Jinja Python package, but there are also many differences. Unfortunately, this means jinjar is not a drop-in replacement for Jinja – you might need to adapt existing Jinja templates for the jinjar engine.
The most fundamental difference between jinjar and Jinja is:
This is described in more detail in the Variables section below.
Before starting, let’s create a few R objects for rendering example templates.
library(jinjar)
# length-1 vector
<- "My Webpage"
title
# vector
<- c("User A", "User B", "User C")
users
# list
<- list(
godzilla Name = "Godzilla",
Born = 1952,
Birthplace = "Japan"
)
# data frame
<- data.frame(
navigation caption = c("Home", "Blog"),
href = c("index.html", "blog.html")
)
# HTML special characters
<- 'Dwayne "The Rock" Johnson' name
A jinjar template is simply a text file, and when rendered the output is also a text file (e.g. HTML, SQL, LaTeX).
A template contains variables and/or expressions, which get replaced with values when a template is rendered; and tags, which control the logic of the template.
Below is a minimal template that illustrates a few basics using the default jinjar configuration. We will cover the details later in this document:
<!DOCTYPE html>
<html lang="en">
<head>
<title>{{ title }}</title>
</head>
<body>
<ul id="navigation">
{% for item in navigation -%}<li><a href="{{ item.href }}">{{ item.caption }}</a></li>
{% endfor -%}</ul>
{# a comment #}</body>
</html>
<!DOCTYPE html>
<html lang="en">
<head>
<title>My Webpage</title>
</head>
<body>
<ul id="navigation">
<li><a href="index.html">Home</a></li>
<li><a href="blog.html">Blog</a></li>
</ul>
</body>
</html>
The following example shows the default configuration settings, but you can adjust the syntax configuration as desired using jinjar_config()
.
There are a few kinds of delimiters. The default delimiters are configured as follows:
{% ... %}
for Statements{{ ... }}
for Expressions to print to the template output{# ... #}
for Comments not included in the template outputLine Statements are also possible, though they don’t have default prefix characters. To use them, set line_statement
when creating the jinjar_config()
.
When writing a template, we refer to variables that act as data placeholders. We define their values when rendering the template.
Although we pass R objects to render()
, it is helpful to understand that these are encoded as JSON objects before the template is rendered.
R object | JSON object | Template example |
---|---|---|
Length-1 vector | Scalar | {{ foo }} |
Vector | Array | {{ foo.1 }} |
List | Object | {{ foo.bar }} |
Data frame | Array of objects | {{ foo.1.bar }} |
You can use dot (.
) notation to access data nested within a variable. An array element is accessed by its numeric index (e.g. foo.1
) and an object value is accessed by its key (e.g. foo.bar
).
Note: In R, the dot is a valid character in an object name (e.g. my.data
). However, this causes ambiguity when accessing nested data values. For this reason, each dot is replaced with an underscore when the data is encoded as JSON (e.g. my.data
becomes my_data
).
The double-brace syntax is used to print the value of the variable (e.g. {{ foo }}
). To use the variable in other contexts (e.g. control structures), then these braces are omitted (e.g. {% for bar in foo %}
).
If a template variable has not been defined, then an error occurs. However, you can use the default(foo, bar)
function to specify a fallback value.
In the default configuration, whitespace (e.g. spaces, tabs, newlines) is left unchanged in the rendered output. For example, in the default configuration we get:
<div>
{% if true %}
yay
{% endif %}</div>
<div>
yay
</div>
By setting trim_blocks = TRUE
when creating the jinjar_config()
, the first newline after a control block is automatically removed. Setting lstrip_blocks = TRUE
removes any whitespace from the beginning of the line until the start of each block. With both options enabled, the above example becomes:
<div>
{% if true %}
yay
{% endif %}</div>
<div>
yay</div>
Instead of changing the global configuration, you can manually trim whitespace at a more finegrained level.
-
) after the opening delimiter, this removes any whitespace from the beginning of the line until the start of the block (i.e. the same as the lstrip_blocks
feature).-
) before the closing delimiter, this removes any whitespace (including newlines) until the next non-whitespace character (i.e. slightly different from the trim_blocks
feature).This can be activated for control blocks, comments, or variable expressions:
<div>
{% if true -%}
yay
{%- endif -%}</div>
<div>
yay</div>
If line statements are enabled (see jinjar_config()
), it’s possible to mark a line as a statement. For example, if the line statement prefix is configured to #
, you can do:
<ul id="navigation">
# for item in navigation<li><a href="{{ item.href }}">{{ item.caption }}</a></li>
# endfor</ul>
<ul id="navigation">
<li><a href="index.html">Home</a></li>
<li><a href="blog.html">Blog</a></li>
</ul>
A control structure refers to all those things that control the flow of a program. With the default syntax, control structures appear inside {% ... %}
blocks.
A for-loop allows you to iterate over each element in a vector:
{% for user in users -%}
{{ loop.index1 }}. {{ user }} {%- endfor -%}
1. User A
2. User B
3. User C
or loop over key-value pairs in a named list:
<dl>
{% for key, value in godzilla %}<dt>{{ key }}</dt>
<dd>{{ value }}</dd>
{% endfor -%}</dl>
<dl>
<dt>Birthplace</dt>
<dd>Japan</dd>
<dt>Born</dt>
<dd>1952</dd>
<dt>Name</dt>
<dd>Godzilla</dd>
</dl>
As described in Variables, a data frame is translated to an array of JSON objects. Therefore a nested combination of the above two loops could theoretically be used. In practice, it is much more common to iterate over rows and access the individual elements by their attributes:
<ul id="navigation">
{% for item in navigation -%}<li><a href="{{ item.href }}">{{ item.caption }}</a></li>
{% endfor -%}</ul>
<ul id="navigation">
<li><a href="index.html">Home</a></li>
<li><a href="blog.html">Blog</a></li>
</ul>
While inside a for-loop block, you can access some special variables:
Variable | Description |
---|---|
loop.index |
The current iteration (0-based). |
loop.index1 |
The current iteration (1-based). |
loop.is_first |
True if first iteration. |
loop.is_last |
True if last iteration. |
loop.parent |
In nested loops, the parent loop variable. |
Conditional branches are written using if
, else if
and else
statements, which evaluate Expressions.
{% if length(users) > 5 -%}
{% for user in users -%}* {{ user }}
{% endfor %}
{% else if length(users) > 0 -%}
{{ join(users, ", ") }}.
{% else -%}
No users found. {% endif %}
User A, User B, User C.
Using the set
statement, you can assign values to variables.
{% set name="world" -%}
Hello {{ name }}!
Hello world!
The extends
tag can be used for template inheritance. See Template Inheritance in vignette("auxiliary-templates")
.
The include
tag inserts the rendered contents of an auxiliary template. See Template Inclusion in vignette("auxiliary-templates")
.
Basic expressions are supported in templates.
The simplest form of expressions are literals, which represent fixed values.
As described in Variables, the template is rendered using data stored in JSON format. For this reason, literals must also be specified in JSON format. The following types of literals are supported:
true
or false
.
null
.Here is example usage for each type:
String: {{ "A string" }}
Integer: {{ 3 }}
Numeric: {{ 3.14 }} or {{ 1.6e-19 }}
Boolean: {{ true }} or {{ false }}
List: {{ [1, 2, 3] }}
Object: {{ {"a": 1, "b": 2} }}
Null: {{ null }}
String: A string
Integer: 3
Numeric: 3.14 or 1.6e-19
Boolean: true or false
List: [1,2,3]
Object: {"a":1,"b":2}
Null:
You can perform simple arithmetic using standard operators:
1 + 1: {{ 1 + 1 }}
3 - 2: {{ 3 - 2 }}
2 * 2: {{ 2 * 2 }}
1 / 2: {{ 1 / 2 }}
2 ^ 3: {{ 2 ^ 3 }}
7 % 3: {{ 7 % 3 }}
1 + 1: 2
3 - 2: 1
2 * 2: 4
1 / 2: 0.5
2 ^ 3: 8
7 % 3: 1
You can perform comparisons:
1 == 1: {{ 1 == 1 }}
1 != 1: {{ 1 != 1 }}
2 > 1: {{ 2 > 1 }}
2 >= 1: {{ 2 >= 1 }}
2 < 1: {{ 2 < 1 }}
2 <= 1: {{ 2 <= 1 }}
1 == 1: true
1 != 1: false
2 > 1: true
2 >= 1: true
2 < 1: false
2 <= 1: false
Within expressions and control structures, you can use the Boolean operators: and
, or
, and not
.
true and false: {{ true and false }}
true or false: {{ true or false }}
not false: {{ not false }}
true and false: false
true or false: true
not false: true
You can also check if a value is contained within a list using in
:
{{ 1 in [1, 2, 3] }}
true
You can check if a value exists by passing the variable name as a string:
users does exist: {{ exists("users") }}
abc doesn't exist: {{ exists("abc") }}
users does exist: true
abc doesn't exist: false
Similarly, you can check if a value exists within a JSON object, by passing the key as a string:
Birthplace does exist: {{ existsIn(godzilla, "Birthplace") }}
Weight doesn't exist: {{ existsIn(godzilla, "Weight") }}
Birthplace does exist: true
Weight doesn't exist: false
Concisely handle missing values using the default()
function:
{{ default(godzilla.Weight, 20000) }}
20000
You can also check the data type of a variable or literal:
{{ isString("a string") }}
{{ isInteger(3) }}
{{ isFloat(3.14) }}
{{ isNumber(3) }} and {{ isNumber(3.14) }}
{{ isBoolean(false) }}
{{ isArray([1, 2, 3]) }}
{{ isObject({"a": 1, "b": 2}) }}
true
true
true
true and true
true
true
true
You can convert strings to numeric types, using the int()
or float()
functions:
{{ int("2") }}
{{ float("2.5") }}
2
2.5
When generating HTML from templates, there’s always a risk that a variable will include characters that affect the resulting HTML. The special characters are: <
, >
, &
and "
.
In jinjar, it’s your responsibility to manually escape variables, using the escape_html()
function. You should escape variables that might contain any of the special characters. But if a variable is trusted to contain well-formed HTML, then it should not be escaped (otherwise you could accidentally double-escape the content).
<input type="text" value="{{ escape_html(name) }}">
<input type="text" value="Dwayne "The Rock" Johnson">
SQL databases expect string literals to be wrapped in single-quotes, while other types of literals (e.g., numbers) are not quoted. This is cumbersome to achieve when writing a template, so the quote_sql()
function provides this functionality.
Important: quote_sql()
does not provide any protection against SQL injection attacks.
WHERE title = {{ quote_sql(title) }} AND year = {{ quote_sql(godzilla.Born) }}
WHERE title = 'My Webpage' AND year = 1952
When passed an array, quote_sql()
will quote each element and return a comma-separated list. This is particularly helpful when using the SQL IN
operator.
WHERE user IN ({{ quote_sql(users) }})
WHERE user IN ('User A', 'User B', 'User C')
You can check if an integer is even or odd, or divisible by some other integer. This could be used to make alternating row colors.
{{ even(42) }}
{{ odd(42) }}
{{ divisibleBy(42, 7) }}
true
false
true
You can round floating point numbers to a specific precision:
{{ round(3.1415, 0) }}
{{ round(3.1415, 3) }}
3
3.142
Translate a string to lower case or upper case:
{{ lower("Hello") }}
{{ upper("Hello") }}
hello
HELLO
Escape special characters for use in HTML content (see HTML Escaping):
<input type="text" value="{{ escape_html(name) }}">
<input type="text" value="Dwayne "The Rock" Johnson">
Quote string data for use as string literals in a SQL query (see SQL Quoting):
WHERE user IN ({{ quote_sql(users) }})
WHERE user IN ('User A', 'User B', 'User C')
Get the number of list elements:
length(): {{ length([3,1,2]) }}
length(): 3
Get the first or last elements:
first(): {{ first([3,1,2]) }}
last(): {{ last([3,1,2]) }}
first(): 3
last(): 2
Get the minimum or maximum elements:
min(): {{ min([3,1,2]) }}
max(): {{ max([3,1,2]) }}
min(): 1
max(): 3
Sort the list into ascending order:
sort(): {{ sort([3,1,2]) }}
sort(): [1,2,3]
Join a list with a separator:
{{ join([1,2,3], " + ") }}
{{ join(users, ", ") }}
1 + 2 + 3
User A, User B, User C
Generate a list as a range of integers:
{% for i in range(4) %}{{ loop.index1 }}{% endfor %}
1234
Access elements using a dynamic index with at()
. Note that the index is zero-based.
{% set x = [1,2,3] -%}
{% set i = 2 -%}
{{ x.2 }}
{{ at(x, i) }}
3
3
Access values using a dynamic key with at()
:
{% set x = {"a": 1, "b": 2} -%}
{% set key = "b" -%}
{{ x.b }}
{{ at(x, key) }}
2
2
Comments
To comment-out some lines, preventing them from appearing in the rendered document, use the comment syntax (default:
{# ... #}
). This is useful for debugging or documenting the template.