- Workflow Description Language
- Language Specification
- Namespaces
- Scope
- Optional Parameters & Type Constraints
- Scatter / Gather
- Variable Resolution
- Computing Inputs
- Type Coercion
- Standard Library
- File stdout()
- File stderr()
- Array[String] read_lines(String|File)
- Array[Array[String]] read_tsv(String|File)
- Map[String, String] read_map(String|File)
- Object read_object(String|File)
- Array[Object] read_objects(String|File)
- mixed read_json(String|File)
- Int read_int(String|File)
- String read_string(String|File)
- Float read_float(String|File)
- Boolean read_boolean(String|File)
- File write_lines(Array[String])
- File write_tsv(Array[Array[String]])
- File write_map(Map[String, String])
- File write_object(Object)
- File write_objects(Array[Object])
- File write_json(mixed)
- Float size(File, [String])
- String sub(String, String, String)
- Array[Int] range(Int)
- Array[Array[X]] transpose(Array[Array[X]])
- Array[Pair(X,Y)] zip(Array[X], Array[Y])
- Array[Pair(X,Y)] cross(Array[X], Array[Y])
- Integer length(Array[X])
- Array[String] prefix(String, Array[X])
- X select_first(Array[X?])
- Array[X] select_all(Array[X?])
- Boolean defined(X?)
- String basename(String)
- Int floor(Float), Int ceil(Float) and Int round(Float)
- Data Types & Serialization
- Serialization of Task Inputs
- De-serialization of Task Outputs
WDL is meant to be a human readable and writable way to express tasks and workflows. The "Hello World" tool in WDL would look like this:
task hello {
String pattern
File in
command {
egrep '${pattern}' '${in}'
}
runtime {
docker: "broadinstitute/my_image"
}
output {
Array[String] matches = read_lines(stdout())
}
}
workflow wf {
call hello
}
This describes a task, called 'hello', which has two parameters (String pattern
and File in
). A task
definition is a way of encapsulating a UNIX command and environment and presenting them as functions. Tasks have both inputs and outputs. Inputs are declared as declarations at the top of the task
definition, while outputs are defined in the output
section.
The user must provide a value for these two parameters in order for this task to be runnable. Implementations of WDL should accept their inputs as JSON format. For example, the above task needs values for two parameters: String pattern
and File in
:
Variable | Value |
---|---|
wf.hello.pattern | ^[a-z]+$ |
wf.hello.in | /file.txt |
Or, in JSON format:
{
"wf.hello.pattern": "^[a-z]+$",
"wf.hello.in": "/file.txt"
}
Running the wf
workflow with these parameters would yield a command line from the call hello
:
egrep '^[a-z]+$' '/file.txt'
A simple workflow that runs this task in parallel would look like this:
workflow example {
Array[File] files
scatter(path in files) {
call hello {input: in=path}
}
}
The inputs to this workflow would be example.files
and example.hello.pattern
.
17 August 2015
- Added concept of fully-qualified-name as well as namespace identifier.
- Changed task definitions to have all inputs as declarations.
- Changed command parameters (
${
...}
) to accept expressions and fewer "declarative" elements- command parameters also are required to evaluate to primitive types
- Added a
output
section to workflows - Added a lot of functions to the standard library for serializing/deserializing WDL values
- Specified scope, namespace, and variable resolution semantics
These are common among many of the following sections
$ws = (0x20 | 0x9 | 0xD | 0xA)+
$identifier = [a-zA-Z][a-zA-Z0-9_]+
$string = "([^\\\"\n]|\\[\\"\'nrbtfav\?]|\\[0-7]{1,3}|\\x[0-9a-fA-F]+|\\[uU]([0-9a-fA-F]{4})([0-9a-fA-F]{4})?)*"
$string = '([^\\\'\n]|\\[\\"\'nrbtfav\?]|\\[0-7]{1,3}|\\x[0-9a-fA-F]+|\\[uU]([0-9a-fA-F]{4})([0-9a-fA-F]{4})?)*'
$boolean = 'true' | 'false'
$integer = [1-9][0-9]*|0[xX][0-9a-fA-F]+|0[0-7]*
$float = (([0-9]+)?\.([0-9]+)|[0-9]+\.|[0-9]+)([eE][-+]?[0-9]+)?
$string
can accept the following between single or double-quotes:
- Any character not in set:
\\
,"
(or'
for single-quoted string),\n
- An escape sequence starting with
\\
, followed by one of the following characters:\\
,"
,'
,[nrbtfav]
,?
- An escape sequence starting with
\\
, followed by 1 to 3 digits of value 0 through 7 inclusive. This specifies an octal escape code. - An escape sequence starting with
\\x
, followed by hexadecimal characters0-9a-fA-F
. This specifies a hexidecimal escape code. - An escape sequence starting with
\\u
or\\U
followed by either 4 or 8 hexadecimal characters0-9a-fA-F
. This specifies a unicode code point
All inputs and outputs must be typed. The following primitive types exist in WDL:
Int i = 0 # An integer value
Float f = 27.3 # A floating point number
Boolean b = true # A boolean true/false
String s = "hello, world" # A string value
File f = "/path/to/file" # A file
In addition, the following compound types can be constructed, parameterized by other types. In the examples below P
represents any of the primitive types above, and X
and Y
represent any valid type (even nested compound types):
Array[X] xs = [x1, x2, x3] # An array of Xs
Map[P,Y] p_to_y = { p1: y1, p2: y2, p3: y3 } # A map from Ps to Ys
Pair[X,Y] x_and_y = (x, y) # A pair of one X and one Y
Object o = { "field1": f1, "field2": f2 } # Object keys are always `String`s
Some examples of types:
File
Array[File]
Pair[Int, Array[String]]
Map[String, String]
Types can also have a postfix quantifier (either ?
or +
):
?
means that the value is optional. It can only be used in calls or functions that accept optional values.+
can only be applied toArray
types, and it signifies that the array is required to be non-empty.
For more details on the postfix quantifiers, see the section on Optional Parameters & Type Constraints
For more information on type and how they are used to construct commands and define outputs of tasks, see the Data Types & Serialization section.
$fully_qualified_name = $identifier ('.' $identifier)*
$namespaced_identifier = $identifier ('.' $identifier)*
A fully qualified name is the unique identifier of any particular call
or call input or output. For example:
other.wdl
task foobar {
File in
command {
sh setup.sh ${in}
}
output {
File results = stdout()
}
}
main.wdl
import "other.wdl" as other
task test {
String my_var
command {
./script ${my_var}
}
output {
File results = stdout()
}
}
workflow wf {
Array[String] arr = ["a", "b", "c"]
call test
call test as test2
call other.foobar
output {
test.results
foobar.results
}
scatter(x in arr) {
call test as scattered_test {
input: my_var=x
}
}
}
The following fully-qualified names would exist within workflow wf
in main.wdl:
wf
- References top-level workflowwf.test
- References the first call to tasktest
wf.test2
- References the second call to tasktest
(aliased as test2)wf.test.my_var
- References theString
input of first call to tasktest
wf.test.results
- References theFile
output of first call to tasktest
wf.test2.my_var
- References theString
input of second call to tasktest
wf.test2.results
- References theFile
output of second call to tasktest
wf.foobar.results
- References theFile
output of the call toother.foobar
wf.foobar.input
- References theFile
input of the call toother.foobar
wf.arr
- References theArray[String]
declaration on the workflowwf.scattered_test
- References the scattered version ofcall test
wf.scattered_test.my_var
- References anArray[String]
for each element used asmy_var
when running the scattered version ofcall test
.wf.scattered_test.results
- References anArray[File]
which are the accumulated results from scatteringcall test
wf.scattered_test.1.results
- References anFile
from the second invocation (0-indexed) ofcall test
within the scatter block. This particular invocation used value "b" formy_var
A namespaced identifier has the same syntax as a fully-qualified name. It is interpreted as the left-hand side being the name of a namespace and then the right-hand side being the name of a workflow, task, or namespace within that namespace. Consider this workflow:
import "other.wdl" as ns
workflow wf {
call ns.ns2.task
}
Here, ns.ns2.task
is a namespace identifier (see the Call Statement section for more details). Namespace identifiers, like fully-qualified names are left-associative, which means ns.ns2.task
is interpreted as ((ns.ns2).task)
, which means ns.ns2
would have to resolve to a namespace so that .task
could be applied. If ns2
was a task definition within ns
, then this namespaced identifier would be invalid.
$declaration = $type $identifier ('=' $expression)?
Declarations are declared at the top of any scope.
In a task definition, declarations are interpreted as inputs to the task that are not part of the command line itself.
If a declaration does not have an initialization, then the value is expected to be provided by the user before the workflow or task is run.
Some examples of declarations:
File x
String y = "abc"
Float pi = 3 + .14
Map[String, String] m
A declaration may also refer to elements that are outputs of tasks. For example:
task test {
String var
command {
./script ${var}
}
output {
String value = read_string(stdout())
}
}
task test2 {
Array[String] array
command {
./script ${write_lines(array)}
}
output {
Int value = read_int(stdout())
}
}
workflow wf {
call test as x {input: var="x"}
call test as y {input: var="y"}
Array[String] strs = [x.value, y.value]
call test2 as z {input: array=strs}
}
strs
in this case would not be defined until both call test as x
and call test as y
have successfully completed. Before that's the case, strs
is undefined. If any of the two tasks fail, then evaluation of strs
should return an error to indicate that the call test2 as z
operation should be skipped.
$expression = '(' $expression ')'
$expression = $expression '.' $expression
$expression = $expression '[' $expression ']'
$expression = $expression '(' ($expression (',' $expression)*)? ')'
$expression = '!' $expression
$expression = '+' $expression
$expression = '-' $expression
$expression = if $expression then $expression else $expression
$expression = $expression '*' $expression
$expression = $expression '%' $expression
$expression = $expression '/' $expression
$expression = $expression '+' $expression
$expression = $expression '-' $expression
$expression = $expression '<' $expression
$expression = $expression '=<' $expression
$expression = $expression '>' $expression
$expression = $expression '>=' $expression
$expression = $expression '==' $expression
$expression = $expression '!=' $expression
$expression = $expression '&&' $expression
$expression = $expression '||' $expression
$expression = '{' ($expression ':' $expression)* '}'
$expression = '[' $expression* ']'
$expression = $string | $integer | $float | $boolean | $identifier
Below are the valid results for operators on types. Any combination not in the list will result in an error.
LHS Type | Operators | RHS Type | Result | Semantics |
---|---|---|---|---|
Boolean |
== |
Boolean |
Boolean |
|
Boolean |
!= |
Boolean |
Boolean |
|
Boolean |
> |
Boolean |
Boolean |
|
Boolean |
>= |
Boolean |
Boolean |
|
Boolean |
< |
Boolean |
Boolean |
|
Boolean |
<= |
Boolean |
Boolean |
|
Boolean |
` | ` | Boolean |
|
Boolean |
&& |
Boolean |
Boolean |
|
File |
+ |
File |
File |
Append file paths |
File |
== |
File |
Boolean |
|
File |
!= |
File |
Boolean |
|
File |
+ |
String |
File |
|
File |
== |
String |
Boolean |
|
File |
!= |
String |
Boolean |
|
Float |
+ |
Float |
Float |
|
Float |
- |
Float |
Float |
|
Float |
* |
Float |
Float |
|
Float |
/ |
Float |
Float |
|
Float |
% |
Float |
Float |
|
Float |
== |
Float |
Boolean |
|
Float |
!= |
Float |
Boolean |
|
Float |
> |
Float |
Boolean |
|
Float |
>= |
Float |
Boolean |
|
Float |
< |
Float |
Boolean |
|
Float |
<= |
Float |
Boolean |
|
Float |
+ |
Int |
Float |
|
Float |
- |
Int |
Float |
|
Float |
* |
Int |
Float |
|
Float |
/ |
Int |
Float |
|
Float |
% |
Int |
Float |
|
Float |
== |
Int |
Boolean |
|
Float |
!= |
Int |
Boolean |
|
Float |
> |
Int |
Boolean |
|
Float |
>= |
Int |
Boolean |
|
Float |
< |
Int |
Boolean |
|
Float |
<= |
Int |
Boolean |
|
Float |
+ |
String |
String |
|
Int |
+ |
Float |
Float |
|
Int |
- |
Float |
Float |
|
Int |
* |
Float |
Float |
|
Int |
/ |
Float |
Float |
|
Int |
% |
Float |
Float |
|
Int |
== |
Float |
Boolean |
|
Int |
!= |
Float |
Boolean |
|
Int |
> |
Float |
Boolean |
|
Int |
>= |
Float |
Boolean |
|
Int |
< |
Float |
Boolean |
|
Int |
<= |
Float |
Boolean |
|
Int |
+ |
Int |
Int |
|
Int |
- |
Int |
Int |
|
Int |
* |
Int |
Int |
|
Int |
/ |
Int |
Int |
Integer division |
Int |
% |
Int |
Int |
Integer division, return remainder |
Int |
== |
Int |
Boolean |
|
Int |
!= |
Int |
Boolean |
|
Int |
> |
Int |
Boolean |
|
Int |
>= |
Int |
Boolean |
|
Int |
< |
Int |
Boolean |
|
Int |
<= |
Int |
Boolean |
|
Int |
+ |
String |
String |
|
String |
+ |
Float |
String |
|
String |
+ |
Int |
String |
|
String |
+ |
String |
String |
|
String |
== |
String |
Boolean |
|
String |
!= |
String |
Boolean |
|
String |
> |
String |
Boolean |
|
String |
>= |
String |
Boolean |
|
String |
< |
String |
Boolean |
|
String |
<= |
String |
Boolean |
|
- |
Float |
Float |
||
+ |
Float |
Float |
||
- |
Int |
Int |
||
+ |
Int |
Int |
||
! |
Boolean |
Boolean |
This is an operator that takes three arguments, a condition expression, an if-true expression and an if-false expression. The condition is always evaluated. If the condition is true then the if-true value is evaluated and returned. If the condition is false, the if-false expression is evaluated and returned. The return type of the if-then-else should be the same, regardless of which side is evaluated or runtime problems might occur.
Examples:
- Choose whether to say "good morning" or "good afternoon":
Boolean morning = ...
String greeting = "good " + if morning then "morning" else "afternoon"
- Choose how much memory to use for a task:
Int array_length = length(array)
runtime {
memory: if array_length > 100 then "16GB" else "8GB"
}
Precedence | Operator type | Associativity | Example |
---|---|---|---|
12 | Grouping | n/a | (x) |
11 | Member Access | left-to-right | x.y |
10 | Index | left-to-right | x[y] |
9 | Function Call | left-to-right | x(y,z,...) |
8 | Logical NOT | right-to-left | !x |
Unary Plus | right-to-left | +x | |
Unary Negation | right-to-left | -x | |
7 | Multiplication | left-to-right | x*y |
Division | left-to-right | x/y | |
Remainder | left-to-right | x%y | |
6 | Addition | left-to-right | x+y |
Subtraction | left-to-right | x-y | |
5 | Less Than | left-to-right | x<y |
Less Than Or Equal | left-to-right | x<=y | |
Greater Than | left-to-right | x>y | |
Greater Than Or Equal | left-to-right | x>=y | |
4 | Equality | left-to-right | x==y |
Inequality | left-to-right | x!=y | |
3 | Logical AND | left-to-right | x&&y |
2 | Logical OR | left-to-right | x||y |
1 | Assignment | right-to-left | x=y |
The syntax x.y
refers to member access. x
must be an object or task in a workflow. A Task can be thought of as an object where the attributes are the outputs of the task.
workflow wf {
Object obj
Object foo
# This would cause a syntax error,
# because foo is defined twice in the same namespace.
call foo {
input: var=obj.attr # Object attribute
}
call foo as foo2 {
input: var=foo.out # Task output
}
}
The syntax x[y]
is for indexing maps and arrays. If x
is an array, then y
must evaluate to an integer. If x
is a map, then y
must evaluate to a key in that map.
Given a Pair x
, the left and right elements of that type can be accessed using the syntax x.left
and x.right
.
Function calls, in the form of func(p1, p2, p3, ...)
, are either standard library functions or engine-defined functions.
In this current iteration of the spec, users cannot define their own functions.
Arrays values can be specified using Python-like syntax, as follows:
Array[String] a = ["a", "b", "c"]
Array[Int] b = [0,1,2]
Maps values can be specified using a similar Python-like sytntax:
Map[Int, Int] = {1: 10, 2: 11}
Map[String, Int] = {"a": 1, "b": 2}
Pair values can be specified inside of a WDL using another Python-like syntax, as follows:
Pair[Int, String] twenty_threes = (23, "twenty-three")
Pair values can also be specified within the workflow inputs JSON with a Left
and Right
value specified using JSON style syntax. For example, given a workflow wf_hello
and workflow-level variable twenty_threes
, it could be declared in the workflow inputs JSON as follows:
{
"wf_hello.twenty_threes": { "Left": 23, "Right": "twenty-three" }
}
$document = ($import | $task | $workflow)+
$document
is the root of the parse tree and it consists of one or more import statement, task, or workflow definition
A WDL file may contain import statements to include WDL code from other sources
$import = 'import' $ws+ $string ($ws+ 'as' $ws+ $identifier)?
The import statement specifies that $string
which is to be interpted as a URI which points to a WDL file. The engine is responsible for resolving the URI and downloading the contents. The contents of the document in each URI must be WDL source code.
Every imported WDL file requires a namespace which can be specified using an identifier (via the as $identifier
syntax). If you do not explicitly specify a namespace identifier then the default namespace is the filename of the imported WDL, minus the .wdl extension.
For all imported WDL files, the tasks and workflows imported from that file will only be accessible through that assigned namespace.
import "http://example.com/lib/analysis_tasks" as analysis
import "http://example.com/lib/stdlib"
workflow wf {
File bam_file
# file_size is from "http://example.com/lib/stdlib"
call stdlib.file_size {
input: file=bam_file
}
call analysis.my_analysis_task {
input: size=file_size.bytes, file=bam_file
}
}
Engines should at the very least support the following protocols for import URIs:
http://
andhttps://
file://
- no protocol (which should be interpreted as
file://
A task is a declarative construct with a focus on constructing a command from a template. The command specification is interpreted in an engine specific way, though a typical case is that a command is a UNIX command line which would be run in a Docker image.
Tasks also define their outputs, which is essential for building dependencies between tasks. Any other data specified in the task definition (e.g. runtime information and meta-data) is optional.
$task = 'task' $ws+ $identifier $ws* '{' $ws* $declaration* $task_sections $ws* '}'
For example, task name { ... }
. Inside the curly braces defines the sections.
The task has one or more sections:
$task_sections = ($command | $runtime | $task_output | $parameter_meta | $meta)+
Additional requirement: Exactly one
$command
section needs to be defined, preferably as the first section.
$command = 'command' $ws* '{' (0xA | 0xD)* $command_part+ $ws+ '}'
$command = 'command' $ws* '<<<' (0xA | 0xD)* $command_part+ $ws+ '>>>'
A command is a task section that starts with the keyword 'command', and is enclosed in curly braces or <<<
>>>
. The body of the command specifies the literal command line to run with placeholders ($command_part_var
) for the parts of the command line that needs to be filled in.
$command_part = $command_part_string | $command_part_var
$command_part_string = ^'${'+
$command_part_var = '${' $var_option* $expression '}'
The parser should read characters from the command line until it reaches a ${
character sequence. This is interpreted as a literal string ($command_part_string
).
The parser should interpret any variable enclosed in ${
...}
as a $command_part_var
.
The $expression
usually references declarations at the task level. For example:
task test {
String flags
command {
ps ${flags}
}
}
In this case flags
within the ${
...}
is an expression. The $expression
can also be more complex, like a function call: write_lines(some_array_value)
NOTE: the
$expression
in this context can only evaluate to a primitive type (e.g. notArray
,Map
, orObject
). The only exception to this rule is whensep
is specified as one of the$var_option
fields
As another example, consider how the parser would parse the following command:
grep '${start}...${end}' ${input}
This command would be parsed as:
grep '
- command_part_string${start}
- command_part_var...
- command_part_string${end}
- command_part_var'
- command_part_string${input}
- command_part_var
$var_option = $var_option_key $ws* '=' $ws* $var_option_value
$var_option_key = 'sep' | 'true' | 'false' | 'quote' | 'default'
$var_option_value = $expression
The $var_option
is a set of key-value pairs for any additional and less-used options that need to be set on a parameter.
'sep' is interpreted as the separator string used to join multiple parameters together. sep
is only valid if the expression evaluates to an Array
.
For example, if there were a declaration Array[Int] ints = [1,2,3]
, the command python script.py ${sep=',' numbers}
would yield the command line:
python script.py 1,2,3
Alternatively, if the command were python script.py ${sep=' ' numbers}
it would parse to:
python script.py 1 2 3
Additional Requirements:
- sep MUST accept only a string as its value
'true' and 'false' are only used for type Boolean and they specify what the parameter returns when the Boolean is true or false, respectively.
For example, ${true='--enable-foo', false='--disable-foo' Boolean yes_or_no}
would evaluate to either --enable-foo
or --disable-foo
based on the value of yes_or_no.
If either value is left out, then it's equivalent to specifying the empty string. If the parameter is ${true='--enable-foo' Boolean yes_or_no}
, and a value of false is specified for this parameter, then the parameter will evaluate to the empty string.
Additional Requirement:
true
andfalse
values MUST be strings.true
andfalse
are only allowed if the type isBoolean
This specifies the default value if no other value is specified for this parameter.
task default_test {
String? s
command {
./my_cmd ${default="foobar" s}
}
}
This task takes an optional String
parameter and if a value is not specified, then the value of foobar
will be used instead.
Additional Requirements:
- The type of the expression must match the type of the parameter
- If 'default' is specified, the
$type_postfix_quantifier
for the variable's type MUST be?
Sometimes a command is sufficiently long enough or might use {
characters that using a different set of delimiters would make it more clear. In this case, enclose the command in <<<
...>>>
, as follows:
task heredoc {
File in
command<<<
python <<CODE
with open("${in}") as fp:
for line in fp:
if not line.startswith('#'):
print(line.strip())
CODE
>>>
}
Parsing of this command should be the same as the prior section describes.
Any text inside of the command
section, after instantiated, should have all common leading whitespace removed. In the task heredoc
example in the previous section, if the user specifies a value of /path/to/file
as the value for File in
, then the command should be:
python <<CODE
with open("/path/to/file") as fp:
for line in fp:
if not line.startswith('#'):
print(line.strip())
CODE
The 2-spaces that were common to each line were removed.
If the user mixes tabs and spaces, the behavior is undefined. A warning is suggested, and perhaps a convention of 4 spaces per tab. Other implementations might return an error in this case.
The outputs section defines which of the files and values should be exported after a successful run of this tool.
$task_output = 'output' $ws* '{' ($ws* $task_output_kv $ws*)* '}'
$task_output_kv = $type $identifier $ws* '=' $ws* $string
The outputs section contains typed variable definitions and a binding to the variable that they export.
The left-hand side of the equality defines the type and name of the output.
The right-hand side defines the path to the file that contains that variable definition.
For example, if a task's output section looks like this:
output {
Int threshold = read_int("threshold.txt")
}
Then the task is expecting a file called "threshold.txt" in the current working directory where the task was executed. Inside of that file must be one line that contains only an integer and whitespace. See the Data Types & Serialization section for more details.
The filename strings may also contain variable definitions themselves (see the String Interpolation section below for more details):
output {
Array[String] quality_scores = read_lines("${sample_id}.scores.txt")
}
If this is the case, then sample_id
is considered an input to the task.
As with inputs, the outputs can reference previous outputs in the same block. The only requirement is that the output being referenced must be specified before the output which uses it.
output {
String a = "a"
String ab = a + "b"
}
Globs can be used to define outputs which contain many files. The glob function generates an array of File outputs:
output {
Array[File] output_bams = glob("*.bam")
}
Within tasks, any string literal can use string interpolation to access the value of any of the task's inputs. The most obvious example of this is being able to define an output file which is named as function of its input. For example:
task example {
String prefix
File bam
command {
python analysis.py --prefix=${prefix} ${bam}
}
output {
File analyzed = "${prefix}.out"
File bam_sibling = "${bam}.suffix"
}
}
Any ${identifier}
inside of a string literal must be replaced with the value of the identifier. If prefix were specified as foobar
, then "${prefix}.out"
would be evaluated to "foobar.out"
.
$runtime = 'runtime' $ws* '{' ($ws* $runtime_kv $ws*)* '}'
$runtime_kv = $identifier $ws* '=' $ws* $expression
The runtime section defines key/value pairs for runtime information needed for this task. Individual backends will define which keys they will inspect so a key/value pair may or may not actually be honored depending on how the task is run.
Values can be any expression and it is up to the engine to reject keys and/or values that do not make sense in that context. For example, consider the following WDL:
task test {
command {
python script.py
}
runtime {
docker: ["ubuntu:latest", "broadinstitute/scala-baseimage"]
}
}
The value for the docker
runtime attribute in this case is an array of values. The parser should accept this. Some engines might interpret it as an "either this image or that image" or could reject it outright.
Since values are expressions, they can also reference variables in the task:
task test {
String ubuntu_version
command {
python script.py
}
runtime {
docker: "ubuntu:" + ubuntu_version
}
}
Most key/value pairs are arbitrary. However, the following keys have recommended conventions:
Location of a Docker image for which this task ought to be run. This can have a format like ubuntu:latest
or broadinstitute/scala-baseimage
in which case it should be interpreted as an image on DockerHub (i.e. it is valid to use in a docker pull
command).
task docker_test {
String arg
command {
python process.py ${arg}
}
runtime {
docker: "ubuntu:latest"
}
}
Memory requirements for this task. Two kinds of values are supported for this attributes:
Int
- Intepreted as bytesString
- This should be a decimal value with suffixes likeB
,KB
,MB
or binary suffixesKiB
,MiB
. For example:6.2 GB
,5MB
,2GiB
.
task memory_test {
String arg
command {
python process.py ${arg}
}
runtime {
memory: "2GB"
}
}
$parameter_meta = 'parameter_meta' $ws* '{' ($ws* $parameter_meta_kv $ws*)* '}'
$parameter_meta_kv = $identifier $ws* '=' $ws* $string
This purely optional section contains key/value pairs where the keys are names of parameters and the values are string descriptions for those parameters.
Additional requirement: Any key in this section MUST correspond to a parameter in the command line
$meta = 'meta' $ws* '{' ($ws* $meta_kv $ws*)* '}'
$meta_kv = $identifier $ws* '=' $ws* $string
This purely optional section contains key/value pairs for any additional meta data that should be stored with the task. For example, perhaps author or contact email.
task hello_world {
command {echo hello world}
}
task one_and_one {
String pattern
File infile
command {
grep ${pattern} ${infile}
}
output {
File filtered = stdout()
}
}
task runtime_meta {
String memory_mb
String sample_id
String param
String sample_id
command {
java -Xmx${memory_mb}M -jar task.jar -id ${sample_id} -param ${param} -out ${sample_id}.out
}
output {
File results = "${sample_id}.out"
}
runtime {
docker: "broadinstitute/baseimg"
}
parameter_meta {
memory_mb: "Amount of memory to allocate to the JVM"
param: "Some arbitrary parameter"
sample_id: "The ID of the sample in format foo_bar_baz"
}
meta {
author: "Joe Somebody"
email: "[email protected]"
}
}
task bwa_mem_tool {
Int threads
Int min_seed_length
Int min_std_max_min
File reference
File reads
command {
bwa mem -t ${threads} \
-k ${min_seed_length} \
-I ${sep=',' min_std_max_min+} \
${reference} \
${sep=' ' reads+} > output.sam
}
output {
File sam = "output.sam"
}
runtime {
docker: "broadinstitute/baseimg"
}
}
Notable pieces in this example is ${sep=',' min_std_max_min+}
which specifies that min_std_max_min can be one or more integers (the +
after the variable name indicates that it can be one or more). If an Array[Int]
is passed into this parameter, then it's flattened by combining the elements with the separator character (sep=','
).
This task also defines that it exports one file, called 'sam', which is the stdout of the execution of bwa mem.
The 'docker' portion of this task definition specifies which that this task must only be run on the Docker image specified.
task wc2_tool {
File file1
command {
wc ${file1}
}
output {
Int count = read_int(stdout())
}
}
workflow count_lines4_wf {
Array[File] files
scatter(f in files) {
call wc2_tool {
input: file1=f
}
}
output {
wc2_tool.count
}
}
In this example, it's all pretty boilerplate, declarative code, except for some language-y like features, like firstline(stdout)
and append(list_of_count, wc2-tool.count)
. These both can be implemented fairly easily if we allow for custom function definitions. Parsing them is no problem. Implementation would be fairly simple and new functions would not be hard to add. Alternatively, this could be something like JavaScript or Python snippets that we run.
This task should produce a command line like this:
tmap mapall \
stage1 map1 --min-seq-length 20 \
map2 --min-seq-length 20 \
stage2 map1 --max-seq-length 20 --min-seq-length 10 --seed-length 16 \
map2 --max-seed-hits -1 --max-seq-length 20 --min-seq-length 10
Task definition would look like this:
task tmap_tool {
Array[String] stages
File reads
command {
tmap mapall ${sep=' ' stages} < ${reads} > output.sam
}
output {
File sam = "output.sam"
}
}
For this particular case where the command line is itself a mini DSL, The best option at that point is to allow the user to type in the rest of the command line, which is what ${sep=' ' stages+}
is for. This allows the user to specify an array of strings as the value for stages
and then it concatenates them together with a space character
Variable | Value |
---|---|
reads | /path/to/fastq |
stages | ["stage1 map1 --min-seq-length 20 map2 --min-seq-length 20", "stage2 map1 --max-seq-length 20 --min-seq-length 10 --seed-length 16 map2 --max-seed-hits -1 --max-seq-length 20 --min-seq-length 10"] |
$workflow = 'workflow' $ws* '{' $ws* $workflow_element* $ws* '}'
$workflow_element = $call | $loop | $conditional | $declaration | $scatter | $parameter_meta | $meta
A workflow is defined as the keyword workflow
and the body being in curly braces.
An example of a workflow that runs one task (not defined here) would be:
workflow wf {
Array[File] files
Int threshold
Map[String, String] my_map
call analysis_job {
input: search_paths=files, threshold=threshold, gender_lookup=my_map
}
}
$call = 'call' $ws* $namespaced_identifier $ws+ ('as' $identifier)? $ws* $call_body?
$call_body = '{' $ws* $inputs? $ws* '}'
$inputs = 'input' $ws* ':' $ws* $variable_mappings
$variable_mappings = $variable_mapping_kv (',' $variable_mapping_kv)*
$variable_mapping_kv = $identifier $ws* '=' $ws* $expression
A workflow may call other tasks/workflows via the call
keyword. The $namespaced_identifier
is the reference to which task to run. Most commonly, it's simply the name of a task (see examples below), but it can also use .
as a namespace resolver.
See the section on Fully Qualified Names & Namespaced Identifiers for details about how the $namespaced_identifier
ought to be interpreted
All call
statements must be uniquely identifiable. By default, the call's unique identifier is the task name (e.g. call foo
would be referenced by name foo
). However, if one were to call foo
twice in a workflow, each subsequent call
statement will need to alias itself to a unique name using the as
clause: call foo as bar
.
A call
statement may reference a workflow too (e.g. call other_workflow
). In this case, the $inputs
section specifies a subset of the workflow's inputs and must specify fully qualified names.
import "lib.wdl" as lib
workflow wf {
call my_task
call my_task as my_task_alias
call my_task as my_task_alias2 {
input: threshold=2
}
call lib.other_task
}
The $call_body
is optional and is meant to specify how to satisfy a subset of the the task or workflow's input parameters as well as a way to map tasks outputs to variables defined in the visible scopes.
A $variable_mapping
in the $inputs
section maps parameters in the task to expressions. These expressions usually reference outputs of other tasks, but they can be arbitrary expressions.
As an example, here is a workflow in which the second task requires an output from the first task:
task task1 {
command {
python do_stuff.py
}
output {
File results = stdout()
}
}
task task2 {
File foobar
command {
python do_stuff2.py ${foobar}
}
output {
File results = stdout()
}
}
workflow wf {
call task1
call task2 {
input: foobar=task1.results
}
}
Workflows can also be called inside of workflows.
main.wdl
import "sub_wdl.wdl" as sub
workflow main_workflow {
call sub.wf_hello { input: wf_hello_input = "sub world" }
output {
String main_output = wf_hello.salutation
}
}
sub_wdl.wdl
task hello {
String addressee
command {
echo "Hello ${addressee}!"
}
runtime {
docker: "ubuntu:latest"
}
output {
String salutation = read_string(stdout())
}
}
workflow wf_hello {
String wf_hello_input
call hello {input: addressee = wf_hello_input }
output {
String salutation = hello.salutation
}
}
Note that because a wdl file can only contain 1 workflow, sub workflows can only be used through imports. Otherwise, calling a workflow or a task is equivalent syntactically. Inputs are specified and outputs retrieved the same way as they are for task calls.
$scatter = 'scatter' $ws* '(' $ws* $scatter_iteration_statment $ws* ')' $ws* $scatter_body
$scatter_iteration_statment = $identifier $ws* 'in' $ws* $expression
$scatter_body = '{' $ws* $workflow_element* $ws* '}'
A "scatter" clause defines that everything in the body ($scatter_body
) can be run in parallel. The clause in parentheses ($scatter_iteration_statement
) declares which collection to scatter over and what to call each element.
The $scatter_iteration_statement
has two parts: the "item" and the "collection". For example, scatter(x in y)
would define x
as the item, and y
as the collection. The item is always an identifier, while the collection is an expression that MUST evaluate to an Array
type. The item will represent each item in that expression. For example, if y
evaluated to an Array[String]
then x
would be a String
.
The $scatter_body
defines a set of scopes that will execute in the context of this scatter block.
For example, if $expression
is an array of integers of size 3, then the body of the scatter clause can be executed 3-times in parallel. $identifier
would refer to each integer in the array.
scatter(i in integers) {
call task1{input: num=i}
call task2{input: num=task1.output}
}
In this example, task2
depends on task1
. Variable i
has an implicit index
attribute to make sure we can access the right output from task1
. Since both task1 and task2 run N times where N is the length of the array integers
, any scalar outputs of these tasks is now an array.
$conditional = 'if' '(' $expression ')' '{' $workflow_element* '}'
Conditionals only execute the body if the expression evaluates to true.
- When a call's output is referenced outside the same containing
if
it will need to be handled as an optional type. E.g.
workflow foo {
# Call 'x', producing a Boolean output:
call x
Boolean x_out = x.out
# Call 'y', producing an Int output, in a conditional block:
if (x_out) {
call y
Int y_out = y.out
}
# Outside the if block, we have to handle this output as optional:
Int? y_out_maybe = y.out
# Call 'z' which takes an optional Int input:
call z { input: optional_int = y_out_maybe }
}
- Optional types can be coalesced by using the
select_all
andselect_first
array functions:
workflow foo {
Array[Int] scatter_range = [1, 2, 3, 4, 5]
scatter (i in scatter_range) {
call x { input: i = i }
if (x.validOutput) {
Int x_out = x.out
}
}
# Because it was declared inside the scatter and the if-block, the type of x_out is different here:
Array[Int?] x_out_maybes = x_out
# We can select only the valid elements with select_all:
Array[Int] x_out_valids = select_all(x_out_maybes)
# Or we can select the first valid element:
Int x_out_first = select_first(x_out_maybes)
}
- When conditional blocks are nested, referenced outputs are only ever single-level conditionals (i.e. we never produce
Int??
or deeper):
workflow foo {
Boolean b
Boolean c
if(b) {
if(c) {
call x
Int x_out = x.out
}
}
Int? x_out_maybe = x_out # Even though it's within two 'if's, we don't need Int??
# Call 'y' which takes an Int input:
call y { input: int_input = select_first([x_out_maybe, 5]) } # The select_first produces an Int, not an Int?
}
$wf_parameter_meta = 'parameter_meta' $ws* '{' ($ws* $wf_parameter_meta_kv $ws*)* '}'
$wf_parameter_meta_kv = $identifier $ws* '=' $ws* $string
This purely optional section contains key/value pairs where the keys are names of parameters and the values are string descriptions for those parameters.
Additional requirement: Any key in this section MUST correspond to a worflow input
As an example:
parameter_meta {
memory_mb: "Amount of memory to allocate to the JVM"
param: "Some arbitrary parameter"
sample_id: "The ID of the sample in format foo_bar_baz"
}
$wf_meta = 'meta' $ws* '{' ($ws* $wf_meta_kv $ws*)* '}'
$wf_meta_kv = $identifier $ws* '=' $ws* $string
This purely optional section contains key/value pairs for any additional meta data that should be stored with the workflow. For example, perhaps author or contact email.
As an example:
meta {
author: "Joe Somebody"
email: "[email protected]"
}
Each workflow
definition can specify an output
section. This section lists outputs as Type name = expression
, just like task outputs.
Workflow outputs also follow the same syntax rules as task outputs. They are expressions which can reference all call outputs, workflow inputs, intermediate values and previous workflow outputs. e.g:
task t {
Int i
command {
# do something
}
output {
String out = "out"
}
}
workflow w {
String w_input = "some input"
call t
call t as u
output {
String t_out = t.out
String u_out = u.out
String input_as_output = w_input
String previous_output = u_out
}
}
Note that they can't reference call inputs (eg we cannot use Int i = t.i
as a workflow output). However this can be achieved by also declaring the desired call input as an output of the call.
When declaring a workflow output that points to a call inside a scatter, the aggregated call is used, just like any expression that references it from outside the scatter. e.g:
task t {
command {
# do something
}
output {
String out = "out"
}
}
workflow w {
Array[Int] arr = [1, 2]
scatter(i in arr) {
call t
}
output {
Array[String] t_out = t.out
}
}
In this example t_out
has an Array[String]
result type, because call t
is inside a scatter.
If the output {...}
section is omitted from a top-level workfow then the workflow engine should include all outputs from all calls in its final output.
However, if a workflow is intended to be called as a subworkflow, it is required that outputs are named and specified using expressions in the outputs block, just like task outputs. The rationale here is:
- To present the same interface when calling subworkflows as when calling tasks.
- To make it easy for callers of subworkflows to find out exactly what outputs the call is creating.
- In case of nested subworkflows, to give the outputs at the top level a simple fixed name rather than a long qualified name like
a.b.c.d.out
(which is liable to change if the underlying implementation ofc
changes, for example).
Import statements can be used to pull in tasks/workflows from other locations as well as to create namespaces. In the simplest case, an import statement adds the tasks/workflows that are imported into the specified namespace. For example:
tasks.wdl
task x {
command { python script.py }
}
task y {
command { python script2.py }
}
workflow.wdl
import "tasks.wdl" as pyTasks
workflow wf {
call pyTasks.x
call pyTasks.y
}
Tasks x
and y
are inside the namespace pyTasks
, which is different from the wf
namespace belonging to the primary workflow. However, if no namespace is specified for tasks.wdl:
workflow.wdl
import "tasks.wdl"
workflow wf {
call tasks.x
call tasks.y
}
Now everything inside of tasks.wdl
must be accessed through the default namespace tasks
.
Each namespace may contain namespaces, tasks, and at most one workflow. The names of the contained namespaces, tasks, and workflow need to be unique within that namespace. For example, one cannot import two workflows while they have the same namespace identifier. Additionally, a workflow and a namespace both named foo
cannot exist inside a common namespace. Similarly there cannot be a task foo
in a workflow also named foo
.
However, you can import two workflows with different namespace identifiers that have identically named tasks. For example, you can import namespaces foo
and bar
, both of which contain a task baz
, and you can call foo.baz
and bar.baz
from the same primary workflow.
Scopes are defined as:
workflow {...}
blockscall
blockswhile(expr) {...}
blocksif(expr) {...}
blocksscatter(x in y) {...}
blocks
Inside of any scope, variables may be declared. The variables declared in that scope are visible to any sub-scope, recursively. For example:
task my_task {
Int x
File f
command {
my_cmd --integer=${var} ${f}
}
}
workflow wf {
Array[File] files
Int x = 2
scatter(file in files) {
Int x = 3
call my_task {
Int x = 4
input: var=x, f=file
}
}
}
my_task
will use x=4
to set the value for var
in its command line. However, my_task
also needs a value for x
which is defined at the task level. Since my_task
has two inputs (x
and var
), and only one of those is set in the call my_task
declaration, the value for my_task.x
still needs to be provided by the user when the workflow is run.
Types can be optionally suffixed with a ?
or +
in certain cases.
?
means that the parameter is optional. A user does not need to specify a value for the parameter in order to satisfy all the inputs to the workflow.+
applies only toArray
types and it represents a constraint that theArray
value must containe one-or-more elements.
task test {
Array[File] a
Array[File]+ b
Array[File]? c
#File+ d <-- can't do this, + only applies to Arrays
command {
/bin/mycmd ${sep=" " a}
/bin/mycmd ${sep="," b}
/bin/mycmd ${write_lines(c)}
}
}
workflow wf {
call test
}
If you provided these values for inputs:
var | value |
---|---|
wf.test.a | ["1", "2", "3"] |
wf.test.b | [] |
The workflow engine should reject this because wf.test.b
is required to have at least one element. If we change it to:
var | value |
---|---|
wf.test.a | ["1", "2", "3"] |
wf.test.b | ["x"] |
This would be valid input because wf.test.c
is not required. Given these values, the command would be instantiated as:
/bin/mycmd 1 2 3
/bin/mycmd x
/bin/mycmd
If our inputs were:
var | value |
---|---|
wf.test.a | ["1", "2", "3"] |
wf.test.b | ["x","y"] |
wf.test.c | ["a","b","c","d"] |
Then the command would be instantiated as:
/bin/mycmd 1 2 3
/bin/mycmd x,y
/bin/mycmd /path/to/c.txt
Sometimes, optional parameters need a string prefix. Consider this task:
task test {
String? val
command {
python script.py --val=${val}
}
}
Since val
is optional, this command line can be instantiated in two ways:
python script.py --val=foobar
Or
python script.py --val=
The latter case is very likely an error case, and this --val=
part should be left off if a value for val
is omitted. To solve this problem, modify the expression inside the template tag as follows:
python script.py ${"--val=" + val}
The scatter
block is meant to parallelize a series of identical tasks but give them slightly different inputs. The simplest example is:
task inc {
Int i
command <<<
python -c "print(${i} + 1)"
>>>
output {
Int incremented = read_int(stdout())
}
}
workflow wf {
Array[Int] integers = [1,2,3,4,5]
scatter(i in integers) {
call inc{input: i=i}
}
}
Running this workflow (which needs no inputs), would yield a value of [2,3,4,5,6]
for wf.inc
. While task inc
itself returns an Int
, when it is called inside a scatter block, that type becomes an Array[Int]
.
Any task that's downstream from the call to inc
and outside the scatter block must accept an Array[Int]
:
task inc {
Int i
command <<<
python -c "print(${i} + 1)"
>>>
output {
Int incremented = read_int(stdout())
}
}
task sum {
Array[Int] ints
command <<<
python -c "print(${sep="+" ints})"
>>>
output {
Int sum = read_int(stdout())
}
}
workflow wf {
Array[Int] integers = [1,2,3,4,5]
scatter (i in integers) {
call inc {input: i=i}
}
call sum {input: ints = inc.increment}
}
This workflow will output a value of 20
for wf.sum.sum
. This works because call inc
will output an Array[Int]
because it is in the scatter block.
However, from inside the scope of the scatter block, the output of call inc
is still an Int
. So the following is valid:
workflow wf {
Array[Int] integers = [1,2,3,4,5]
scatter(i in integers) {
call inc {input: i=i}
call inc as inc2 {input: i=inc.incremented}
}
call sum {input: ints = inc2.increment}
}
In this example, inc
and inc2
are being called in serial where the output of one is fed to another. inc2 would output the array [3,4,5,6,7]
Inside of expressions, variables are resolved differently depending on if the expression is in a task
declaration or a workflow
declaration
Inside a task, resolution is trivial: The variable referenced MUST be a declaration of the task. For example:
task my_task {
Array[String] strings
command {
python analyze.py --strings-file=${write_lines(strings)}
}
}
Inside of this task, there exists only one expression: write_lines(strings)
. In here, when the expression evaluator tries to resolve strings
, which must be a declaration of the task (in this case it is).
In a workflow, resolution works by traversing the scope heirarchy starting from expression that references the variable.
workflow wf {
String s = "wf_s"
String t = "t"
call my_task {
String s = "my_task_s"
input: in0 = s+"-suffix", in1 = t+"-suffix"
}
}
In this example, there are two expressions: s+"-suffix"
and t+"-suffix"
. s
is resolved as "my_task_s"
and t
is resolved as "t"
.
Both tasks and workflows have a typed inputs that must be satisfied in order to run. The following sections describe how to compute inputs for task
and workflow
declarations
Tasks define all their inputs as declarations at the top of the task definition.
task test {
String s
Int i
Float f
command {
./script.sh -i ${i} -f ${f}
}
}
In this example, s
, i
, and f
are inputs to this task. Even though the command line does not reference ${s}
. Implementations of WDL engines may display a warning or report an error in this case, since s
isn't used.
Workflows have declarations, like tasks, but a workflow must also account for all calls to sub-tasks when determining inputs.
Workflows also return their inputs as fully qualified names. Tasks only return the names of the variables as inputs (as they're guaranteed to be unique within a task). However, since workflows can call the same task twice, names might collide. The general algorithm for computing inputs going something like this:
- Take all inputs to all
call
statements in the workflow - Subtract out all inputs that are satisfied through the
input:
section - Add in all declarations which don't have a static value defined
Consider the following workflow:
task t1 {
String s
Int x
command {
./script --action=${s} -x${x}
}
output {
Int count = read_int(stdout())
}
}
task t2 {
String s
Int t
Int x
command {
./script2 --action=${s} -x${x} --other=${t}
}
output {
Int count = read_int(stdout())
}
}
task t3 {
Int y
File ref_file # Do nothing with this
command {
python -c "print(${y} + 1)"
}
output {
Int incr = read_int(stdout())
}
}
workflow wf {
Int int_val
Int int_val2 = 10
Array[Int] my_ints
File ref_file
call t1 {
input: x=int_val
}
call t2 {
input: x=int_val, t=t1.count
}
scatter(i in my_ints) {
call t3 {
input: y=i, ref=ref_file
}
}
}
The inputs to wf
would be:
wf.t1.s
as aString
wf.t2.s
as aString
wf.int_val
as anInt
wf.my_ints
as anArray[Int]
wf.ref_file
as aFile
Once workflow inputs are computed (see previous section), the value for each of the fully-qualified names needs to be specified per invocation of the workflow. Workflow inputs are specified in JSON or YAML format. In JSON, the inputs to the workflow in the previous section can be:
{
"wf.t1.s": "some_string",
"wf.t2.s": "some_string",
"wf.int_val": 3,
"wf.my_ints": [5,6,7,8],
"wf.ref_file": "/path/to/file.txt"
}
It's important to note that the type in JSON must be coercable to the WDL type. For example wf.int_val
expects an integer, but if we specified it in JSON as "wf.int_val": "3"
, this coercion from string to integer is not valid and would result in a type error. See the section on Type Coercion for more details.
WDL values can be created from either JSON values or from native language values. The below table references String-like, Integer-like, etc to refer to values in a particular programming language. For example, "String-like" could mean a java.io.String
in the Java context or a str
in Python. An "Array-like" could refer to a Seq
in Scala or a list
in Python.
WDL Type | Can Accept | Notes / Constraints |
---|---|---|
String |
JSON String | |
String-like | ||
String |
Identity coercion | |
File |
||
File |
JSON String | Interpreted as a file path |
String-like | Interpreted as file path | |
String |
Interpreted as file path | |
File |
Identity Coercion | |
Int |
JSON Number | Use floor of the value for non-integers |
Integer-like | ||
Int |
Identity coercion | |
Float |
JSON Number | |
Float-like | ||
Float |
Identity coercion | |
Boolean |
JSON Boolean | |
Boolean-like | ||
Boolean |
Identity coercion | |
Array[T] |
JSON Array | Elements must be coercable to T |
Array-like | Elements must be coercable to T |
|
Map[K, V] |
JSON Object | keys and values must be coercable to K and V , respectively |
Map-like | keys and values must be coercable to K and V , respectively |
Returns a File
reference to the stdout that this task generated.
Returns a File
reference to the stderr that this task generated.
Given a file-like object (String
, File
) as a parameter, this will read each line as a string and return an Array[String]
representation of the lines in the file.
The order of the lines in the returned Array[String]
must be the order in which the lines appear in the file-like object.
This task would grep
through a file and return all strings that matched the pattern:
task do_stuff {
String pattern
File file
command {
grep '${pattern}' ${file}
}
output {
Array[String] matches = read_lines(stdout())
}
}
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
the read_tsv()
function takes one parameter, which is a file-like object (String
, File
) and returns an Array[Array[String]]
representing the table from the TSV file.
If the parameter is a String
, this is assumed to be a local file path relative to the current working directory of the task.
For example, if I write a task that outputs a file to ./results/file_list.tsv
, and my task is defined as:
task do_stuff {
File file
command {
python do_stuff.py ${file}
}
output {
Array[Array[String]] output_table = read_tsv("./results/file_list.tsv")
}
}
Then when the task finishes, to fulfull the outputs_table
variable, ./results/file_list.tsv
must be a valid TSV file or an error will be reported.
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
Given a file-like object (String
, File
) as a parameter, this will read each line from a file and expect the line to have the format col1\tcol2
. In other words, the file-like object must be a two-column TSV file.
This task would grep
through a file and return all strings that matched the pattern:
The following task would write a two-column TSV to standard out and that would be interpreted as a Map[String, String]
:
task do_stuff {
String flags
File file
command {
./script --flags=${flags} ${file}
}
output {
Map[String, String] mapping = read_map(stdout())
}
}
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
Given a file-like object that contains a 2-row and n-column TSV file, this function will turn that into an Object.
task test {
command <<<
python <<CODE
print('\t'.join(["key_{}".format(i) for i in range(3)]))
print('\t'.join(["value_{}".format(i) for i in range(3)]))
CODE
>>>
output {
Object my_obj = read_object(stdout())
}
}
The command will output to stdout the following:
key_1\tkey_2\tkey_3
value_1\tvalue_2\tvalue_3
Which would be turned into an Object
in WDL that would look like this:
Attribute | Value |
---|---|
key_1 | "value_1" |
key_2 | "value_2" |
key_3 | "value_3" |
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
Given a file-like object that contains a 2-row and n-column TSV file, this function will turn that into an Object.
task test {
command <<<
python <<CODE
print('\t'.join(["key_{}".format(i) for i in range(3)]))
print('\t'.join(["value_{}".format(i) for i in range(3)]))
print('\t'.join(["value_{}".format(i) for i in range(3)]))
print('\t'.join(["value_{}".format(i) for i in range(3)]))
CODE
>>>
output {
Array[Object] my_obj = read_objects(stdout())
}
}
The command will output to stdout the following:
key_1\tkey_2\tkey_3
value_1\tvalue_2\tvalue_3
value_1\tvalue_2\tvalue_3
value_1\tvalue_2\tvalue_3
Which would be turned into an Array[Object]
in WDL that would look like this:
Index | Attribute | Value |
---|---|---|
0 | key_1 | "value_1" |
key_2 | "value_2" | |
key_3 | "value_3" | |
1 | key_1 | "value_1" |
key_2 | "value_2" | |
key_3 | "value_3" | |
2 | key_1 | "value_1" |
key_2 | "value_2" | |
key_3 | "value_3" |
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
the read_json()
function takes one parameter, which is a file-like object (String
, File
) and returns a data type which matches the data structure in the JSON file. The mapping of JSON type to WDL type is:
JSON Type | WDL Type |
---|---|
object | Map[String, ?] |
array | Array[?] |
number | Int or Float |
string | String |
boolean | Boolean |
null | ??? |
If the parameter is a String
, this is assumed to be a local file path relative to the current working directory of the task.
For example, if I write a task that outputs a file to ./results/file_list.json
, and my task is defined as:
task do_stuff {
File file
command {
python do_stuff.py ${file}
}
output {
Map[String, String] output_table = read_json("./results/file_list.json")
}
}
Then when the task finishes, to fulfull the output_table
variable, ./results/file_list.json
must be a valid TSV file or an error will be reported.
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
The read_int()
function takes a file path which is expected to contain 1 line with 1 integer on it. This function returns that integer.
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed.
The read_string()
function takes a file path which is expected to contain 1 line with 1 string on it. This function returns that string.
No trailing newline characters should be included
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
The read_float()
function takes a file path which is expected to contain 1 line with 1 floating point number on it. This function returns that float.
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
The read_boolean()
function takes a file path which is expected to contain 1 line with 1 Boolean value (either "true" or "false" on it). This function returns that Boolean value.
If the entire contents of the file can not be read for any reason, the calling task or workflow will be considered to have failed. Examples of failure include but are not limted to not having access to the file, resource limitations (e.g. memory) when reading the file, and implementation imposed file size limits.
Given something that's compatible with Array[String]
, this writes each element to it's own line on a file. with newline \n
characters as line separators.
task example {
Array[String] array = ["first", "second", "third"]
command {
./script --file-list=${write_lines(array)}
}
}
If this task were run, the command might look like:
./script --file-list=/local/fs/tmp/array.txt
And /local/fs/tmp/array.txt
would contain:
first
second
third
Given something that's compatible with Array[Array[String]]
, this writes a TSV file of the data structure.
task example {
Array[String] array = [["one", "two", "three"], ["un", "deux", "trois"]]
command {
./script --tsv=${write_tsv(array)}
}
}
If this task were run, the command might look like:
./script --tsv=/local/fs/tmp/array.tsv
And /local/fs/tmp/array.tsv
would contain:
one\ttwo\tthree
un\tdeux\ttrois
Given something that's compatible with Map[String, String]
, this writes a TSV file of the data structure.
task example {
Map[String, String] map = {"key1": "value1", "key2": "value2"}
command {
./script --map=${write_map(map)}
}
}
If this task were run, the command might look like:
./script --tsv=/local/fs/tmp/map.tsv
And /local/fs/tmp/map.tsv
would contain:
key1\tvalue1
key2\tvalue2
Given any Object
, this will write out a 2-row, n-column TSV file with the object's attributes and values.
task test {
Object input
command <<<
/bin/do_work --obj=${write_object(input)}
>>>
output {
File results = stdout()
}
}
if input
were to have the value:
Attribute | Value |
---|---|
key_1 | "value_1" |
key_2 | "value_2" |
key_3 | "value_3" |
The command would instantiate to:
/bin/do_work --obj=/path/to/input.tsv
Where /path/to/input.tsv
would contain:
key_1\tkey_2\tkey_3
value_1\tvalue_2\tvalue_3
Given any Array[Object]
, this will write out a 2+ row, n-column TSV file with each object's attributes and values.
task test {
Array[Object] in
command <<<
/bin/do_work --obj=${write_objects(in)}
>>>
output {
File results = stdout()
}
}
if in
were to have the value:
Index | Attribute | Value |
---|---|---|
0 | key_1 | "value_1" |
key_2 | "value_2" | |
key_3 | "value_3" | |
1 | key_1 | "value_4" |
key_2 | "value_5" | |
key_3 | "value_6" | |
2 | key_1 | "value_7" |
key_2 | "value_8" | |
key_3 | "value_9" |
The command would instantiate to:
/bin/do_work --obj=/path/to/input.tsv
Where /path/to/input.tsv
would contain:
key_1\tkey_2\tkey_3
value_1\tvalue_2\tvalue_3
value_4\tvalue_5\tvalue_6
value_7\tvalue_8\tvalue_9
Given something with any type, this writes the JSON equivalent to a file. See the table in the definition of read_json()
task example {
Map[String, String] map = {"key1": "value1", "key2": "value2"}
command {
./script --map=${write_json(map)}
}
}
If this task were run, the command might look like:
./script --tsv=/local/fs/tmp/map.json
And /local/fs/tmp/map.json
would contain:
{
"key1": "value1"
"key2": "value2"
}
Given a File
and a String
(optional), returns the size of the file in Bytes or in the unit specified by the second argument.
task example {
File input_file
command {
echo "this file is 22 bytes" > created_file
}
output {
Float input_file_size = size(input_file)
Float created_file_size = size("created_file") # 22.0
Float created_file_size_in_KB = size("created_file", "K") # 0.022
}
}
Supported units are KiloByte ("K", "KB"), MegaByte ("M", "MB"), GigaByte ("G", "GB"), TeraByte ("T", "TB") as well as their binary version "Ki" ("KiB"), "Mi" ("MiB"), "Gi" ("GiB"), "Ti" ("TiB"). Default unit is Bytes ("B").
Given 3 String parameters input
, pattern
, replace
, this function will replace any occurrence matching pattern
in input
by replace
.
pattern
is expected to be a regular expression. Details of regex evaluation will depend on the execution engine running the WDL.
Example 1:
String chocolike = "I like chocolate when it's late"
String chocolove = sub(chocolike, "like", "love") # I love chocolate when it's late
String chocoearly = sub(chocolike, "late", "early") # I like chocoearly when it's early
String chocolate = sub(chocolike, "late$", "early") # I like chocolate when it's early
}
The sub function will also accept input
and replace
parameters that can be coerced to a String (e.g. File). This can be useful to swap the extension of a filename for example
Example 2:
task example {
File input_file = "my_input_file.bam"
String output_file_name = sub(input_file, "\\.bam$", ".index") # my_input_file.index
command {
echo "I want an index instead" > ${output_file_name}
}
output {
File outputFile = output_file_name
}
}
Given an integer argument, the range
function creates an array of integers of length equal to the given argument. For example range(3)
provides the array: (0, 1, 2)
.
Given a two dimensional array argument, the transpose
function transposes the two dimensional array according to the standard matrix transpose rules. For example transpose( ((0, 1, 2), (3, 4, 5)) )
will return the rotated two-dimensional array: ((0, 3), (1, 4), (2, 5))
.
Given any two Object types, the zip
function returns the dot product of those Object types in the form of a Pair object.
Pair[Int, String] p = (0, "z")
Array[Int] xs = [ 1, 2, 3 ]
Array[String] ys = [ "a", "b", "c" ]
Array[String] zs = [ "d", "e" ]
Array[Pair[Int, String]] zipped = zip(xs, ys) # i.e. zipped = [ (1, "a"), (2, "b"), (3, "c") ]
Given any two Object types, the cross
function returns the cross product of those Object types in the form of a Pair object.
Pair[Int, String] p = (0, "z")
Array[Int] xs = [ 1, 2, 3 ]
Array[String] ys = [ "a", "b", "c" ]
Array[String] zs = [ "d", "e" ]
Array[Pair[Int, String]] crossed = cross(xs, zs) # i.e. crossed = [ (1, "d"), (1, "e"), (2, "d"), (2, "e"), (3, "d"), (3, "e") ]
Given an Array, the length
function returns the number of elements in the Array as an Integer.
Array[Int] xs = [ 1, 2, 3 ]
Array[String] ys = [ "a", "b", "c" ]
Array[String] zs = [ ]
Integer xlen = length(xs) # 3
Integer ylen = length(ys) # 3
Integer zlen = length(zs) # 0
Given a String and an Array[X] where X is a primitive type, the prefix
function returns an array of strings comprised
of each element of the input array prefixed by the specified prefix string. For example:
Array[String] env = ["key1=value1", "key2=value2", "key3=value3"]
Array[String] env_param = prefix("-e ", env) # ["-e key1=value1", "-e key2=value2", "-e key3=value3"]
Array[Integer] env2 = [1, 2, 3]
Array[String] env2_param = prefix("-f ", env2) # ["-f 1", "-f 2", "-f 3"]
Given an array of optional values, select_first
will select the first defined value and return it. Note that this is a runtime check and requires that at least one defined value will exist: if no defined value is found when select_first is evaluated, the workflow will fail.
Given an array of optional values, select_all
will select only those elements which are defined.
This function will return false
if the argument is an unset optional value. It will return true
in all other cases.
- This function returns the basename of a file path passed to it:
basename("/path/to/file.txt")
returns"file.txt"
. - Also supports an optional parameter, suffix to remove:
basename("/path/to/file.txt", ".txt")
returns"file"
.
- These functions convert a Float value into an Int by:
- floor: Round down to the next lower integer
- ceil: Round up to the next higher integer
- round: Round to the nearest integer based on standard rounding rules
Tasks and workflows are given values for their input parameters in order to run. The type of each of those input parameters are declarations on the task
or workflow
. Those input parameters can be any valid type:
Primitive Types:
- String
- Int
- Float
- File
- Boolean
Compound Types:
- Array
- Map
- Object
- Pair
When a WDL workflow engine instantiates a command specified in the command
section of a task
, it must serialize all ${...}
tags in the command into primitive types.
For example, if I'm writing a tool that operates on a list of FASTQ files, there are a variety of ways that this list can be passed to that task:
- A file containing one file path per line (e.g.
Rscript analysis.R --files=fastq_list.txt
) - A file containing a JSON list (e.g.
Rscript analysis.R --files=fastq_list.json
) - Enumerated on the command line (e.g. (
Rscript analysis.R 1.fastq 2.fastq 3.fastq
)
Each of these methods has its merits and one method might be better for one tool while another method would be better for another tool.
On the other end, tasks need to be able to communicate data structures back to the workflow engine. For example, let's say this same tool that takes a list of FASTQs wants to return back a Map[File, Int]
representing the number of reads in each FASTQ. A tool might choose to output it as a two-column TSV or as a JSON object and WDL needs to know how to convert that to the proper data type.
WDL provides some standard library functions for converting compound types like Array
into primitive types, like File
.
When a task finishes, the output
section defines how to convert the files and stdout/stderr into WDL types. For example,
task test {
Array[File] files
command {
Rscript analysis.R --files=${sep=',' files}
}
output {
Array[String] strs = read_lines(stdout())
}
}
Here, the expression read_lines(stdout())
says "take the output from stdout, break into lines, and return that result as an Array[String]". See the definition of read_lines and stdout for more details.
Serializing primitive inputs into strings is intuitively easy because the value is just turned into a string and inserted into the command line.
Consider this example:
task output_example {
String s
Int i
Float f
command {
python do_work.py ${s} ${i} ${f}
}
}
If I provide values for the declarations in the task as:
var | value |
---|---|
s | "str" |
i | 2 |
f | 1.3 |
Then, the command would be instantiated as:
python do_work.py str 2 1.3
Compound types, like Array
and Map
must be converted to a primitive type before it can be used in the command. There are many ways to turn a compound types into primitive types, as laid out in following sections
Arrays can be serialized in two ways:
- Array Expansion: elements in the list are flattened to a string with a separator character.
- File Creation: create a file with the elements of the array in it and passing that file as the parameter on the command line.
The array flattening approach can be done if a parameter is specified as ${sep=' ' my_param}
. my_param
must be declared as an Array
of primitive types. When the value of my_param
is specified, then the values are joined together with the separator character (a space in this case). For example:
task test {
Array[File] bams
command {
python script.py --bams=${sep=',' bams}
}
}
If passed an array for the value of bams
:
Element |
---|
/path/to/1.bam |
/path/to/2.bam |
/path/to/3.bam |
Would produce the command python script.py --bams=/path/to/1.bam,/path/to/2.bam,/path/to/1.bam
An array may be turned into a file with each element in the array occupying a line in the file.
task test {
Array[File] bams
command {
sh script.sh ${write_lines(bams)}
}
}
if bams
is given this array:
Element |
---|
/path/to/1.bam |
/path/to/2.bam |
/path/to/3.bam |
Then, the resulting command line could look like:
sh script.sh /jobs/564758/bams
Where /jobs/564758/bams
would contain:
/path/to/1.bam
/path/to/2.bam
/path/to/3.bam
The array may be turned into a JSON document with the file path for the JSON file passed in as the parameter:
task test {
Array[File] bams
command {
sh script.sh ${write_json(bams)}
}
}
if bams
is given this array:
Element |
---|
/path/to/1.bam |
/path/to/2.bam |
/path/to/3.bam |
Then, the resulting command line could look like:
sh script.sh /jobs/564758/bams.json
Where /jobs/564758/bams.json
would contain:
[
"/path/to/1.bam",
"/path/to/2.bam",
"/path/to/3.bam"
]
Map types cannot be serialized on the command line directly and must be serialized through a file
The map type can be serialized as a two-column TSV file and the parameter on the command line is given the path to that file, using the write_map()
function:
task test {
Map[String, Float] sample_quality_scores
command {
sh script.sh ${write_map(sample_quality_scores)}
}
}
if sample_quality_scores
is given this Map[String, Float] as:
Key | Value |
---|---|
sample1 | 98 |
sample2 | 95 |
sample3 | 75 |
Then, the resulting command line could look like:
sh script.sh /jobs/564757/sample_quality_scores.tsv
Where /jobs/564757/sample_quality_scores.tsv
would contain:
sample1\t98
sample2\t95
sample3\t75
The map type can also be serialized as a JSON file and the parameter on the command line is given the path to that file, using the write_json()
function:
task test {
Map[String, Float] sample_quality_scores
command {
sh script.sh ${write_json(sample_quality_scores)}
}
}
if sample_quality_scores is given this map:
Key | Value |
---|---|
sample1 | 98 |
sample2 | 95 |
sample3 | 75 |
Then, the resulting command line could look like:
sh script.sh /jobs/564757/sample_quality_scores.json
Where /jobs/564757/sample_quality_scores.json
would contain:
{
"sample1": 98,
"sample2": 95,
"sample3": 75
}
An object is a more general case of a map where the keys are strings and the values are of arbitrary types and treated as strings. Objects can be serialized with either write_object()
or write_json()
functions:
task test {
Object sample
command {
perl script.pl ${write_object(sample)}
}
}
if sample is provided as:
Attribute | Value |
---|---|
attr1 | value1 |
attr2 | value2 |
attr3 | value3 |
attr4 | value4 |
Then, the resulting command line could look like:
perl script.pl /jobs/564759/sample.tsv
Where /jobs/564759/sample.tsv
would contain:
attr1\tattr2\tattr3\tattr4
value1\tvalue2\tvalue3\tvalue4
task test {
Object sample
command {
perl script.pl ${write_json(sample)}
}
}
if sample is provided as:
Attribute | Value |
---|---|
attr1 | value1 |
attr2 | value2 |
attr3 | value3 |
attr4 | value4 |
Then, the resulting command line could look like:
perl script.pl /jobs/564759/sample.json
Where /jobs/564759/sample.json
would contain:
{
"attr1": "value1",
"attr2": "value2",
"attr3": "value3",
"attr4": "value4",
}
Array[Object]
must guarantee that all objects in the array have the same set of attributes. These can be serialized with either write_objects()
or write_json()
functions, as described in following sections.
an Array[Object]
can be serialized using write_objects()
into a TSV file:
task test {
Array[Object] sample
command {
perl script.pl ${write_objects(sample)}
}
}
if sample is provided as:
Index | Attribute | Value |
---|---|---|
0 | attr1 | value1 |
attr2 | value2 | |
attr3 | value3 | |
attr4 | value4 | |
1 | attr1 | value5 |
attr2 | value6 | |
attr3 | value7 | |
attr4 | value8 |
Then, the resulting command line could look like:
perl script.pl /jobs/564759/sample.tsv
Where /jobs/564759/sample.tsv
would contain:
attr1\tattr2\tattr3\tattr4
value1\tvalue2\tvalue3\tvalue4
value5\tvalue6\tvalue7\tvalue8
an Array[Object]
can be serialized using write_json()
into a JSON file:
task test {
Array[Object] sample
command {
perl script.pl ${write_json(sample)}
}
}
if sample is provided as:
Index | Attribute | Value |
---|---|---|
0 | attr1 | value1 |
attr2 | value2 | |
attr3 | value3 | |
attr4 | value4 | |
1 | attr1 | value5 |
attr2 | value6 | |
attr3 | value7 | |
attr4 | value8 |
Then, the resulting command line could look like:
perl script.pl /jobs/564759/sample.json
Where /jobs/564759/sample.json
would contain:
[
{
"attr1": "value1",
"attr2": "value2",
"attr3": "value3",
"attr4": "value4"
},
{
"attr1": "value5",
"attr2": "value6",
"attr3": "value7",
"attr4": "value8"
}
]
A task's command can only output data as files. Therefore, every de-serialization function in WDL takes a file input and returns a WDL type
De-serialization of primitive types is done through a read_*
function. For example, read_int("file/path")
and read_string("file/path")
.
For example, if I have a task that outputs a String
and an Int
:
task output_example {
String param1
String param2
command {
python do_work.py ${param1} ${param2} --out1=int_file --out2=str_file
}
output {
Int my_int = read_int("int_file")
String my_str = read_string("str_file")
}
}
Both files file_with_int
and file_with_uri
should contain one line with the value on that line. This value is then validated against the type of the variable. If file_with_int
contains a line with the text "foobar", the workflow must fail this task with an error.
Tasks can also output to a file or stdout/stderr an Array
, Map
, or Object
data structure in a two major formats:
- JSON - because it fits naturally with the types within WDL
- Text based / TSV - These are usually simple table and text-based encodings (e.g.
Array[String]
could be serialized by having each element be a line in a file)
Maps are deserialized from:
- Files that contain a JSON Array as their top-level element.
- Any file where it is desirable to interpret each line as an element of the
Array
.
read_lines()
will return an Array[String]
where each element in the array is a line in the file.
This return value can be auto converted to other Array
types. For example:
task test {
command <<<
python <<CODE
import random
for i in range(10):
print(random.randrange(10))
CODE
>>>
output {
Array[Int] my_ints = read_lines(stdout())
}
}
my_ints
would contain ten random integers ranging from 0 to 10.
read_json()
will return whatever data type resides in that JSON file
task test {
command <<<
echo '["foo", "bar"]'
>>>
output {
Array[String] my_array = read_json(stdout())
}
}
This task would assign the array with elements "foo"
and "bar"
to my_array
.
If the echo statement was instead echo '{"foo": "bar"}'
, the engine MUST fail the task for a type mismatch.
Maps are deserialized from:
- Files that contain a JSON Object as their top-level element.
- Files that contain a two-column TSV file.
read_map()
will return an Map[String, String]
where the keys are the first column in the TSV input file and the corresponding values are the second column.
This return value can be auto converted to other Map
types. For example:
task test {
command <<<
python <<CODE
for i in range(3):
print("key_{idx}\t{idx}".format(idx=i)
CODE
>>>
output {
Map[String, Int] my_ints = read_map(stdout())
}
}
This would put a map containing three keys (key_0
, key_1
, and key_2
) and three respective values (0
, 1
, and 2
) as the value of my_ints
read_json()
will return whatever data type resides in that JSON file. If that file contains a JSON object with homogeneous key/value pair types (e.g. string -> int
pairs), then the read_json()
function would return a Map
.
task test {
command <<<
echo '{"foo":"bar"}'
>>>
output {
Map[String, String] my_map = read_json(stdout())
}
}
This task would assign the one key-value pair map in the echo statement to my_map
.
If the echo statement was instead echo '["foo", "bar"]'
, the engine MUST fail the task for a type mismatch.
Objects are deserialized from files that contain a two-row, n-column TSV file. The first row are the object attribute names and the corresponding entries on the second row are the values.
read_object()
will return an Object
where the keys are the first row in the TSV input file and the corresponding values are the second row (corresponding column).
task test {
command <<<
python <<CODE
print('\t'.join(["key_{}".format(i) for i in range(3)]))
print('\t'.join(["value_{}".format(i) for i in range(3)]))
CODE
>>>
output {
Object my_obj = read_object(stdout())
}
}
This would put an object containing three attributes (key_0
, key_1
, and key_2
) and three respective values (value_0
, value_1
, and value_2
) as the value of my_obj
Array[Object]
MUST assume that all objects in the array are homogeneous (they have the same attributes, but the attributes don't have to have the same values)
An Array[Object]
is deserialized from files that contains at least 2 rows and a uniform n-column TSV file. The first row are the object attribute names and the corresponding entries on the subsequent rows are the values
read_object()
will return an Object
where the keys are the first row in the TSV input file and the corresponding values are the second row (corresponding column).
task test {
command <<<
python <<CODE
print('\t'.join(["key_{}".format(i) for i in range(3)]))
print('\t'.join(["value_{}".format(i) for i in range(3)]))
print('\t'.join(["value_{}".format(i) for i in range(3)]))
print('\t'.join(["value_{}".format(i) for i in range(3)]))
CODE
>>>
output {
Array[Object] my_obj = read_objects(stdout())
}
}
This would create an array of three identical Object
s containing three attributes (key_0
, key_1
, and key_2
) and three respective values (value_0
, value_1
, and value_2
) as the value of my_obj