Yatrfile Structure and Features

Suppose you have the following yatrfile.yml in your current working directory:

include:
  - "{{urlbase}}/test/test2.yml"

capture:
  baz: "ls {{glob}}"

macros:
  urlbase: https://raw.githubusercontent.com/mbodenhamer/yatrfiles/master/yatrfiles
  b: bar
  c: "{{b}} baz"
  canard: "false"
  glob: "*.yml"

default: foo

tasks:
  cwd: pwd

  bar:
    - foo
    - "echo {{c}} {{_1|default('xyz')}}"

  verily: "true"

  cond1:
    command: 'echo "{{baz}}"'
    if: "true"

  cond2:
    command: foo
    if: "false"

  cond3:
    command: foo
    ifnot: verily

  cond4:
    command: foo
    ifnot: "{{canard}}"

As this example demonstrates, the primary functionality of a yatrfile is found in five top-level sections: include, capture, macros, tasks, and default. Four other sections, files, settings, import, and declare, are also supported (see files, settings, import, and declare, respectively).

macros

The macros section must be a mapping of macro names to macro definitions. Macro definitions may either be plain strings or Jinja2 templates. Macros that include Jinja2 templates will be rendered according to the values of the macros in terms of which they are defined. For example, in the above macros section, two macros b and c are defined thusly:

b: bar
c: "{{b}} baz"

As such, b resolves to bar and c resolves to bar baz. As the macros section is a mapping, and not a list, there is no inherent order to macro definition. yatr takes care of resolving macros and their dependencies in the right order, provided that there are no cyclic macro definitions (e.g. a macro a defined in terms of b, which is defined in terms of a). If any such cycles exist, the program will exit with an error.

include

The include section must be a list of strings, each of which must be either a filesystem path or a URL specifying the location of another yatrfile. When a yatrfile is “included” in this manner, its macros and tasks are added to the macros and tasks defined by the main yatrfile. Nested includes are supported, following the rule that conflicts in macro or task names are resolved by favoring the definition closest to the main yatrfile.

For example, suppose yatr is invoked on a yatrfile named C.yml, which includes B.yml, which includes A.yml, as follows:

A.yml:

macros:
  a: foo
  b: def
  c: xyz

B.yml:

include:
  - A.yml

macros:
  a: bar
  b: ghi

C.yml:

include:
  - B.yml

macros:
  a: baz

In this case, the macro values would resolve as follows:

$ yatr -f C.yml --dump
a = baz
b = ghi
c = xyz

Name conflicts of tasks from includes are resolved the same way as for macros.

Include paths or URLs may use macros, as the main yatrfile above demonstrates, having an include defined in terms of the urlbase macro. However, any such macros must be defined in the yatrfile itself, and cannot be defined in an included yatrfile or depend on the macros defined in an included yatrfile for their proper resolution.

If an include path is a URL, yatr will attempt to download the file and save it in a cache directory. By default, the cache directory is set to ~/.yatr/, but this may be changed through the --cache-dir option. If the URL file already exists in the cache directory, yatr will load the cached file without downloading. To force yatr to re-download all URL includes specified by the yatrfile, run yatr --pull at the command line.

tasks

Tasks are defined in the tasks section of the yatrfile. Tasks may be defined as a single command string. In this example, the task cwd is simply defined as the system command pwd. If your current working directory happens to be /foo/baz, then:

$ yatr cwd
/foo/baz

Macros are not fully resolved until task runtime. The example yatrfile specifies the inclusion of a file named test2.yml, which defines a task named foo. However, foo is defined in terms of a macro named b, which is not defined in test2.yml. The macro b is defined in the main yatrfile, however, which induces the following behavior:

$ yatr foo
bar

Tasks may also be defined as a list of command strings, to be executed one after the other, as illustrated by bar:

$ yatr bar
bar
bar baz xyz

If the command string is the name of a defined task, then yatr will simply execute that task instead of trying to execute that string as a system command. The bar task will first execute the foo task defined in test2.yml, and then run the echo command.

