blob: 52eeb622e1f7c026a58e476a947526be3bd3a353 [file] [log] [blame]
Jakub Josef4edd7432017-05-10 17:58:56 +02001package com.mirantis.mk
Jakub Josef93f08e22017-06-05 19:14:53 +02002import com.cloudbees.groovy.cps.NonCPS
Jakub Josef4edd7432017-05-10 17:58:56 +02003
4/**
5 *
6 * Jenkins common functions
7 *
8 */
9
10/**
Kirill Mashchenko43001092018-12-25 05:28:53 +040011 * Returns a list of groups which user belongs
12 * @param username String
13 * @return list of groups [String]
Jakub Josefd44b6972018-01-23 17:55:57 +010014 */
Kirill Mashchenko43001092018-12-25 05:28:53 +040015def userGroups(username) {
16 res = []
17 def authorities = Jenkins.instance.securityRealm.loadUserByUsername(username).getAuthorities()
18 authorities.each {
19 res.add(it.toString())
Jakub Josefd44b6972018-01-23 17:55:57 +010020 }
Kirill Mashchenko43001092018-12-25 05:28:53 +040021 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 */
30def 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 */
41def userInGroups(username, groups) {
42 return groups.any{userInGroup(username, it)}
43}
44
45/**
46 * Returns current username from build
47 * @return username String
48 */
49def currentUsername() {
50 username = ''
51 wrap([$class: 'BuildUser']) {
Maxim Rasskazov2b7c3be2019-06-21 14:50:02 +040052 username = env.BUILD_USER_ID ?: 'jenkins'
Kirill Mashchenko43001092018-12-25 05:28:53 +040053 }
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 */
66def 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 */
76def currentUserInGroup(group) {
77 username = currentUsername()
78 return userInGroup(username, group)
Jakub Josefd44b6972018-01-23 17:55:57 +010079}
80
81/**
Jakub Josef4edd7432017-05-10 17:58:56 +020082 * Get Jenkins job running builds
83 * @param jobName job name
84 * @return list of running builds
85 */
Jakub Joseffbe8c7c2017-05-11 13:35:11 +020086@NonCPS
Jakub Josef4edd7432017-05-10 17:58:56 +020087def getJobRunningBuilds(jobName){
Kirill Mashchenko43001092018-12-25 05:28:53 +040088 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 Josef93f08e22017-06-05 19:14:53 +020093}
94
95@NonCPS
96def getRunningBuilds(job){
Kirill Mashchenko43001092018-12-25 05:28:53 +040097 return job.builds.findAll{build -> build.isBuilding()}
Jakub Josef93f08e22017-06-05 19:14:53 +020098}
99
100@NonCPS
101def killStuckBuilds(maxSeconds, job){
Kirill Mashchenko43001092018-12-25 05:28:53 +0400102 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 Josef93f08e22017-06-05 19:14:53 +0200119 }
Kirill Mashchenko43001092018-12-25 05:28:53 +0400120 return result
Jakub Josefd44b6972018-01-23 17:55:57 +0100121}
Richard Felkl838892f2018-06-12 17:58:20 +0200122
123/**
124 * Get Jenkins job object
125 * @param jobName job name
126 * @return job object that matches jobName
127 */
Denis Egorenkob1a40b62019-01-11 18:04:57 +0400128def getJobByName(jobName, regexp=false){
Richard Felkl838892f2018-06-12 17:58:20 +0200129 for(item in Hudson.instance.items) {
Denis Egorenkob1a40b62019-01-11 18:04:57 +0400130 if (regexp && item.name ==~ jobName || item.name == jobName) {
Richard Felkl838892f2018-06-12 17:58:20 +0200131 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 */
141def 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 Egorenko46ff1382018-12-14 15:46:36 +0400152
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
160def getBuildCauseActions(build) {
Denis Egorenko270c5302019-01-10 20:53:01 +0400161 for(action in build.actions) {
162 if (action instanceof hudson.model.CauseAction) {
163 return action.causes
164 }
Denis Egorenko46ff1382018-12-14 15:46:36 +0400165 }
Denis Egorenko270c5302019-01-10 20:53:01 +0400166 return []
Denis Egorenko46ff1382018-12-14 15:46:36 +0400167}
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
175def getGerritBuildContext(build) {
176 def causes = getBuildCauseActions(build)
Denis Egorenko270c5302019-01-10 20:53:01 +0400177 for(cause in causes) {
178 if (cause instanceof com.sonyericsson.hudson.plugins.gerrit.trigger.hudsontrigger.GerritCause) {
179 return cause.context.getOtherBuilds()
180 }
Denis Egorenko46ff1382018-12-14 15:46:36 +0400181 }
Denis Egorenko270c5302019-01-10 20:53:01 +0400182 return []
Denis Egorenko46ff1382018-12-14 15:46:36 +0400183}
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 Egorenko270c5302019-01-10 20:53:01 +0400192@NonCPS
Denis Egorenko46ff1382018-12-14 15:46:36 +0400193def waitForOtherBuilds(LinkedHashMap config){
Denis Egorenko270c5302019-01-10 20:53:01 +0400194 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 Egorenko46ff1382018-12-14 15:46:36 +0400201 def checkBuilds = config.get('checkBuilds')
202 def regexp = config.get('regexp', false)
Denis Egorenko270c5302019-01-10 20:53:01 +0400203
Denis Egorenko46ff1382018-12-14 15:46:36 +0400204 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 Egorenko270c5302019-01-10 20:53:01 +0400212
213 def buildsMap = []
Denis Egorenko46ff1382018-12-14 15:46:36 +0400214 if (waitForBuilds) {
215 def waiting = true
Denis Egorenko270c5302019-01-10 20:53:01 +0400216 print "\u001B[36mWaiting for next jobs: ${waitForBuilds}\u001B[0m"
Denis Egorenko46ff1382018-12-14 15:46:36 +0400217 while(waiting) {
218 waiting = false
219 waitForBuilds.each { job ->
220 if (job.inProgress) {
221 waiting = true
Denis Egorenko270c5302019-01-10 20:53:01 +0400222 } else {
223 buildInfo = [
224 'jobName': job.fullDisplayName.tokenize(' ')[0],
225 'jobNumber': job.number,
226 ]
227 buildsMap.add(buildInfo)
Denis Egorenko46ff1382018-12-14 15:46:36 +0400228 }
229 }
230 }
231 }
Denis Egorenko270c5302019-01-10 20:53:01 +0400232 return buildsMap
Denis Egorenko46ff1382018-12-14 15:46:36 +0400233}
azvyagintseveb817352019-11-08 13:30:17 +0200234
235/**
236 * Check dependency jobs passed successfully
237
Владислав Наумов90caa882020-08-12 17:20:12 +0200238 * @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 * ]
azvyagintseveb817352019-11-08 13:30:17 +0200244 */
Владислав Наумов90caa882020-08-12 17:20:12 +0200245def checkDependencyJobs(block = true, allowNotBuilt = false) {
azvyagintseveb817352019-11-08 13:30:17 +0200246 def common = new com.mirantis.mk.Common()
Владислав Наумов90caa882020-08-12 17:20:12 +0200247 def acceptedStatuses = ['SUCCESS']
248 if (allowNotBuilt) {
249 acceptedStatuses.add('NOT_BUILT')
250 }
251
azvyagintseveb817352019-11-08 13:30:17 +0200252 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){
Владислав Наумов90caa882020-08-12 17:20:12 +0200258 common.infoMsg("Here is dependency jobs-list: ${depList} , accepted job statuses are: ${acceptedStatuses}")
azvyagintseveb817352019-11-08 13:30:17 +0200259 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'
Владислав Наумов90caa882020-08-12 17:20:12 +0200264 if (!acceptedStatuses.contains(env.getProperty(triggerResult))) {
azvyagintseveb817352019-11-08 13:30:17 +0200265 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}
Владислав Наумов34e9ed82021-02-02 17:04:56 +0100280
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 */
289def 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}
Владислав Наумов468bc742021-04-08 14:23:43 +0200307
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
315def 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}