Skip to content

Commit cfe6b1d

Browse files
committed
Fix issue overriding nested params from cli #1923
1 parent 330f044 commit cfe6b1d

File tree

4 files changed

+86
-1
lines changed

4 files changed

+86
-1
lines changed

modules/nextflow/src/main/groovy/nextflow/config/ConfigParser.groovy

+4-1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ import com.google.common.hash.Hashing
2424
import groovy.transform.PackageScope
2525
import nextflow.ast.NextflowXform
2626
import nextflow.exception.ConfigParseException
27+
import nextflow.extension.Bolts
2728
import nextflow.file.FileHelper
2829
import nextflow.util.Duration
2930
import nextflow.util.MemoryUnit
@@ -195,7 +196,9 @@ class ConfigParser {
195196
}
196197

197198
ConfigParser setParams(Map vars) {
198-
this.paramVars = vars
199+
// deep clone the map to prevent side-effect
200+
// see https://github.com/nextflow-io/nextflow/issues/1923
201+
this.paramVars = Bolts.deepClone(vars)
199202
return this
200203
}
201204

modules/nextflow/src/test/groovy/nextflow/config/ConfigBuilderTest.groovy

+28
Original file line numberDiff line numberDiff line change
@@ -1876,7 +1876,35 @@ class ConfigBuilderTest extends Specification {
18761876
[foo:1, bar:[:]] | [bar: [x:10, y:20]] | [foo: 1, bar: [x:10, y:20]]
18771877
[foo:1, bar:[x:1, y:2]] | [bar: [x:10, y:20]] | [foo: 1, bar: [x:10, y:20]]
18781878
[foo:1, bar:[x:1, y:2]] | [foo: 2, bar: [x:10, y:20]] | [foo: 2, bar: [x:10, y:20]]
1879+
}
18791880
1881+
def 'prevent config side effects' () {
1882+
given:
1883+
def folder = Files.createTempDirectory('test')
1884+
and:
1885+
def config = folder.resolve('nf.config')
1886+
config.text = '''\
1887+
params.test.foo = "foo_def"
1888+
params.test.bar = "bar_def"
1889+
'''.stripIndent()
1890+
1891+
when:
1892+
def cfg1 = new ConfigBuilder()
1893+
.setOptions( new CliOptions(userConfig: [config.toString()]))
1894+
.build()
1895+
then:
1896+
cfg1.params.test.foo == "foo_def"
1897+
cfg1.params.test.bar == "bar_def"
1898+
1899+
1900+
when:
1901+
def cfg2 = new ConfigBuilder()
1902+
.setOptions( new CliOptions(userConfig: [config.toString()]))
1903+
.setCmdRun( new CmdRun(params: ['test.foo': 'CLI_FOO'] ))
1904+
.build()
1905+
then:
1906+
cfg2.params.test.foo == "CLI_FOO"
1907+
cfg2.params.test.bar == "bar_def"
18801908
}
18811909
18821910
}

modules/nf-commons/src/main/nextflow/extension/Bolts.groovy

+22
Original file line numberDiff line numberDiff line change
@@ -911,4 +911,26 @@ class Bolts {
911911
else
912912
return self.substring(0,Math.min(self.size()-max, max)) + suffix
913913
}
914+
915+
static <T extends Serializable> T deepClone(T obj) {
916+
final buffer = new ByteArrayOutputStream()
917+
final oos = new ObjectOutputStream(buffer)
918+
oos.writeObject(obj)
919+
oos.flush()
920+
921+
final inputStream = new ByteArrayInputStream(buffer.toByteArray())
922+
return (T) new ObjectInputStream(inputStream).readObject()
923+
}
924+
925+
@CompileDynamic
926+
static <T extends Map> T deepClone(T map) {
927+
final result = map.clone()
928+
for( def key : map.keySet() ) {
929+
def value = map.get(key)
930+
if( value instanceof Map ) {
931+
map.put(key, deepClone(value))
932+
}
933+
}
934+
return (T)result
935+
}
914936
}

modules/nf-commons/src/test/nextflow/extension/BoltsTest.groovy

+32
Original file line numberDiff line numberDiff line change
@@ -333,4 +333,36 @@ class BoltsTest extends Specification {
333333
Bolts.redact('12345', 3, 'xx') == '12xx'
334334
}
335335

336+
def 'should deep clone obj' () {
337+
given:
338+
Map map = [foo: 1, bar: [x: 2, y: 3]]
339+
340+
when:
341+
def copy = Bolts.deepClone((Serializable)map)
342+
then:
343+
copy == map
344+
345+
when:
346+
copy.bar.x = 20
347+
then:
348+
copy.bar.x == 20
349+
map.bar.x == 2
350+
}
351+
352+
def 'should deep clone map' () {
353+
given:
354+
Map map = [foo: 1, bar: [x: 2, y: 3]]
355+
356+
when:
357+
def copy = Bolts.deepClone((Map)map)
358+
then:
359+
copy == map
360+
361+
when:
362+
copy.bar.x = 20
363+
then:
364+
copy.bar.x == 20
365+
map.bar.x == 2
366+
}
367+
336368
}

0 commit comments

Comments
 (0)