Metadata-Version: 2.1
Name: yte
Version: 1.3.0
Summary: A YAML template engine with Python expressions
Home-page: https://github.com/koesterlab/yte
License: MIT
Author: Johannes Köster
Author-email: johannes.koester@tu-dortmund.de
Requires-Python: >=3.7
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Requires-Dist: dpath (>=2.0,<3.0)
Requires-Dist: plac (>=1.3.4,<2.0.0)
Requires-Dist: pyyaml (>=6.0,<7.0)
Project-URL: Repository, https://github.com/koesterlab/yte
Description-Content-Type: text/markdown

# YTE - A YAML template engine with Python expressions

[![test coverage: 100%](https://img.shields.io/badge/test%20coverage-100%25-green)](https://github.com/koesterlab/yte/blob/main/pyproject.toml#L30)
![GitHub Workflow Status](https://img.shields.io/github/workflow/status/koesterlab/yte/CI)
![PyPI](https://img.shields.io/pypi/v/yte)
[![Conda Recipe](https://img.shields.io/badge/recipe-yte-green.svg)](https://anaconda.org/conda-forge/yte)
[![Conda Downloads](https://img.shields.io/conda/dn/conda-forge/yte.svg)](https://anaconda.org/conda-forge/yte)
[![Conda Version](https://img.shields.io/conda/vn/conda-forge/yte.svg)](https://github.com/conda-forge/yte-feedstock)


YTE is a template engine for YAML format that utilizes the YAML structure in combination with Python expressions for enabling to dynamically build YAML documents.

## Syntax

The key idea of YTE is to rely on the YAML structure to enable conditionals, loops and other arbitrary Python expressions to dynamically render YAML files.
Python expressions are thereby declared by prepending them with a `?` anywhere in the YAML.
Any such value will be automatically evaluated by YTE, yielding plain YAML as a result.
Importantly, YTE templates are still valid YAML files (for YAML, the `?` expressions are just strings).

### Examples

#### Conditionals

##### Template

```yaml
?if True:
  foo: 1
?elif False:
  bar: 2
?else:
  bar: 1
```

##### Rendered

```yaml
foo: 1
```

##### Template

```yaml
?if True:
  - a
  - b
```

##### Rendered

```yaml
- a
- b
```

##### Template

```yaml
- foo
- bar
- ?if True:
    baz
  ?else:
    bar
```

##### Rendered


```yaml
- foo
- bar
- baz
```


#### Loops

##### Template

```yaml
?for i in range(2):
  '?f"key:{i}"': 1  # When expressions in keys or values contain colons, they need to be additionally quoted.
  ?if i == 1:
      foo: true
```

##### Rendered

```yaml
"key:0": 1
"key:1": 1
foo: true
```

#### Accessing already rendered document parts

A globally available object `doc` (a wrapper around a Python dict)
enables to access parts of the document that have already been rendered above.
This way, one can often avoid variable definitions (see below).
In addition to normal dict access, the object allows to search (`doc.dpath_search`) and access (`doc.dpath_get`) its contents via [dpath](https://github.com/dpath-maintainers/dpath-python) queries.
Simple dpath get queries can also be performed by putting the dpath query directly into the square bracket operator of the `doc` object (the logic in that case is as follows: first, the given value is tried as plain key, if that fails, `doc.dpath_get` is tried as a fallback); see example below.

##### Template

```yaml
foo: 1
bar:
  a: 2
  # dict access
  b: ?doc["foo"] + doc["bar"]["a"]
  # implicit simple dpath get query
  c: ?doc["bar/a"]
  # explicit dpath queries
  d: ?doc.dpath_get("foo") + doc.dpath_get("bar/a")
```

##### Rendered

```yaml
foo: 1
bar:
  a: 2
  b: 3
  c: 2
  c: 3
```

#### Variable definitions

##### Template

```yaml
# The special keyword __variables__ allows to define variables that can be reused below.
# It can be used anywhere in the YAML, also repeatedly and inside of ifs or loops
# with the restriction of not having duplicate __variables__ keys on the same level.

# The usage of __variables__ can be disabled via the API.

__variables__:
  first: foo
  second: 1.5
  # apart from constant values as defined above, also Python expressions are allowed:
  third: ?2 * 3 


a: ?first
b: ?second
c:
  ?for x in range(3):
    __variables__:
      y: ?x * 2
    ?if True:  
      - ?y
```

##### Rendered

```yaml
a: foo
b: 1.5
c:
- 0
- 2
- 4
```

#### Arbitrary definitions

##### Template

```yaml
  # The special keyword __definitions__ allows to define custom statements.
  # It can be used anywhere in the YAML, also repeatedly and inside of ifs or loops
  # with the restriction of not having duplicate __definitions__ keys on the same level.

  # The usage of __definitions__ can be disabled via the API.

  __definitions__:
    - from itertools import product
    - someval = 2
    - |
      def squared(value):
          return value ** 2

  ?for item in product([1, 2], ["a", "b"]):
    - ?f"{item}"

  ?if True:
    - ?squared(2) * someval
    - someval: ?someval
```

##### Rendered

```yaml
- 1-a
- 1-b
- 2-a
- 2-b
- 4
- someval: 2

## Usage

### Command line interface

YTE comes with a command line interface.
To render any YTE template, just issue

```bash
yte < the-template.yaml > the-rendered-version.yaml
```

### Python API

Alternatively, you can invoke YTE via its Python API:

```python
from yte import process_yaml

# set some variables as a Python dictionary
variables = ...

# render a string and obtain the result as a Python dict
result = process_yaml("""
?for i in range(10):
  - ?f"item-{i}"
""", variables=variables)

# render a file and obtain the result as a Python dict
with open("the-template.yaml", "r") as template:
    result = process_yaml(template, variables=variables)

# render a file and write the result as valid YAML
with open("the-template.yaml", "r") as template, open("the-rendered-version.yaml", "w") as outfile:
    result = process_yaml(template, outfile=outfile, variables=variables)


# render a file while disabling the __definitions__ feature
with open("the-template.yaml", "r") as template:
    result = process_yaml(template, variables=variables, disable_features=["definitions"])

# render a file while disabling the __variables__ feature
with open("the-template.yaml", "r") as template:
    result = process_yaml(template, variables=variables, disable_features=["variables"])

# render a file while disabling the __variables__ and __definitions__ feature
with open("the-template.yaml", "r") as template:
    result = process_yaml(template, variables=variables, disable_features=["variables", "definitions"])
```

## Comparison with other engines

Lots of template engines are available, for example the famous generic [jinja2](https://jinja.palletsprojects.com).
The reasons to generate a YAML specific engine are

1. The YAML syntax can be exploited to simplify template expression syntax, and make it feel less foreign (i.e. fewer special characters for control flow needed) while increasing human readability.
2. Whitespace handling (which is important with YAML since it has a semantic there) becomes unnecessary (e.g. with jinja2, some [tuning](https://radeksprta.eu/posts/control-whitespace-in-ansible-templates) is required to obtain proper YAML rendering).

Of course, YTE is not the first YAML specific template engine.
Others include

* [Yglu](https://yglu.io)
* [Emrichen](https://github.com/con2/emrichen)

The main difference between YTE and these two is that YTE extends YAML with plain Python syntax instead of introducing another specialized language.
Of course, the choice is also a matter of taste.

