hooks.py
657 lines
| 19.0 KiB
| text/x-python
|
PythonLexer
/ vcsserver / hooks.py
r0 | # -*- coding: utf-8 -*- | |||
# RhodeCode VCSServer provides access to different vcs backends via network. | ||||
r352 | # Copyright (C) 2014-2018 RhodeCode GmbH | |||
r0 | # | |||
# This program is free software; you can redistribute it and/or modify | ||||
# it under the terms of the GNU General Public License as published by | ||||
# the Free Software Foundation; either version 3 of the License, or | ||||
# (at your option) any later version. | ||||
# | ||||
# This program is distributed in the hope that it will be useful, | ||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
# | ||||
# You should have received a copy of the GNU General Public License | ||||
# along with this program; if not, write to the Free Software Foundation, | ||||
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | ||||
r171 | import io | |||
r276 | import os | |||
r171 | import sys | |||
import logging | ||||
r0 | import collections | |||
import importlib | ||||
r407 | import base64 | |||
r171 | ||||
r0 | from httplib import HTTPConnection | |||
import mercurial.scmutil | ||||
import mercurial.node | ||||
import simplejson as json | ||||
r369 | from vcsserver import exceptions, subprocessio, settings | |||
r0 | ||||
r171 | log = logging.getLogger(__name__) | |||
r0 | ||||
class HooksHttpClient(object): | ||||
connection = None | ||||
def __init__(self, hooks_uri): | ||||
self.hooks_uri = hooks_uri | ||||
def __call__(self, method, extras): | ||||
connection = HTTPConnection(self.hooks_uri) | ||||
body = self._serialize(method, extras) | ||||
r407 | try: | |||
connection.request('POST', '/', body) | ||||
except Exception: | ||||
log.error('Connection failed on %s', connection) | ||||
raise | ||||
r0 | response = connection.getresponse() | |||
return json.loads(response.read()) | ||||
def _serialize(self, hook_name, extras): | ||||
data = { | ||||
'method': hook_name, | ||||
'extras': extras | ||||
} | ||||
return json.dumps(data) | ||||
class HooksDummyClient(object): | ||||
def __init__(self, hooks_module): | ||||
self._hooks_module = importlib.import_module(hooks_module) | ||||
def __call__(self, hook_name, extras): | ||||
with self._hooks_module.Hooks() as hooks: | ||||
return getattr(hooks, hook_name)(extras) | ||||
class RemoteMessageWriter(object): | ||||
"""Writer base class.""" | ||||
r222 | def write(self, message): | |||
r0 | raise NotImplementedError() | |||
class HgMessageWriter(RemoteMessageWriter): | ||||
"""Writer that knows how to send messages to mercurial clients.""" | ||||
def __init__(self, ui): | ||||
self.ui = ui | ||||
def write(self, message): | ||||
# TODO: Check why the quiet flag is set by default. | ||||
old = self.ui.quiet | ||||
self.ui.quiet = False | ||||
self.ui.status(message.encode('utf-8')) | ||||
self.ui.quiet = old | ||||
class GitMessageWriter(RemoteMessageWriter): | ||||
"""Writer that knows how to send messages to git clients.""" | ||||
def __init__(self, stdout=None): | ||||
self.stdout = stdout or sys.stdout | ||||
def write(self, message): | ||||
self.stdout.write(message.encode('utf-8')) | ||||
r407 | class SvnMessageWriter(RemoteMessageWriter): | |||
"""Writer that knows how to send messages to svn clients.""" | ||||
def __init__(self, stderr=None): | ||||
# SVN needs data sent to stderr for back-to-client messaging | ||||
self.stderr = stderr or sys.stderr | ||||
def write(self, message): | ||||
self.stderr.write(message.encode('utf-8')) | ||||
r0 | def _handle_exception(result): | |||
exception_class = result.get('exception') | ||||
r171 | exception_traceback = result.get('exception_traceback') | |||
if exception_traceback: | ||||
log.error('Got traceback from remote call:%s', exception_traceback) | ||||
r0 | if exception_class == 'HTTPLockedRC': | |||
r490 | raise exceptions.RepositoryLockedException()(*result['exception_args']) | |||
r509 | elif exception_class == 'HTTPBranchProtected': | |||
raise exceptions.RepositoryBranchProtectedException()(*result['exception_args']) | ||||
r0 | elif exception_class == 'RepositoryError': | |||
r490 | raise exceptions.VcsException()(*result['exception_args']) | |||
r0 | elif exception_class: | |||
raise Exception('Got remote exception "%s" with args "%s"' % | ||||
(exception_class, result['exception_args'])) | ||||
def _get_hooks_client(extras): | ||||
if 'hooks_uri' in extras: | ||||
protocol = extras.get('hooks_protocol') | ||||
r213 | return HooksHttpClient(extras['hooks_uri']) | |||
r0 | else: | |||
return HooksDummyClient(extras['hooks_module']) | ||||
def _call_hook(hook_name, extras, writer): | ||||
r407 | hooks_client = _get_hooks_client(extras) | |||
log.debug('Hooks, using client:%s', hooks_client) | ||||
result = hooks_client(hook_name, extras) | ||||
r276 | log.debug('Hooks got result: %s', result) | |||
r0 | writer.write(result['output']) | |||
_handle_exception(result) | ||||
return result['status'] | ||||
def _extras_from_ui(ui): | ||||
r276 | hook_data = ui.config('rhodecode', 'RC_SCM_DATA') | |||
if not hook_data: | ||||
# maybe it's inside environ ? | ||||
r333 | env_hook_data = os.environ.get('RC_SCM_DATA') | |||
if env_hook_data: | ||||
hook_data = env_hook_data | ||||
r334 | extras = {} | |||
if hook_data: | ||||
extras = json.loads(hook_data) | ||||
r0 | return extras | |||
r509 | def _rev_range_hash(repo, node, check_heads=False): | |||
r223 | ||||
commits = [] | ||||
r509 | revs = [] | |||
r432 | start = repo[node].rev() | |||
r509 | end = len(repo) | |||
for rev in range(start, end): | ||||
revs.append(rev) | ||||
r223 | ctx = repo[rev] | |||
commit_id = mercurial.node.hex(ctx.node()) | ||||
branch = ctx.branch() | ||||
commits.append((commit_id, branch)) | ||||
r509 | parent_heads = [] | |||
if check_heads: | ||||
parent_heads = _check_heads(repo, start, end, revs) | ||||
return commits, parent_heads | ||||
def _check_heads(repo, start, end, commits): | ||||
changelog = repo.changelog | ||||
parents = set() | ||||
for new_rev in commits: | ||||
for p in changelog.parentrevs(new_rev): | ||||
if p == mercurial.node.nullrev: | ||||
continue | ||||
if p < start: | ||||
parents.add(p) | ||||
for p in parents: | ||||
branch = repo[p].branch() | ||||
# The heads descending from that parent, on the same branch | ||||
parent_heads = set([p]) | ||||
reachable = set([p]) | ||||
for x in xrange(p + 1, end): | ||||
if repo[x].branch() != branch: | ||||
continue | ||||
for pp in changelog.parentrevs(x): | ||||
if pp in reachable: | ||||
reachable.add(x) | ||||
parent_heads.discard(pp) | ||||
parent_heads.add(x) | ||||
# More than one head? Suggest merging | ||||
if len(parent_heads) > 1: | ||||
return list(parent_heads) | ||||
return [] | ||||
r223 | ||||
r276 | def repo_size(ui, repo, **kwargs): | |||
extras = _extras_from_ui(ui) | ||||
return _call_hook('repo_size', extras, HgMessageWriter(ui)) | ||||
def pre_pull(ui, repo, **kwargs): | ||||
extras = _extras_from_ui(ui) | ||||
return _call_hook('pre_pull', extras, HgMessageWriter(ui)) | ||||
def pre_pull_ssh(ui, repo, **kwargs): | ||||
r334 | extras = _extras_from_ui(ui) | |||
if extras and extras.get('SSH'): | ||||
r276 | return pre_pull(ui, repo, **kwargs) | |||
return 0 | ||||
def post_pull(ui, repo, **kwargs): | ||||
extras = _extras_from_ui(ui) | ||||
return _call_hook('post_pull', extras, HgMessageWriter(ui)) | ||||
def post_pull_ssh(ui, repo, **kwargs): | ||||
r334 | extras = _extras_from_ui(ui) | |||
if extras and extras.get('SSH'): | ||||
r276 | return post_pull(ui, repo, **kwargs) | |||
return 0 | ||||
r170 | def pre_push(ui, repo, node=None, **kwargs): | |||
r509 | """ | |||
Mercurial pre_push hook | ||||
""" | ||||
r170 | extras = _extras_from_ui(ui) | |||
r509 | detect_force_push = extras.get('detect_force_push') | |||
r170 | ||||
rev_data = [] | ||||
if node and kwargs.get('hooktype') == 'pretxnchangegroup': | ||||
branches = collections.defaultdict(list) | ||||
r509 | commits, _heads = _rev_range_hash(repo, node, check_heads=detect_force_push) | |||
for commit_id, branch in commits: | ||||
r170 | branches[branch].append(commit_id) | |||
r509 | for branch, commits in branches.items(): | |||
r170 | old_rev = kwargs.get('node_last') or commits[0] | |||
rev_data.append({ | ||||
'old_rev': old_rev, | ||||
'new_rev': commits[-1], | ||||
'ref': '', | ||||
'type': 'branch', | ||||
'name': branch, | ||||
}) | ||||
r509 | for push_ref in rev_data: | |||
push_ref['multiple_heads'] = _heads | ||||
r170 | extras['commit_ids'] = rev_data | |||
return _call_hook('pre_push', extras, HgMessageWriter(ui)) | ||||
r0 | ||||
r276 | def pre_push_ssh(ui, repo, node=None, **kwargs): | |||
r510 | extras = _extras_from_ui(ui) | |||
if extras.get('SSH'): | ||||
r276 | return pre_push(ui, repo, node, **kwargs) | |||
return 0 | ||||
def pre_push_ssh_auth(ui, repo, node=None, **kwargs): | ||||
r509 | """ | |||
Mercurial pre_push hook for SSH | ||||
""" | ||||
r276 | extras = _extras_from_ui(ui) | |||
if extras.get('SSH'): | ||||
permission = extras['SSH_PERMISSIONS'] | ||||
if 'repository.write' == permission or 'repository.admin' == permission: | ||||
return 0 | ||||
# non-zero ret code | ||||
return 1 | ||||
return 0 | ||||
r223 | def post_push(ui, repo, node, **kwargs): | |||
r509 | """ | |||
Mercurial post_push hook | ||||
""" | ||||
r223 | extras = _extras_from_ui(ui) | |||
commit_ids = [] | ||||
branches = [] | ||||
bookmarks = [] | ||||
tags = [] | ||||
r0 | ||||
r509 | commits, _heads = _rev_range_hash(repo, node) | |||
for commit_id, branch in commits: | ||||
r223 | commit_ids.append(commit_id) | |||
if branch not in branches: | ||||
branches.append(branch) | ||||
r0 | ||||
r223 | if hasattr(ui, '_rc_pushkey_branches'): | |||
bookmarks = ui._rc_pushkey_branches | ||||
r0 | ||||
extras['commit_ids'] = commit_ids | ||||
r223 | extras['new_refs'] = { | |||
'branches': branches, | ||||
'bookmarks': bookmarks, | ||||
'tags': tags | ||||
} | ||||
r0 | ||||
return _call_hook('post_push', extras, HgMessageWriter(ui)) | ||||
r276 | def post_push_ssh(ui, repo, node, **kwargs): | |||
r509 | """ | |||
Mercurial post_push hook for SSH | ||||
""" | ||||
r276 | if _extras_from_ui(ui).get('SSH'): | |||
return post_push(ui, repo, node, **kwargs) | ||||
return 0 | ||||
r221 | def key_push(ui, repo, **kwargs): | |||
if kwargs['new'] != '0' and kwargs['namespace'] == 'bookmarks': | ||||
# store new bookmarks in our UI object propagated later to post_push | ||||
ui._rc_pushkey_branches = repo[kwargs['key']].bookmarks() | ||||
return | ||||
r276 | ||||
r0 | # backward compat | |||
log_pull_action = post_pull | ||||
# backward compat | ||||
log_push_action = post_push | ||||
def handle_git_pre_receive(unused_repo_path, unused_revs, unused_env): | ||||
""" | ||||
Old hook name: keep here for backward compatibility. | ||||
This is only required when the installed git hooks are not upgraded. | ||||
""" | ||||
pass | ||||
def handle_git_post_receive(unused_repo_path, unused_revs, unused_env): | ||||
""" | ||||
Old hook name: keep here for backward compatibility. | ||||
This is only required when the installed git hooks are not upgraded. | ||||
""" | ||||
pass | ||||
HookResponse = collections.namedtuple('HookResponse', ('status', 'output')) | ||||
def git_pre_pull(extras): | ||||
""" | ||||
Pre pull hook. | ||||
:param extras: dictionary containing the keys defined in simplevcs | ||||
:type extras: dict | ||||
:return: status code of the hook. 0 for success. | ||||
:rtype: int | ||||
""" | ||||
if 'pull' not in extras['hooks']: | ||||
return HookResponse(0, '') | ||||
stdout = io.BytesIO() | ||||
try: | ||||
status = _call_hook('pre_pull', extras, GitMessageWriter(stdout)) | ||||
except Exception as error: | ||||
status = 128 | ||||
stdout.write('ERROR: %s\n' % str(error)) | ||||
return HookResponse(status, stdout.getvalue()) | ||||
def git_post_pull(extras): | ||||
""" | ||||
Post pull hook. | ||||
:param extras: dictionary containing the keys defined in simplevcs | ||||
:type extras: dict | ||||
:return: status code of the hook. 0 for success. | ||||
:rtype: int | ||||
""" | ||||
if 'pull' not in extras['hooks']: | ||||
return HookResponse(0, '') | ||||
stdout = io.BytesIO() | ||||
try: | ||||
status = _call_hook('post_pull', extras, GitMessageWriter(stdout)) | ||||
except Exception as error: | ||||
status = 128 | ||||
stdout.write('ERROR: %s\n' % error) | ||||
return HookResponse(status, stdout.getvalue()) | ||||
r170 | def _parse_git_ref_lines(revision_lines): | |||
rev_data = [] | ||||
for revision_line in revision_lines or []: | ||||
old_rev, new_rev, ref = revision_line.strip().split(' ') | ||||
ref_data = ref.split('/', 2) | ||||
if ref_data[1] in ('tags', 'heads'): | ||||
rev_data.append({ | ||||
'old_rev': old_rev, | ||||
'new_rev': new_rev, | ||||
'ref': ref, | ||||
'type': ref_data[1], | ||||
'name': ref_data[2], | ||||
}) | ||||
return rev_data | ||||
def git_pre_receive(unused_repo_path, revision_lines, env): | ||||
r0 | """ | |||
Pre push hook. | ||||
:param extras: dictionary containing the keys defined in simplevcs | ||||
:type extras: dict | ||||
:return: status code of the hook. 0 for success. | ||||
:rtype: int | ||||
""" | ||||
extras = json.loads(env['RC_SCM_DATA']) | ||||
r170 | rev_data = _parse_git_ref_lines(revision_lines) | |||
r0 | if 'push' not in extras['hooks']: | |||
return 0 | ||||
r509 | empty_commit_id = '0' * 40 | |||
detect_force_push = extras.get('detect_force_push') | ||||
for push_ref in rev_data: | ||||
r510 | # store our git-env which holds the temp store | |||
push_ref['git_env'] = [ | ||||
(k, v) for k, v in os.environ.items() if k.startswith('GIT')] | ||||
r509 | push_ref['pruned_sha'] = '' | |||
if not detect_force_push: | ||||
# don't check for forced-push when we don't need to | ||||
continue | ||||
type_ = push_ref['type'] | ||||
new_branch = push_ref['old_rev'] == empty_commit_id | ||||
if type_ == 'heads' and not new_branch: | ||||
old_rev = push_ref['old_rev'] | ||||
new_rev = push_ref['new_rev'] | ||||
cmd = [settings.GIT_EXECUTABLE, 'rev-list', | ||||
old_rev, '^{}'.format(new_rev)] | ||||
stdout, stderr = subprocessio.run_command( | ||||
cmd, env=os.environ.copy()) | ||||
# means we're having some non-reachable objects, this forced push | ||||
# was used | ||||
if stdout: | ||||
push_ref['pruned_sha'] = stdout.splitlines() | ||||
r170 | extras['commit_ids'] = rev_data | |||
r0 | return _call_hook('pre_push', extras, GitMessageWriter()) | |||
def git_post_receive(unused_repo_path, revision_lines, env): | ||||
""" | ||||
Post push hook. | ||||
:param extras: dictionary containing the keys defined in simplevcs | ||||
:type extras: dict | ||||
:return: status code of the hook. 0 for success. | ||||
:rtype: int | ||||
""" | ||||
extras = json.loads(env['RC_SCM_DATA']) | ||||
if 'push' not in extras['hooks']: | ||||
return 0 | ||||
r170 | rev_data = _parse_git_ref_lines(revision_lines) | |||
r0 | ||||
git_revs = [] | ||||
# N.B.(skreft): it is ok to just call git, as git before calling a | ||||
# subcommand sets the PATH environment variable so that it point to the | ||||
# correct version of the git executable. | ||||
empty_commit_id = '0' * 40 | ||||
r223 | branches = [] | |||
tags = [] | ||||
r0 | for push_ref in rev_data: | |||
type_ = push_ref['type'] | ||||
r223 | ||||
r0 | if type_ == 'heads': | |||
if push_ref['old_rev'] == empty_commit_id: | ||||
r223 | # starting new branch case | |||
if push_ref['name'] not in branches: | ||||
branches.append(push_ref['name']) | ||||
r0 | ||||
# Fix up head revision if needed | ||||
r369 | cmd = [settings.GIT_EXECUTABLE, 'show', 'HEAD'] | |||
r0 | try: | |||
r370 | subprocessio.run_command(cmd, env=os.environ.copy()) | |||
r0 | except Exception: | |||
r369 | cmd = [settings.GIT_EXECUTABLE, 'symbolic-ref', 'HEAD', | |||
r0 | 'refs/heads/%s' % push_ref['name']] | |||
r170 | print("Setting default branch to %s" % push_ref['name']) | |||
r370 | subprocessio.run_command(cmd, env=os.environ.copy()) | |||
r0 | ||||
r370 | cmd = [settings.GIT_EXECUTABLE, 'for-each-ref', | |||
'--format=%(refname)', 'refs/heads/*'] | ||||
stdout, stderr = subprocessio.run_command( | ||||
cmd, env=os.environ.copy()) | ||||
heads = stdout | ||||
r0 | heads = heads.replace(push_ref['ref'], '') | |||
r434 | heads = ' '.join(head for head | |||
in heads.splitlines() if head) or '.' | ||||
r370 | cmd = [settings.GIT_EXECUTABLE, 'log', '--reverse', | |||
'--pretty=format:%H', '--', push_ref['new_rev'], | ||||
'--not', heads] | ||||
stdout, stderr = subprocessio.run_command( | ||||
cmd, env=os.environ.copy()) | ||||
git_revs.extend(stdout.splitlines()) | ||||
r0 | elif push_ref['new_rev'] == empty_commit_id: | |||
# delete branch case | ||||
git_revs.append('delete_branch=>%s' % push_ref['name']) | ||||
else: | ||||
r223 | if push_ref['name'] not in branches: | |||
branches.append(push_ref['name']) | ||||
r369 | cmd = [settings.GIT_EXECUTABLE, 'log', | |||
r0 | '{old_rev}..{new_rev}'.format(**push_ref), | |||
'--reverse', '--pretty=format:%H'] | ||||
r370 | stdout, stderr = subprocessio.run_command( | |||
cmd, env=os.environ.copy()) | ||||
git_revs.extend(stdout.splitlines()) | ||||
r0 | elif type_ == 'tags': | |||
r223 | if push_ref['name'] not in tags: | |||
tags.append(push_ref['name']) | ||||
r0 | git_revs.append('tag=>%s' % push_ref['name']) | |||
extras['commit_ids'] = git_revs | ||||
r223 | extras['new_refs'] = { | |||
'branches': branches, | ||||
'bookmarks': [], | ||||
'tags': tags, | ||||
} | ||||
r0 | ||||
if 'repo_size' in extras['hooks']: | ||||
try: | ||||
_call_hook('repo_size', extras, GitMessageWriter()) | ||||
except: | ||||
pass | ||||
return _call_hook('post_push', extras, GitMessageWriter()) | ||||
r407 | ||||
r436 | def _get_extras_from_txn_id(path, txn_id): | |||
extras = {} | ||||
try: | ||||
cmd = ['svnlook', 'pget', | ||||
'-t', txn_id, | ||||
'--revprop', path, 'rc-scm-extras'] | ||||
stdout, stderr = subprocessio.run_command( | ||||
cmd, env=os.environ.copy()) | ||||
extras = json.loads(base64.urlsafe_b64decode(stdout)) | ||||
except Exception: | ||||
log.exception('Failed to extract extras info from txn_id') | ||||
return extras | ||||
r407 | def svn_pre_commit(repo_path, commit_data, env): | |||
path, txn_id = commit_data | ||||
branches = [] | ||||
tags = [] | ||||
r436 | if env.get('RC_SCM_DATA'): | |||
extras = json.loads(env['RC_SCM_DATA']) | ||||
else: | ||||
# fallback method to read from TXN-ID stored data | ||||
extras = _get_extras_from_txn_id(path, txn_id) | ||||
if not extras: | ||||
return 0 | ||||
r407 | ||||
extras['commit_ids'] = [] | ||||
extras['txn_id'] = txn_id | ||||
extras['new_refs'] = { | ||||
'branches': branches, | ||||
'bookmarks': [], | ||||
'tags': tags, | ||||
} | ||||
r436 | ||||
r407 | return _call_hook('pre_push', extras, SvnMessageWriter()) | |||
r436 | def _get_extras_from_commit_id(commit_id, path): | |||
extras = {} | ||||
try: | ||||
cmd = ['svnlook', 'pget', | ||||
'-r', commit_id, | ||||
'--revprop', path, 'rc-scm-extras'] | ||||
stdout, stderr = subprocessio.run_command( | ||||
cmd, env=os.environ.copy()) | ||||
extras = json.loads(base64.urlsafe_b64decode(stdout)) | ||||
except Exception: | ||||
log.exception('Failed to extract extras info from commit_id') | ||||
return extras | ||||
r407 | def svn_post_commit(repo_path, commit_data, env): | |||
""" | ||||
commit_data is path, rev, txn_id | ||||
""" | ||||
path, commit_id, txn_id = commit_data | ||||
branches = [] | ||||
tags = [] | ||||
r436 | if env.get('RC_SCM_DATA'): | |||
extras = json.loads(env['RC_SCM_DATA']) | ||||
else: | ||||
# fallback method to read from TXN-ID stored data | ||||
extras = _get_extras_from_commit_id(commit_id, path) | ||||
if not extras: | ||||
return 0 | ||||
r407 | ||||
extras['commit_ids'] = [commit_id] | ||||
extras['txn_id'] = txn_id | ||||
extras['new_refs'] = { | ||||
'branches': branches, | ||||
'bookmarks': [], | ||||
'tags': tags, | ||||
} | ||||
if 'repo_size' in extras['hooks']: | ||||
try: | ||||
_call_hook('repo_size', extras, SvnMessageWriter()) | ||||
r436 | except Exception: | |||
r407 | pass | |||
return _call_hook('post_push', extras, SvnMessageWriter()) | ||||