zuul/zuul/model.py

580 lines
17 KiB
Python

# Copyright 2012 Hewlett-Packard Development Company, L.P.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
import re
import time
class ChangeQueue(object):
def __init__(self, queue_name):
self.name = ''
self.queue_name = queue_name
self.projects = []
self._jobs = set()
self.queue = []
def __repr__(self):
return '<ChangeQueue %s: %s>' % (self.queue_name, self.name)
def getJobs(self):
return self._jobs
def addProject(self, project):
if project not in self.projects:
self.projects.append(project)
names = [x.name for x in self.projects]
names.sort()
self.name = ', '.join(names)
self._jobs |= set(project.getJobs(self.queue_name))
def enqueueChange(self, change):
if self.queue:
self.queue[-1].change_behind = change
change.change_ahead = self.queue[-1]
self.queue.append(change)
change.queue = self
def dequeueChange(self, change):
if change in self.queue:
self.queue.remove(change)
def mergeChangeQueue(self, other):
for project in other.projects:
self.addProject(project)
class Job(object):
def __init__(self, name):
self.name = name
self.failure_message = None
self.success_message = None
self.parameter_function = None
self.event_filters = []
def __str__(self):
return self.name
def __repr__(self):
return '<Job %s>' % (self.name)
def copy(self, other):
self.failure_message = other.failure_message
self.success_message = other.failure_message
self.event_filters = other.event_filters[:]
def eventMatches(self, event):
if not self.event_filters:
return True
for ef in self.event_filters:
if ef.matches(event):
return True
return False
class Build(object):
def __init__(self, job, uuid):
self.job = job
self.uuid = uuid
self.base_url = None
self.url = None
self.number = None
self.result = None
self.build_set = None
self.launch_time = time.time()
def __repr__(self):
return '<Build %s of %s>' % (self.uuid, self.job.name)
def formatDescription(self):
concurrent_changes = ''
concurrent_builds = ''
other_builds = ''
for change in self.build_set.other_changes:
concurrent_changes += '<li><a href="{change.url}">\
{change.number},{change.patchset}</a></li>'.format(
change=change)
change = self.build_set.change
for build in self.build_set.getBuilds():
if build.base_url:
concurrent_builds += """\
<li>
<a href="{build.base_url}">
{build.job.name} #{build.number}</a>: {build.result}
</li>
""".format(build=build)
else:
concurrent_builds += """\
<li>
{build.job.name}: {build.result}
</li>""".format(build=build)
if self.build_set.previous_build_set:
build = self.build_set.previous_build_set.getBuild(self.job.name)
if build:
other_builds += """\
<li>
Preceded by: <a href="{build.base_url}">
{build.job.name} #{build.number}</a>
</li>
""".format(build=build)
if self.build_set.next_build_set:
build = self.build_set.next_build_set.getBuild(self.job.name)
if build:
other_builds += """\
<li>
Succeeded by: <a href="{build.base_url}">
{build.job.name} #{build.number}</a>
</li>
""".format(build=build)
result = self.build_set.result
if change.number:
ret = """\
<p>
Triggered by change:
<a href="{change.url}">{change.number},{change.patchset}</a><br/>
Branch: <b>{change.branch}</b><br/>
Pipeline: <b>{change.queue_name}</b>
</p>"""
else:
ret = """\
<p>
Triggered by reference:
{change.ref}</a><br/>
Old revision: <b>{change.oldrev}</b><br/>
New revision: <b>{change.newrev}</b><br/>
Pipeline: <b>{change.queue_name}</b>
</p>"""
if concurrent_changes:
ret += """\
<p>
Other changes tested concurrently with this change:
<ul>{concurrent_changes}</ul>
</p>
"""
if concurrent_builds:
ret += """\
<p>
All builds for this change set:
<ul>{concurrent_builds}</ul>
</p>
"""
if other_builds:
ret += """\
<p>
Other build sets for this change:
<ul>{other_builds}</ul>
</p>
"""
if result:
ret += """\
<p>
Reported result: <b>{result}</b>
</p>
"""
ret = ret.format(**locals())
return ret
class JobTree(object):
""" A JobTree represents an instance of one Job, and holds JobTrees
whose jobs should be run if that Job succeeds. A root node of a
JobTree will have no associated Job. """
def __init__(self, job):
self.job = job
self.job_trees = []
def addJob(self, job):
if job not in [x.job for x in self.job_trees]:
t = JobTree(job)
self.job_trees.append(t)
return t
def getJobs(self):
jobs = []
for x in self.job_trees:
jobs.append(x.job)
jobs.extend(x.getJobs())
return jobs
def getJobTreeForJob(self, job):
if self.job == job:
return self
for tree in self.job_trees:
ret = tree.getJobTreeForJob(job)
if ret:
return ret
return None
class Project(object):
def __init__(self, name):
self.name = name
self.job_trees = {} # Queue -> JobTree
def __str__(self):
return self.name
def __repr__(self):
return '<Project %s>' % (self.name)
def addQueue(self, name):
self.job_trees[name] = JobTree(None)
return self.job_trees[name]
def hasQueue(self, name):
if name in self.job_trees:
return True
return False
def getJobTreeForQueue(self, name):
return self.job_trees.get(name, None)
def getJobs(self, queue_name):
tree = self.getJobTreeForQueue(queue_name)
if not tree:
return []
return tree.getJobs()
class BuildSet(object):
def __init__(self, change):
self.change = change
self.other_changes = []
self.builds = {}
self.result = None
self.next_build_set = None
self.previous_build_set = None
def addBuild(self, build):
self.builds[build.job.name] = build
build.build_set = self
# The change isn't enqueued until after it's created
# so we don't know what the other changes ahead will be
# until jobs start.
if not self.other_changes:
next_change = self.change.change_ahead
while next_change:
self.other_changes.append(next_change)
next_change = next_change.change_ahead
def getBuild(self, job_name):
return self.builds.get(job_name)
def getBuilds(self):
keys = self.builds.keys()
keys.sort()
return [self.builds.get(x) for x in keys]
class Change(object):
def __init__(self, queue_name, project, event):
self.queue_name = queue_name
self.project = project
self.branch = None
self.number = None
self.url = None
self.patchset = None
self.refspec = None
self.ref = None
self.oldrev = None
self.newrev = None
self.event = event
self.reported = False
if event.change_number:
self.branch = event.branch
self.number = event.change_number
self.url = event.change_url
self.patchset = event.patch_number
self.refspec = event.refspec
if event.ref:
self.ref = event.ref
self.oldrev = event.oldrev
self.newrev = event.newrev
self.build_sets = []
self.change_ahead = None
self.change_behind = None
self.current_build_set = BuildSet(self)
self.build_sets.append(self.current_build_set)
def _id(self):
if self.number:
return '%s,%s' % (self.number, self.patchset)
return self.newrev
def __repr__(self):
return '<Change 0x%x %s>' % (id(self), self._id())
def equals(self, other):
if self.number:
if (self.number == other.number and
self.patchset == other.patchset):
return True
return False
if self.ref:
if (self.ref == other.ref and
self.newrew == other.newrev):
return True
return False
return False
def _filterJobs(self, jobs):
return filter(lambda job: job.eventMatches(self.event), jobs)
def formatStatus(self, indent=0, html=False):
indent_str = ' ' * indent
ret = ''
if html and self.url is not None:
ret += '%sProject %s change <a href="%s">%s</a>\n' % (indent_str,
self.project.name,
self.url,
self._id())
else:
ret += '%sProject %s change %s\n' % (indent_str,
self.project.name,
self._id())
for job in self._filterJobs(self.project.getJobs(self.queue_name)):
build = self.current_build_set.getBuild(job.name)
if build:
result = build.result
else:
result = None
job_name = job.name
if html:
if build:
url = build.url
else:
url = None
if url is not None:
job_name = '<a href="%s">%s</a>' % (url, job_name)
ret += '%s %s: %s' % (indent_str, job_name, result)
ret += '\n'
if self.change_ahead:
ret += '%sWaiting on:\n' % (indent_str)
ret += self.change_ahead.formatStatus(indent + 2, html)
return ret
def formatReport(self):
ret = ''
if self.didAllJobsSucceed():
ret += 'Build successful\n\n'
else:
ret += 'Build failed\n\n'
for job in self._filterJobs(self.project.getJobs(self.queue_name)):
build = self.current_build_set.getBuild(job.name)
result = build.result
url = build.url
if not url:
url = job.name
ret += '- %s : %s\n' % (url, result)
return ret
def setReportedResult(self, result):
self.current_build_set.result = result
def resetAllBuilds(self):
old = self.current_build_set
self.current_build_set.result = 'CANCELED'
self.current_build_set = BuildSet(self)
old.next_build_set = self.current_build_set
self.current_build_set.previous_build_set = old
self.build_sets.append(self.current_build_set)
def addBuild(self, build):
self.current_build_set.addBuild(build)
def setResult(self, build):
if build.result != 'SUCCESS':
# Get a JobTree from a Job so we can find only its dependent jobs
root = self.project.getJobTreeForQueue(self.queue_name)
tree = root.getJobTreeForJob(build.job)
for job in tree.getJobs():
fakebuild = Build(job, None)
fakebuild.result = 'SKIPPED'
self.addBuild(fakebuild)
def _findJobsToRun(self, job_trees):
torun = []
for tree in job_trees:
job = tree.job
if not job.eventMatches(self.event):
continue
result = None
if job:
build = self.current_build_set.getBuild(job.name)
if build:
result = build.result
else:
# There is no build for the root of this job tree,
# so we should run it.
torun.append(job)
# If there is no job, this is a null job tree, and we should
# run all of its jobs.
if result == 'SUCCESS' or not job:
torun.extend(self._findJobsToRun(tree.job_trees))
return torun
def findJobsToRun(self):
tree = self.project.getJobTreeForQueue(self.queue_name)
if not tree:
return []
return self._findJobsToRun(tree.job_trees)
def areAllJobsComplete(self):
tree = self.project.getJobTreeForQueue(self.queue_name)
for job in self._filterJobs(tree.getJobs()):
build = self.current_build_set.getBuild(job.name)
if not build or not build.result:
return False
return True
def didAllJobsSucceed(self):
tree = self.project.getJobTreeForQueue(self.queue_name)
for job in self._filterJobs(tree.getJobs()):
build = self.current_build_set.getBuild(job.name)
if not build:
return False
if build.result != 'SUCCESS':
return False
return True
def delete(self):
if self.change_behind:
self.change_behind.change_ahead = None
class TriggerEvent(object):
def __init__(self):
self.data = None
# common
self.type = None
self.project_name = None
# patchset-created, comment-added, etc.
self.change_number = None
self.change_url = None
self.patch_number = None
self.refspec = None
self.approvals = []
self.branch = None
self.comment = None
# ref-updated
self.ref = None
self.oldrev = None
self.newrew = None
def __repr__(self):
ret = '<TriggerEvent %s %s' % (self.type, self.project_name)
if self.branch:
ret += " %s" % self.branch
if self.change_number:
ret += " %s,%s" % (self.change_number, self.patch_number)
if self.approvals:
ret += ' ' + ', '.join(
['%s:%s' % (a['type'], a['value']) for a in self.approvals])
ret += '>'
return ret
class EventFilter(object):
def __init__(self, types=[], branches=[], refs=[], approvals={},
comment_filters=[]):
self._types = types
self._branches = branches
self._refs = refs
self.types = [re.compile(x) for x in types]
self.branches = [re.compile(x) for x in branches]
self.refs = [re.compile(x) for x in refs]
self.comment_filters = [re.compile(x) for x in comment_filters]
self.approvals = approvals
def __repr__(self):
ret = '<EventFilter'
if self._types:
ret += ' types: %s' % ', '.join(self._types)
if self._branches:
ret += ' branches: %s' % ', '.join(self._branches)
if self._refs:
ret += ' refs: %s' % ', '.join(self._refs)
if self.approvals:
ret += ' approvals: %s' % ', '.join(
['%s:%s' % a for a in self.approvals.items()])
ret += '>'
return ret
def matches(self, event):
def normalizeCategory(name):
name = name.lower()
return re.sub(' ', '-', name)
# event types are ORed
matches_type = False
for etype in self.types:
if etype.match(event.type):
matches_type = True
if self.types and not matches_type:
return False
# branches are ORed
matches_branch = False
for branch in self.branches:
if branch.match(event.branch):
matches_branch = True
if self.branches and not matches_branch:
return False
# refs are ORed
matches_ref = False
for ref in self.refs:
if ref.match(event.ref):
matches_ref = True
if self.refs and not matches_ref:
return False
# comment_filters are ORed
matches_comment_filter = False
for comment_filter in self.comment_filters:
if (event.comment is not None and
comment_filter.search(event.comment)):
matches_comment_filter = True
if self.comment_filters and not matches_comment_filter:
return False
# approvals are ANDed
for category, value in self.approvals.items():
matches_approval = False
for eapproval in event.approvals:
if (normalizeCategory(eapproval['description']) == category and
int(eapproval['value']) == int(value)):
matches_approval = True
if not matches_approval:
return False
return True