Jakub Josef | 4edd743 | 2017-05-10 17:58:56 +0200 | [diff] [blame] | 1 | package com.mirantis.mk |
Jakub Josef | 93f08e2 | 2017-06-05 19:14:53 +0200 | [diff] [blame] | 2 | import com.cloudbees.groovy.cps.NonCPS |
Jakub Josef | 4edd743 | 2017-05-10 17:58:56 +0200 | [diff] [blame] | 3 | |
| 4 | /** |
| 5 | * |
| 6 | * Jenkins common functions |
| 7 | * |
| 8 | */ |
| 9 | |
| 10 | /** |
Kirill Mashchenko | 4300109 | 2018-12-25 05:28:53 +0400 | [diff] [blame] | 11 | * Returns a list of groups which user belongs |
| 12 | * @param username String |
| 13 | * @return list of groups [String] |
Jakub Josef | d44b697 | 2018-01-23 17:55:57 +0100 | [diff] [blame] | 14 | */ |
Kirill Mashchenko | 4300109 | 2018-12-25 05:28:53 +0400 | [diff] [blame] | 15 | def userGroups(username) { |
| 16 | res = [] |
| 17 | def authorities = Jenkins.instance.securityRealm.loadUserByUsername(username).getAuthorities() |
| 18 | authorities.each { |
| 19 | res.add(it.toString()) |
Jakub Josef | d44b697 | 2018-01-23 17:55:57 +0100 | [diff] [blame] | 20 | } |
Kirill Mashchenko | 4300109 | 2018-12-25 05:28:53 +0400 | [diff] [blame] | 21 | return res |
| 22 | } |
| 23 | |
| 24 | /** |
| 25 | * Check if user belongs to group |
| 26 | * @param username String |
| 27 | * @param group String |
| 28 | * @return boolean result |
| 29 | */ |
| 30 | def userInGroup(username, group) { |
| 31 | def authorities = userGroups(username) |
| 32 | return authorities.any{it==group} |
| 33 | } |
| 34 | |
| 35 | /** |
| 36 | * Check if user belongs to at least one of given groups |
| 37 | * @param username String |
| 38 | * @param groups [String] |
| 39 | * @return boolean result |
| 40 | */ |
| 41 | def userInGroups(username, groups) { |
| 42 | return groups.any{userInGroup(username, it)} |
| 43 | } |
| 44 | |
| 45 | /** |
| 46 | * Returns current username from build |
| 47 | * @return username String |
| 48 | */ |
| 49 | def currentUsername() { |
| 50 | username = '' |
| 51 | wrap([$class: 'BuildUser']) { |
Maxim Rasskazov | 2b7c3be | 2019-06-21 14:50:02 +0400 | [diff] [blame] | 52 | username = env.BUILD_USER_ID ?: 'jenkins' |
Kirill Mashchenko | 4300109 | 2018-12-25 05:28:53 +0400 | [diff] [blame] | 53 | } |
| 54 | if (username) { |
| 55 | return username |
| 56 | } else { |
| 57 | throw new Exception('cant get current username') |
| 58 | } |
| 59 | } |
| 60 | |
| 61 | /** |
| 62 | * Check if current user belongs to at least one of given groups |
| 63 | * @param groups [String] |
| 64 | * @return boolean result |
| 65 | */ |
| 66 | def currentUserInGroups(groups) { |
| 67 | username = currentUsername() |
| 68 | return userInGroups(username, groups) |
| 69 | } |
| 70 | |
| 71 | /** |
| 72 | * Check if current user belongs to group |
| 73 | * @param group String |
| 74 | * @return boolean result |
| 75 | */ |
| 76 | def currentUserInGroup(group) { |
| 77 | username = currentUsername() |
| 78 | return userInGroup(username, group) |
Jakub Josef | d44b697 | 2018-01-23 17:55:57 +0100 | [diff] [blame] | 79 | } |
| 80 | |
| 81 | /** |
Jakub Josef | 4edd743 | 2017-05-10 17:58:56 +0200 | [diff] [blame] | 82 | * Get Jenkins job running builds |
| 83 | * @param jobName job name |
| 84 | * @return list of running builds |
| 85 | */ |
Jakub Josef | fbe8c7c | 2017-05-11 13:35:11 +0200 | [diff] [blame] | 86 | @NonCPS |
Jakub Josef | 4edd743 | 2017-05-10 17:58:56 +0200 | [diff] [blame] | 87 | def getJobRunningBuilds(jobName){ |
Kirill Mashchenko | 4300109 | 2018-12-25 05:28:53 +0400 | [diff] [blame] | 88 | def job = Jenkins.instance.items.find{it -> it.name.equals(jobName)} |
| 89 | if(job){ |
| 90 | return job.builds.findAll{build -> build.isBuilding()} |
| 91 | } |
| 92 | return [] |
Jakub Josef | 93f08e2 | 2017-06-05 19:14:53 +0200 | [diff] [blame] | 93 | } |
| 94 | |
| 95 | @NonCPS |
| 96 | def getRunningBuilds(job){ |
Kirill Mashchenko | 4300109 | 2018-12-25 05:28:53 +0400 | [diff] [blame] | 97 | return job.builds.findAll{build -> build.isBuilding()} |
Jakub Josef | 93f08e2 | 2017-06-05 19:14:53 +0200 | [diff] [blame] | 98 | } |
| 99 | |
| 100 | @NonCPS |
| 101 | def killStuckBuilds(maxSeconds, job){ |
Kirill Mashchenko | 4300109 | 2018-12-25 05:28:53 +0400 | [diff] [blame] | 102 | def common = new com.mirantis.mk.Common() |
| 103 | def result = true |
| 104 | def runningBuilds = getRunningBuilds(job) |
| 105 | def jobName = job.name |
| 106 | for(int j=0; j < runningBuilds.size(); j++){ |
| 107 | int durationInSeconds = (System.currentTimeMillis() - runningBuilds[j].getTimeInMillis())/1000.0 |
| 108 | if(durationInSeconds > maxSeconds){ |
| 109 | result = false |
| 110 | def buildId = runningBuilds[j].id |
| 111 | common.infoMsg("Aborting ${jobName}-${buildId} which is running for ${durationInSeconds}s") |
| 112 | try{ |
| 113 | runningBuilds[j].finish(hudson.model.Result.ABORTED, new java.io.IOException("Aborting build by long running jobs killer")); |
| 114 | result = true |
| 115 | }catch(e){ |
| 116 | common.errorMsg("Error occured during aborting build: Exception: ${e}") |
| 117 | } |
| 118 | } |
Jakub Josef | 93f08e2 | 2017-06-05 19:14:53 +0200 | [diff] [blame] | 119 | } |
Kirill Mashchenko | 4300109 | 2018-12-25 05:28:53 +0400 | [diff] [blame] | 120 | return result |
Jakub Josef | d44b697 | 2018-01-23 17:55:57 +0100 | [diff] [blame] | 121 | } |
Richard Felkl | 838892f | 2018-06-12 17:58:20 +0200 | [diff] [blame] | 122 | |
| 123 | /** |
| 124 | * Get Jenkins job object |
| 125 | * @param jobName job name |
| 126 | * @return job object that matches jobName |
| 127 | */ |
Denis Egorenko | b1a40b6 | 2019-01-11 18:04:57 +0400 | [diff] [blame] | 128 | def getJobByName(jobName, regexp=false){ |
Richard Felkl | 838892f | 2018-06-12 17:58:20 +0200 | [diff] [blame] | 129 | for(item in Hudson.instance.items) { |
Denis Egorenko | b1a40b6 | 2019-01-11 18:04:57 +0400 | [diff] [blame] | 130 | if (regexp && item.name ==~ jobName || item.name == jobName) { |
Richard Felkl | 838892f | 2018-06-12 17:58:20 +0200 | [diff] [blame] | 131 | return item |
| 132 | } |
| 133 | } |
| 134 | } |
| 135 | |
| 136 | /** |
| 137 | * Get Jenkins job parameters |
| 138 | * @param jobName job name |
| 139 | * @return HashMap with parameter names as keys and their values as values |
| 140 | */ |
| 141 | def getJobParameters(jobName){ |
| 142 | def job = getJobByName(jobName) |
| 143 | def prop = job.getProperty(ParametersDefinitionProperty.class) |
| 144 | def params = new java.util.HashMap<String,String>() |
| 145 | if(prop != null) { |
| 146 | for(param in prop.getParameterDefinitions()) { |
| 147 | params.put(param.name, param.defaultValue) |
| 148 | } |
| 149 | } |
| 150 | return params |
| 151 | } |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 152 | |
| 153 | /** |
| 154 | * Get list of causes actions for given build |
| 155 | * |
| 156 | * @param build Job build object (like, currentBuild.rawBuild) |
| 157 | * @return list of causes actions for given build |
| 158 | */ |
| 159 | @NonCPS |
| 160 | def getBuildCauseActions(build) { |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 161 | for(action in build.actions) { |
| 162 | if (action instanceof hudson.model.CauseAction) { |
| 163 | return action.causes |
| 164 | } |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 165 | } |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 166 | return [] |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 167 | } |
| 168 | |
| 169 | /** |
| 170 | * Get list of builds, triggered by Gerrit with given build |
| 171 | * @param build Job build object (like, currentBuild.rawBuild) |
| 172 | * @return list of builds with names and numbers |
| 173 | */ |
| 174 | @NonCPS |
| 175 | def getGerritBuildContext(build) { |
| 176 | def causes = getBuildCauseActions(build) |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 177 | for(cause in causes) { |
| 178 | if (cause instanceof com.sonyericsson.hudson.plugins.gerrit.trigger.hudsontrigger.GerritCause) { |
| 179 | return cause.context.getOtherBuilds() |
| 180 | } |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 181 | } |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 182 | return [] |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 183 | } |
| 184 | |
| 185 | /** |
| 186 | * Wait for other jobs |
| 187 | * @param config config parameter: |
| 188 | * builds - List of job build objects, which should be checked |
| 189 | * checkBuilds - List of job names or regexps, which should be used to check provided builds list |
| 190 | * regexp - Wheither to use regexp or simple string matching |
| 191 | */ |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 192 | @NonCPS |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 193 | def waitForOtherBuilds(LinkedHashMap config){ |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 194 | def context = config.get('context', 'gerrit') |
| 195 | def builds = [] |
| 196 | if (context == 'gerrit') { |
| 197 | builds = getGerritBuildContext(currentBuild.rawBuild) |
| 198 | } else if (context == 'custom') { |
| 199 | builds = config.get('builds') |
| 200 | } |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 201 | def checkBuilds = config.get('checkBuilds') |
| 202 | def regexp = config.get('regexp', false) |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 203 | |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 204 | def waitForBuilds = builds.findAll { build -> |
| 205 | def jobName = build.fullDisplayName.tokenize(' ')[0] |
| 206 | if (regexp) { |
| 207 | checkBuilds.find { jobName ==~ it } |
| 208 | } else { |
| 209 | jobName in checkBuilds |
| 210 | } |
| 211 | } |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 212 | |
| 213 | def buildsMap = [] |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 214 | if (waitForBuilds) { |
| 215 | def waiting = true |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 216 | print "\u001B[36mWaiting for next jobs: ${waitForBuilds}\u001B[0m" |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 217 | while(waiting) { |
| 218 | waiting = false |
| 219 | waitForBuilds.each { job -> |
| 220 | if (job.inProgress) { |
| 221 | waiting = true |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 222 | } else { |
| 223 | buildInfo = [ |
| 224 | 'jobName': job.fullDisplayName.tokenize(' ')[0], |
| 225 | 'jobNumber': job.number, |
| 226 | ] |
| 227 | buildsMap.add(buildInfo) |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 228 | } |
| 229 | } |
| 230 | } |
| 231 | } |
Denis Egorenko | 270c530 | 2019-01-10 20:53:01 +0400 | [diff] [blame] | 232 | return buildsMap |
Denis Egorenko | 46ff138 | 2018-12-14 15:46:36 +0400 | [diff] [blame] | 233 | } |
azvyagintsev | eb81735 | 2019-11-08 13:30:17 +0200 | [diff] [blame] | 234 | |
| 235 | /** |
| 236 | * Check dependency jobs passed successfully |
| 237 | |
Владислав Наумов | 90caa88 | 2020-08-12 17:20:12 +0200 | [diff] [blame] | 238 | * @param block (bool) Block child jobs in case of parent dependencies failed |
| 239 | * @param allowNotBuilt (bool) Approve not_built status of the dependency job |
| 240 | * @return (map)[ |
| 241 | * status: (bool) True if there are no failed dependencies |
| 242 | * log: (string) Verbose description |
| 243 | * ] |
azvyagintsev | eb81735 | 2019-11-08 13:30:17 +0200 | [diff] [blame] | 244 | */ |
Владислав Наумов | 90caa88 | 2020-08-12 17:20:12 +0200 | [diff] [blame] | 245 | def checkDependencyJobs(block = true, allowNotBuilt = false) { |
azvyagintsev | eb81735 | 2019-11-08 13:30:17 +0200 | [diff] [blame] | 246 | def common = new com.mirantis.mk.Common() |
Владислав Наумов | 90caa88 | 2020-08-12 17:20:12 +0200 | [diff] [blame] | 247 | def acceptedStatuses = ['SUCCESS'] |
| 248 | if (allowNotBuilt) { |
| 249 | acceptedStatuses.add('NOT_BUILT') |
| 250 | } |
| 251 | |
azvyagintsev | eb81735 | 2019-11-08 13:30:17 +0200 | [diff] [blame] | 252 | depList = [] |
| 253 | if (env.TRIGGER_DEPENDENCY_KEYS){ |
| 254 | common.infoMsg('Job may depends on parent jobs, check if dependency jobs exist...') |
| 255 | depKeys = env.TRIGGER_DEPENDENCY_KEYS.toString() |
| 256 | depList = depKeys.split() |
| 257 | if (depList){ |
Владислав Наумов | 90caa88 | 2020-08-12 17:20:12 +0200 | [diff] [blame] | 258 | common.infoMsg("Here is dependency jobs-list: ${depList} , accepted job statuses are: ${acceptedStatuses}") |
azvyagintsev | eb81735 | 2019-11-08 13:30:17 +0200 | [diff] [blame] | 259 | for (String item : depList) { |
| 260 | prjName = item.replaceAll('[^a-zA-Z0-9]+', '_') |
| 261 | triggerResult = 'TRIGGER_' + prjName.toUpperCase() + '_BUILD_RESULT' |
| 262 | triggerJobName = 'TRIGGER_' + prjName.toUpperCase() + '_BUILD_NAME' |
| 263 | triggerJobBuild = 'TRIGGER_' + prjName.toUpperCase() + '_BUILD_NUMBER' |
Владислав Наумов | 90caa88 | 2020-08-12 17:20:12 +0200 | [diff] [blame] | 264 | if (!acceptedStatuses.contains(env.getProperty(triggerResult))) { |
azvyagintsev | eb81735 | 2019-11-08 13:30:17 +0200 | [diff] [blame] | 265 | msg = "Dependency job ${env.getProperty(triggerJobName)} #${env.getProperty(triggerJobBuild)} is ${env.getProperty(triggerResult)}" |
| 266 | common.warningMsg(msg) |
| 267 | if (block){ |
| 268 | currentBuild.result = 'NOT_BUILT' |
| 269 | currentBuild.description = msg |
| 270 | } |
| 271 | return [status: false, log: msg, jobs: depList] |
| 272 | } |
| 273 | } |
| 274 | } |
| 275 | } else { |
| 276 | common.infoMsg('There is no job-dependencies') |
| 277 | } |
| 278 | return [status: true, log: '', jobs: depList] |
| 279 | } |
Владислав Наумов | 34e9ed8 | 2021-02-02 17:04:56 +0100 | [diff] [blame] | 280 | |
| 281 | /** |
| 282 | * Return jenkins infra metadata according to specified jenkins intstance |
| 283 | |
| 284 | * @param jenkinsServerURL (string) URL to jenkins server in form: env.JENKINS_URL |
| 285 | * @return (map)[ |
| 286 | * jenkins_service_user: (string) name of jenkins user needed for gerrit ops |
| 287 | * ] |
| 288 | */ |
| 289 | def getJenkinsInfraMetadata(jenkinsServerURL) { |
| 290 | def meta = [ |
| 291 | jenkins_service_user: '', |
| 292 | ] |
| 293 | |
| 294 | switch (jenkinsServerURL) { |
| 295 | case 'https://ci.mcp.mirantis.net/': |
| 296 | meta['jenkins_service_user'] = 'mcp-jenkins' |
| 297 | break |
| 298 | case 'https://mcc-ci.infra.mirantis.net/': |
| 299 | meta['jenkins_service_user'] = 'mcc-ci-jenkins' |
| 300 | break |
| 301 | default: |
| 302 | error("Failed to detect jenkins service user, supported jenkins platforms: 'https://ci.mcp.mirantis.net/' 'https://mcc-ci.infra.mirantis.net/'") |
| 303 | } |
| 304 | |
| 305 | return meta |
| 306 | } |
Владислав Наумов | 468bc74 | 2021-04-08 14:23:43 +0200 | [diff] [blame] | 307 | |
| 308 | /** |
| 309 | * Get list of all jenkins workers matched desired label |
| 310 | * |
| 311 | * @param labelString (string) desired worker label |
| 312 | * @return (list) all workers, currently matched label |
| 313 | */ |
| 314 | @NonCPS |
| 315 | def getWorkers(String labelString = null) { |
| 316 | def workerLabel = hudson.model.labels.LabelAtom.get(labelString) |
| 317 | def workers = [] |
| 318 | hudson.model.Hudson.instance.slaves.each { |
| 319 | if (it.getComputer().isOnline()) { |
| 320 | if (workerLabel) { |
| 321 | if (workerLabel in it.getAssignedLabels()) { |
| 322 | workers << it.name |
| 323 | } |
| 324 | } else { |
| 325 | // if labelString is null, getting all workers |
| 326 | workers << it.name |
| 327 | } |
| 328 | } |
| 329 | } |
| 330 | return workers |
| 331 | } |