Jinja-style templating for LaTeX documents by Curvenote
Project description
Curvenote Template
Curvenote Template is a command line tool (cli) for rendering LaTeX documents from jinja-style templates. This package uses jinja2 as the template engine with a modified enviroment and syntax that plays well with LaTeX's markup.
This allows you to build LaTeX documents driven by content, data and a template. We built this package while developing our template based PDF/LaTeX export system for Curvenote, where it is used to create documents from the templates on the Curvenote Community Template Repo.
Installation
Install the package into your virtual environment using pip:
pip install curvenote-template
and confirm correct installation by typing:
curvenote_template --version
Note: The cli command name uses and underscore in place of a hypen
An example
As we are dealing with content and data, the cli accepts paths to specific files or folders rather than accepting arguments inline. Here is a minimal example:
Given these 3 files:
# data.yml
title: Exploring Outer Space
author:
name: Ana Space
email: ana@outer.space
% content.tex
Outer space is the expanse that exists beyond Earth and between celestial bodies. Outer space is not completely empty—it is a hard vacuum containing a low density of particles, predominantly a plasma of hydrogen and helium, as well as electromagnetic radiation, magnetic fields, neutrinos, dust, and cosmic rays.
% template.tex
\documentclass{article}
\title{[-title-]}
\author{[-author.name-] ([-author.email-])}
\begin{document}
\maketitle
[-CONTENT-]
\vskip 1cm
The End!
\end{document}
We can render a LaTeX document with the following command:
curvenote_template build-lite output.tex data.yml content.tex template.tex
To produce a .tex
file with the following contents:
% output.tex
\documentclass{article}
\title{Exploring Outer Space}
\author{Ana Space (ana@outer.space)}
\begin{document}
\maketitle
% content.tex
Outer space is the expanse that exists beyond Earth and between celestial bodies. Outer space is not completely empty—it is a hard vacuum containing a low density of particles, predominantly a plasma of hydrogen and helium, as well as electromagnetic radiation, magnetic fields, neutrinos, dust, and cosmic rays.
\vskip 1cm
The End!
\end{document}
Which when compiled produces the following document:
The document layout is flexible and will be based on structure provided in the template.tex
file, where the modified jinja syntax ([-
, -]
) is used to expand variables from the matching DocModel provided in data.yml
.
[-CONTENT-]
is a special variable that will expand to the entire contents of content.tex
This example only shows variable expansion ([-myvar-]
) but the full jinja2
environment is available with control flow, filters and many python commands.
The build-lite
command shown above is not opinionated and can be used to render any template with a matching DocModel data structure.
CLI Overview
Get help from the command line tool at any time using the --help
option.
curvenote_template --help
Note: the CLI uses
typer
which provides shell completion option as standard. These are list in help messages and installation is recommended, but these commands are not central to use of the tool.
Command overview
The following commands are available on the cli.
build-lite
build-lite
is an unopinionated rendering command which as in the example above will allow you to render any template given content and a DocModel.
curvenote_template build-lite --help
Usage: curvenote_template build-lite [OPTIONS] TARGET DOCMODEL_FILE
CONTENT_FILE TEMPLATE_FILE
Arguments:
TARGET Name of a local file to write the rendered content to. If
TARGET exists it will be replaced. [required]
DOCMODEL_FILE Path to a YAML file containing the DocModel (a free-form
dict) required to render the template. [required]
CONTENT_FILE Path to a file containing the main content to render
[required]
TEMPLATE_FILE Path to a file with a compatible LaTeX template e.g.
mytemplate.tex.The template should align with the data
structure given by the DocModel [required]
Options:
--bib-file FILE Path to an optional bib file.This will be copied as-
is into the target folder.
--lipsum / --no-lipsum If specified will patch the document with
'\usepackage{lipsum}'.For use in template testing
where `example/content.tex` uses the lipsum package.
[default: no-lipsum]
--help Show this message and exit.
The "DocModel" in this case is a bit of an overstatement as it is just a free-form python dictionary defined in the data.yml
file as shown above.
As you build your template, you can decide on the structure of the data in this file and keep it aligned with the variables you access from within the template. To find out more see Creating Templates
build
build
is an opinionated rendering command intended for use with Curvenote content and templates specifically.
curvenote_template build --help
Usage: curvenote_template build [OPTIONS] TARGET_FOLDER CONTENT_PATH
[TEMPLATE_PATH]
Arguments:
TARGET_FOLDER Local folder in which to construct the Latex assets. If
TARGET exists itand all files will be removed and a new
empty folder structure created [required]
CONTENT_PATH Path to a folder with containing content to render. Folder
shoud contain the following files: - main.tex -
main.bib - data.yaml Along with any additional graphics
assets [required]
[TEMPLATE_PATH] Path to a Curvenote compatible LaTeX template folder. This
is intended for use with local Curvenote templates or in
template development.Omitting this option will use the
built in template.
Options:
--lipsum / --no-lipsum If specified will patch the document with
'\usepackage{lipsum}'.For use in template testing
where `example/content.tex` uses the lipsum
package. [default: no-lipsum]
--strict / --no-strict If true, then missing required tagged content or
options will halt the process. [default: no-
strict]
--no-copy / --no-no-copy If true, then image assets will not be copied into
the target folder. [default: no-no-copy]
--help Show this message and exit.
When exporting LaTeX from Curvenote's API custom environments and commands are included by default. These require certain packages to be loaded and definitions to be included in the final document. build
will include these definition files and expect certain structure to be present in the DocModel
when rendering.
As build
is not generally applicable outside of Curvenote templates, we'll not discuss the details further here. For more information check the Curvenote Open Template Repo.
Note: The Curvenote API can also respond with vanilla LaTeX, but this is not the default case for rendering. For more information on programmatically accessing the Curvenote API, see the Curvenote python client.
validate
validate
is a dry run command which will validate a Curvenote template. This is very simple validation at the moment and we expect this to be extended.
curvenote_template validate --help
Usage: curvenote_template validate [OPTIONS] TEMPLATE_PATH
Arguments:
TEMPLATE_PATH Local folder containing the Curvenote compatible template to
validate [required]
Creating Templates
This cli tool uses a customized jinja2
environment. We explain the custom syntax below and how to use that in conjunction with the content.tex
and data.yml
files in a bit more detail than shown in the previous example.
However, to get the most out of this tool, understanding how jinja2
works and the features it provides will help a lot. The Jinja Template Designer Documentation is a great resource, covering all the features of the jinja2
language in the context of HTML rendering and the standard jinja2
syntax.
That guide along with the information included below for LaTeX rendering and our custom syntax should give you everything you need to know to produce your own templates.
Jinja
About jinja
templates:
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. The template syntax is heavily inspired by Django and Python.
The cli uses a jinja2
environment with the following modifications.
Syntax
customized | standard jinja2 | |
---|---|---|
Statements | [# #] |
{% %} |
Expressions | [- -] |
{{ }} |
Comments | %# #% |
{# #} |
A minimal LaTeX example illustrating these would be:
\documentclass{article}
\begin{document}
\section{Famous People}
%# Print a list of famous people defined in the context dictionary #%
\begin{itemize}
[# for person in famous_people #]
\item [-person.name-], [-person.job-] [# if person.email #]([-person.email-])[# endif #]
[# endfor #]
\end{itemize}
\end{document}
Which will print out a list of famous people's names, jobs and emails, if we have them.
Other environment differences
In addition to the custom syntax we also set the following options:
option | our setting | jinja default | effect |
---|---|---|---|
trim_blocks | True | False | If this is set to True the first newline after a block is removed (block, not variable tag!) |
autoescape | False | True | If set to False the XML/HTML autoescaping feature is disabled |
auto_reload | True | False | Will always check template location for changes and recompiles the template as needed |
undefined | SlientUndefined | None | Ignore any undefined variables in the template, render anyways without affected blocks or variables |
keep_trailing_newline | True | False | Preserve the trailing newline when rendering templates, important in LaTeX |
jinja
provide a whole host of functions, tests and filters at global scope. We have extended further this by adding the python __builtins__
providing additional commonly used python functions within the jinja
rendering context.
Building a DocModel
We use the term DocModel in our documentation to refer to the dictionary of data passed to a jinja
template for rendering, loaded from a data.yml
file. jinja
docs call this the Context Dictionary. It is easy to relate this to the yml
file that you need to create to use the cli.
The fields at the root level of the file are available as variables in the jinja
context at global scope.
# data.yml
title: Outer Space
author: Ana Cosmo
% template.tex
...
\title{[-title-]}
\author{[-author-]}
...
You can add comments to your template, and these will be removed at render time. Just in case, the comments are also valid LaTeX comments so should not affect your build even i one did leak through
% template.tex
% this is a LaTeX comment
%# this is a template comment and will be removed at render, but it is also a valid LaTeX comment #%
These variables themselves can be nested data structures of dictionaries, lists, strings, numbers and booleans
# data.yml
authors:
- name: Ana Cosmo
email: ana@thecosmos.org
- name: Bill Saturn
email: bill@gasgiant.com
tags:
- space
- planets
- cosmos
% template.tex
...
[# for author in authors #]
\author{[-author.name-] ([-author.email-])}
[# endfor #]
The first author is [-authors[0].name-]
The last author is [-authors[-1].name-]
tags: [-tags|join(', ')-]
%# main content goes here #%
[-CONTENT-]
...
[-CONTENT-]
is a special variable that will expand to the entire contents of content.tex
We'll not go into more jinja
features here, as they are covered in the Jinja Template Designer Documentation -- simply replace the standard jinja
syntax with our customized syntax
and bear in mind the rules of LaTeX and constructing sophisicated templates is pretty straightforward with practice.
Curvenote Templates
To look at some of the templates we've developed at Curvenote, or for documentation on how to create template for Curvenote please visit the Open Template Repo and check the documentation there.
We use this cli tool for development, testing and validation of any template submitted to that repo.
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
File details
Details for the file curvenote_template-0.2.5.tar.gz
.
File metadata
- Download URL: curvenote_template-0.2.5.tar.gz
- Upload date:
- Size: 44.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.4.2 importlib_metadata/4.8.1 pkginfo/1.7.1 requests/2.26.0 requests-toolbelt/0.9.1 tqdm/4.62.3 CPython/3.8.8
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | a60bc83a0c5238b932420ade8f496619eb7b01315b8169d7ea47656acfc82927 |
|
MD5 | 3c48d2da2f461296ba41e3c495bc93a6 |
|
BLAKE2b-256 | 78cc5bc532f139995361a1ecfee8248db4814a5f45ad731e51a9e8b0ab917fa4 |
Provenance
File details
Details for the file curvenote_template-0.2.5-py3-none-any.whl
.
File metadata
- Download URL: curvenote_template-0.2.5-py3-none-any.whl
- Upload date:
- Size: 50.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.4.2 importlib_metadata/4.8.1 pkginfo/1.7.1 requests/2.26.0 requests-toolbelt/0.9.1 tqdm/4.62.3 CPython/3.8.8
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 17f2c6fd58acf93f6c965744e0bfb7eb2b27f6414ddeda11dbe509f1f526676e |
|
MD5 | 07fa5312775538486205267f9958247d |
|
BLAKE2b-256 | a9a9cf9bd2fa23ff6c5061b049c4d64caeb4faa944844662dd5585675cae5d49 |