| # 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 urllib |
| import logging |
| import voluptuous as vs |
| from zuul.source import BaseSource |
| from zuul.model import Project |
| from zuul.driver.gerrit.gerritmodel import GerritRefFilter |
| from zuul.driver.util import scalar_or_list, to_list |
| from zuul.lib.dependson import find_dependency_headers |
| |
| |
| class GerritSource(BaseSource): |
| name = 'gerrit' |
| log = logging.getLogger("zuul.source.Gerrit") |
| |
| def __init__(self, driver, connection, config=None): |
| hostname = connection.canonical_hostname |
| super(GerritSource, self).__init__(driver, connection, |
| hostname, config) |
| |
| def getRefSha(self, project, ref): |
| return self.connection.getRefSha(project, ref) |
| |
| def isMerged(self, change, head=None): |
| return self.connection.isMerged(change, head) |
| |
| def canMerge(self, change, allow_needs): |
| return self.connection.canMerge(change, allow_needs) |
| |
| def postConfig(self): |
| pass |
| |
| def getChange(self, event, refresh=False): |
| return self.connection.getChange(event, refresh) |
| |
| change_re = re.compile(r"/(\#\/c\/)?(\d+)[\w]*") |
| |
| def getChangeByURL(self, url): |
| try: |
| parsed = urllib.parse.urlparse(url) |
| except ValueError: |
| return None |
| path = parsed.path |
| if parsed.fragment: |
| path += '#' + parsed.fragment |
| m = self.change_re.match(path) |
| if not m: |
| return None |
| try: |
| change_no = int(m.group(2)) |
| except ValueError: |
| return None |
| query = "change:%s" % (change_no,) |
| results = self.connection.simpleQuery(query) |
| if not results: |
| return None |
| change = self.connection._getChange( |
| results[0]['number'], results[0]['currentPatchSet']['number']) |
| return change |
| |
| def getChangesDependingOn(self, change, projects): |
| changes = [] |
| if not change.uris: |
| return changes |
| queries = set() |
| for uri in change.uris: |
| queries.add('message:%s' % uri) |
| query = '(' + ' OR '.join(queries) + ')' |
| results = self.connection.simpleQuery(query) |
| seen = set() |
| for result in results: |
| for match in find_dependency_headers(result['commitMessage']): |
| found = False |
| for uri in change.uris: |
| if uri in match: |
| found = True |
| break |
| if not found: |
| continue |
| key = (result['number'], result['currentPatchSet']['number']) |
| if key in seen: |
| continue |
| seen.add(key) |
| change = self.connection._getChange( |
| result['number'], result['currentPatchSet']['number']) |
| changes.append(change) |
| return changes |
| |
| def getCachedChanges(self): |
| for x in self.connection._change_cache.values(): |
| for y in x.values(): |
| yield y |
| |
| def getProject(self, name): |
| p = self.connection.getProject(name) |
| if not p: |
| p = Project(name, self) |
| self.connection.addProject(p) |
| return p |
| |
| def getProjectOpenChanges(self, project): |
| return self.connection.getProjectOpenChanges(project) |
| |
| def getProjectBranches(self, project, tenant): |
| return self.connection.getProjectBranches(project, tenant) |
| |
| def getGitUrl(self, project): |
| return self.connection.getGitUrl(project) |
| |
| def _getGitwebUrl(self, project, sha=None): |
| return self.connection._getGitwebUrl(project, sha) |
| |
| def getRequireFilters(self, config): |
| f = GerritRefFilter( |
| connection_name=self.connection.connection_name, |
| open=config.get('open'), |
| current_patchset=config.get('current-patchset'), |
| statuses=to_list(config.get('status')), |
| required_approvals=to_list(config.get('approval')), |
| ) |
| return [f] |
| |
| def getRejectFilters(self, config): |
| f = GerritRefFilter( |
| connection_name=self.connection.connection_name, |
| reject_approvals=to_list(config.get('approval')), |
| ) |
| return [f] |
| |
| def getRefForChange(self, change): |
| partial = change[-2:] |
| return "refs/changes/%s/%s/.*" % (partial, change) |
| |
| |
| approval = vs.Schema({'username': str, |
| 'email': str, |
| 'older-than': str, |
| 'newer-than': str, |
| }, extra=vs.ALLOW_EXTRA) |
| |
| |
| def getRequireSchema(): |
| require = {'approval': scalar_or_list(approval), |
| 'open': bool, |
| 'current-patchset': bool, |
| 'status': scalar_or_list(str)} |
| |
| return require |
| |
| |
| def getRejectSchema(): |
| reject = {'approval': scalar_or_list(approval)} |
| |
| return reject |