The bar task also illustrates another feature of yatr: command-line arguments may be passed to tasks for execution. For example:

$ yatr bar foo
bar
bar baz foo

Unless, explicitly re-defined, the macro _1 denotes the first task command-line argument, _2 denotes the second task command-line argument, and so on. Default values may be specified using the Jinja2 default filter, as is illustrated in the definition of bar.

default

The default section, if specified, must contain the name of a task to be run if no task names are provided at the command line. In this example, the default task is set to foo:

default: foo

As such, running yatr at the command line is equivalent to running yatr foo:

$ yatr
bar

If no default task is defined, and if yatr is invoked without any arguments, then yatr will exit after printing usage information.

capture

The capture section defines a special type of macro, specifying a mapping from a macro name to a system command whose captured output is to be the value of the macro. Values of capture mappings cannot contain task references, though they may contain references to other macros. In the main example above, the yatrfile defines a capture macro named baz, whose definition is ls {{glob}}. In the macro section, glob is defined as *.yml. Thus, if yatr is invoked in the example working directory, the value of baz will resolve to A.yml  B.yml  C.yml  D.yml  yatrfile.yml.

files

The files section defines another special type of macro, associating names with filesystem paths. Each associated path must either be a filesystem path or a URL specifying the location of a file. As with the include and import sections, if the path is a URL, the file will be downloaded to the cache directory and the associated name will contain the path of the cached file. If the file already exists in the cache directory, no download will be performed, unless yatr --pull is run. The files section has the same limited support for macros as the include and import sections.

For example, consider the following yatrfile:

files:
  test1: "{{urlbase}}/test/test1.txt"

macros:
  urlbase: https://raw.githubusercontent.com/mbodenhamer/yatrfiles/master/yatrfiles
  tmpdir: "{{_1}}"

tasks:
  foo: 'cp "{{test1}}" "{{tmpdir}}/test1.txt"'
  bar: 'cat "{{tmpdir}}/test1.txt"'

Invoking yatr will cause test1.txt to be downloaded to the cache directory. Running the tasks defined in this yatrfile produces the following behavior:

$ yatr foo /tmp
$ yatr bar /tmp
foo

The first invocation of yatr downloads test1.txt to the cache directory, and copies the file to /tmp. The second invocation dumps the contents of the copied file to stdout.

settings

The top-level section settings allows the global execution behavior of yatr to be modified in various ways. For example, the silent setting, if set to true, will suppress all system command output at the console. Such behavior is disabled by default.

An example of setting setting values in a yatrfile can be found in D.yml, which includes the example yatrfile discussed in Yatrfile Structure and Features:

include:
  - yatrfile.yml

settings:
  silent: true

In the example above, running yatr foo led to the output bar being printed to the console. However, invoking the same task through D.yml will result in no output being printed:

$ yatr -f D.yml foo

However, any setting can be set or overridden at the command line by supplying the -s option:

$ yatr -f D.yml -s silent=false foo
bar

For Boolean-type settings, such as silent, any of the following strings may be used to denote True, regardless of capitalization: yes, true, 1. Likewise, any of the following strings may be used to denote False, regardless of capitalization: no, false, 0.

The following table lists the available settings:

Name Description
exit_on_error If true, halt execution if task command exits with non-zero status; true by default (Boolean string)
loop_count_macro The name of the macro that contains the current loop iteration number (string)
preview_conditionals If true, specially demarcate if and ifnot commands when -p is supplied; true by default (Boolean string)
silent If true, suppress task output; false by default (Boolean string)

import

The import feature enables the functionality of yatr to be extended when necessary, while preserving the simplicity of the default YAML specification for the majority of use cases in which yatr’s default capabilities are sufficient.

The import section must be a list of strings, each of which must be either a filesystem path or a URL specifying the location of a Python module. Strings containing Python module names (such as would be found in a Python import statement) are also supported. Each module so imported must contain a top-level variable named env, which must be an instance of the Env class (see yatr.env module). Modules to be imported in this manner are called “extension modules” (not to be confused with Python extension modules written in C/C++).

