jjb: Add lttng-modules jobs for Ubuntu noble
[lttng-ci.git] / scripts / lttng-modules / master.groovy
CommitLineData
f3d8604b 1/**
073dc82c 2 * Copyright (C) 2016-2020 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
3a01c580
MJ
26class InvalidKVersionException extends Exception {
27 public InvalidKVersionException(String message) {
591756e5
MJ
28 super(message)
29 }
30}
31
3a01c580
MJ
32class EmptyKVersionException extends Exception {
33 public EmptyKVersionException(String message) {
591756e5
MJ
34 super(message)
35 }
36}
37
3a01c580 38class VanillaKVersion implements Comparable<VanillaKVersion> {
f3d8604b 39
3a01c580
MJ
40 Integer major = 0
41 Integer majorB = 0
42 Integer minor = 0
43 Integer patch = 0
44 Integer rc = Integer.MAX_VALUE
f3d8604b 45
3a01c580 46 VanillaKVersion() {}
f3d8604b 47
3a01c580 48 VanillaKVersion(version) {
f3d8604b
MJ
49 this.parse(version)
50 }
51
3a01c580
MJ
52 static VanillaKVersion minKVersion() {
53 return new VanillaKVersion("v0.0.0")
54 }
55
56 static VanillaKVersion maxKVersion() {
57 return new VanillaKVersion("v" + Integer.MAX_VALUE + ".0.0")
58 }
59
60 static VanillaKVersion factory(version) {
61 return new VanillaKVersion(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
69 this.rc = Integer.MAX_VALUE
70
591756e5 71 if (!version) {
3a01c580 72 throw new EmptyKVersionException("Empty kernel version")
591756e5
MJ
73 }
74
f3d8604b
MJ
75 def match = version =~ /^v(\d+)\.(\d+)(\.(\d+))?(\.(\d+))?(-rc(\d+))?$/
76 if (!match) {
3a01c580 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
99 // RC
100 if (match.group(8) != null) {
101 this.rc = Integer.parseInt(match.group(8))
102 }
103 }
104
105 // Return true if this version is a release candidate
106 Boolean isRC() {
107 return this.rc != Integer.MAX_VALUE
108 }
109
e9b44189 110 // Return true if both version are of the same stable branch
3a01c580 111 Boolean isSameStable(VanillaKVersion o) {
e9b44189
MJ
112 if (this.major != o.major) {
113 return false
114 }
115 if (this.majorB != o.majorB) {
116 return false
117 }
118 if (this.minor != o.minor) {
119 return false
120 }
121
122 return true
123 }
124
3a01c580 125 @Override int compareTo(VanillaKVersion o) {
f3d8604b 126 if (this.major != o.major) {
e9b44189 127 return Integer.compare(this.major, o.major)
f3d8604b
MJ
128 }
129 if (this.majorB != o.majorB) {
e9b44189 130 return Integer.compare(this.majorB, o.majorB)
f3d8604b
MJ
131 }
132 if (this.minor != o.minor) {
e9b44189 133 return Integer.compare(this.minor, o.minor)
f3d8604b
MJ
134 }
135 if (this.patch != o.patch) {
e9b44189 136 return Integer.compare(this.patch, o.patch)
f3d8604b
MJ
137 }
138 if (this.rc != o.rc) {
e9b44189 139 return Integer.compare(this.rc, o.rc)
f3d8604b
MJ
140 }
141
142 // Same version
143 return 0;
144 }
145
146 String toString() {
147 String vString = "v${this.major}"
148
149 if (this.majorB > 0) {
150 vString = vString.concat(".${this.majorB}")
151 }
152
153 vString = vString.concat(".${this.minor}")
154
155 if (this.patch > 0) {
156 vString = vString.concat(".${this.patch}")
157 }
158
159 if (this.rc > 0 && this.rc < Integer.MAX_VALUE) {
160 vString = vString.concat("-rc${this.rc}")
161 }
162 return vString
163 }
164}
165
3a01c580
MJ
166class UbuntuKVersion implements Comparable<UbuntuKVersion> {
167
168 Integer major = 0
169 Integer minor = 0
170 Integer patch = 0
171 Integer umajor = 0
172 Integer uminor = 0
173 String suffix = ""
cba2f848 174 String strLTS = ""
3a01c580
MJ
175 Boolean isLTS = false
176
177 UbuntuKVersion() {}
178
179 UbuntuKVersion(version) {
180 this.parse(version)
181 }
182
183 static UbuntuKVersion minKVersion() {
184 return new UbuntuKVersion("Ubuntu-lts-0.0.0-0.0")
185 }
186
187 static UbuntuKVersion maxKVersion() {
188 return new UbuntuKVersion("Ubuntu-" + Integer.MAX_VALUE + ".0.0-0.0")
189 }
190
191 static UbuntuKVersion factory(version) {
192 return new UbuntuKVersion(version)
193 }
194
195 def parse(version) {
196 this.major = 0
197 this.minor = 0
198 this.patch = 0
199 this.umajor = 0
200 this.uminor = 0
201 this.suffix = "";
202 this.isLTS = false
203
204 if (!version) {
205 throw new EmptyKVersionException("Empty kernel version")
206 }
207
073dc82c 208 //'Ubuntu-hwe-5.8-5.8.0-19.20_20.04.3',
3a01c580
MJ
209 //'Ubuntu-lts-4.8.0-27.29_16.04.1',
210 //'Ubuntu-4.4.0-70.91',
073dc82c 211 def match = version =~ /^Ubuntu-(lts-|hwe-)??(?:\d+\.\d+-)??(\d+)\.(\d+)\.(\d+)-(\d+)\.(\d+)(.*)??$/
3a01c580
MJ
212 if (!match) {
213 throw new InvalidKVersionException("Invalid kernel version: ${version}")
214 }
215
cba2f848
MJ
216 if (match.group(1) != null) {
217 this.isLTS = true
218 this.strLTS = match.group(1)
219 }
3a01c580
MJ
220
221 // Major
222 this.major = Integer.parseInt(match.group(2))
223
224 // Minor
225 this.minor = Integer.parseInt(match.group(3))
226
227 // Patch level
228 this.patch = Integer.parseInt(match.group(4))
229
230 // Ubuntu major
231 this.umajor = Integer.parseInt(match.group(5))
232
233 // Ubuntu minor
234 this.uminor = Integer.parseInt(match.group(6))
235
236 if (match.group(7) != null) {
237 this.suffix = match.group(7)
238 }
239 }
240
241 // Return true if this version is a release candidate
242 Boolean isRC() {
243 return false
244 }
245
246 // Return true if both version are of the same stable branch
247 Boolean isSameStable(UbuntuKVersion o) {
248 if (this.isLTS != o.isLTS) {
249 return false
250 }
251 if (this.major != o.major) {
252 return false
253 }
254 if (this.minor != o.minor) {
255 return false
256 }
257 if (this.patch != o.patch) {
258 return false
259 }
260
261 return true
262 }
263
264 @Override int compareTo(UbuntuKVersion o) {
265 if (this.major != o.major) {
266 return Integer.compare(this.major, o.major)
267 }
268 if (this.minor != o.minor) {
269 return Integer.compare(this.minor, o.minor)
270 }
271 if (this.patch != o.patch) {
272 return Integer.compare(this.patch, o.patch)
273 }
274 if (this.umajor != o.umajor) {
275 return Integer.compare(this.umajor, o.umajor)
276 }
277 if (this.uminor != o.uminor) {
278 return Integer.compare(this.uminor, o.uminor)
279 }
280 if (this.isLTS != o.isLTS) {
281 if (o.isLTS) {
282 return 1
283 } else {
284 return -1
285 }
286 }
287
288 // Same version
289 return 0;
290 }
291
292 String toString() {
293 String vString = "Ubuntu-"
294
295 if (this.isLTS) {
cba2f848 296 vString = vString.concat("${this.strLTS}")
3a01c580
MJ
297 }
298
46bde213
MJ
299 // The tag pattern changed for HWE kernels >= 5.0
300 if (this.isLTS && this.major >= 5) {
073dc82c
MJ
301 vString = vString.concat("${this.major}.${this.minor}-${this.major}.${this.minor}.${this.patch}-${this.umajor}.${this.uminor}${this.suffix}")
302 } else {
303 vString = vString.concat("${this.major}.${this.minor}.${this.patch}-${this.umajor}.${this.uminor}${this.suffix}")
304 }
3a01c580
MJ
305
306 return vString
307 }
308}
309
f3d8604b
MJ
310
311// Retrieve parameters of the current build
28b10e79 312def mbranch = build.getEnvironment(listener).get('GIT_BRANCH').minus('origin/')
f3d8604b
MJ
313def maxConcurrentBuild = build.buildVariableResolver.resolve('maxConcurrentBuild')
314def kgitrepo = build.buildVariableResolver.resolve('kgitrepo')
591756e5
MJ
315def kverfloor_raw = build.buildVariableResolver.resolve('kverfloor')
316def kverceil_raw = build.buildVariableResolver.resolve('kverceil')
e9b44189 317def kverfilter = build.buildVariableResolver.resolve('kverfilter')
28b10e79 318def kverrc = build.buildVariableResolver.resolve('kverrc')
3a01c580 319def uversion = build.buildVariableResolver.resolve('uversion')
f3d8604b 320def job = Hudson.instance.getJob(build.buildVariableResolver.resolve('kbuildjob'))
483859f3 321def currentJobName = build.project.getFullDisplayName()
5a196804 322def gitmodpath = build.getEnvironment(listener).get('WORKSPACE') + "/src/lttng-modules"
591756e5 323
f3d8604b
MJ
324// Get the out variable
325def config = new HashMap()
326def bindings = getBinding()
327config.putAll(bindings.getVariables())
328def out = config['out']
329
f3d8604b 330
5a196804
MJ
331// Get the lttng-modules git url
332def gitmodrepo = Git.open(new File(gitmodpath))
333def mgitrepo = gitmodrepo.getRepository().getConfig().getString("remote", "origin", "url")
334
f3d8604b 335// Get tags from git repository
5a196804 336def refs = Git.lsRemoteRepository().setTags(true).setRemote(kgitrepo).call()
f3d8604b
MJ
337
338// Get kernel versions to build
339def kversions = []
340def kversionsRC = []
3a01c580
MJ
341def matchStrs = []
342def blacklist = []
343def kversionFactory = ""
344
345if (uversion != null) {
346 kversionFactory = new UbuntuKVersion()
347 switch (uversion) {
74901baa
MJ
348 case 'jammy':
349 matchStrs = [
350 ~/^refs\/tags\/(Ubuntu-5\.15\.0-\d{1,3}?\.[\d]+)$/,
1af1b414 351 ~/^refs\/tags\/(Ubuntu-hwe-6\.2-6\.2\.0-.*_22\.04\.\d+)$/,
b6ebf05e 352 ~/^refs\/tags\/(Ubuntu-hwe-6\.5-6\.5\.0-.*_22\.04\.\d+)$/,
74901baa
MJ
353 ]
354 break
355
9e5757c6
MJ
356 case 'focal':
357 matchStrs = [
358 ~/^refs\/tags\/(Ubuntu-5\.4\.0-\d{1,3}?\.[\d]+)$/,
74901baa 359 ~/^refs\/tags\/(Ubuntu-hwe-5\.13-5\.13\.0-.*_20\.04\.\d+)$/,
9e5757c6
MJ
360 ]
361 break
362
009efde7
KS
363 case 'noble':
364 matchStrs = [
365 ~/^refs\/tags\/(Ubuntu-6\.8\.0-\d{1,3}?\.[\d]+)$/,
366 ]
367 break
368
3a01c580 369 default:
57bdee9e 370 println "Unsupported Ubuntu version: ${uversion}"
3a01c580
MJ
371 throw new InterruptedException()
372 break
373 }
374} else {
375 // Vanilla
376 kversionFactory = new VanillaKVersion()
377 matchStrs = [
378 ~/^refs\/tags\/(v[\d\.]+(-rc(\d+))?)$/,
379 ]
2cfd47ab
MJ
380 blacklist = [
381 'v3.2.3',
382 ]
3a01c580 383}
f3d8604b 384
3a01c580
MJ
385// Parse kernel versions
386def kverfloor = ""
387try {
388 kverfloor = kversionFactory.factory(kverfloor_raw)
389} catch (EmptyKVersionException e) {
390 kverfloor = kversionFactory.minKVersion()
391}
f3d8604b 392
3a01c580
MJ
393def kverceil = ""
394try {
395 kverceil = kversionFactory.factory(kverceil_raw)
396} catch (EmptyKVersionException e) {
397 kverceil = kversionFactory.maxKVersion()
398}
399
400// Build a sorted list of versions to build
401for (ref in refs) {
402 for (matchStr in matchStrs) {
403 def match = ref.getName() =~ matchStr
404 if (match && !blacklist.contains(match.group(1))) {
405 def v = kversionFactory.factory(match.group(1))
406
407 if ((v >= kverfloor) && (v < kverceil)) {
408 if (v.isRC()) {
409 kversionsRC.add(v)
410 } else {
411 kversions.add(v)
412 }
f3d8604b
MJ
413 }
414 }
415 }
416}
417
cff1edbb 418// The filtering step assumes the version lists are sorted
f3d8604b
MJ
419kversions.sort()
420kversionsRC.sort()
421
e9b44189
MJ
422switch (kverfilter) {
423 case 'stable-head':
424 // Keep only the head of each stable branch
425 println('Filter kernel versions to keep only the latest point release of each stable branch.')
426
427 for (i = 0; i < kversions.size(); i++) {
428 def curr = kversions[i]
429 def next = i < kversions.size() - 1 ? kversions[i + 1] : null
430
431 if (next != null) {
432 if (curr.isSameStable(next)) {
433 kversions.remove(i)
434 i--
435 }
436 }
437 }
438 break
439
3da11679 440 case 'lts-head':
cff1edbb
MJ
441 // Keep only the head of each LTS branch and the latest non-RC tag
442 println('Filter kernel versions to keep only the latest point release of each lts branch and the current stable.')
443
3da11679 444 def lts_kversions = []
242e40f0
KS
445 // Old LTS entries are kept so that "lts-head" is still meaningful in kernel
446 // version ranges that are supported by lttng-modules but no longer supported
447 // upstream, eg. lttng-modules stable-2.13 supports >= 3.0
448 lts_kversions.add(kversionFactory.factory("v3.0")) // LTS until October 2013
449 lts_kversions.add(kversionFactory.factory("v3.2")) // LTS until May 2018
450 lts_kversions.add(kversionFactory.factory("v3.4")) // LTS until October 2016
451 lts_kversions.add(kversionFactory.factory("v3.10")) // LTS until November 2017
452 lts_kversions.add(kversionFactory.factory("v3.12")) // LTS until May 2017
453 lts_kversions.add(kversionFactory.factory("v3.14")) // LTS until August 2016
454 lts_kversions.add(kversionFactory.factory("v3.16")) // LTS until October 2014
455 lts_kversions.add(kversionFactory.factory("v3.18")) // LTS until January 2017
456 lts_kversions.add(kversionFactory.factory("v4.1")) // LTS until May 2018
3da11679
MJ
457 lts_kversions.add(kversionFactory.factory("v4.4")) // SLTS until 2026
458 lts_kversions.add(kversionFactory.factory("v4.9")) // LTS until January 2023
459 lts_kversions.add(kversionFactory.factory("v4.14")) // LTS until January 2024
460 lts_kversions.add(kversionFactory.factory("v4.19")) // LTS until December 2024
461 lts_kversions.add(kversionFactory.factory("v5.4")) // LTS until December 2025
462 lts_kversions.add(kversionFactory.factory("v5.10")) // LTS until December 2026
6ab31912 463 lts_kversions.add(kversionFactory.factory("v5.15")) // LTS until December 2026
3da11679 464 lts_kversions.add(kversionFactory.factory("v6.1")) // LTS until December 2026
6ab31912 465 lts_kversions.add(kversionFactory.factory("v6.6")) // LTS until December 2026
cff1edbb
MJ
466
467 // First filter the head of each branch
468 for (i = 0; i < kversions.size(); i++) {
469 def curr = kversions[i]
470 def next = i < kversions.size() - 1 ? kversions[i + 1] : null
471
472 if (next != null) {
473 if (curr.isSameStable(next)) {
474 kversions.remove(i)
475 i--
476 }
477 }
478 }
479
480 for (i = 0; i < kversions.size(); i++) {
481 def curr = kversions[i]
482
483 // Keep the newest tag
484 if (i == kversions.size() - 1) {
485 break
486 }
487
488 // Prune non-LTS versions
3da11679
MJ
489 def keep = false
490 for (j = 0; j < lts_kversions.size(); j++) {
491 if (curr.isSameStable(lts_kversions[j])) {
492 keep = true
493 break
494 }
495 }
496
497 if (!keep) {
cff1edbb
MJ
498 kversions.remove(i)
499 i--
500 }
501 }
502 break
503
e9b44189
MJ
504 default:
505 // No filtering of kernel versions
506 println('No kernel versions filtering selected.')
507 break
508}
509
28b10e79
MJ
510if (kverrc == "true") {
511 // If the last RC version is newer than the last stable, add it to the build list
512 if (kversionsRC.size() > 0 && kversionsRC.last() > kversions.last()) {
513 kversions.add(kversionsRC.last())
514 }
f3d8604b
MJ
515}
516
f3d8604b
MJ
517println "Building the following kernel versions:"
518for (k in kversions) {
519 println k
520}
521
522// Debug: Stop build here
523//throw new InterruptedException()
524
525def joburl = HyperlinkNote.encodeTo('/' + job.url, job.fullDisplayName)
526
527def allBuilds = []
528def ongoingBuild = []
529def failedRuns = []
530def isFailed = false
483859f3 531def similarJobQueued = 0;
f3d8604b
MJ
532
533// Loop while we have kernel versions remaining or jobs running
534while ( kversions.size() != 0 || ongoingBuild.size() != 0 ) {
535
536 if(ongoingBuild.size() < maxConcurrentBuild.toInteger() && kversions.size() != 0) {
537 def kversion = kversions.pop()
538 def job_params = [
28b10e79 539 new StringParameterValue('mversion', mbranch),
5a196804 540 new StringParameterValue('mgitrepo', mgitrepo),
a1ae361e 541 new StringParameterValue('ktag', kversion.toString()),
f3d8604b
MJ
542 new StringParameterValue('kgitrepo', kgitrepo),
543 ]
544
545 // Launch the parametrized build
546 def param_build = job.scheduleBuild2(0, new Cause.UpstreamCause(build), new ParametersAction(job_params))
28b10e79 547 println "triggering ${joburl} for the ${mbranch} branch on kernel ${kversion}"
f3d8604b
MJ
548
549 // Add it to the ongoing build queue
550 ongoingBuild.push(param_build)
551
552 } else {
553
554 println "Waiting... Queued: " + kversions.size() + " Running: " + ongoingBuild.size()
555 try {
3a01c580 556 Thread.sleep(10000)
f3d8604b
MJ
557 } catch(e) {
558 if (e in InterruptedException) {
559 build.setResult(hudson.model.Result.ABORTED)
560 throw new InterruptedException()
561 } else {
562 throw(e)
563 }
564 }
565
3a01c580 566 // Abort job if a newer instance is queued
b7efa3aa
MJ
567 if (!currentJobName.contains("gerrit")) {
568 similarJobQueued = Hudson.instance.queue.items.count{it.task.getFullDisplayName() == currentJobName}
569 if (similarJobQueued > 0) {
570 println "Abort, a newer instance of the job was queued"
483859f3
JR
571 build.setResult(hudson.model.Result.ABORTED)
572 throw new InterruptedException()
b7efa3aa 573 }
483859f3
JR
574 }
575
f3d8604b
MJ
576 def i = ongoingBuild.iterator()
577 while ( i.hasNext() ) {
578 currentBuild = i.next()
579 if ( currentBuild.isCancelled() || currentBuild.isDone() ) {
580 // Remove from queue
581 i.remove()
582
583 // Print results
584 def matrixParent = currentBuild.get()
585 allBuilds.add(matrixParent)
a1ae361e 586 def kernelStr = matrixParent.buildVariableResolver.resolve("ktag")
f3d8604b
MJ
587 println "${matrixParent.fullDisplayName} (${kernelStr}) completed with status ${matrixParent.result}"
588
589 // Process child runs of matrixBuild
590 def childRuns = matrixParent.getRuns()
591 for ( childRun in childRuns ) {
592 println "\t${childRun.fullDisplayName} (${kernelStr}) completed with status ${childRun.result}"
593 if (childRun.result != Result.SUCCESS) {
594 failedRuns.add(childRun)
595 isFailed = true
596 }
597 }
598 }
599 }
600 }
601}
602
603// Get log of failed runs
604for (failedRun in failedRuns) {
605 println "---START---"
606 failedRun.writeWholeLogTo(out)
607 println "---END---"
608}
609
610println "---Build report---"
611for (b in allBuilds) {
a1ae361e 612 def kernelStr = b.buildVariableResolver.resolve("ktag")
f3d8604b 613 println "${b.fullDisplayName} (${kernelStr}) completed with status ${b.result}"
7e02032c 614 // Cleanup builds
7e942863
MJ
615 try {
616 b.delete()
617 } catch (all) {}
f3d8604b
MJ
618}
619
620// Mark this build failed if any child build has failed
621if (isFailed) {
c5c05f73 622 build.setResult(hudson.model.Result.FAILURE)
f3d8604b
MJ
623}
624
625// EOF
This page took 0.093134 seconds and 4 git commands to generate.