42

I have a declarative pipeline script for my multibranch project in which I would like to read a text file and store the result as a string variable to be accessed by a later step in the pipeline. Using the snippet generator I tried to do something like this:

filename = readFile 'output.txt'

For which filename would be my string.

I get an error in the Jenkins console output:

org.codehaus.groovy.control.MultipleCompilationErrorsException: startup failed:
WorkflowScript: 30: Expected a step @ line 30, column 5.
            filename = readFile 'output.txt'

Do I need to use a withEnv step to set the output of readFile to a Jenkins environment variable? If so, how?

Thanks

Gerold Broser
  • 14,080
  • 5
  • 48
  • 107
Dalton Sweeney
  • 1,108
  • 3
  • 11
  • 24

5 Answers5

68

The error is due to that you're only allowed to use pipeline steps inside the steps directive. One workaround that I know is to use the script step and wrap arbitrary pipeline script inside of it and save the result in the environment variable so that it can be used later.

So in your case:

pipeline {
    agent any
    stages {
        stage("foo") {
            steps {
                script {
                    env.FILENAME = readFile 'output.txt'
                }
                echo "${env.FILENAME}"
            }
        }
    }
}
Jon S
  • 15,846
  • 4
  • 44
  • 45
  • Strange, when run this (Jenkins 2.32.2) I get the expected result (added `writeFile file: 'output.txt', text: 'asdsad'` just before `readFile`, I get `[Pipeline] echo asdsad` as expected). Are you sure there is any contents in the file? – Jon S Mar 01 '17 at 20:05
  • Ya it's blank and that was due to some other issue in my logic, but I think your answer is what I was looking for. Thanks – Dalton Sweeney Mar 01 '17 at 20:26
  • 4
    one hint: you may want to trim like this `readFile('output.txt').trim()` – hao Jul 20 '18 at 09:03
  • 1
    Note that you cannot assign a `credential(...)` to a variable like this. It renders out to a string `"@credentials(=BLAHBLAH>"` and will then silently not work. – shuckc Jan 15 '19 at 12:49
  • Yes, the env variable isn't backed by a normal map, instead it operates 100% with strings, i.e. all values which are assigned to keys in the env variable are converted to strings (using toString() i guess). – Jon S Jan 15 '19 at 13:18
28

According to the documentation, you can also set global environment variables if you later want to use the value of the variable in other parts of your script. In your case, it would be setting it in the root pipeline:

pipeline {
  ...
  environment {
    FILENAME = readFile ...
  }
  ...
}
Adam Link
  • 2,783
  • 4
  • 30
  • 44
12

We got around this by adding functions to the environment step, i.e.:

environment {
    ENVIRONMENT_NAME = defineEnvironment() 
}
...
def defineEnvironment() {
    def branchName = "${env.BRANCH_NAME}"
    if (branchName == "master") {
        return 'staging'
    }
    else {
        return 'test'
    }
}
Jansky
  • 1,455
  • 1
  • 17
  • 33
2

A complete example for scripted pipepline:

       stage('Build'){
            withEnv(["GOPATH=/ws","PATH=/ws/bin:${env.PATH}"]) {
                sh 'bash build.sh'
            }
        }
Graham
  • 7,431
  • 18
  • 59
  • 84
Qiang Li
  • 1,099
  • 11
  • 8
  • 1
    This question doesn't say anything about running a build script, or adding to PATH. – daemone Feb 27 '18 at 08:50
  • 1
    True, but this is the only answer with scripted syntax, and this is a fine piece of example code. – Kyle Feb 21 '20 at 14:34
2

I can' t comment yet but, just a hint: use try/catch clauses to avoid breaking the pipeline (if you are sure the file exists, disregard)

    pipeline {
        agent any
            stages {
                stage("foo") {
                    steps {
                        script {
                            try {                    
                                env.FILENAME = readFile 'output.txt'
                                echo "${env.FILENAME}"
                            }
                            catch(Exception e) {
                                //do something, e.g. echo 'File not found'
                            }
                        }
                   }
    }

Another hint (this was commented by @hao, and think is worth to share): you may want to trim like this readFile('output.txt').trim()

Raúl Salinas-Monteagudo
  • 3,412
  • 1
  • 24
  • 22
Paulo Vinícius
  • 326
  • 2
  • 9