Frederick Rueger
Frederick Rueger

Reputation: 113

Simple parallel execution in Jenkins for an array

I'm having trouble running a Jenkins job using Groovy, and it seems super simple but I'm 1) a noob at Java/Groovy and 2) all the examples I can find are not quite what I need to do.

I found the following similar question: Jenkins Groovy Parallel Variable not working, but that exact code results in an error:

java.lang.UnsupportedOperationException: Calling public static java.util.List org.codehaus.groovy.runtime.DefaultGroovyMethods.collect(java.lang.Object,groovy.lang.Closure) on a CPS-transformed closure is not yet supported (JENKINS-26481); encapsulate in a @NonCPS method, or use Java-style loops

Here's my pseudo-code of what I want to do:

String[] arr = [ "one","two","three",'four','five' ]
echo "Running commands: ${arr}"
int top = arr.size()
echo "top is ${top}"
parallel arr.each {
    echo "${it}"
}

I've tried closures, but they throw an error and honestly I barely know how to use them. The following code

String[] arr = [ "one","two","three",'four','five' ]
echo "Running commands: ${arr}"

parallel arr.collect { item ->
    { -> 
        echo "${item}"
    }
}

throws the same "java.lang.UnsupportedOperationException" as the related question above.

I'm trying to learn how to properly utilize Groovy with Jenkins, but it's been rough. Just looking for any help I can get at this point.

Thank you.

Upvotes: 11

Views: 25533

Answers (1)

Ken
Ken

Reputation: 1985

EDIT: In modern Jenkins (tested in 2.150.1) the original bug listed above is fixed, and .each works. You can do the following for parallel execution:

String[] arr = [ "one","two","three",'four','five' ]

def stepsForParallel = [:]
arr.each {
    def stepName = "running ${it}"
    stepsForParallel[stepName] = { ->           
        echo "${it}"
    }
}
parallel stepsForParallel

ORIGINAL: It's a pretty annoying limitation, but currently you can't use .each in a pipeline script (as documented here: https://issues.jenkins-ci.org/browse/JENKINS-26481)

You need to do an actual loop, like

String[] arr = [ "one","two","three",'four','five' ]
echo "Running commands: ${arr}"
int top = arr.size()
echo "top is ${top}"
for (it in arr) {
    echo "${it}"
}

If actually need parallel execution the code will look more like:

String[] arr = [ "one","two","three",'four','five' ]
echo "Running commands: ${arr}"
int top = arr.size()
echo "top is ${top}"
def stepsForParallel = [:]

for (int i = 0; i < arr.size(); i++) {
    def it = arr[i]
    def stepName = "running ${it}"
    stepsForParallel[stepName] = { ->           
        echo "${it}"
    }
}

parallel stepsForParallel

The other stackoverflow you linked to uses the Build Flow plugin, which is the predecessor to the Pipeline plugin that you're using. That's why the same code doesn't work.

Upvotes: 13

Related Questions