forked from typedb/typedb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathJenkinsfile
319 lines (263 loc) · 10 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
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
#!groovy
import static Constants.*
def isMainBranch() {
return env.BRANCH_NAME in ['master', 'stable']
}
// Jenkins normally serializes every variable in a Jenkinsfile so it can pause and resume jobs.
// This method contains variables representing 'jobs', which cannot be serialized.
// The `@NonCPS` annotation stops Jenkins trying to serialize the variables in this method.
@NonCPS
def stopAllRunningBuildsForThisJob() {
def job = Jenkins.instance.getItemByFullName(env.JOB_NAME)
for (build in job.builds) {
if (build.isBuilding() && build.getNumber() != env.BUILD_NUMBER.toInteger()) {
build.doStop()
}
}
}
class Constants {
// In order to add a new integration test, create a new sub-folder under `grakn-test` with two executable scripts,
// `load.sh` and `validate.sh`. Add the name of the folder to the list `INTEGRATION_TESTS` below.
static final INTEGRATION_TESTS = ["test-snb", "test-biomed"]
static final LONG_RUNNING_INSTANCE_ADDRESS = '172.31.22.83'
}
String statusHeader(String message) {
return "${message} on ${env.BRANCH_NAME}: ${env.JOB_NAME} #${env.BUILD_NUMBER}"
}
String statusNotification(String message, String format='slack') {
def user = sh(returnStdout: true, script: "git show --format=\"%aN\" | head -n 1").trim()
String author = "authored by - ${user}"
if (format == 'slack') {
String link = "(<${env.BUILD_URL}|Open>)"
return "${statusHeader(message)} ${link}\n${author}"
} else if (format == 'html') {
String link = "<a href=\"${env.BUILD_URL}\">Open</a>"
return "<p>${statusHeader(message)} ${link}</p><p>${author}</p>"
} else {
throw new RuntimeException("Unrecognised format ${format}")
}
}
def slackGithub(String message, String color = null) {
slackSend channel: "#github", color: color, message: statusNotification(message)
}
def runIntegrationTest(String workspace, String moduleName) {
String modulePath = "${workspace}/grakn-test/${moduleName}"
stage(moduleName) {
withPath("${modulePath}:${modulePath}/src/main/bash") {
withGrakn {
timeout(180) {
stage('Load') {
sh "load.sh"
}
}
timeout(360) {
stage('Validate') {
sh "validate.sh"
}
}
}
}
}
}
def withGrakn(Closure closure) {
//Stages allow you to organise and group things within Jenkins
try {
timeout(15) {
stage('Start Grakn') {
sh 'init-grakn.sh'
}
}
closure()
} finally {
archiveArtifacts artifacts: "${env.PACKAGE}/logs/grakn.log"
archiveArtifacts artifacts: "${env.PACKAGE}/logs/grakn-postprocessing.log"
archiveArtifacts artifacts: "${env.PACKAGE}/logs/cassandra.log"
}
}
def graknNode(Closure closure) {
//Everything is wrapped in a try catch so we can handle any test failures
//If one test fails then all the others will stop. I.e. we fail fast
node {
String workspace = pwd()
withPath("${workspace}/grakn-test/test-integration/src/test/bash") {
try {
closure(workspace)
} finally {
stage('Tear Down') {
sh 'tear-down.sh'
}
}
}
}
}
def archiveArtifactsS3 (String artifacts) {
step([$class: 'S3BucketPublisher',
consoleLogLevel: 'INFO',
pluginFailureResultConstraint: 'FAILURE',
entries: [[
sourceFile: '${artifacts}',
bucket: 'performance-logs.grakn.ai',
selectedRegion: 'eu-west-1',
noUploadOnFailure: true,
managedArtifacts: true,
flatten: true,
showDirectlyInBrowser: true,
keepForever: true
]],
profileName: 'use-iam',
dontWaitForConcurrentBuildCompletion: false,
])
}
def withPath(String path, Closure closure) {
return withEnv(["PATH+EXTRA=${path}"], closure)
}
def ssh(String command) {
sh "ssh -o StrictHostKeyChecking=no -l ubuntu ${LONG_RUNNING_INSTANCE_ADDRESS} ${command}"
}
def buildGrakn() {
sh "build-grakn.sh ${env.BRANCH_NAME}"
}
def shouldRunAllTests() {
return isMainBranch()
}
def shouldDeployLongRunningInstance() {
return env.BRANCH_NAME == 'stable'
}
def mvn(String args) {
sh "mvn --batch-mode ${args}"
}
Closure createTestJob(split, i, testTimeout) {
return { workspace ->
checkout scm
def mavenVerify = 'clean verify -P janus -U -Djetty.log.level=WARNING -Djetty.log.appender=STDOUT -DMaven.test.failure.ignore=true -Dsurefire.rerunFailingTestsCount=1'
/* Write includesFile or excludesFile for tests. Split record provided by splitTests. */
/* Tell Maven to read the appropriate file. */
if (split.includes) {
writeFile file: "${workspace}/parallel-test-includes-${i}.txt", text: split.list.join("\n")
mavenVerify += " -Dsurefire.includesFile=${workspace}/parallel-test-includes-${i}.txt"
} else {
writeFile file: "${workspace}/parallel-test-excludes-${i}.txt", text: split.list.join("\n")
mavenVerify += " -Dsurefire.excludesFile=${workspace}/parallel-test-excludes-${i}.txt"
}
try {
/* Call the Maven build with tests. */
timeout(testTimeout) {
stage('Run Janus test profile') {
mvn mavenVerify
archiveArtifacts artifacts: "grakn-dist/target/grakn-dist*.tar.gz"}
}
} finally {
/* Archive the test results */
junit "**/TEST*.xml"
}
}
}
//Add all tests to job map
void addTests(Map<String, Closure> jobs) {
/* Request the test groupings. Based on previous test results. */
/* see https://wiki.jenkins-ci.org/display/JENKINS/Parallel+Test+Executor+Plugin and demo on github
/* Using arbitrary parallelism of 4 and "generateInclusions" feature added in v1.8. */
def splits = splitTests parallelism: [$class: 'CountDrivenParallelism', size: 4], generateInclusions: true
def numSplits = splits.size()
// change timeout based on how many splits we have. e.g. 1 split = 120min, 3 splits = 60min
def testTimeout = 30 + 90 / numSplits;
splits.eachWithIndex { split, i ->
def job = createTestJob(split, i, testTimeout)
addJob(jobs, "split-${i}", job)
}
}
void addJob(Map<String, Closure> jobs, String name, Closure closure) {
jobs[name] = {
graknNode { workspace ->
withEnv(["PACKAGE=${name}"]) {
closure(workspace)
}
}
}
}
// Main script to run
def runBuild() {
//This sets properties in the Jenkins server.
properties([
pipelineTriggers([
issueCommentTrigger('.*!rtg.*')
]),
buildDiscarder(logRotator(numToKeepStr: '30', artifactNumToKeepStr: '7'))
])
if (!isMainBranch()) {
stopAllRunningBuildsForThisJob()
//Keep fewer artifacts for PRs
properties([
buildDiscarder(logRotator(numToKeepStr: '7', artifactNumToKeepStr: '1'))
])
}
// This is a map that we fill with jobs to perform in parallel, name -> job closure
jobs = [:]
addTests(jobs)
if (shouldRunAllTests()) {
// Build grakn so it can be used by benchmarks and integration tests
graknNode { workspace ->
checkout scm
stage('Build Grakn') {
buildGrakn()
archiveArtifacts artifacts: "grakn-dist/target/grakn-dist*.tar.gz"
// Stash the built distribution so other nodes can access it
stash includes: 'grakn-dist/target/grakn-dist*.tar.gz', name: 'dist'
}
}
addJob(jobs, 'benchmarks') { workspace ->
checkout scm
unstash 'dist'
timeout(60) {
stage('Run the benchmarks') {
mvn "clean test -P janus -Dtest=*Benchmark -DfailIfNoTests=false -Dcheckstyle.skip=true -Dfindbugs.skip=true -Dpmd.skip=true"
archiveArtifacts artifacts: 'grakn-test/test-integration/benchmarks/*.json'
// TODO: re-enable and fix archiving in S3
// archiveArtifactsS3 artifacts: 'grakn-test/test-integration/benchmarks/*.json'
}
}
}
INTEGRATION_TESTS.each { String moduleName ->
// Add each integration test as a parallel job
addJob(jobs, moduleName) { workspace ->
checkout scm
unstash 'dist'
runIntegrationTest(workspace, moduleName)
}
}
}
// Execute all jobs in parallel
parallel(jobs)
graknNode { workspace ->
checkout scm
// only deploy long-running instance on stable branch if all tests pass
if (shouldDeployLongRunningInstance()) {
unstash 'dist'
stage('Deploy Grakn') {
sshagent(credentials: ['jenkins-aws-ssh']) {
sh "scp -o StrictHostKeyChecking=no grakn-dist/target/grakn-dist*.tar.gz ubuntu@${LONG_RUNNING_INSTANCE_ADDRESS}:~/"
sh "scp -o StrictHostKeyChecking=no scripts/repeat-query ubuntu@${LONG_RUNNING_INSTANCE_ADDRESS}:~/"
ssh "'bash -s' < scripts/start-long-running-instance.sh"
}
}
}
slackGithub "Build Success", "good"
}
}
try {
runBuild()
} catch (Exception e) {
node {
String message = "Build Failure"
if (isMainBranch() && currentBuild.getPreviousBuild().getResult().toString() == "SUCCESS") {
emailext (
subject: statusHeader(message),
body: statusNotification(message, 'html'),
mimeType: 'text/html',
recipientProviders: [[$class: 'CulpritsRecipientProvider']]
)
}
slackGithub message, "danger"
}
throw e
}