The following is an example of a yatr extension module:

from yatr import Env
env = Env()

@env.jinja_filter('foo')
def foo(value, **kwargs):
    return '{}_foo'.format(value)

@env.jinja_function('bar')
def bar(value, **kwargs):
    return '{}_bar'.format(value)

@env.task('barfoo', display=('path', 'baz1'))
def bar_foo(env, *args, **kwargs):
    import os
    print(os.path.join(kwargs['path'], kwargs['baz1']))

This particular extension module defines a custom Jinja2 function (see Custom Jinja2 Functions) and a custom Jinja2 filter (see Custom Jinja2 Filters). A task is also defined in terms of a Python function. Macros can theoretically be defined in an extension module through use of the yatr API, but the straightforward manner of macro declaration facilitated by the standard yatrfile YAML syntax makes the use of include directives a much more efficient and user-friendly alternative.

In this example extension module, a Jinja2 function bar is defined that appends “_bar” to its first argument. Likewise, a Jinja2 filter foo is defined that appends “_foo” to its first argument. Because yatr supplies the current execution environment to custom Jinja 2 filters and functions by way of a keyword argument named env, all such filters and functions defined in extension modules should accept **kwargs as the final argument, even if the kwargs variable is not used within the body of the filter or function itself.

Here is an example yatrfile that uses the extension module defined above:

import:
  - test8.py

macros:
  shebang: "#!/bin/bash"
  bar: "{{env('YATR_BAR', 'baz')}}"
  path: "{{env('PATH')}}"
  baz1: "{{'baz'|foo}}"
  baz2: "{{bar('foo')}}"

tasks:
  foo:
    - echo foo
    - echo bar
    - echo baz

  bar:
    - "echo {{bar}}"

  path:
    - "echo {{path}}"

  baz: "echo {{baz1}} {{baz2}}"

In addition to bar and foo, this yatrfile also makes use of the built-in custom Jinja2 function env (see env()). The task baz is defined in terms of macros that make use of bar and foo. Invoking the task produces the following output:

$ yatr baz
baz_foo foo_bar

In addition to custom Jinja2 functions and filters, extension modules can also be used to define tasks that execute as Python callables. In this example, the extension module defines a function named bar_foo that will be defined in the yatr execution environment as a task named barfoo. Extension tasks have access to all defined macro values through the first parameter, env (see yatr.env module). Moreover, any extension tasks defined using the @env.task decorator will also receive all defined macros through the *args and **kwargs arguments: *args will be populated with any positional argument macros that are defined (i.e., _1, _2, _3, etc.), and **kwargs will be populated with all defined macro values that are not positional argument macros. While these values can also be accessed via env, the *args and **kwargs parameters are notable in that they represent the current execution environment. In cases where macros in different sections are defined with the same name, using **kwargs enables the programmer to access the actual execution value for that name without having to replicate yatr’s macro precedence logic in the extension function.

In this example, suppose the value of the environment variable PATH is set to /foo/bar. In such case, executing the extension task barfoo produces the following output:

$ yatr barfoo
/foo/bar/baz_foo

When executing extension tasks defined via the @env.task decorator, yatr will treat any function that does not raise an exception as exiting with return code 0. Likewise, a function that raises an exception is treated as exiting with return code 1. If extension functions are defined without using the @env.task decorator, the programmer should ensure that the function returns either 0 or 1, as appropriate.

The preview and verbose options (-p and -v) also work with extension functions. By default, yatr will print the function name, along with the full contents of *args and **kwargs. As many more macros may be defined than are used in the extension function, the optional display keyword argument may be provided in the @env.task decorator, allowing the programmer to specify only those keyword arguments to be displayed. In the above example, executing the barfoo extension task with verbose output would produce the following behavior:

$ yatr -v barfoo
bar_foo(baz1=baz_foo, path=/foo/bar)
/foo/bar/baz_foo

declare

