1 enum KernelVersioning {
2 MAJOR,MINOR,REVISION,BUILD
5 class BasicVersion implements Comparable<BasicVersion> {
13 // Default Constructor
16 // Parse a version string of format X.Y.Z.W-A
17 BasicVersion(String version, String ref) {
21 if (version.contains('-')) {
23 token = version.tokenize('-')
24 tokenVersion = token[0]
25 if (token[1]?.isInteger()) {
26 rc = token[1].toInteger()
29 tokenVersion = version
32 tokenVersion = tokenVersion.tokenize('.')
34 def tagEnum = KernelVersioning.MAJOR
36 if (it?.isInteger()) {
38 case KernelVersioning.MAJOR:
39 major = it.toInteger()
40 tagEnum = KernelVersioning.MINOR
42 case KernelVersioning.MINOR:
43 minor = it.toInteger()
44 tagEnum = KernelVersioning.REVISION
46 case KernelVersioning.REVISION:
47 revision = it.toInteger()
48 tagEnum = KernelVersioning.BUILD
50 case KernelVersioning.BUILD:
51 build = it.toInteger()
55 println("Unsupported version extension")
56 println("Trying to parse: ${version}")
57 println("Invalid sub version value: ${it}")
58 //TODO: throw exception for jenkins
85 int compareTo(BasicVersion kernelVersion) {
86 return major <=> kernelVersion.major ?: minor <=> kernelVersion.minor ?: revision <=> kernelVersion.revision ?: build <=> kernelVersion.build ?: rc <=> kernelVersion.rc
90 def kernelTagCutOff = new BasicVersion("2.6.36", "")
91 def modulesBranches = ["master", "stable-2.5", "stable-2.6"]
93 //def modulesBranches = ["master","stable-2.5","stable-2.6", "stable-2.4"]
95 def linuxURL = "git://git.kernel.org/pub/scm/linux/kernel/git/stable/linux-stable.git"
96 def modulesURL = "https://github.com/lttng/lttng-modules.git"
98 // Linux specific variable
99 String linuxCheckoutTo = "linux-source"
100 String recipeCheckoutTo = "recipe"
101 String modulesCheckoutTo = "lttng-modules"
103 def linuxGitReference = "/home/jenkins/gitcache/linux-stable.git"
105 // Check if we are on jenkins
106 // Useful for outside jenkins devellopment related to groovy only scripting
107 def isJenkinsInstance = binding.variables.containsKey('JENKINS_HOME')
109 // Fetch tags and format
110 // Split the string into sections based on |
111 // And pipe the results together
112 String process = "git ls-remote -t $linuxURL | cut -c42- | sort -V"
113 def out = new StringBuilder()
114 def err = new StringBuilder()
115 Process result = process.tokenize( '|' ).inject( null ) { p, c ->
122 result.waitForProcessOutput(out,err)
124 if ( result.exitValue() == 0 ) {
125 def branches = out.readLines().collect {
126 // Scrap special string tag
127 it.replaceAll("\\^\\{\\}", '')
130 branches = branches.unique()
133 branches.each { branch ->
134 def stripBranch = branch.replaceAll("rc", '').replaceAll(/refs\/tags\/v/,'')
135 BasicVersion kVersion = new BasicVersion(stripBranch, branch)
136 versions.add(kVersion)
139 // Sort the version via Comparable implementation of KernelVersion
140 versions = versions.sort()
142 // Find the version cutoff
143 def cutoffPos = versions.findIndexOf{(it.major >= kernelTagCutOff.major) && (it.minor >= kernelTagCutOff.minor) && (it.revision >= kernelTagCutOff.revision) && (it.build >= kernelTagCutOff.build) && (it.rc >= kernelTagCutOff.rc)}
145 // If error set cutoff on last so no job are created
146 if (cutoffPos == -1) {
147 cutoffPos = versions.size()
149 // Get last version and include only last rc
152 last = versions.last()
154 int i = versions.size()-1
155 while (i > -1 && versions[i].rc != -1 ) {
160 lastNoRcPos = versions.size()
163 String modulesPrefix = "lttng-modules"
164 String kernelPrefix = "dsl-kernel"
165 String separator = "-"
168 println("CutOff index")
172 // Actual job creation
173 for (int i = cutoffPos; i < versions.size() ; i++) {
175 // Only create for valid build
176 if ( (i < lastNoRcPos && versions[i].rc == -1) || (i >= lastNoRcPos)) {
177 println ("Preparing job for")
179 String jobName = kernelPrefix + separator + versions[i].print()
181 // Generate modules job based on supported modules jobs
183 modulesBranches.each { branch ->
184 modulesJob[branch] = modulesPrefix + separator + branch + separator + jobName
189 if (isJenkinsInstance) {
190 matrixJob("${jobName}") {
191 using("linux-master")
197 branch(versions[i].gitRefs)
199 relativeTargetDir(linuxCheckoutTo)
200 reference(linuxGitReference)
205 downstream(it.value, 'SUCCESS')
210 // Corresponding Module job
211 modulesJob.each { job ->
212 println("\t" + job.key + " " + job.value)
213 if (isJenkinsInstance) {
214 matrixJob(job.value) {
222 branch(versions[i].gitRefs)
224 relativeTargetDir(linuxCheckoutTo)
225 reference(linuxGitReference)
233 relativeTargetDir(modulesCheckoutTo)
237 copyArtifacts("${jobName}/arch=\$arch,label=kernel", "linux-artifact/**", '', false, false) {
238 latestSuccessful(true) // Latest successful build
240 shell(readFileFromWorkspace('lttng-modules/lttng-modules-dsl-master.sh'))
248 // Trigger generations
249 def dslTriggerKernel = """\
250 import hudson.model.*
251 import jenkins.model.*
252 import hudson.AbortException
253 import hudson.console.HyperlinkNote
254 import java.util.concurrent.CancellationException
255 import java.util.Random
258 Random random = new Random()
259 def jobs = hudson.model.Hudson.instance.items
261 def jobStartWithKernel = "KERNELPREFIX"
262 def jobStartWithModule = "MODULEPREFIX"
269 def jobName = job.getName()
270 if (jobName.startsWith(jobStartWithKernel)) {
271 counter = counter + 1
272 def lastBuild = job.getLastBuild()
273 if (lastBuild == null || lastBuild.result != Result.SUCCESS) {
276 println("\tAlready built")
281 println "Kernel total "+ counter
282 println "Kernel to build "+ toBuild.size()
285 def kernelEnabledNode = 0
286 hudson.model.Hudson.instance.nodes.each { node ->
287 if (node.getLabelString().contains("kernel")){
291 println "Nb of live kernel enabled build node "+ kernelEnabledNode
293 def ongoingBuild = []
294 def q = jenkins.model.Jenkins.getInstance().getQueue()
296 while (toBuild.size() != 0) {
297 // Throttle the build with both the number of current parent task and queued
298 // task.Look for both kernel and downstream module from previous kernel.
299 def queuedTask = q.getItems().findAll {
300 it.task.getParent().name.startsWith(jobStartWithKernel) ||
301 it.task.getParent().name.startsWith(jobStartWithModule)
303 if ((ongoingBuild.size() <= kernelEnabledNode.intdiv(2)) && (queuedTask.size() < limitQueue)) {
304 def job = toBuild.pop()
305 ongoingBuild.push(job.scheduleBuild2(0))
306 println "\t trigering " + HyperlinkNote.encodeTo('/' + job.url, job.fullDisplayName)
308 println "Currently " + ongoingBuild.size() + " build currently on execution. Limit: " + kernelEnabledNode.intdiv(2)
309 println "Currently " + queuedTask.findAll{it.task.getParent().name.startsWith(jobStartWithModule)}.size() + " module jobs are queued. Limit: " + limitQueue
310 println "Currently " + queuedTask.findAll{it.task.getParent().name.startsWith(jobStartWithKernel)}.size() + " kernel jobs are queued. Limit: " + limitQueue
311 Thread.sleep(random.nextInt(60000))
312 ongoingBuild.removeAll{ it.isCancelled() || it.isDone() }
317 throw new AbortException("Some job failed")
320 def dslTriggerModule = """\
321 import hudson.model.*
322 import hudson.AbortException
323 import hudson.console.HyperlinkNote
324 import java.util.concurrent.CancellationException
325 import java.util.Random
328 Random random = new Random()
329 def jobs = hudson.model.Hudson.instance.items
331 def jobStartWith = "JOBPREFIX"
338 def jobName = job.getName()
339 if (jobName.startsWith(jobStartWith)) {
340 counter = counter + 1
346 def kernelEnabledNode = 0
347 hudson.model.Hudson.instance.nodes.each { node ->
348 if (node.getLabelString().contains("kernel")){
353 def ongoingBuild = []
354 def q = jenkins.model.Jenkins.getInstance().getQueue()
356 while (toBuild.size() != 0) {
357 // Throttle the build with both the number of current parent task and queued
358 // task.Look for both kernel and downstream module from previous kernel.
359 def queuedTask = q.getItems().findAll {
360 it.task.getParent().name.startsWith(jobStartWithKernel) ||
361 it.task.getParent().name.startsWith(jobStartWithModule)
364 if ((ongoingBuild.size() <= kernelEnabledNode.intdiv(2)) && (queuedTask.size() < limitQueue)) {
365 def job = toBuild.pop()
366 ongoingBuild.push(job.scheduleBuild2(0))
367 println "\\t trigering " + HyperlinkNote.encodeTo('/' + job.url, job.fullDisplayName)
369 Thread.sleep(random.nextInt(60000))
370 ongoingBuild.removeAll{ it.isCancelled() || it.isDone() }
375 throw new AbortException("Some job failed")
379 dslTriggerKernel = dslTriggerKernel.replaceAll("KERNELPREFIX", kernelPrefix)
380 dslTriggerKernel = dslTriggerKernel.replaceAll("MODULEPREFIX", modulesPrefix)
381 if (isJenkinsInstance) {
382 freeStyleJob("dsl-trigger-kernel") {
384 systemGroovyCommand(dslTriggerKernel)
391 modulesBranches.each { branch ->
392 dslTriggerModule = dslTriggerModule.replaceAll("JOBPREFIX",modulesPrefix + separator + branch + separator)
393 freeStyleJob("dsl-trigger-module-${branch}") {
395 systemGroovyCommand(dslTriggerModule)