Preparing for strict syntax
This page explains how to update Nextflow scripts and config files to adhere to the Nextflow language specification, also known as the strict syntax.
If you are still using DSL1, see Migrating from DSL1 to learn how to migrate your Nextflow pipelines to DSL2 before consulting this guide.
To enable strict syntax, set the NXF_SYNTAX_PARSER environment variable to v2:
export NXF_SYNTAX_PARSER=v2
The strict syntax is enabled by default. You can disable it by setting the environment variable NXF_SYNTAX_PARSER=v1.
Overview
The strict syntax is a subset of DSL2. While DSL2 allows any Groovy syntax, the strict syntax allows only a subset of Groovy syntax for Nextflow scripts and config files. This new specification enables more specific error reporting, ensures more consistent code, and will allow the Nextflow language to evolve independently of Groovy.
The strict syntax will eventually become the only way to write Nextflow code, and new language features will be implemented only in the strict syntax, with few exceptions. Therefore, it is important to prepare for the strict syntax in order to maintain compatibility with future versions of Nextflow and be able to use new language features.
This page outlines the key differences between DSL2 and the strict syntax. The extent of required changes will vary depending on the amount of custom Groovy code used within your scripts and config files.
Removed syntax
Import declarations
In Groovy, the import declaration can be used to import external classes:
import groovy.json.JsonSlurper
def json = new JsonSlurper().parseText(json_file.text)
In the strict syntax, use the fully qualified name to reference the class:
def json = new groovy.json.JsonSlurper().parseText(json_file.text)
Class declarations
Some users use classes in Nextflow to define helper functions or custom types. Helper functions should be defined as standalone functions in Nextflow. Custom types should be moved to the lib directory.
Enums, a special type of class, are supported, but they cannot be included across modules at this time.
Record types will be addressed in a future version of the Nextflow language specification.
Mixing script declarations and statements
In the strict syntax, a script may contain any of the following top-level declarations:
- Feature flags
- Include declarations
- Parameter declarations
- Workflows
- Processes
- Functions
- Output block
Alternatively, a script may contain only statements, also known as a code snippet:
println 'Hello world!'
Code snippets are treated as an implicit entry workflow:
workflow {
println 'Hello world!'
}
Script declarations and statements cannot be mixed at the same level. All statements must reside within script declarations unless the script is a code snippet:
process hello {
// ...
}
// incorrect -- move into entry workflow
// println 'Hello world!'
// correct
workflow {
println 'Hello world!'
}
Mixing statements and script declarations was necessary in DSL1 and optional in DSL2. However, it is no longer supported in the strict syntax in order to simplify the language and to ensure that top-level statements are not executed when the script is included as a module.
Assignment expressions
In Groovy, variables can be assigned in an expression:
hello(x = 1, y = 2)
In the strict syntax, assignments are allowed only as statements:
x = 1
y = 2
hello(x, y)
In Groovy, variables can be incremented and decremented in an expression:
hello(x++, y--)
In the strict syntax, use += and -= instead:
x += 1
y -= 1
hello(x, y)
For and while loops
In Groovy, loop statements, such as for and while, are supported:
for (rseqc_module in ['read_distribution', 'inner_distance', 'tin']) {
if (rseqc_modules.contains(rseqc_module))
rseqc_modules.remove(rseqc_module)
}
In the strict syntax, use higher-order functions, such as the each method, instead:
['read_distribution', 'inner_distance', 'tin'].each { rseqc_module ->
if (rseqc_modules.contains(rseqc_module))
rseqc_modules.remove(rseqc_module)
}
Lists, maps, and sets provide several functions (e.g., collect, find, findAll, inject) for iteration. See Groovy standard library for more information.
Switch statements
In Groovy, switch statements are used for pattern matching on a value:
switch (aligner) {
case 'bowtie2':
// ...
break
case 'bwamem':
// ...
break
case 'dragmap':
// ...
break
case 'snap':
// ...
break
default:
// ...
}
In the strict syntax, use if-else statements instead:
if (aligner == 'bowtie2') {
// ...
} else if (aligner == 'bwamem') {
// ...
} else if (aligner == 'dragmap') {
// ...
} else if (aligner == 'snap') {
// ...
} else {
// ...
}
Spread operator
In Groovy, the spread operator can be used to flatten a nested list:
ch.map { meta, bambai -> [meta, *bambai] }
In the strict syntax, enumerate the list elements explicitly:
// alternative 1
ch.map { meta, bambai -> [meta, bambai[0], bambai[1]] }
// alternative 2
ch.map { meta, bambai ->
def (bam, bai) = bambai
[meta, bam, bai]
}
Implicit environment variables
In Nextflow DSL1 and DSL2, environment variables can be referenced directly in strings:
println "PWD = ${PWD}"
In the strict syntax, use System.getenv() instead:
println "PWD = ${System.getenv('PWD')}"
The env() function should be used instead of System.getenv():
println "PWD = ${env('PWD')}"
Restricted syntax
The following patterns are still supported but have been restricted. That is, some syntax variants have been removed. In Nextflow DSL2, include declarations can have an In the strict syntax, these clauses are no longer supported. Params should be passed to workflows, processes, and functions as explicit inputs: Where the In Groovy, variables can be declared in many different ways: In the strict syntax, variables must be declared with < AddedInVersion version="25.10.0" /> Local variables can be declared with a type annotation: Groovy-style type annotations are still supported. However, the language server and Groovy supports a wide variety of strings, including multi-line strings, dynamic strings, slashy strings, multi-line dynamic slashy strings, and more. The strict syntax supports single- and double-quoted strings, multi-line strings, and slashy strings. Slashy strings cannot be interpolated: Use a double-quoted string instead: Slashy strings cannot span multiple lines: Use a multi-line string instead: Dollar slashy strings are not supported: Use a multi-line string instead: In Groovy, there are two ways to perform type conversions or casts: In the strict syntax, only hard casts are supported. Use an explicit method to cast a value to a different type if one is available. For example, to parse a string as a number: In Nextflow DSL2, the name of a process In the strict syntax, the name must be specified with quotes: In Nextflow DSL1 and DSL2, the process In the strict syntax, the Workflow handlers (i.e. The strict syntax does not allow statements to be mixed with script declarations, so workflow handlers must be defined in the entry workflow: Workflow handlers can be specified as sections in the entry workflow: See Workflow handlers for details.Include declarations
addParams or params clause as described in Module parameters:params.message = 'Hola'
params.target = 'Mundo'
include { sayHello } from './some/module' addParams(message: 'Ciao')
workflow {
sayHello()
}include { sayHello } from './some/module'
params.message = 'Hola'
params.target = 'Mundo'
workflow {
sayHello('Ciao', params.target)
}sayHello workflow is defined as follows:workflow sayHello {
take:
message
target
main:
// ...
}Variable declarations
def a = 1
final b = 2
def c = 3, d = 4
def (e, f) = [5, 6]
String str = 'hello'
def Map meta = [:]def and must not specify a type:def a = 1
def b = 2
def (c, d) = [3, 4]
def (e, f) = [5, 6]
def str = 'hello'
def meta = [:]def a: Integer = 1
def b: Integer = 2
def (c: Integer, d: Integer) = [3, 4]
def (e: Integer, f: Integer) = [5, 6]
def str: String = 'hello'
def meta: Map = [:]nextflow lint will automatically convert them to Nextflow-style type annotations when formatting code. Groovy-style type annotations will not be supported in a future version.Strings
def id = 'SRA001'
assert 'SRA001.fastq' ~= /${id}\.f(?:ast)?q/def id = 'SRA001'
assert 'SRA001.fastq' ~= "${id}\\.f(?:ast)?q"/
Patterns in the code,
Symbols dance to match and find,
Logic unconfined.
/"""
Patterns in the code,
Symbols dance to match and find,
Logic unconfined.
"""$/
echo "Hello world!"
/$"""
echo "Hello world!"
"""Type conversions
def map = (Map) readJson(json) // soft cast
def map = readJson(json) as Map // hard castdef x = '42' as Integer
def x = '42'.toInteger() // preferredProcess env inputs and outputs
env input/output can be specified with or without quotes:process my_task {
input:
env FOO
env 'BAR'
// ...
}process my_task {
input:
env 'FOO'
env 'BAR'
// ...
}Implicit process script section
script: section label can almost always be omitted:process greet {
input:
val greeting
"""
echo '${greeting}!'
"""
}script: label can be omitted only if there are no other sections:process hello {
"""
echo 'Hello world!'
"""
}
process greet {
input:
val greeting
script:
"""
echo '${greeting}!'
"""
}Workflow onComplete/onError handlers
workflow.onComplete and workflow.onError) can be defined in several different ways in a script, but are typically defined as top-level statements and without an equals sign:workflow.onComplete {
println "Pipeline completed at: $workflow.complete"
println "Execution status: ${ workflow.success ? 'OK' : 'failed' }"
}workflow {
// ...
workflow.onComplete = {
println "Pipeline completed at: $workflow.complete"
println "Execution status: ${ workflow.success ? 'OK' : 'failed' }"
}
}workflow {
main:
// ...
onComplete:
println "Pipeline completed at: $workflow.complete"
println "Execution status: ${ workflow.success ? 'OK' : 'failed' }"
}
Deprecated syntax
The following patterns are deprecated, and the strict syntax reports warnings for them. These warnings will become errors in the future. Channel factories should be accessed using the See channel and [Channel<E>stdlib-types-channel for more information. In Groovy, a closure with no parameters is assumed to have a single parameter named In the strict syntax, the closure parameter should be explicitly declared: The process channel vs Channelchannel namespace instead of the Channel type:Channel.of(1, 2, 3) // incorrect
channel.of(1, 2, 3) // correctImplicit closure parameter
it:ch.map { it * 2 }ch.map { v -> v * 2 } // correct
ch.map { it -> it * 2 } // also correctProcess shell section
shell section is deprecated. Use the script section instead. The strict syntax provides error checking to help distinguish between Nextflow variables and Bash variables.
Best practices
The following patterns are discouraged and may become warnings or errors in future Nextflow versions. The language server can detect these patterns, but does not report them by default.
To enable these checks, set Nextflow > Error reporting mode to paranoid in the extension settings. While params can be used anywhere in the pipeline code, they are only intended to be used in the entry workflow and the As a best practice, processes and workflows should receive params as explicit inputs: The process when section is discouraged. As a best practice, conditional logic should be implemented in the calling workflow (e.g. using an Using params outside the entry workflow
output block.process myproc {
input:
val myproc_args
// ...
}
workflow myflow {
take:
myflow_args
// ...
}
workflow {
myproc(params.myproc_args)
myflow(params.myflow_args)
}Process when section
if statement or filter operator) instead of the process definition.
Configuration syntax
To enable the strict syntax, set the NXF_SYNTAX_PARSER environment variable to v2:
export NXF_SYNTAX_PARSER=v2
See Configuration for a comprehensive description of the configuration language.
Currently, Nextflow parses config files as Groovy scripts, allowing the use of scripting constructs like variables, helper functions, try-catch blocks, and conditional logic for dynamic configuration:
def getHostname() {
// ...
}
def hostname = getHostname()
if (hostname == 'small') {
params.max_memory = 32.GB
params.max_cpus = 8
}
else if (hostname == 'large') {
params.max_memory = 128.GB
params.max_cpus = 32
}
The strict config syntax does not support functions, and only allows statements (e.g., variables and if statements) within closures. The same dynamic configuration can be achieved by using a dynamic include:
includeConfig ({
def hostname = // ...
if (hostname == 'small')
return 'small.config'
else if (hostname == 'large')
return 'large.config'
else
return '/dev/null'
}())
The include source is a closure that is immediately invoked. It includes a different config file based on the return value of the closure. Including /dev/null is equivalent to including nothing.
Each conditional configuration is defined in a separate config file:
// small.config
params.max_memory = 32.GB
params.max_cpus = 8
// large.config
params.max_memory = 128.GB
params.max_cpus = 32
Preserving Groovy code
There are two ways to preserve Groovy code:
- Move the code to the
libdirectory - Create a plugin
Any Groovy code can be moved into the lib directory, which supports the full Groovy language. This approach is useful for temporarily preserving some Groovy code until it can be updated later and incorporated into a Nextflow script. See The lib directory for more information.
For Groovy code that is complicated or if it depends on third-party libraries, it may be better to create a plugin. Plugins can define custom functions that can be included by Nextflow scripts like a module. Furthermore, plugins can be easily re-used across different pipelines. See Developing plugins for more information on how to develop plugins.