The declare section must be a list of strings, each of which must be the name of a macro. The function of this section is best explained by example. Suppose a macro named foo is included as part of the definition of either a macro or a task. If foo is not defined in the yatrfile (or any included yatrfiles), the yatrfile will fail validation (see Commands, particularly –validate). If foo is included in the declare section, however, yatr will effectively ignore the macro and allow the yatrfile to validate.

The declare section is necessary for validating yatrfiles that make use of certain macros that are only defined at runtime. Yatr will automatically handle cases of builtin runtime-defined macros (such as _1), and these do not need to be included in the declare section. However, any runtime-defined macros that are not builtin to yatr will need to be included in the declare section in order for the yatrfile to validate successfully. An example of using the declare section is included in List Macros and For Loops.

Custom Jinja2 Functions

The following functions are defined by default for use in Jinja2 templates.

commands()

The commands function takes a single argument and prints the commands corresponding to the execution of the task whose name is the argument. For example, suppose one is using the example yatrfile of the import section above in order to run a --render command on the following template file (template.j2):

{{shebang}}
{{commands('foo')}}

One could then render the template like so:

yatr -i template.j2 -o template.bash --render

The resulting output file (template.bash) would look like:

#!/bin/bash
echo foo
echo bar
echo baz

env()

The env function takes either one or two arguments. In either case, the first argument must be the name of an environment variable. The env function will return the value of this environment variable if it is defined. If the environment variable is undefined and only one argument is supplied to env, the function will raise an exception and halt execution of the task. On the other hand, if a second argument is supplied to env, it will be returned in the case that the environment variable in question is undefined.

For example, consider the example yatrfile of the import section above. The home macro is defined in terms of the environment variable PATH. In the practically-inconceivable case that PATH is not defined, yatr will exit with an exception when loading this yatrfile. On the other hand, in an environment in which YATR_BAR is not defined, the program will behave as follows:

$ yatr bar
baz
$ YATR_BAR=foo yatr bar
foo

Custom Jinja2 Filters

There are currently no custom Jinja2 filters defined by default for use in Jinja2 templates, but some will probably be added in future releases.

Conditional Task Execution

Tasks may be defined to execute conditionally upon the successful execution of a command, using the keys if and ifnot. If these or other command options are used, the command itself must be explicitly identified by use of the command key. These principles are illustrated in the cond1, cond2, cond3, and cond4 tasks:

$ yatr cond1
A.yml  B.yml  C.yml  D.yml  yatrfile.yml
$ yatr cond2
$ yatr cond3
$ yatr cond4
bar

The values supplied to if and ifnot may be anything that would otherwise constitute a valid task definition. If a value is supplied for if, the command will be executed only if the return code of the test command is zero. Likewise, if a value is supplied for ifnot, the command will be executed only if the return code of the test command is non-zero.

List Macros and For Loops

In most use cases, macros will either be plain strings or Jinja2 templates. However, there are some cases in which it is useful to have a list of strings or macros defined itself as a macro. To define such a “list macro”, simply use YAML list syntax in the macro definition. For example, consider the following yatrfile:

declare:
  - count

macros:
  a: x
  b: 
    - "{{a}}"
    - "y"
  c: 
    - w
    - z

tasks:
  foo:
    command: "echo {{a}} {{u}} {{v}} {{_n}}"
    for:
      var:
        - u
        - v
      in:
        - b
        - c

  bar:
    command: "echo {{x}} {{count}}"
    for:
      var: x
      in: [1, 2, 3, 4]

The macro a is a plain string, but both b and c are list macros. List macros can be used for iteration via for loops, as is illustrated by the definitions of the tasks named foo and bar.

The for key requires two sub-keys, var and in. The var sub-key defines the iteration variable(s), while the in sub-key specifies the lists or list macros over which to iterate. In the case that var is a string value, for specifies a simple and intuitive for loop over the values specified by in. The value of in may either be the name of a list macro, as in the task named foo, or a list literal, as in the task named bar. In the case that var is a list, for specifies a loop over the Cartesian product of the lists specified by in. The task named foo illustrates a 2x2 Cartesian product, while the task named bar illustrates a simple for loop.

