Processes
In Nextflow, a process is a specialized function for executing scripts in a scalable and portable manner.
Here is an example process definition:
process hello {
    output:
    path 'hello.txt'
    script:
    """
    echo 'Hello world!' > hello.txt
    """
}
See Process for a full description of the process syntax.
Script
The script section defines, as a string expression, the script that is executed by the process.
A process may contain only one script, and if the script guard is not explicitly declared, the script must be the final statement in the process definition.
The script string is executed as a Bash script in the host environment. It can be any command or script that you would normally execute on the command line or in a Bash script. Naturally, the script may only use commands that are available in the host environment.
The script section can be a simple string or a multi-line string. The latter approach makes it easier to write scripts with multiple commands spanning multiple lines. For example:
process blast {
  """
  blastp -db $db -query query.fa -outfmt 6 > blast_result
  cat blast_result | head -n 10 | cut -f 2 > top_hits
  blastdbcmd -db $db -entry_batch top_hits > sequences
  """
}
As explained in the script tutorial section, strings can be defined using single-quotes or double-quotes, and multi-line strings are defined by three single-quote or three double-quote characters.
There is a subtle but important difference between them. Like in Bash, strings delimited by a " character support variable substitutions, while strings delimited by ' do not.
In the above code fragment, the $db variable is replaced by the actual value defined elsewhere in the pipeline script.
Warning
Since Nextflow uses the same Bash syntax for variable substitutions in strings, you must manage them carefully depending on whether you want to evaluate a Nextflow variable or a Bash variable.
When you need to access a system environment variable in your script, you have two options.
If you don’t need to access any Nextflow variables, you can define your script section with single-quotes:
process echo_path {
  '''
  echo "The path is: $PATH"
  '''
}
Otherwise, you can define your script with double-quotes and escape the system environment variables by prefixing them with a back-slash \ character, as shown in the following example:
process blast {
  """
  blastp -db \$DB -query query.fa -outfmt 6 > blast_result
  cat blast_result | head -n $MAX | cut -f 2 > top_hits
  blastdbcmd -db \$DB -entry_batch top_hits > sequences
  """
}
In this example, $MAX is a Nextflow variable that must be defined elsewhere in the pipeline script. Nextflow replaces it with the actual value before executing the script. Meanwhile, $DB is a Bash variable that must exist in the execution environment, and Bash will replace it with the actual value during execution.
Scripts à la carte
The process script is interpreted by Nextflow as a Bash script by default, but you are not limited to Bash.
You can use your favourite scripting language (Perl, Python, R, etc), or even mix them in the same pipeline.
A pipeline may be composed of processes that execute very different tasks. With Nextflow, you can choose the scripting language that best fits the task performed by a given process. For example, for some processes R might be more useful than Perl, whereas for others you may need to use Python because it provides better access to a library or an API, etc.
To use a language other than Bash, simply start your process script with the corresponding shebang. For example:
process perl_task {
    """
    #!/usr/bin/perl
    print 'Hi there!' . '\n';
    """
}
process python_task {
    """
    #!/usr/bin/python
    x = 'Hello'
    y = 'world!'
    print "%s - %s" % (x,y)
    """
}
workflow {
    perl_task()
    python_task()
}
Tip
Since the actual location of the interpreter binary file can differ across platforms, it is wise to use the env command followed by the interpreter name, e.g. #!/usr/bin/env perl, instead of the absolute path, in order to make your script more portable.
Conditional scripts
The script section is like a function that returns a string. This means that you can write arbitrary code to determine the script, as long as the final statement is a string.
If-else statements based on task inputs can be used to produce a different script. For example:
mode = 'tcoffee'
process align {
    input:
    path sequences
    script:
    if( mode == 'tcoffee' )
        """
        t_coffee -in $sequences > out_file
        """
    else if( mode == 'mafft' )
        """
        mafft --anysymbol --parttree --quiet $sequences > out_file
        """
    else if( mode == 'clustalo' )
        """
        clustalo -i $sequences -o out_file
        """
    else
        error "Invalid alignment mode: ${mode}"
}
In the above example, the process will execute one of several scripts depending on the value of the mode parameter. By default it will execute the tcoffee command.
Template
Process scripts can be externalized to template files, which allows them to be reused across different processes and tested independently from the pipeline execution.
A template can be used in place of an embedded script using the template function in the script section:
process hello {
    input:
    val STR
    script:
    template 'hello.sh'
}
workflow {
    channel.of('this', 'that') | hello
}
By default, Nextflow looks for the template script in the templates directory located alongside the Nextflow script in which the process is defined. An absolute path can be used to specify a different location. However, this practice is discouraged because it hinders pipeline portability.
An example template script is provided below:
#!/bin/bash
echo "process started at `date`"
echo $STR
echo "process completed"
Variables prefixed with the dollar character ($) are interpreted as Nextflow variables when the template script is executed by Nextflow and Bash variables when executed directly. For example, the above script can be executed from the command line by providing each input as an environment variable:
STR='Hello!' bash templates/my_script.sh
The following caveats should be considered:
- Template scripts are recommended only for Bash scripts. Languages that do not prefix variables with - $(e.g. Python and R) can’t be executed directly as a template script.
- Variables escaped with - \$will be interpreted as Bash variables when executed by Nextflow, but will not be interpreted as variables when executed from the command line. This practice should be avoided to ensure that the template script behaves consistently.
- Template variables are evaluated even if they are commented out in the template script. If a template variable is missing, it will cause the pipeline to fail regardless of where it occurs in the template. 
Tip
Template scripts are generally discouraged due to the caveats described above. The best practice for using a custom script is to embed it in the process definition at first and move it to a separate file with its own command line interface once the code matures.
Shell
Deprecated since version 24.11.0-edge: Use the script section instead. Consider using the strict syntax, which provides error checking to help distinguish between Nextflow variables and Bash variables in the process script.
The shell section is a string expression that defines the script that is executed by the process. It is an alternative to the Script definition with one important difference: it uses the exclamation mark ! character, instead of the usual dollar $ character, to denote Nextflow variables.
This way, it is possible to use both Nextflow and Bash variables in the same script without having to escape the latter, which makes process scripts easier to read and maintain. For example:
process hello {
    input:
    val str
    shell:
    '''
    echo "User $USER says !{str}"
    '''
}
workflow {
    channel.of('Hello', 'Hola', 'Bonjour') | hello
}
In the above example, $USER is treated as a Bash variable, while !{str} is treated as a Nextflow variable.
Note
- Shell script definitions require the use of single-quote - 'delimited strings. When using double-quote- "delimited strings, dollar variables are interpreted as Nextflow variables as usual. See String interpolation.
- Variables prefixed with - !must always be enclosed in curly brackets, i.e.- !{str}is a valid variable whereas- !stris ignored.
- Shell scripts support the use of the Template mechanism. The same rules are applied to the variables defined in the template script. 
Native execution
The exec section executes the given code without launching a job.
For example:
process hello {
    input:
    val name
    exec:
    println "Hello Mr. $name"
}
workflow {
    channel.of('a', 'b', 'c') | hello
}
will display:
Hello Mr. b
Hello Mr. a
Hello Mr. c
A native process is very similar to a function. However, it provides additional capabilities such as parallelism, caching, and progress logging.
Stub
New in version 20.11.0-edge.
You can define a command stub, which replaces the actual process command when the -stub-run or -stub command-line option is enabled:
process salmon_index {
    input:
    path transcriptome
    output:
    path 'index'
    script:
    """
    salmon index --threads $task.cpus -t $transcriptome -i index
    """
    stub:
    """
    mkdir index
    touch index/seq.bin
    touch index/info.json
    touch index/refseq.bin
    """
}
The stub section can be defined before or after the script section. When the pipeline is executed with the -stub-run option and a process’s stub is not defined, the script section is executed.
This feature makes it easier to quickly prototype the workflow logic without using the real commands. The developer can use it to provide a dummy script that mimics the execution of the real one in a quicker manner. In other words, it is a way to perform a dry-run.
Inputs
The input section defines the input of a process, similar to function arguments. A process may have at most one input section, which must contain at least one input declaration.
The input section follows the syntax shown below:
input:
  <input qualifier> <input name>
