jjb: lttng-modules: fix RT jobs failure
[lttng-ci.git] / scripts / lttng-modules / master-rt.groovy
CommitLineData
f3d8604b 1/**
5a196804 2 * Copyright (C) 2016-2018 - Michael Jeanson <mjeanson@efficios.com>
f3d8604b
MJ
3 *
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <http://www.gnu.org/licenses/>.
16 */
17
18import hudson.model.*
19import hudson.AbortException
20import hudson.console.HyperlinkNote
21import java.util.concurrent.CancellationException
22import org.eclipse.jgit.api.Git
23import org.eclipse.jgit.lib.Ref
24
25
af3990ed
MJ
26class InvalidKVersionException extends Exception {
27 public InvalidKVersionException(String message) {
28 super(message)
29 }
30}
31
32class EmptyKVersionException extends Exception {
33 public EmptyKVersionException(String message) {
34 super(message)
35 }
36}
f3d8604b 37
af3990ed 38class RTKVersion implements Comparable<RTKVersion> {
f3d8604b 39
af3990ed
MJ
40 Integer major = 0
41 Integer majorB = 0
42 Integer minor = 0
43 Integer patch = 0
44 Integer rt = 0
f3d8604b 45
af3990ed
MJ
46 RTKVersion() {}
47
48 RTKVersion(version) {
f3d8604b
MJ
49 this.parse(version)
50 }
51
af3990ed
MJ
52 static RTKVersion minKVersion() {
53 return new RTKVersion("v0.0.0-rt0-rebase")
54 }
55
56 static RTKVersion maxKVersion() {
57 return new RTKVersion("v" + Integer.MAX_VALUE + ".0.0-rt0-rebase")
58 }
59
60 static RTKVersion factory(version) {
61 return new RTKVersion(version)
62 }
63
f3d8604b
MJ
64 def parse(version) {
65 this.major = 0
66 this.majorB = 0
67 this.minor = 0
68 this.patch = 0
2c1d386a 69 this.rt = 0
f3d8604b 70
af3990ed
MJ
71 if (!version) {
72 throw new EmptyKVersionException("Empty kernel version")
73 }
74
2c1d386a 75 def match = version =~ /^v(\d+)\.(\d+)(\.(\d+))?(\.(\d+))?(-rt(\d+)-rebase)$/
f3d8604b 76 if (!match) {
af3990ed 77 throw new InvalidKVersionException("Invalid kernel version: ${version}")
f3d8604b
MJ
78 }
79
80 Integer offset = 0;
81
82 // Major
83 this.major = Integer.parseInt(match.group(1))
84 if (this.major <= 2) {
85 offset = 2
86 this.majorB = Integer.parseInt(match.group(2))
87 }
88
89 // Minor
90 if (match.group(2 + offset) != null) {
91 this.minor = Integer.parseInt(match.group(2 + offset))
92 }
93
94 // Patch level
95 if (match.group(4 + offset) != null) {
96 this.patch = Integer.parseInt(match.group(4 + offset))
97 }
98
2c1d386a
MJ
99 // RT
100 this.rt = Integer.parseInt(match.group(8))
f3d8604b
MJ
101 }
102
af3990ed
MJ
103 // Return true if both version are of the same stable branch
104 Boolean isSameStable(RTKVersion o) {
105 if (this.major != o.major) {
106 return false
107 }
108 if (this.majorB != o.majorB) {
109 return false
110 }
111 if (this.minor != o.minor) {
112 return false
113 }
114
115 return true
116 }
117
118 @Override int compareTo(RTKVersion o) {
f3d8604b 119 if (this.major != o.major) {
af3990ed 120 return Integer.compare(this.major, o.major)
f3d8604b
MJ
121 }
122 if (this.majorB != o.majorB) {
af3990ed 123 return Integer.compare(this.majorB, o.majorB)
f3d8604b
MJ
124 }
125 if (this.minor != o.minor) {
af3990ed 126 return Integer.compare(this.minor, o.minor)
f3d8604b
MJ
127 }
128 if (this.patch != o.patch) {
af3990ed 129 return Integer.compare(this.patch, o.patch)
f3d8604b 130 }
af3990ed
MJ
131 if (this.rt != o.rt) {
132 return Integer.compare(this.rt, o.rt)
f3d8604b
MJ
133 }
134
135 // Same version
136 return 0;
137 }
138
139 String toString() {
140 String vString = "v${this.major}"
141
142 if (this.majorB > 0) {
143 vString = vString.concat(".${this.majorB}")
144 }
145
146 vString = vString.concat(".${this.minor}")
147
148 if (this.patch > 0) {
149 vString = vString.concat(".${this.patch}")
150 }
151
2c1d386a
MJ
152 if (this.rt > 0) {
153 vString = vString.concat("-rt${this.rt}-rebase")
f3d8604b
MJ
154 }
155 return vString
156 }
157}
158
159
160// Retrieve parameters of the current build
28b10e79 161def mbranch = build.getEnvironment(listener).get('GIT_BRANCH').minus('origin/')
f3d8604b
MJ
162def maxConcurrentBuild = build.buildVariableResolver.resolve('maxConcurrentBuild')
163def kgitrepo = build.buildVariableResolver.resolve('kgitrepo')
af3990ed
MJ
164def kverfloor_raw = build.buildVariableResolver.resolve('kverfloor')
165def kverceil_raw = build.buildVariableResolver.resolve('kverceil')
166def kverfilter = build.buildVariableResolver.resolve('kverfilter')
f3d8604b 167def job = Hudson.instance.getJob(build.buildVariableResolver.resolve('kbuildjob'))
bcf4d9d0 168def currentJobName = build.project.getFullDisplayName()
5a196804 169def gitmodpath = build.getEnvironment(listener).get('WORKSPACE') + "/src/lttng-modules"
af3990ed 170
f3d8604b
MJ
171// Get the out variable
172def config = new HashMap()
173def bindings = getBinding()
174config.putAll(bindings.getVariables())
175def out = config['out']
176
f3d8604b 177
5a196804
MJ
178// Get the lttng-modules git url
179def gitmodrepo = Git.open(new File(gitmodpath))
180def mgitrepo = gitmodrepo.getRepository().getConfig().getString("remote", "origin", "url")
181
f3d8604b 182// Get tags from git repository
5a196804 183def refs = Git.lsRemoteRepository().setTags(true).setRemote(kgitrepo).call()
f3d8604b
MJ
184
185// Get kernel versions to build
186def kversions = []
c9c7c76b 187def tagMatchStrs = [
af3990ed
MJ
188 ~/^refs\/tags\/(v[\d\.]+(-rt(\d+)-rebase))$/,
189]
190def blacklist = [
c9c7c76b
MJ
191 ~/v4\.11\.8-rt5-rebase/,
192 ~/v4\.11\.9-rt6-rebase/,
193 ~/v4\.11\.9-rt7-rebase/,
194 ~/v4\.11\.12-rt8-rebase/,
195 ~/v4\.11\.12-rt9-rebase/,
196 ~/v4\.11\.12-rt10-rebase/,
197 ~/v4\.11\.12-rt11-rebase/,
198 ~/v4\.11\.12-rt12-rebase/,
199 ~/v4\.11\.12-rt13-rebase/,
e468fc14 200 ~/v4\.19.*-rebase/,
c9c7c76b
MJ
201 ~/v3\.6.*-rebase/,
202 ~/v3\.8.*-rebase/,
af3990ed
MJ
203]
204
205def kversionFactory = new RTKVersion()
206
207// Parse kernel versions
208def kverfloor = ""
209try {
210 kverfloor = kversionFactory.factory(kverfloor_raw)
211} catch (EmptyKVersionException e) {
212 kverfloor = kversionFactory.minKVersion()
213}
f3d8604b 214
af3990ed
MJ
215def kverceil = ""
216try {
217 kverceil = kversionFactory.factory(kverceil_raw)
218} catch (EmptyKVersionException e) {
219 kverceil = kversionFactory.maxKVersion()
220}
221
222// Build a sorted list of versions to build
223for (ref in refs) {
c9c7c76b
MJ
224 for (tagMatchStr in tagMatchStrs) {
225 def tagMatch = ref.getName() =~ tagMatchStr
f3d8604b 226
c9c7c76b
MJ
227 if (tagMatch) {
228 def kversion_raw = tagMatch.group(1)
229 def blacklisted = false
230
231 // Check if the kversion is blacklisted
232 for (blackMatchStr in blacklist) {
233 def blackMatch = kversion_raw =~ blackMatchStr
234
235 if (blackMatch) {
236 blacklisted = true
237 break;
238 }
239 }
240
241 if (!blacklisted) {
242 def v = kversionFactory.factory(kversion_raw)
243
244 if ((v >= kverfloor) && (v < kverceil)) {
245 kversions.add(v)
246 }
af3990ed 247 }
f3d8604b
MJ
248 }
249 }
250}
251
252kversions.sort()
f3d8604b 253
c9c7c76b
MJ
254//println "Pre filtering kernel versions:"
255//for (k in kversions) {
256// println k
257//}
258
af3990ed
MJ
259switch (kverfilter) {
260 case 'stable-head':
261 // Keep only the head of each stable branch
262 println('Filter kernel versions to keep only the latest point release of each stable branch.')
263
264 for (i = 0; i < kversions.size(); i++) {
265 def curr = kversions[i]
266 def next = i < kversions.size() - 1 ? kversions[i + 1] : null
267
268 if (next != null) {
269 if (curr.isSameStable(next)) {
270 kversions.remove(i)
271 i--
272 }
273 }
274 }
275 break
276
277 default:
278 // No filtering of kernel versions
279 println('No kernel versions filtering selected.')
280 break
281}
282
283
f3d8604b
MJ
284println "Building the following kernel versions:"
285for (k in kversions) {
286 println k
287}
288
289// Debug: Stop build here
290//throw new InterruptedException()
291
292def joburl = HyperlinkNote.encodeTo('/' + job.url, job.fullDisplayName)
293
294def allBuilds = []
295def ongoingBuild = []
296def failedRuns = []
297def isFailed = false
af3990ed 298def similarJobQueued = 0;
f3d8604b
MJ
299
300// Loop while we have kernel versions remaining or jobs running
301while ( kversions.size() != 0 || ongoingBuild.size() != 0 ) {
302
303 if(ongoingBuild.size() < maxConcurrentBuild.toInteger() && kversions.size() != 0) {
304 def kversion = kversions.pop()
305 def job_params = [
28b10e79 306 new StringParameterValue('mversion', mbranch),
5a196804 307 new StringParameterValue('mgitrepo', mgitrepo),
a1ae361e 308 new StringParameterValue('ktag', kversion.toString()),
f3d8604b 309 new StringParameterValue('kgitrepo', kgitrepo),
ba7bcca2 310 new StringParameterValue('distroversion', ''),
f3d8604b
MJ
311 ]
312
313 // Launch the parametrized build
314 def param_build = job.scheduleBuild2(0, new Cause.UpstreamCause(build), new ParametersAction(job_params))
28b10e79 315 println "triggering ${joburl} for the ${mbranch} branch on kernel ${kversion}"
f3d8604b
MJ
316
317 // Add it to the ongoing build queue
318 ongoingBuild.push(param_build)
319
320 } else {
321
322 println "Waiting... Queued: " + kversions.size() + " Running: " + ongoingBuild.size()
323 try {
af3990ed 324 Thread.sleep(10000)
f3d8604b
MJ
325 } catch(e) {
326 if (e in InterruptedException) {
327 build.setResult(hudson.model.Result.ABORTED)
328 throw new InterruptedException()
329 } else {
330 throw(e)
331 }
332 }
333
af3990ed 334 // Abort job if a newer instance is queued
bcf4d9d0
JR
335 similarJobQueued = Hudson.instance.queue.items.count{it.task.getFullDisplayName() == currentJobName}
336 if ( similarJobQueued > 0 ) {
bcf4d9d0
JR
337 build.setResult(hudson.model.Result.ABORTED)
338 throw new InterruptedException()
339 }
340
f3d8604b
MJ
341 def i = ongoingBuild.iterator()
342 while ( i.hasNext() ) {
343 currentBuild = i.next()
344 if ( currentBuild.isCancelled() || currentBuild.isDone() ) {
345 // Remove from queue
346 i.remove()
347
348 // Print results
349 def matrixParent = currentBuild.get()
350 allBuilds.add(matrixParent)
a1ae361e 351 def kernelStr = matrixParent.buildVariableResolver.resolve("ktag")
f3d8604b
MJ
352 println "${matrixParent.fullDisplayName} (${kernelStr}) completed with status ${matrixParent.result}"
353
354 // Process child runs of matrixBuild
355 def childRuns = matrixParent.getRuns()
356 for ( childRun in childRuns ) {
357 println "\t${childRun.fullDisplayName} (${kernelStr}) completed with status ${childRun.result}"
358 if (childRun.result != Result.SUCCESS) {
359 failedRuns.add(childRun)
360 isFailed = true
361 }
362 }
363 }
364 }
365 }
366}
367
368// Get log of failed runs
369for (failedRun in failedRuns) {
370 println "---START---"
371 failedRun.writeWholeLogTo(out)
372 println "---END---"
373}
374
375println "---Build report---"
376for (b in allBuilds) {
a1ae361e 377 def kernelStr = b.buildVariableResolver.resolve("ktag")
f3d8604b 378 println "${b.fullDisplayName} (${kernelStr}) completed with status ${b.result}"
7e02032c 379 // Cleanup builds
7e942863
MJ
380 try {
381 b.delete()
382 } catch (all) {}
f3d8604b
MJ
383}
384
385// Mark this build failed if any child build has failed
386if (isFailed) {
c5c05f73 387 build.setResult(hudson.model.Result.FAILURE)
f3d8604b
MJ
388}
389
390// EOF
This page took 0.060334 seconds and 4 git commands to generate.