In Exercise 1.1 we will create a simple declarative pipeline directly within the Jenkins interface.
First create a personal folder in Jenkins to hold the pipelines you create in this workshop:
- Log into Jenkins
- Click on New Item
- Type you name into the Enter an item name text box, click on Folder, and then click on the OK button.
- Click on Save in the folder configuration screen to accept the default options.
Now create a new pipeline inside of your personal folder:
- Click on the create new jobs link.
- Type SimplePipeline into the Enter an item name (example: MyFirstPipeline) text box, click on Pipeline, and then click on the OK button.
- Copy and paste the following code into the Pipeline Script text box:
pipeline {
agent any
stages {
stage('Say Hello') {
steps {
echo 'Hello World!'
}
}
}
}
- Click on Save and then click on Build Now to run your pipeline.
In Exercise 1.2 we will update the pipeline we created in 1.1 to execute in a docker container. To update the pipeline:
- In the
steps
block add the following after theecho
step:
sh 'mvn -v'
-
Execute your job by clicking on Build Now and check the Console Log. The build will fail with
mvn: not found
-
Click on configure and update the
agent
portion of the pipeline to read:
agent {
docker { image 'maven:3.5.2-jdk-9' }
}
- Execute your job by clicking on Build Now and check the Console Log to see how Jenkins pulls the appropriate docker image and runs your build insde the container created from that image. Also note that you are able to compile against JDK 9 regardless what JDK the Jenkins agent is using.
Before going on to the next exercise let's revert our pipeline to using:
agent any
And remove the sh 'mvn -v'
step
For Exercise 1.3 we are going to update our SimplePipeline job to demonstrate how to use environmental variables.
At the top of the pipeline insert the following code between the agent
and stages
blocks:
environment {
MY_NAME = 'Mary'
}
Then update the echo 'Hello World!'
line to read echo "Hello ${MY_NAME}!"
and run your build again to view the results.
We can also use environmental variables to import credentials. To demonstrate we will add the following line to our environment
block:
TEST_USER = credentials('test-user')
We will also add the following echo
steps within the steps
of the Say Hello stage
:
echo "${TEST_USER_USR}"
echo "${TEST_USER_PSW}"
Note: After executing the build look at the console output and make note of the fact that the credential user name and password are masked when output via the echo command.
In Exercise 1.4 we will alter our pipeline to accept external input in the form of a Parameter.
At the top of your pipeline insert the following block of code between the environment
and stages
blocks:
parameters {
string(name: 'Name', defaultValue: 'whoever you are',
description: 'Who should I say hi to?')
}
Then update the echo "Hello ${MY_NAME}!'
line to read echo "Hello ${params.Name}!"
and run your build again to view the results.
Note: Jenkins UI won't update properly when you save the pipeline to show the Build with parameters
option so you need to run a build, view the results, and then return to the project to see the updated option.
For Exercise 1.5 we are going to add a new stage after the Say Hello stage that will demonstrate how to ask interactively for user input. The declarative input
directive blocks the stage
from executing and acquiring an agent. If the input
is approved, the stage will then continue.
Insert the following stage
block into your pipeline:
stage('Deploy') {
input {
message "Should we continue?"
}
steps {
echo "Continuing with deployment"
}
}
Note: To keep Jenkins from waiting indefinitely for a user response your should set a timeout
for the stage
like shown below:
stage('Deploy') {
options {
timeout(time: 1, unit: 'MINUTES')
}
input {
message "Should we continue?"
}
steps {
echo "Continuing with deployment"
}
}
What happens if your input step times out? Post Actions are designed to handle a variety of conditions (not only failures) that could occur outside the standard pipeline flow.
In this example we will add a Post Action to our Deploy stage to handle a time out (aborted run). Modify your Deploy stage to look like:
stage('Deploy') {
options {
timeout(time: 1, unit: 'MINUTES')
}
input {
message "Should we continue?"
}
steps {
echo "Continuing with deployment"
}
post {
aborted {
echo 'Why didn\'t you push my button?'
}
}
}
On the next build wait for the input time and you will see the following line in your console output: Why didn't you push my button?
.
Note: After you have executed this pipeline several times you might want to remove the Deploy
stage from your pipeline so that you will not have to manually approve it each time it runs.
In Exercise 1.7 we are going to add a stage to our pipeline that uses a Shared Library to import functionality that allows us to say hi.
More information on using Shared Libraries is available here: https://jenkins.io/doc/book/pipeline/shared-libraries/
Add the following line at the top of your pipeline above the pipeline
line:
library 'SharedLibs'
Then add the following stage after the Deploy stage:
stage('Shared Lib') {
steps {
helloWorld("Craig")
}
}
The helloWorld
function we are calling can be seen at: https://github.com/PipelineHandsOn/shared-libraries/blob/master/vars/helloWorld.groovy
In this exercise we are going to add another stage to our pipeline that runs two steps in parallel on two different docker based agents (one running Java 7 and one running Java 8). The following code also includes sleep
steps to demonstrate what happens when parallel steps complete execution at different times:
Important Note The following code demonstrates a new set of features added to Declarative Pipeline in Version 1.2 (parallel stages) and 1.2.1 (failFast inside of a parallel stage).
stage('Testing') {
parallel {
stage('Java 7') {
agent { docker 'openjdk:7-jdk-alpine' }
failFast true
steps {
sh 'java -version'
sleep time: 1, unit: 'MINUTES'
}
}
stage('Java 8') {
agent { docker 'openjdk:8-jdk-alpine' }
steps {
sh 'java -version'
sleep time: 2, unit: 'MINUTES'
}
}
}
}
Note: If your build breaks double check your pipeline script to make sure that the agent at the top of of the pipeline was reverted back to agent any
as described in exercise 1.2.
Finally we will use the Blue Ocean Pipeline Editor to create a simple declarative pipeline using the following steps:
- Click on the Open Blue Ocean button in the left side navigation bar
- Click on the New Pipeline button
- Click on one of the options in the Where do you store your code? section (Github for this course)
- Enter your Github token
- Select the Organization in which the repository that you want to create the Jenkinsfile in exists
- Select New Pipeline
- Choose the Repository
- Click on Create Pipeline
Once the pipeline has created Blue Ocean will open the editor screen. We will create a few simple steps using the following instructions (feel free to veer of course and try all of the options available):
- Click on the + icon next to the pipeline's Start node
- Click into Name your stage and enter a name
- Click on + Add step
- Click on Shell script
- Type
mvn -v
into the text box - Click on Save to save the pipeline and execute it
- Enter a commit message into the Save Pipeline pop up and click Save & Run
After your pipeline executes you can click on the pencil icon to continue editing your pipeline.