blob: 8926e768e8a9a484f17474ce46f5acb74ad05868 [file] [log] [blame]
James E. Blairee743612012-05-29 14:49:32 -07001# Copyright 2012 Hewlett-Packard Development Company, L.P.
2#
3# Licensed under the Apache License, Version 2.0 (the "License"); you may
4# not use this file except in compliance with the License. You may obtain
5# a copy of the License at
6#
7# http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12# License for the specific language governing permissions and limitations
13# under the License.
14
15import re
James E. Blairff986a12012-05-30 14:56:51 -070016import time
James E. Blairee743612012-05-29 14:49:32 -070017
James E. Blair1e8dd892012-05-30 09:15:05 -070018
James E. Blair4aea70c2012-07-26 14:23:24 -070019class Pipeline(object):
20 """A top-level pipeline such as check, gate, post, etc."""
21 def __init__(self, name):
22 self.name = name
23 self.job_trees = {} # project -> JobTree
24 self.manager = None
25
James E. Blaird09c17a2012-08-07 09:23:14 -070026 def __repr__(self):
27 return '<Pipeline %s>' % self.name
28
James E. Blair4aea70c2012-07-26 14:23:24 -070029 def setManager(self, manager):
30 self.manager = manager
31
32 def addProject(self, project):
33 job_tree = JobTree(None) # Null job == job tree root
34 self.job_trees[project] = job_tree
35 return job_tree
36
37 def getProjects(self):
38 return self.job_trees.keys()
39
40 def getJobTree(self, project):
41 tree = self.job_trees.get(project)
42 return tree
43
44 def getJobs(self, changeish):
James E. Blaird09c17a2012-08-07 09:23:14 -070045 tree = self.getJobTree(changeish.project)
James E. Blair4aea70c2012-07-26 14:23:24 -070046 if not tree:
47 return []
48 return changeish.filterJobs(tree.getJobs())
49
50 def _findJobsToRun(self, job_trees, changeish):
51 torun = []
52 if changeish.change_ahead:
53 # Only run jobs if any 'hold' jobs on the change ahead
54 # have completed successfully.
55 if self.isHoldingFollowingChanges(changeish.change_ahead):
56 return []
57 for tree in job_trees:
58 job = tree.job
59 result = None
60 if job:
61 if not job.changeMatches(changeish):
62 continue
63 build = changeish.current_build_set.getBuild(job.name)
64 if build:
65 result = build.result
66 else:
67 # There is no build for the root of this job tree,
68 # so we should run it.
69 torun.append(job)
70 # If there is no job, this is a null job tree, and we should
71 # run all of its jobs.
72 if result == 'SUCCESS' or not job:
73 torun.extend(self._findJobsToRun(tree.job_trees, changeish))
74 return torun
75
76 def findJobsToRun(self, changeish):
James E. Blaird09c17a2012-08-07 09:23:14 -070077 tree = self.getJobTree(changeish.project)
James E. Blair4aea70c2012-07-26 14:23:24 -070078 if not tree:
79 return []
80 return self._findJobsToRun(tree.job_trees, changeish)
81
82 def areAllJobsComplete(self, changeish):
83 for job in self.getJobs(changeish):
84 build = changeish.current_build_set.getBuild(job.name)
85 if not build or not build.result:
86 return False
87 return True
88
89 def didAllJobsSucceed(self, changeish):
90 for job in self.getJobs(changeish):
91 build = changeish.current_build_set.getBuild(job.name)
92 if not build:
93 return False
94 if build.result != 'SUCCESS':
95 return False
96 return True
97
98 def didAnyJobFail(self, changeish):
99 for job in self.getJobs(changeish):
100 build = changeish.current_build_set.getBuild(job.name)
101 if build and build.result == 'FAILURE':
102 return True
103 return False
104
105 def isHoldingFollowingChanges(self, changeish):
106 for job in self.getJobs(changeish):
107 if not job.hold_following_changes:
108 continue
109 build = changeish.current_build_set.getBuild(job.name)
110 if not build:
111 return True
112 if build.result != 'SUCCESS':
113 return True
114 if not changeish.change_ahead:
115 return False
116 return self.isHoldingFollowingChanges(changeish.change_ahead)
117
118 def formatStatus(self, changeish, indent=0, html=False):
119 indent_str = ' ' * indent
120 ret = ''
121 if html and changeish.url is not None:
122 ret += '%sProject %s change <a href="%s">%s</a>\n' % (
123 indent_str,
124 changeish.project.name,
125 changeish.url,
126 changeish._id())
127 else:
128 ret += '%sProject %s change %s\n' % (indent_str,
129 changeish.project.name,
130 changeish._id())
131 for job in self.getJobs(changeish):
132 build = changeish.current_build_set.getBuild(job.name)
133 if build:
134 result = build.result
135 else:
136 result = None
137 job_name = job.name
138 if html:
139 if build:
140 url = build.url
141 else:
142 url = None
143 if url is not None:
144 job_name = '<a href="%s">%s</a>' % (url, job_name)
145 ret += '%s %s: %s' % (indent_str, job_name, result)
146 ret += '\n'
147 if changeish.change_ahead:
148 ret += '%sWaiting on:\n' % (indent_str)
149 ret += self.formatStatus(changeish.change_ahead,
150 indent + 2, html)
151 return ret
152
153 def formatReport(self, changeish):
154 ret = ''
155 if self.didAllJobsSucceed(changeish):
156 ret += 'Build successful\n\n'
157 else:
158 ret += 'Build failed\n\n'
159
160 for job in self.getJobs(changeish):
161 build = changeish.current_build_set.getBuild(job.name)
162 result = build.result
163 if result == 'SUCCESS' and job.success_message:
164 result = job.success_message
165 elif result == 'FAILURE' and job.failure_message:
166 result = job.failure_message
167 url = build.url
168 if not url:
169 url = job.name
170 ret += '- %s : %s\n' % (url, result)
171 return ret
172
173 def formatDescription(self, build):
174 concurrent_changes = ''
175 concurrent_builds = ''
176 other_builds = ''
177
178 for change in build.build_set.other_changes:
179 concurrent_changes += '<li><a href="{change.url}">\
180 {change.number},{change.patchset}</a></li>'.format(
181 change=change)
182
183 change = build.build_set.change
184
185 for build in build.build_set.getBuilds():
186 if build.base_url:
187 concurrent_builds += """\
188<li>
189 <a href="{build.base_url}">
190 {build.job.name} #{build.number}</a>: {build.result}
191</li>
192""".format(build=build)
193 else:
194 concurrent_builds += """\
195<li>
196 {build.job.name}: {build.result}
197</li>""".format(build=build)
198
199 if build.build_set.previous_build_set:
200 other_build = build.build_set.previous_build_set.getBuild(
201 build.job.name)
202 if other_build:
203 other_builds += """\
204<li>
205 Preceded by: <a href="{build.base_url}">
206 {build.job.name} #{build.number}</a>
207</li>
208""".format(build=other_build)
209
210 if build.build_set.next_build_set:
211 other_build = build.build_set.next_build_set.getBuild(
212 build.job.name)
213 if other_build:
214 other_builds += """\
215<li>
216 Succeeded by: <a href="{build.base_url}">
217 {build.job.name} #{build.number}</a>
218</li>
219""".format(build=other_build)
220
221 result = build.build_set.result
222
James E. Blairbe765db2012-08-07 08:36:20 -0700223 if hasattr(change, 'number'):
James E. Blair4aea70c2012-07-26 14:23:24 -0700224 ret = """\
225<p>
226 Triggered by change:
227 <a href="{change.url}">{change.number},{change.patchset}</a><br/>
228 Branch: <b>{change.branch}</b><br/>
229 Pipeline: <b>{self.name}</b>
230</p>"""
231 else:
232 ret = """\
233<p>
234 Triggered by reference:
235 {change.ref}</a><br/>
236 Old revision: <b>{change.oldrev}</b><br/>
237 New revision: <b>{change.newrev}</b><br/>
238 Pipeline: <b>{self.name}</b>
239</p>"""
240
241 if concurrent_changes:
242 ret += """\
243<p>
244 Other changes tested concurrently with this change:
245 <ul>{concurrent_changes}</ul>
246</p>
247"""
248 if concurrent_builds:
249 ret += """\
250<p>
251 All builds for this change set:
252 <ul>{concurrent_builds}</ul>
253</p>
254"""
255
256 if other_builds:
257 ret += """\
258<p>
259 Other build sets for this change:
260 <ul>{other_builds}</ul>
261</p>
262"""
263 if result:
264 ret += """\
265<p>
266 Reported result: <b>{result}</b>
267</p>
268"""
269
270 ret = ret.format(**locals())
271 return ret
272
273 def setResult(self, changeish, build):
274 if build.result != 'SUCCESS':
275 # Get a JobTree from a Job so we can find only its dependent jobs
276 root = self.getJobTree(changeish.project)
277 tree = root.getJobTreeForJob(build.job)
278 for job in tree.getJobs():
279 fakebuild = Build(job, None)
280 fakebuild.result = 'SKIPPED'
281 changeish.addBuild(fakebuild)
282
283
James E. Blairee743612012-05-29 14:49:32 -0700284class ChangeQueue(object):
James E. Blair4aea70c2012-07-26 14:23:24 -0700285 """DependentPipelines have multiple parallel queues shared by
286 different projects; this is one of them. For instance, there may
287 a queue shared by interrelated projects foo and bar, and a second
288 queue for independent project baz. Pipelines have one or more
289 PipelineQueues."""
290 def __init__(self, pipeline):
291 self.pipeline = pipeline
James E. Blairee743612012-05-29 14:49:32 -0700292 self.name = ''
James E. Blairee743612012-05-29 14:49:32 -0700293 self.projects = []
294 self._jobs = set()
295 self.queue = []
296
James E. Blair9f9667e2012-06-12 17:51:08 -0700297 def __repr__(self):
James E. Blair4aea70c2012-07-26 14:23:24 -0700298 return '<ChangeQueue %s: %s>' % (self.pipeline.name, self.name)
James E. Blairee743612012-05-29 14:49:32 -0700299
300 def getJobs(self):
301 return self._jobs
302
303 def addProject(self, project):
304 if project not in self.projects:
305 self.projects.append(project)
306 names = [x.name for x in self.projects]
307 names.sort()
308 self.name = ', '.join(names)
James E. Blair4aea70c2012-07-26 14:23:24 -0700309 self._jobs |= set(self.pipeline.getJobTree(project).getJobs())
James E. Blairee743612012-05-29 14:49:32 -0700310
311 def enqueueChange(self, change):
312 if self.queue:
313 self.queue[-1].change_behind = change
314 change.change_ahead = self.queue[-1]
315 self.queue.append(change)
316 change.queue = self
317
318 def dequeueChange(self, change):
319 if change in self.queue:
320 self.queue.remove(change)
321
322 def mergeChangeQueue(self, other):
323 for project in other.projects:
324 self.addProject(project)
325
James E. Blair1e8dd892012-05-30 09:15:05 -0700326
James E. Blair4aea70c2012-07-26 14:23:24 -0700327class Project(object):
328 def __init__(self, name):
329 self.name = name
330
331 def __str__(self):
332 return self.name
333
334 def __repr__(self):
335 return '<Project %s>' % (self.name)
336
337
James E. Blairee743612012-05-29 14:49:32 -0700338class Job(object):
339 def __init__(self, name):
James E. Blair222d4982012-07-16 09:31:19 -0700340 # If you add attributes here, be sure to add them to the copy method.
James E. Blairee743612012-05-29 14:49:32 -0700341 self.name = name
342 self.failure_message = None
343 self.success_message = None
James E. Blaire5a847f2012-07-10 15:29:14 -0700344 self.parameter_function = None
James E. Blair222d4982012-07-16 09:31:19 -0700345 self.hold_following_changes = False
James E. Blaire421a232012-07-25 16:59:21 -0700346 self.branches = []
347 self._branches = []
James E. Blairee743612012-05-29 14:49:32 -0700348
349 def __str__(self):
350 return self.name
351
352 def __repr__(self):
353 return '<Job %s>' % (self.name)
354
James E. Blairb0954652012-06-01 11:32:01 -0700355 def copy(self, other):
356 self.failure_message = other.failure_message
James E. Blair15437412012-07-18 08:45:55 -0700357 self.success_message = other.success_message
James E. Blair222d4982012-07-16 09:31:19 -0700358 self.parameter_function = other.parameter_function
359 self.hold_following_changes = other.hold_following_changes
James E. Blaire421a232012-07-25 16:59:21 -0700360 self.branches = other.branches[:]
361 self._branches = other._branches[:]
James E. Blairb0954652012-06-01 11:32:01 -0700362
James E. Blaire421a232012-07-25 16:59:21 -0700363 def changeMatches(self, change):
364 if not self.branches:
James E. Blaire5a847f2012-07-10 15:29:14 -0700365 return True
James E. Blaire421a232012-07-25 16:59:21 -0700366 for branch in self.branches:
367 if branch.match(change.branch):
James E. Blaire5a847f2012-07-10 15:29:14 -0700368 return True
369 return False
370
James E. Blair1e8dd892012-05-30 09:15:05 -0700371
James E. Blairee743612012-05-29 14:49:32 -0700372class JobTree(object):
373 """ A JobTree represents an instance of one Job, and holds JobTrees
374 whose jobs should be run if that Job succeeds. A root node of a
375 JobTree will have no associated Job. """
376
377 def __init__(self, job):
378 self.job = job
379 self.job_trees = []
380
381 def addJob(self, job):
382 if job not in [x.job for x in self.job_trees]:
383 t = JobTree(job)
384 self.job_trees.append(t)
385 return t
386
387 def getJobs(self):
388 jobs = []
389 for x in self.job_trees:
390 jobs.append(x.job)
391 jobs.extend(x.getJobs())
392 return jobs
393
394 def getJobTreeForJob(self, job):
395 if self.job == job:
396 return self
397 for tree in self.job_trees:
398 ret = tree.getJobTreeForJob(job)
399 if ret:
400 return ret
401 return None
402
James E. Blair1e8dd892012-05-30 09:15:05 -0700403
James E. Blair4aea70c2012-07-26 14:23:24 -0700404class Build(object):
405 def __init__(self, job, uuid):
406 self.job = job
407 self.uuid = uuid
408 self.base_url = None
409 self.url = None
410 self.number = None
411 self.result = None
412 self.build_set = None
413 self.launch_time = time.time()
James E. Blairee743612012-05-29 14:49:32 -0700414
415 def __repr__(self):
James E. Blair4aea70c2012-07-26 14:23:24 -0700416 return '<Build %s of %s>' % (self.uuid, self.job.name)
James E. Blairee743612012-05-29 14:49:32 -0700417
James E. Blair1e8dd892012-05-30 09:15:05 -0700418
James E. Blair7e530ad2012-07-03 16:12:28 -0700419class BuildSet(object):
James E. Blair11700c32012-07-05 17:50:05 -0700420 def __init__(self, change):
421 self.change = change
422 self.other_changes = []
James E. Blair7e530ad2012-07-03 16:12:28 -0700423 self.builds = {}
James E. Blair11700c32012-07-05 17:50:05 -0700424 self.result = None
425 self.next_build_set = None
426 self.previous_build_set = None
James E. Blair7e530ad2012-07-03 16:12:28 -0700427
428 def addBuild(self, build):
429 self.builds[build.job.name] = build
430 build.build_set = self
431
James E. Blair11700c32012-07-05 17:50:05 -0700432 # The change isn't enqueued until after it's created
433 # so we don't know what the other changes ahead will be
434 # until jobs start.
435 if not self.other_changes:
436 next_change = self.change.change_ahead
437 while next_change:
438 self.other_changes.append(next_change)
439 next_change = next_change.change_ahead
440
James E. Blair7e530ad2012-07-03 16:12:28 -0700441 def getBuild(self, job_name):
442 return self.builds.get(job_name)
443
James E. Blair11700c32012-07-05 17:50:05 -0700444 def getBuilds(self):
445 keys = self.builds.keys()
446 keys.sort()
447 return [self.builds.get(x) for x in keys]
448
James E. Blair7e530ad2012-07-03 16:12:28 -0700449
James E. Blair4aea70c2012-07-26 14:23:24 -0700450class Changeish(object):
451 """Something like a change; either a change or a ref"""
452 is_reportable = False
James E. Blair32663402012-06-01 10:04:18 -0700453
James E. Blair4aea70c2012-07-26 14:23:24 -0700454 def __init__(self, project):
455 self.project = project
James E. Blair7e530ad2012-07-03 16:12:28 -0700456 self.build_sets = []
James E. Blairee743612012-05-29 14:49:32 -0700457 self.change_ahead = None
458 self.change_behind = None
James E. Blair11700c32012-07-05 17:50:05 -0700459 self.current_build_set = BuildSet(self)
460 self.build_sets.append(self.current_build_set)
James E. Blairee743612012-05-29 14:49:32 -0700461
James E. Blair0dc8ba92012-07-16 14:23:52 -0700462 def equals(self, other):
James E. Blair4aea70c2012-07-26 14:23:24 -0700463 raise NotImplementedError()
James E. Blair0dc8ba92012-07-16 14:23:52 -0700464
James E. Blair4aea70c2012-07-26 14:23:24 -0700465 def filterJobs(self, jobs):
James E. Blaire421a232012-07-25 16:59:21 -0700466 return filter(lambda job: job.changeMatches(self), jobs)
James E. Blaire5a847f2012-07-10 15:29:14 -0700467
James E. Blairee743612012-05-29 14:49:32 -0700468 def resetAllBuilds(self):
James E. Blair11700c32012-07-05 17:50:05 -0700469 old = self.current_build_set
470 self.current_build_set.result = 'CANCELED'
471 self.current_build_set = BuildSet(self)
472 old.next_build_set = self.current_build_set
473 self.current_build_set.previous_build_set = old
James E. Blair7e530ad2012-07-03 16:12:28 -0700474 self.build_sets.append(self.current_build_set)
James E. Blairee743612012-05-29 14:49:32 -0700475
476 def addBuild(self, build):
James E. Blair7e530ad2012-07-03 16:12:28 -0700477 self.current_build_set.addBuild(build)
James E. Blairee743612012-05-29 14:49:32 -0700478
James E. Blairee743612012-05-29 14:49:32 -0700479 def delete(self):
480 if self.change_behind:
481 self.change_behind.change_ahead = None
Clark Boylanba2efee2012-07-17 10:46:13 -0700482 self.change_behind = None
483 self.queue.dequeueChange(self)
James E. Blairee743612012-05-29 14:49:32 -0700484
James E. Blair1e8dd892012-05-30 09:15:05 -0700485
James E. Blair4aea70c2012-07-26 14:23:24 -0700486class Change(Changeish):
487 is_reportable = True
488
489 def __init__(self, project):
490 super(Change, self).__init__(project)
491 self.branch = None
492 self.number = None
493 self.url = None
494 self.patchset = None
495 self.refspec = None
496
497 self.reported = False
498 self.needs_change = None
499 self.needed_by_changes = []
500 self.is_current_patchset = True
501 self.can_merge = False
502 self.is_merged = False
503
504 def _id(self):
James E. Blairbe765db2012-08-07 08:36:20 -0700505 return '%s,%s' % (self.number, self.patchset)
James E. Blair4aea70c2012-07-26 14:23:24 -0700506
507 def __repr__(self):
508 return '<Change 0x%x %s>' % (id(self), self._id())
509
510 def equals(self, other):
511 if (self.number == other.number and
512 self.patchset == other.patchset):
513 return True
514 return False
515
516 def setReportedResult(self, result):
517 self.current_build_set.result = result
518
519
520class Ref(Changeish):
521 is_reportable = False
522
523 def __init__(self, project):
James E. Blairbe765db2012-08-07 08:36:20 -0700524 super(Ref, self).__init__(project)
James E. Blair4aea70c2012-07-26 14:23:24 -0700525 self.ref = None
526 self.oldrev = None
527 self.newrev = None
528
James E. Blairbe765db2012-08-07 08:36:20 -0700529 def _id(self):
530 return self.newrev
531
James E. Blair4aea70c2012-07-26 14:23:24 -0700532 def equals(self, other):
533 if (self.ref == other.ref and
534 self.newrev == other.newrev):
535 return True
536 return False
537
538
James E. Blairee743612012-05-29 14:49:32 -0700539class TriggerEvent(object):
540 def __init__(self):
541 self.data = None
James E. Blair32663402012-06-01 10:04:18 -0700542 # common
James E. Blairee743612012-05-29 14:49:32 -0700543 self.type = None
544 self.project_name = None
James E. Blair32663402012-06-01 10:04:18 -0700545 # patchset-created, comment-added, etc.
James E. Blairee743612012-05-29 14:49:32 -0700546 self.change_number = None
Clark Boylanfc56df32012-06-28 15:25:57 -0700547 self.change_url = None
James E. Blairee743612012-05-29 14:49:32 -0700548 self.patch_number = None
James E. Blaira03262c2012-05-30 09:41:16 -0700549 self.refspec = None
James E. Blairee743612012-05-29 14:49:32 -0700550 self.approvals = []
551 self.branch = None
Clark Boylanb9bcb402012-06-29 17:44:05 -0700552 self.comment = None
James E. Blair32663402012-06-01 10:04:18 -0700553 # ref-updated
James E. Blairee743612012-05-29 14:49:32 -0700554 self.ref = None
James E. Blair32663402012-06-01 10:04:18 -0700555 self.oldrev = None
James E. Blair89cae0f2012-07-18 11:18:32 -0700556 self.newrev = None
James E. Blairee743612012-05-29 14:49:32 -0700557
James E. Blair9f9667e2012-06-12 17:51:08 -0700558 def __repr__(self):
James E. Blairee743612012-05-29 14:49:32 -0700559 ret = '<TriggerEvent %s %s' % (self.type, self.project_name)
James E. Blair1e8dd892012-05-30 09:15:05 -0700560
James E. Blairee743612012-05-29 14:49:32 -0700561 if self.branch:
562 ret += " %s" % self.branch
563 if self.change_number:
564 ret += " %s,%s" % (self.change_number, self.patch_number)
565 if self.approvals:
James E. Blair1e8dd892012-05-30 09:15:05 -0700566 ret += ' ' + ', '.join(
567 ['%s:%s' % (a['type'], a['value']) for a in self.approvals])
James E. Blairee743612012-05-29 14:49:32 -0700568 ret += '>'
569
570 return ret
571
James E. Blair4aea70c2012-07-26 14:23:24 -0700572 def getChange(self, project, trigger):
James E. Blaire421a232012-07-25 16:59:21 -0700573 # TODO: make the scheduler deal with events (which may have
574 # changes) rather than changes so that we don't have to create
575 # "fake" changes for events that aren't associated with changes.
576
577 if self.change_number:
James E. Blair4aea70c2012-07-26 14:23:24 -0700578 change = trigger.getChange(self.change_number, self.patch_number)
James E. Blaire421a232012-07-25 16:59:21 -0700579 if self.ref:
James E. Blair4aea70c2012-07-26 14:23:24 -0700580 change = Ref(project)
James E. Blaire421a232012-07-25 16:59:21 -0700581 change.ref = self.ref
582 change.oldrev = self.oldrev
583 change.newrev = self.newrev
584
585 return change
586
James E. Blair1e8dd892012-05-30 09:15:05 -0700587
James E. Blairee743612012-05-29 14:49:32 -0700588class EventFilter(object):
James E. Blaire5a847f2012-07-10 15:29:14 -0700589 def __init__(self, types=[], branches=[], refs=[], approvals={},
Clark Boylanb9bcb402012-06-29 17:44:05 -0700590 comment_filters=[]):
James E. Blairee743612012-05-29 14:49:32 -0700591 self._types = types
592 self._branches = branches
593 self._refs = refs
594 self.types = [re.compile(x) for x in types]
595 self.branches = [re.compile(x) for x in branches]
596 self.refs = [re.compile(x) for x in refs]
Clark Boylanb9bcb402012-06-29 17:44:05 -0700597 self.comment_filters = [re.compile(x) for x in comment_filters]
James E. Blairee743612012-05-29 14:49:32 -0700598 self.approvals = approvals
599
James E. Blair9f9667e2012-06-12 17:51:08 -0700600 def __repr__(self):
James E. Blairee743612012-05-29 14:49:32 -0700601 ret = '<EventFilter'
James E. Blair1e8dd892012-05-30 09:15:05 -0700602
James E. Blairee743612012-05-29 14:49:32 -0700603 if self._types:
604 ret += ' types: %s' % ', '.join(self._types)
605 if self._branches:
606 ret += ' branches: %s' % ', '.join(self._branches)
607 if self._refs:
608 ret += ' refs: %s' % ', '.join(self._refs)
609 if self.approvals:
James E. Blair1e8dd892012-05-30 09:15:05 -0700610 ret += ' approvals: %s' % ', '.join(
611 ['%s:%s' % a for a in self.approvals.items()])
James E. Blairee743612012-05-29 14:49:32 -0700612 ret += '>'
613
614 return ret
615
616 def matches(self, event):
617 def normalizeCategory(name):
618 name = name.lower()
619 return re.sub(' ', '-', name)
620
621 # event types are ORed
622 matches_type = False
623 for etype in self.types:
624 if etype.match(event.type):
625 matches_type = True
626 if self.types and not matches_type:
627 return False
628
629 # branches are ORed
630 matches_branch = False
631 for branch in self.branches:
632 if branch.match(event.branch):
633 matches_branch = True
634 if self.branches and not matches_branch:
635 return False
636
637 # refs are ORed
638 matches_ref = False
639 for ref in self.refs:
640 if ref.match(event.ref):
641 matches_ref = True
642 if self.refs and not matches_ref:
643 return False
644
Clark Boylanb9bcb402012-06-29 17:44:05 -0700645 # comment_filters are ORed
646 matches_comment_filter = False
647 for comment_filter in self.comment_filters:
648 if (event.comment is not None and
649 comment_filter.search(event.comment)):
650 matches_comment_filter = True
651 if self.comment_filters and not matches_comment_filter:
652 return False
653
James E. Blairee743612012-05-29 14:49:32 -0700654 # approvals are ANDed
655 for category, value in self.approvals.items():
656 matches_approval = False
657 for eapproval in event.approvals:
658 if (normalizeCategory(eapproval['description']) == category and
659 int(eapproval['value']) == int(value)):
660 matches_approval = True
James E. Blair1e8dd892012-05-30 09:15:05 -0700661 if not matches_approval:
662 return False
James E. Blairee743612012-05-29 14:49:32 -0700663 return True