An input declaration consists of a qualifier and a name. The input qualifier defines the type of data to be received. This information is used by Nextflow to apply the semantic rules associated with each qualifier, and handle it properly depending on the target execution platform (grid, cloud, etc).
When a process is invoked in a workflow, it must be provided a channel or dataflow value for each input in the process input section, similar to calling a function with specific arguments. The examples provided in the following sections demonstrate how a process is invoked.
The following input qualifiers are available:
- val: Access the input value by name in the process script.
- path: Handle the input value as a path, staging the file properly in the execution context.
- env: Use the input value to set an environment variable in the process script.
- stdin: Forward the input value to the process- stdinspecial file.
- tuple: Handle a group of input values having any of the above qualifiers.
- each: Execute the process for each element in the input collection.
See process reference for the full list of input methods and options.
Input variables (val)
The val qualifier accepts any data type. It can be accessed in the process script by using the specified input name, as shown in the following example:
process echo {
  input:
  val x
  script:
  """
  echo "process job $x"
  """
}
workflow {
  def num = channel.of(1,2,3)
  echo(num)
}
In the above example, the process is executed three times: once for each value emitted by the num channel. The resulting output is similar to the one shown below:
process job 3
process job 1
process job 2
Note
Processes do not necessarily process items in the order that they are received. In the above example, the value 3 was processed before the others.
Note
When the process declares exactly one input, the pipe | operator can be used to provide inputs to the process, instead of passing it as a parameter. Both methods have identical semantics:
process echo {
  input:
  val x
  script:
  """
  echo "process job $x"
  """
}
workflow {
  channel.of(1,2,3) | echo
}
Input files (path)
The path qualifier allows you to provide input files to the process execution context. Nextflow will stage the files into the process execution directory, and they can be accessed in the script by using the specified input name. For example:
process blast {
  input:
  path query_file
  script:
  """
  blastp -query ${query_file} -db nr
  """
}
workflow {
  def proteins = channel.fromPath( '/some/path/*.fa' )
  blast(proteins)
}
In the above example, all the files ending with the suffix .fa are sent over the channel proteins. These files are received by the process, which executes a BLAST query on each of them.
It’s worth noting that in the above example, the name of the file in the file-system is not used. You can access the file without even knowing its name, because you can reference it in the process script by the input name.
There may be cases where your task needs to use a file whose name is fixed, i.e. it does not have to change along with the actual provided file. In this case, you can specify a fixed name with the name attribute in the input file parameter definition, as shown in the following example:
input:
path query_file, name: 'query.fa'
or, using a shorter syntax:
input:
path 'query.fa'
The previous example can be re-written as shown below:
process blast {
  input:
  path 'query.fa'
  script:
  """
  blastp -query query.fa -db nr
  """
}
workflow {
  def proteins = channel.fromPath( '/some/path/*.fa' )
  blast(proteins)
}
In this example, each file received by the process is staged with the name query.fa in a different execution context (i.e. the folder where a task is executed).
Tip
This feature allows you to execute the process command multiple times without worrying about the file names changing. In other words, Nextflow helps you write pipeline tasks that are self-contained and decoupled from the execution environment. As a best practice, you should avoid referencing files in your process script other than those defined in your input section.
Channel factories like channel.fromPath produce file objects, but a path input can also accept a string literal path. The string value should be an absolute path, i.e. it must be prefixed with a / character or a supported URI protocol (file://, http://, s3://, etc), and it cannot contain special characters (\n, etc).
process cat {
  input:
  path x
  script:
  """
  cat $x
  """
}
workflow {
  cat('/some/data/file.txt')
}
Note
Process path inputs have nearly the same interface as described in Path, with one difference which is relevant when files are staged into a subdirectory. Given the following input:
path x, name: 'my-dir/file.txt'
In this case, x.name returns the file name with the parent directory (e.g. my-dir/file.txt), whereas normally it would return the file name (e.g. file.txt). You can use x.fileName.name to get the file name.
Multiple input files
A path input can also accept a collection of files instead of a single value. In this case, the input variable will be a list.
When the input has a fixed file name and a collection of files is received by the process, the file name will be appended with a numerical suffix representing its ordinal position in the list. For example:
process blast {
    input:
    path 'seq'
    script:
    """
    echo seq*
    """
}
workflow {
    def fasta = channel.fromPath( "/some/path/*.fa" ).buffer(size: 3)
    blast(fasta)
}
will output:
seq1 seq2 seq3
seq1 seq2 seq3
...
The target input file name may contain the * and ? wildcards, which can be used to control the name of staged files. The following table shows how the wildcards are replaced depending on the cardinality of the received input collection.
| Arity | Name pattern | Staged file names | 
|---|---|---|
| any | 
 | named as the source file | 
| one | 
 | 
 | 
| one | 
 | 
 | 
| one | 
 | 
 | 
| many | 
 | 
 | 
| many | 
 | 
 | 
| many | 
 | 
 | 
| many | 
 | named as the source file, created in  | 
| many | 
 | named as the source file, created in a progressively indexed subdirectory e.g.  | 
| many | 
 | (as above) | 
The following example shows how a wildcard can be used in the input file definition:
process blast {
    input:
    path 'seq?.fa'
    script:
    """
    cat seq1.fa seq2.fa seq3.fa
    """
}
workflow {
    def fasta = channel.fromPath( "/some/path/*.fa" ).buffer(size: 3)
    blast(fasta)
}
Note
Rewriting input file names according to a named pattern is an extra feature and not at all required. The normal file input syntax introduced in the Input files (path) section is valid for collections of multiple files as well. To handle multiple input files while preserving the original file names, use a variable identifier or the * wildcard.
New in version 23.09.0-edge.
The arity option can be used to enforce the expected number of files, either as a number or a range.
For example:
input:
path('one.txt', arity: '1')         // exactly one file is expected
path('pair_*.txt', arity: '2')      // exactly two files are expected
path('many_*.txt', arity: '1..*')   // one or more files are expected
When a task is executed, Nextflow will check whether the received files for each path input match the declared arity, and fail if they do not. When the arity is '1', the corresponding input variable will be a single file; otherwise, it will be a list of files.
Dynamic input file names
When the input file name is specified by using the name option or a string literal, you can also use other input values as variables in the file name string. For example:
process grep {
  input:
  val x
  path "${x}.fa"
  script:
  """
  cat ${x}.fa | grep '>'
  """
}
In the above example, the input file name is determined by the current value of the x input value.
This approach allows input files to be staged in the task directory with a name that is coherent with the current execution context.
Tip
In most cases, you won’t need to use dynamic file names, because each task is executed in its own directory, and input files are automatically staged into this directory by Nextflow. This behavior guarantees that input files with the same name won’t overwrite each other. The above example is useful specifically when there are potential file name conflicts within a single task.
Input environment variables (env)
The env qualifier allows you to define an environment variable in the process execution context based on the input value. For example:
process echo_env {
    input:
    env 'HELLO'
    script:
    '''
    echo "$HELLO world!"
    '''
}
workflow {
    channel.of('hello', 'hola', 'bonjour', 'ciao') | echo_env
}
hello world!
ciao world!
bonjour world!
hola world!
Standard input (stdin)
The stdin qualifier allows you to forward the input value to the standard input of the process script. For example:
process cat {
  input:
  stdin
  script:
  """
  cat -
  """
}
workflow {
  channel.of('hello', 'hola', 'bonjour', 'ciao')
    | map { v -> v + '\n' }
    | cat
}
will output:
hola
bonjour
ciao
hello
Input tuples (tuple)
The tuple qualifier groups multiple values into a single input definition. Each element in the tuple is associated with a corresponding element in the tuple definition. For example:
process cat {
    input:
    tuple val(id), path('input.txt')
    script:
    """
    echo "Processing $id"
    cat input.txt > copy
    """
}
workflow {
  channel.of( [1, 'alpha.txt'], [2, 'beta.txt'], [3, 'delta.txt'] ) | cat
}
In the above example, the tuple input consists of the value x and the file input.txt.
A tuple definition may contain any of the following qualifiers, as previously described: val, env, path and stdin. Files specified with the path qualifier are treated exactly the same as standalone path inputs.
Input repeaters (each)
The each qualifier allows you to repeat the execution of a process for each item in a collection, each time a new value is received. For example:
process align {
  input:
  path seq
  each mode
  script:
  """
  t_coffee -in $seq -mode $mode > result
  """
}
workflow {
  sequences = channel.fromPath('*.fa')
  methods = ['regular', 'espresso', 'psicoffee']
  align(sequences, methods)
}
In the above example, each time a file of sequences is emitted from the sequences channel, the process executes three tasks, each running a T-coffee alignment with a different value for the mode parameter. This behavior is useful when you need to repeat the same task over a given set of parameters.
Input repeaters can be applied to files as well. For example:
process align {
  input:
  path seq
  each mode
  each path(lib)
  script:
  """
  t_coffee -in $seq -mode $mode -lib $lib > result
  """
}
workflow {
  sequences = channel.fromPath('*.fa')
  methods = ['regular', 'espresso']
  libraries = [ file('PQ001.lib'), file('PQ002.lib'), file('PQ003.lib') ]
  align(sequences, methods, libraries)
}
In the above example, each sequence input file emitted by the sequences channel triggers six alignment tasks, three with the regular method against each library file, and three with the espresso method.
Note
When multiple repeaters are defined, the process is executed for each combination of them.
Multiple inputs
A process can declare multiple inputs, which allows it to accept inputs from multiple dataflow sources.
Warning
Do not supply more than one channel when calling a process with multiple inputs. Invoking a process with multiple channels can lead to non-deterministic behavior. All additional inputs should be dataflow values.
When a process is defined with multiple inputs, it waits for a value from each input and launches a new task with the combined values. When one of the inputs is a channel, the process repeats until all values in the channel are consumed. If the channel is empty, the process will not launch any tasks.
For example:
process echo {
  input:
  val x
  val y
  script:
  """
  echo $x and $y
  """
}
workflow {
  x = channel.value(1)
  y = channel.of('a', 'b', 'c')
  echo(x, y)
}
The above example executes the echo process three times. The dataflow value x is reused for each value in y. It outputs:
1 and a
1 and b
1 and c
See also: Singleton outputs.
Outputs
The output section defines the outputs of a process, similar to a function return. A process may have at most one output section, which must contain at least one output declaration.
The output section follows the syntax shown below:
output:
  <output qualifier> <output name> [, <option>: <option value>]
An output declaration consists of a qualifier and a name. Some optional attributes can also be specified.
When a process is invoked, each process output is returned as a channel (except under certain conditions). The examples provided in the following sections demonstrate how to access the outputs of a process.
The following output qualifiers are available:
- val: Emit the variable with the specified name.
- path: Emit a file produced by the process with the specified name.
- env: Emit the variable defined in the process environment with the specified name.
- stdout: Emit the- stdoutof the executed process.
- tuple: Emit multiple values.
- eval: Emit the result of a script or command evaluated in the task execution context.
Refer to the process reference for the full list of available output methods and options.
Output variables (val)
The val qualifier allows you to output any Nextflow variable defined in the process. A common use case is to output a variable that was defined in the input section, as shown in the following example:
process echo {
  input:
  each x
  output:
  val x
  script:
  """
  echo $x > file
  """
}
workflow {
  methods = ['prot', 'dna', 'rna']
  received = echo(methods)
  received.view { method -> "Received: $method" }
}
The output value can be a value literal, an input variable, any other Nextflow variable in the process scope, or a value expression. For example:
process cat {
  input:
  path infile
  output:
  val x
  val 'BB11'
  val "${infile.baseName}.out"
  script:
  x = infile.name
  """
  cat $x > file
  """
}
workflow {
  ch_dummy = channel.fromPath('*').first()
  (ch_var, ch_str, ch_exp) = cat(ch_dummy)
  ch_var.view { var -> "ch_var: $var" }
  ch_str.view { str -> "ch_str: $str" }
  ch_exp.view { exp -> "ch_exp: $exp" }
}
Output files (path)
The path qualifier allows you to output one or more files produced by the process. For example:
process random_number {
  output:
  path 'result.txt'
  script:
  '''
  echo $RANDOM > result.txt
  '''
}
workflow {
  numbers = random_number()
  numbers.view { file -> "Received: ${file.text}" }
}
In the above example, the random_number process creates a file named result.txt which contains a random number. Since a path output with the same name is declared, that file is emitted by the corresponding process output. A downstream process or operator with a compatible input will be able to receive it.
Refer to the process reference for the list of available options for path outputs.
Multiple output files
When an output file name contains a * or ? wildcard character, it is interpreted as a glob path matcher. This allows you to capture multiple files into a list and emit the list as a single value. For example:
process split_letters {
    output:
    path 'chunk_*'
    script:
    """
    printf 'Hola' | split -b 1 - chunk_
    """
}
workflow {
    split_letters
        | flatten
        | view { chunk -> "File: ${chunk.name} => ${chunk.text}" }
}
It prints:
File: chunk_aa => H
File: chunk_ab => o
File: chunk_ac => l
File: chunk_ad => a
By default, all the files matching the specified glob pattern are emitted as a single list. However, as the above example demonstrates, the flatten operator can be used to transform the list of files into a channel that emits each file individually.
Some caveats on glob pattern behavior:
- Input files are not included (unless - includeInputsis- true)
- Directories are included, unless the - **pattern is used to recurse through directories
Warning
Although the input files matching an output glob pattern are not included in the resulting output, these files may still be transferred from the task scratch directory to the original task work directory. Therefore, to avoid unnecessary file copies, avoid using loose wildcards when defining output files, e.g. path '*'. Instead, use a prefix or a suffix to restrict the set of matching files to only the expected ones, e.g. path 'prefix_*.sorted.bam'.
Read more about glob syntax at the following link What is a glob?
New in version 23.09.0-edge.
The arity option can be used to enforce the expected number of files, either as a number or a range.
For example:
output:
path('one.txt', arity: '1')         // exactly one file is expected
path('pair_*.txt', arity: '2')      // exactly two files are expected
path('many_*.txt', arity: '1..*')   // one or more files are expected
When a task completes, Nextflow will check whether the produced files for each path output match the declared arity, and fail if they do not. When the arity is '1', the corresponding output will be a single file; otherwise, it will be a list of files.
Dynamic output file names
When an output file name needs to be expressed dynamically, it is possible to define it using a dynamic string which references variables in the input section or in the script global context. For example:
process align {
  input:
  val species
  path seq
  output:
  path "${species}.aln"
  script:
  """
  t_coffee -in $seq > ${species}.aln
  """
}
In the above example, each process execution produces an alignment file whose name depends on the actual value of the species input.
Tip
The management of output files in Nextflow is often misunderstood.
With other tools it is generally necessary to organize the output files into some kind of directory structure or to guarantee a unique file name scheme, so that result files don’t overwrite each other and so they can be referenced unequivocally by downstream tasks.
With Nextflow, in most cases, you don’t need to manage the naming of output files, because each task is executed in its own unique directory, so files produced by different tasks can’t overwrite each other. Also, metadata can be associated with outputs by using the tuple output qualifier, instead of including them in the output file name.
One example in which you’d need to manage the naming of output files is when you use the publishDir directive to have output files also in a specific path of your choice. If two tasks have the same filename for their output and you want them to be in the same path specified by publishDir, the last task to finish will overwrite the output of the task that finished before. You can dynamically change that by adding the saveAs option to your publishDir directive.
To sum up, the use of output files with static names over dynamic ones is preferable whenever possible, because it will result in simpler and more portable code.
Output environment variables (env)
The env qualifier allows you to output a variable defined in the process execution environment:
process seq {
    output:
    env 'RESULT'
    script:
    '''
    RESULT=$(seq 5)
    '''
}
workflow {
    seq | view
}
Standard output (stdout)
The stdout qualifier allows you to output the stdout of the executed process:
process hello {
    output:
    stdout
    script:
    """
    echo "Hello world!"
    """
}
workflow {
    hello | view { message -> "I say... $message" }
}
Eval output (eval)
New in version 24.02.0-edge.
The eval qualifier allows you to capture the standard output of an arbitrary command evaluated the task shell interpreter context:
process hello {
    output:
    eval('echo Hello world!')
    script:
    """
    true
    """
}
workflow {
    hello | view
}
Only one-line Bash commands are supported. You can use a semi-colon ; to specify multiple Bash commands on a single line, and many interpreters can execute arbitrary code on the command line, e.g. python -c 'print("Hello world!")'.
If the command fails, the task will also fail. In Bash, you can append || true to a command to suppress any command failure.
Output tuples (tuple)
The tuple qualifier outputs multiple values in a single output as a tuple. For example:
process blast {
  input:
    val species
    path query
  output:
    tuple val(species), path('result')
  script:
    """
    blast -db nr -query $query > result
    """
}
workflow {
  ch_species = channel.of('human', 'cow', 'horse')
  ch_query = channel.fromPath('*.fa')
  blast(ch_species, ch_query)
}
In the above example, a blast task is executed for each pair of species and query that are received. Each task produces a new tuple containing the value for species and the file result.
A tuple definition may contain any of the following qualifiers, as previously described: val, path, env and stdout. Files specified with the path qualifier are treated exactly the same as standalone path inputs.
Note
While parentheses for input and output qualifiers are generally optional, they are required when specifying elements in an input/output tuple.
Here’s an example with a single path output (parentheses optional):
process hello {
    output:
    path 'result.txt', hidden: true
    script:
    """
    echo 'another new line' >> result.txt
    """
}
And here’s an example with a tuple output (parentheses required):
process hello {
    output:
    tuple path('last_result.txt'), path('result.txt', hidden: true)
    script:
    """
    echo 'another new line' >> result.txt
    echo 'another new line' > last_result.txt
    """
}
Naming outputs
The emit option is used to define the name of a process output, which can be used to access the output by name in the calling workflow. For example:
process hello_bye {
    output:
    path 'hello.txt', emit: hello
    stdout emit: bye
    script:
    """
    echo "hello" > hello.txt
    echo "bye"
    """
}
workflow {
    hello_bye()
    hello_bye.out.hello.view()
}
See Calling processes and workflows for more details.
Optional outputs
Normally, if a specified output is not produced by the task, the task will fail. Setting optional: true will cause the task to not fail, and instead emit nothing to the corresponding output channel.
output:
path("output.txt"), optional: true
In this example, the process is normally expected to produce an output.txt file, but in this case, if the file is missing, the task will not fail. The output channel will only contain values for those tasks that produced output.txt.
Note
While this option can be used with any process output, it cannot be applied to individual elements of a tuple output. The entire tuple must be optional or not optional.
Singleton outputs
When a process is only supplied with dataflow values, regular values, or no inputs, it returns outputs as dataflow values. For example:
process echo {
  input:
  val greeting
  output:
  val greeting
  exec:
  true
}
process greet {
  input:
  val greeting
  val name
  output:
  val "$greeting, $name!"
  exec:
  true
}
workflow {
  names = channel.of( 'World', 'Mundo', 'Welt' )
  greeting = echo('Hello')
  result = greet(greeting, names)
  result.view()
}
In the above example, the echo process is invoked with a regular value that is wrapped in a dataflow value. As a result, echo returns a dataflow value and greet is executed three times.
If the call to echo was changed to echo( channel.of('hello') ), the process would instead return a channel, and greet would be executed only once.
See also: Multiple inputs.
When
Note
As a best practice, conditional logic should be implemented in the calling workflow (e.g. using an if statement or filter operator) instead of the process definition.
The when section allows you to define a condition that must be satisfied in order to execute the process. The condition can be any expression that returns a boolean value.
It can be useful to enable/disable the process execution depending on the state of various inputs and parameters. For example:
process blast_search {
  input:
  path proteins
  val dbtype
  when:
  proteins.name =~ /^BB11.*/ && dbtype == 'nr'
  script:
  """
  blastp -query $proteins -db nr
  """
}
Directives
Directives are optional settings that affect the execution of the current process.
By default, directives are evaluated when the process is defined. However, if the value is a dynamic string or closure, it will be evaluated separately for each task, which allows task-specific variables like task and val inputs to be used.
Some directives are only supported by specific executors. Refer to the Executors page for more information about each executor.
Refer to the process reference for the full list of process directives. If you are new to Nextflow, here are some commonly-used operators to learn first:
General:
- errorStrategy: strategy for handling task failures 
- tag: a semantic name used to differentiate between task executions of the same process 
Resource requirements:
- cpus: the number of CPUs to request for each task 
- memory: the amount of memory to request for each task 
- time: the amount of walltime to request for each task 
Software dependencies:
Using task directive values
The task object also contains the values of all process directives for the given task, which allows you to access these settings at runtime. For examples:
process hello {
  script:
  """
  some_tool --cpus $task.cpus --mem $task.memory
  """
}
In the above snippet, task.cpus and task.memory hold the values for the cpus directive and memory directive directives, respectively, which were resolved for this task based on the process configuration.
Dynamic directives
A directive can be assigned dynamically, during the process execution, so that its actual value can be evaluated based on the process inputs.
To be defined dynamically, the directive’s value needs to be expressed using a closure. For example:
process hello {
  executor 'sge'
  queue { entries > 100 ? 'long' : 'short' }
  input:
  tuple val(entries), path('data.txt')
  script:
  """
  your_command --here
  """
}
In the above example, the queue directive is evaluated dynamically, depending on the input value entries. When it is larger than 100, jobs will be submitted to the long queue, otherwise the short queue will be used.
All directives can be assigned a dynamic value except the following:
New in version 25.10.
Dynamic directives can be specified without a closure when using the strict syntax:
queue (entries > 100 ? 'long' : 'short')
Directives will be evaluated dynamically if they reference task inputs. Process configuration options must still be specified with a closure to be dynamic.
Dynamic task resources
It’s a very common scenario that different instances of the same process may have very different needs in terms of computing resources. In such situations requesting, for example, an amount of memory too low will cause some tasks to fail. Instead, using a higher limit that fits all the tasks in your execution could significantly decrease the execution priority of your jobs.
Dynamic directives can be used to adjust the requested task resources when a task fails and is re-executed. For example:
process hello {
    memory { 2.GB * task.attempt }
    time { 1.hour * task.attempt }
    errorStrategy { task.exitStatus in 137..140 ? 'retry' : 'terminate' }
    maxRetries 3
    script:
    """
    your_command --here
    """
}
In the above example the memory and execution time limits are defined dynamically. The first time the process is executed the task.attempt is set to 1, thus it will request 2 GB of memory and 1 hour of walltime.
If the task execution fails with an exit status between 137 and 140, the task is re-executed; otherwise, the run is terminated immediately. The re-executed task will have task.attempt set to 2, and will request 4 GB of memory and 2 hours of walltime.
The maxRetries directive sets the maximum number of times the same task can be re-executed.
Tip
Directives with named arguments, such as accelerator and disk, must use a more verbose syntax when they are dynamic. For example:
// static request
disk 375.GB, type: 'local-ssd'
// dynamic request
disk { [request: 375.GB * task.attempt, type: 'local-ssd'] }
Task resources can also be defined in terms of task inputs. For example:
process hello {
    memory { 8.GB + 1.GB * Math.ceil(input_file.size() / 1024 ** 3) }
    input:
    path input_file
    script:
    """
    your_command --here
    """
}
In this example, each task requests 8 GB of memory, plus the size of the input file rounded up to the next GB. This way, each task requests only as much memory as it needs based on the size of the inputs. The specific function that you use should be tuned for each process.
Dynamic task resources with previous execution trace
New in version 24.10.0.
Task resource requests can be updated relative to the trace record metrics of the previous task attempt. The metrics can be accessed through the task.previousTrace variable. For example:
process hello {
    memory { task.attempt > 1 ? task.previousTrace.memory * 2 : (1.GB) }
    errorStrategy { task.exitStatus in 137..140 ? 'retry' : 'terminate' }
    maxRetries 3
    script:
    """
    your_command --here
    """
}
In the above example, the memory is set according to previous trace record metrics. In the first attempt, when no trace metrics are available, it is set to 1 GB. In each subsequent attempt, the requested memory is doubled. See Trace file for more information about trace records.
Dynamic retry with backoff
There are cases in which the required execution resources may be temporary unavailable e.g. network congestion. In these cases immediately re-executing the task will likely result in the identical error. A retry with an exponential backoff delay can better recover these error conditions:
process hello {
  errorStrategy { sleep(Math.pow(2, task.attempt) * 200 as long); return 'retry' }
  maxRetries 5
  script:
  """
  your_command --here
  """
}