It should be noted that the local variables defined by var only exist in the context of the execution of the loop. It should also be noted that the for loop defines a special local variable named _n, which contains the current iteration number. Note that the task named foo is defined in terms of _n. As such:

$ yatr foo
x x w 0
x x z 1
x y w 2
x y z 3

The name of _n may be changed if desired via the loop_count_macro setting. For example:

$ yatr -s loop_count_macro=count bar
1 0
2 1
3 2
4 3

Note that the macro count is included in the declare section (see declare) to allow the yatrfile to validate successfully.

Dictionary Macros

In addition to list macros, yatr also supports the use of dictionary macros. To define such a “dictionary macro”, simply use YAML dictionary syntax in the macro definition. For example, consider the following yatrfile:

macros:
  a: abc
  b:
    a: def
    b: "{{a}}"
  c: "{{b.a}} {{b.b}}"

tasks:
  foo: "echo {{c}}"

The macro b is a dictionary macro that is defined in terms of a. A second string macro, c, is defined in terms of the two items in b. As such:

$ yatr foo
def abc

Calling Tasks with Arguments

Tasks can be called from other tasks by providing the name of a task as the value to the command key. When a task is called in this manner, its macros can also be overridden using the args and kwargs keys. Values in args will override _1, _2, and so on, while values in kwargs will override named macros. These macro overrides only take effect for that specific task call, and do not change macro values globally.

Consider the following example yatrfile:

macros:
  x: 5

tasks:
  x: "echo {{x}} {{_1|default(3)}} {{ARGS[0]|default(10)}}"

  y:
    command: x
    args:
      - 1
      - 2
    kwargs:
      x: 7

  z:
    command: x
    args:
      - 6
    for:
      var: x
      in: [1, 2, 3]

  w:
    - x
    - "y"
    - x

  u:
    - echo foo
    - task:
        command: x
        args:
          - 20
        kwargs:
          x: 30

The task y shows how macro values may be overridden in a task definition:

$ yatr x
5 3 10

$ yatr x 4
5 4 4

$ yatr y 4
7 1 4

In calling x, y overrides _1 and _2 (which is not used by x), but does not affect the builtin macro ARGS (see Builtin Macros). Note that ARGS[0] is equivalent to _1, unless _1 is overridden locally through a task call.

The task z shows that calling tasks in this manner is compatible with for loop functionality:

$ yatr z 4
1 6 4
2 6 4
3 6 4

The task w shows that calling tasks in this manner does not change global macro values:

$ yatr w 4
5 4 4
7 1 4
5 4 4

Tasks can also be defined anonymously within task list definitions using the task keyword, as illustrated by the task u:

$ yatr u
foo
30 20 10

Builtin Macros

The following macros are defined by default:

Name Description
ARGS Command-line task arguments (list of strings)
CURDIR Yatrfile directory path (string)
YATR Invocation of yatr on current yatrfile (string)
YATRFILE Yatrfile path (string)

The use of these macros is illustrated in the following yatrfile (test11.yml):

tasks:
  err: yatr --render -i /foo/bar/baz -o /foo/bar/bazz

  foo: "echo {{_1}} >> {{_2}}"
  bar:
    - foo
    - "yatr -f {{YATRFILE}} foo a {{_2}}"
    - "{{YATR}} foo c {{_2}}"
    - "yatr -f {{CURDIR}}/test11.yml foo d {{ARGS[1]}}"
    - foo

  baz:
    - err
    - foo

For example, suppose bar is invoked in the following manner on an empty file /tmp/foo:

$ yatr bar b /tmp/foo

The file /tmp/foo will now contain the following:

b
a
c
d
b

The other tasks illustrate the use of the exit_on_error setting (see settings). Supposing that neither /foo/bar/baz or /foo/bar/bazz exist on the filesystem, attempting to run baz with default settings will result in an error and foo will not be run. On the other hand, foo will run if baz is invoked like so:

$ yatr -s exit_on_error=false baz a /tmp/baz

If /tmp/baz was an empty file, it will now contain:

a