Skip to content

Commit 7e3868d

Browse files
Script parser (#4613)
--------- Signed-off-by: Ben Sherman <bentshermann@gmail.com> Signed-off-by: Paolo Di Tommaso <paolo.ditommaso@gmail.com> Co-authored-by: Paolo Di Tommaso <paolo.ditommaso@gmail.com>
1 parent 945983d commit 7e3868d

File tree

380 files changed

+58353
-841
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

380 files changed

+58353
-841
lines changed

docs/reference/env-vars.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -185,7 +185,7 @@ The following environment variables control the configuration of the Nextflow ru
185185
`NXF_SYNTAX_PARSER`
186186
: :::{versionadded} 25.02.0-edge
187187
:::
188-
: Set to `'v2'` to use the {ref}`strict syntax <updating-syntax-page>` for Nextflow config files (default: `'v1'`).
188+
: Set to `'v2'` to use the {ref}`strict syntax <updating-syntax-page>` for Nextflow scripts and config files (default: `'v1'`).
189189

190190
`NXF_TEMP`
191191
: Directory where temporary files are stored

docs/updating-syntax.md

+7-3
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,17 @@ This page explains how to update Nextflow scripts and config files to adhere to
88
If you are still using DSL1, see {ref}`dsl1-page` to learn how to migrate your Nextflow pipelines to DSL2 before consulting this guide.
99
:::
1010

11+
(strict-syntax)=
12+
1113
## Preparing for strict syntax
1214

13-
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.
15+
:::{versionadded} 25.02.0-edge
16+
The strict syntax can be enabled in Nextflow by setting `NXF_SYNTAX_PARSER=v2`.
17+
:::
1418

15-
The strict syntax is currently only enforced by the Nextflow language server, which is provided as part of the Nextflow {ref}`vscode-page`. However, the strict syntax will be gradually adopted by the Nextflow CLI in future releases and will eventually be the only way to write Nextflow code.
19+
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.
1620

17-
New language features will be implemented as part of the strict syntax, and not the current DSL2 parser, with few exceptions. Therefore, it is important to prepare for the strict syntax in order to use new language features in the future.
21+
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.
1822

1923
This section describes the key differences between the DSL2 and the strict syntax. In general, the amount of changes that are required depends on the amount of custom Groovy code in your scripts and config files.
2024

docs/vscode.md

+2
Original file line numberDiff line numberDiff line change
@@ -102,6 +102,8 @@ The following settings are available:
102102
`nextflow.paranoidWarnings`
103103
: Enable additional warnings for future deprecations, potential problems, and other discouraged patterns.
104104

105+
(vscode-language-server)=
106+
105107
## Language server
106108

107109
Most of the functionality of the VS Code extension is provided by the [Nextflow language server](https://github.com/nextflow-io/language-server), which implements the [Language Server Protocol (LSP)](https://microsoft.github.io/language-server-protocol/) for Nextflow scripts and config files.

modules/nextflow/src/main/groovy/nextflow/ast/NextflowDSLImpl.groovy

+4-33
Original file line numberDiff line numberDiff line change
@@ -356,44 +356,27 @@ class NextflowDSLImpl implements ASTTransformation {
356356
MethodCallExpression callx
357357
VariableExpression varx
358358

359-
if( (callx=isMethodCallX(stat.expression)) && isThisX(callx.objectExpression) ) {
360-
final name = "_${type}_${callx.methodAsString}"
361-
return stmt( callThisX(name, callx.arguments) )
362-
}
363-
364359
if( (varx=isVariableX(stat.expression)) ) {
365-
final name = "_${type}_${varx.name}"
366-
return stmt( callThisX(name) )
360+
return stmt( callThisX("_${type}_", args(constX(varx.name))) )
367361
}
368362

369363
if( type == WORKFLOW_EMIT ) {
370364
return createAssignX(stat, body, type, uniqueNames)
371365
}
372366

373-
syntaxError(stat, "Workflow malformed parameter definition")
367+
syntaxError(stat, "Invalid workflow ${type}")
374368
return stat
375369
}
376370

377371
protected Statement createAssignX(ExpressionStatement stat, List<Statement> body, String type, Set<String> uniqueNames) {
378372
BinaryExpression binx
379-
MethodCallExpression callx
380-
Expression args=null
381373

382374
if( (binx=isAssignX(stat.expression)) ) {
383375
// keep the statement in body to allow it to be evaluated
384376
body.add(stat)
385377
// and create method call expr to capture the var name in the emission
386378
final left = (VariableExpression)binx.leftExpression
387-
final name = "_${type}_${left.name}"
388-
return stmt( callThisX(name) )
389-
}
390-
391-
if( (callx=isMethodCallX(stat.expression)) && callx.objectExpression.text!='this' && hasTo(callx)) {
392-
// keep the args
393-
args = callx.arguments
394-
// replace the method call expression with a property
395-
stat.expression = new PropertyExpression(callx.objectExpression, callx.method)
396-
// then, fallback to default case
379+
return stmt( callThisX("_${type}_", args(constX(left.name))) )
397380
}
398381

399382
// wrap the expression into a assignment expression
@@ -405,19 +388,7 @@ class NextflowDSLImpl implements ASTTransformation {
405388
body.add(stmt(assign))
406389

407390
// the call method statement for the emit declaration
408-
final name="_${type}_${var}"
409-
callx = args ? callThisX(name, args) : callThisX(name)
410-
return stmt(callx)
411-
}
412-
413-
protected boolean hasTo(MethodCallExpression callX) {
414-
def tupleX = isTupleX(callX.arguments)
415-
if( !tupleX ) return false
416-
if( !tupleX.expressions ) return false
417-
def mapX = isMapX(tupleX.expressions[0])
418-
if( !mapX ) return false
419-
def entry = mapX.getMapEntryExpressions().find { isConstX(it.keyExpression).text=='to' }
420-
return entry != null
391+
return stmt( callThisX("_${type}_", args(constX(var))) )
421392
}
422393

423394
protected String getNextName(Set<String> allNames) {

modules/nextflow/src/main/groovy/nextflow/script/IncludeDef.groovy

+17-10
Original file line numberDiff line numberDiff line change
@@ -16,10 +16,6 @@
1616

1717
package nextflow.script
1818

19-
import nextflow.exception.ScriptCompilationException
20-
import nextflow.plugin.extension.PluginExtensionProvider
21-
import nextflow.plugin.Plugins
22-
2319
import java.nio.file.NoSuchFileException
2420
import java.nio.file.Path
2521

@@ -32,6 +28,10 @@ import groovy.util.logging.Slf4j
3228
import nextflow.NF
3329
import nextflow.Session
3430
import nextflow.exception.IllegalModulePath
31+
import nextflow.exception.ScriptCompilationException
32+
import nextflow.plugin.Plugins
33+
import nextflow.plugin.extension.PluginExtensionProvider
34+
import nextflow.script.parser.v1.ScriptLoaderV1
3535
/**
3636
* Implements a script inclusion
3737
*
@@ -96,11 +96,11 @@ class IncludeDef {
9696
return this
9797
}
9898

99-
/*
100-
* Note: this method invocation is injected during the Nextflow AST manipulation.
101-
* Do not use it explicitly.
99+
/**
100+
* Used internally by the script DSL to include modules
101+
* into a script.
102102
*
103-
* @param ownerParams The params in the owner context
103+
* @param ownerParams The params in the including script context
104104
*/
105105
void load0(ScriptBinding.ParamsMap ownerParams) {
106106
checkValidPath(path)
@@ -109,7 +109,7 @@ class IncludeDef {
109109
return
110110
}
111111
// -- resolve the concrete against the current script
112-
final moduleFile = realModulePath(path)
112+
final moduleFile = realModulePath(path).normalize()
113113
// -- load the module
114114
final moduleScript = loadModule0(moduleFile, resolveParams(ownerParams), session)
115115
// -- add it to the inclusions
@@ -136,10 +136,17 @@ class IncludeDef {
136136
@PackageScope
137137
@Memoized
138138
static BaseScript loadModule0(Path path, Map params, Session session) {
139+
final script = ScriptMeta.getScriptByPath(path)
140+
if( script ) {
141+
script.getBinding().setParams(params)
142+
script.run()
143+
return script
144+
}
145+
139146
final binding = new ScriptBinding() .setParams(params)
140147

141148
// the execution of a library file has as side effect the registration of declared processes
142-
new ScriptParser(session)
149+
new ScriptLoaderV1(session)
143150
.setModule(true)
144151
.setBinding(binding)
145152
.runScript(path)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
/*
2+
* Copyright 2013-2024, Seqera Labs
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package nextflow.script
18+
19+
import java.nio.file.Path
20+
21+
/**
22+
* Interface for parsing and executing a Nextflow script.
23+
*
24+
* @author Ben Sherman <bentshermann@gmail.com>
25+
*/
26+
interface ScriptLoader {
27+
28+
ScriptLoader setEntryName(String name);
29+
30+
ScriptLoader setModule(boolean value);
31+
32+
BaseScript getScript();
33+
34+
Object getResult();
35+
36+
ScriptLoader parse(Path scriptPath);
37+
38+
ScriptLoader runScript();
39+
40+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
/*
2+
* Copyright 2013-2024, Seqera Labs
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*
16+
*/
17+
18+
package nextflow.script
19+
20+
import groovy.transform.CompileStatic
21+
import groovy.util.logging.Slf4j
22+
import nextflow.Session
23+
import nextflow.SysEnv
24+
import nextflow.script.parser.v1.ScriptLoaderV1
25+
import nextflow.script.parser.v2.ScriptLoaderV2
26+
27+
/**
28+
* Factory for creating an instance of {@link ScriptLoader}.
29+
*
30+
* @author Ben Sherman <bentshermann@gmail.com>
31+
*/
32+
@Slf4j
33+
@CompileStatic
34+
class ScriptLoaderFactory {
35+
36+
static ScriptLoader create(Session session) {
37+
final parser = SysEnv.get('NXF_SYNTAX_PARSER', 'v1')
38+
if( parser == 'v1' ) {
39+
return new ScriptLoaderV1(session)
40+
}
41+
if( parser == 'v2' ) {
42+
log.debug "Using script parser v2"
43+
return new ScriptLoaderV2(session)
44+
}
45+
throw new IllegalStateException("Invalid NXF_SYNTAX_PARSER setting -- should be either 'v1' or 'v2'")
46+
}
47+
48+
}

modules/nextflow/src/main/groovy/nextflow/script/ScriptMeta.groovy

+12-6
Original file line numberDiff line numberDiff line change
@@ -48,11 +48,14 @@ class ScriptMeta {
4848

4949
static private Map<BaseScript,ScriptMeta> REGISTRY = new HashMap<>(10)
5050

51+
static private Map<Path,BaseScript> scriptsByPath = new HashMap<>(10)
52+
5153
static private Set<String> resolvedProcessNames = new HashSet<>(20)
5254

5355
@TestOnly
5456
static void reset() {
5557
REGISTRY.clear()
58+
scriptsByPath.clear()
5659
resolvedProcessNames.clear()
5760
}
5861

@@ -61,6 +64,10 @@ class ScriptMeta {
6164
return REGISTRY.get(script)
6265
}
6366

67+
static BaseScript getScriptByPath(Path path) {
68+
return scriptsByPath.get(path)
69+
}
70+
6471
static Set<String> allProcessNames() {
6572
def result = new HashSet()
6673
for( ScriptMeta entry : REGISTRY.values() )
@@ -94,8 +101,8 @@ class ScriptMeta {
94101
get(ExecutionStack.script())
95102
}
96103

97-
/** the script {@link Class} object */
98-
private Class<? extends BaseScript> clazz
104+
/** the script object */
105+
private BaseScript script
99106

100107
/** The location path from where the script has been loaded */
101108
private Path scriptPath
@@ -115,12 +122,12 @@ class ScriptMeta {
115122

116123
Path getModuleDir () { scriptPath?.parent }
117124

118-
String getScriptName() { clazz.getName() }
125+
String getScriptName() { script.getClass().getName() }
119126

120127
boolean isModule() { module }
121128

122129
ScriptMeta(BaseScript script) {
123-
this.clazz = script.class
130+
this.script = script
124131
for( def entry : definedFunctions0(script) ) {
125132
addDefinition(entry)
126133
}
@@ -129,12 +136,11 @@ class ScriptMeta {
129136
/** only for testing */
130137
protected ScriptMeta() {}
131138

132-
@PackageScope
133139
void setScriptPath(Path path) {
140+
scriptsByPath.put(path, script)
134141
scriptPath = path
135142
}
136143

137-
@PackageScope
138144
void setModule(boolean val) {
139145
this.module = val
140146
}

modules/nextflow/src/main/groovy/nextflow/script/ScriptRunner.groovy

+7-7
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@ class ScriptRunner {
4747
/**
4848
* The script interpreter
4949
*/
50-
private ScriptParser scriptParser
50+
private ScriptLoader scriptLoader
5151

5252
/**
5353
* The pipeline file (it may be null when it's provided as string)
@@ -104,7 +104,7 @@ class ScriptRunner {
104104
/**
105105
* @return The interpreted script object
106106
*/
107-
@Deprecated BaseScript getScriptObj() { scriptParser.script }
107+
@Deprecated BaseScript getScriptObj() { scriptLoader.getScript() }
108108

109109
/**
110110
* @return The result produced by the script execution
@@ -225,12 +225,12 @@ class ScriptRunner {
225225
}
226226

227227
protected void parseScript( ScriptFile scriptFile, String entryName ) {
228-
scriptParser = new ScriptParser(session)
228+
scriptLoader = ScriptLoaderFactory.create(session)
229229
.setEntryName(entryName)
230230
// setting module true when running in "inspect" mode to prevent the running the entry workflow
231231
.setModule(ContainerInspectMode.active())
232232
.parse(scriptFile.main)
233-
session.script = scriptParser.script
233+
session.script = scriptLoader.getScript()
234234
}
235235

236236

@@ -241,11 +241,11 @@ class ScriptRunner {
241241
*/
242242
protected run() {
243243
log.debug "> Launching execution"
244-
assert scriptParser, "Missing script instance to run"
244+
assert scriptLoader, "Missing script instance to run"
245245
// -- launch the script execution
246-
scriptParser.runScript()
246+
scriptLoader.runScript()
247247
// -- normalise output
248-
result = normalizeOutput(scriptParser.getResult())
248+
result = normalizeOutput(scriptLoader.getResult())
249249
// -- ignite dataflow network
250250
session.fireDataflowNetwork(preview)
251251
}

0 commit comments

Comments
 (0)