-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathJenkinsfile
160 lines (134 loc) · 5.62 KB
/
Jenkinsfile
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
#!groovy
@Library('utils')
import no.evry.Docker
import no.evry.Config
node('jenkins-docker-3') {
// Make a clean workspace for the new build to prevent multiple concurrent
// builds from polluting the others working directory.
ws {
// Wrap everything in a try/catch block in order to trap exceptions and
// process them before exiting.
try {
// Jenkins pipelines are "dumb" in that regards that it does not do
// anything unless you explicitly tells it so. Here we checkout the
// repository in order to the source code into the workspace.
stage('Checkout')
{
checkout scm
}
// Configure which branches should get the following configurations. The
// configurations are stored in the /build directory and controles the
// Jenkins build and Kubernetes deployment.
def envPatterns = [
[env: 'dev', regex: /^develop$/],
[env: 'test', regex: /^release\/v[0-9]+\.[0-9]+\.[0-9]+$/],
[env: 'prod', regex: /^master$/],
[env: 'prod', regex: /^feature\/submissions$/],
]
config = new Config(this).branchProperties(envPatterns)
def nodeImage = 'node:8-alpine'
def nodeArgs = [
'-v /home/jenkins/.cache/yarn:/home/node/.cache/yarn',
'-e NODE_ENV=development',
].join(' ')
// Install all packge dependencies specificed in package.json. We also
// need to install development dependencies in order to build and test the
// application. We will remove these later in the "Prune" stage.
stage('Install') {
docker.image(nodeImage).inside(nodeArgs) {
sh 'cd app && yarn install --development'
}
}
// Built all frontend assets required for running this application in
// production. Here we are using the Grunt task runner, see Gruntfile for
// how this is otherwise configured.
stage('Build') {
docker.image(nodeImage).inside(nodeArgs) {
sh 'cd app && yarn run grunt'
}
}
// Speed up builds by running multiple tasks in parallel. Keep in mind
// that these tasks can not depend on each other and should not modify the
// sharedstate of the workspace.
parallel (
// nsp (Node Security Project) is a security scanner for Node.js
// packages. It will check against it's database for known
// vulnerabilities and block any builds containing vulnerable packages.
"nsp": {
docker.image(nodeImage).inside(nodeArgs) {
sh 'cd app && yarn run nsp check'
}
},
// Syntax linting checks the style of the source code to make sure the
// code follows a common standard. ESLint is the recommended tool for
// all Node.js and JavaScript projects.
"lint" : {
docker.image(nodeImage).inside(nodeArgs) {
sh 'cd app && yarn run lint'
}
},
// Run the test suite for the application. In any normal project this is
// the part that takes the most time. One should always stribe to
// optimise the tests in order to make them execute faster; no one likes
// to wait for them tests to complete!
"test" : {
mongo = docker.image('mongo:3.4').run()
docker.image(nodeImage).inside([
nodeArgs,
"--link ${mongo.id}:mongo",
"-e MONGO_URI=mongodb://mongo:27017/test",
].join(' ')) {
sh 'cd app && yarn run test'
}
}
)
// Remove all development packages such that the node_modules directory
// only contains packages needed for running the application in
// production. This is to save space and ensure identical packages in
// Jenkins and in production.
stage('Prune') {
docker.image(nodeImage).inside(nodeArgs) {
sh 'cd app && yarn install --production'
}
}
// Build the Docker Image, image tag should be unique to the build to
// prevent building over other images. See the Dockerfile for how the
// Docker Image is otherwise constructed.
stage('Docker Build') {
Docker doc = new Docker(this, [nameOnly: false])
config.DOCKER_TAG = doc.buildTag()
config.DOCKER_IMAGE = doc.image(config.DOCKER_REGISTRY)
image = docker.build(doc.image())
}
// Push the newly built Docker Image to a Docker Registry of your choosing
// in order to reach it later in the deployment stage.
if (config.JENKINS_DEPLOY == 'true') {
stage('Docker Push') {
docker.withRegistry("https://${config.DOCKER_REGISTRY}", config.DOCKER_REGISTRY) {
image.push(config.DOCKER_TAG)
}
}
}
// Deploy the application to Kubernetes. See the Jenkins Utility repo for
// more information about the kubernetesDeploy function and it's configs.
stage("Deploy ${config.K8S_NAMESPACE}@${config.K8S_CLUSTER}") {
boolean dryrun = config.JENKINS_DEPLOY != 'true'
kubernetesDeploy(config, [k8sCluster: config.K8S_CLUSTER, dryrun: dryrun])
}
// Catch abort build interrupts and possible handle them differently. They
// should not be reported as build failures to Slack etc.
} catch (InterruptedException e) {
throw e
// Catch all build failures and report them to Slack etc here.
} catch (e) {
throw e
// Clean up the workspace before exiting. Wait for Jenkins' asynchronous
// resource disposer to pick up before we close the connection to the worker
// node.
} finally {
mongo.stop()
step([$class: 'WsCleanup'])
sleep 10
}
}
}