hook.py
279 lines
| 10.0 KiB
| text/x-python
|
PythonLexer
/ mercurial / hook.py
Matt Mackall
|
r4622 | # hook.py - hook support for mercurial | ||
# | ||||
# Copyright 2007 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Matt Mackall
|
r4622 | |||
Gregory Szorc
|
r25953 | from __future__ import absolute_import | ||
import os | ||||
import sys | ||||
from .i18n import _ | ||||
from . import ( | ||||
demandimport, | ||||
Pulkit Goyal
|
r32642 | encoding, | ||
Gregory Szorc
|
r25953 | error, | ||
extensions, | ||||
Pulkit Goyal
|
r30519 | pycompat, | ||
Gregory Szorc
|
r25953 | util, | ||
) | ||||
Matt Mackall
|
r4622 | |||
Pierre-Yves David
|
r31742 | def _pythonhook(ui, repo, htype, hname, funcname, args, throw): | ||
Matt Mackall
|
r4622 | '''call python hook. hook is callable object, looked up as | ||
name in python module. if callable returns "true", hook | ||||
fails, else passes. if hook raises exception, treated as | ||||
hook failure. exception propagates if throw is "true". | ||||
reason for "true" meaning "hook failed" is so that | ||||
unmodified commands (e.g. mercurial.commands.update) can | ||||
be run as hooks without wrappers to convert return values.''' | ||||
Augie Fackler
|
r21797 | if callable(funcname): | ||
Mads Kiilerich
|
r20548 | obj = funcname | ||
Pulkit Goyal
|
r32614 | funcname = pycompat.sysbytes(obj.__module__ + r"." + obj.__name__) | ||
Mads Kiilerich
|
r20548 | else: | ||
Matt Mackall
|
r4622 | d = funcname.rfind('.') | ||
if d == -1: | ||||
Siddharth Agarwal
|
r26692 | raise error.HookLoadError( | ||
Siddharth Agarwal
|
r28106 | _('%s hook is invalid: "%s" not in a module') | ||
Siddharth Agarwal
|
r26692 | % (hname, funcname)) | ||
Matt Mackall
|
r4622 | modname = funcname[:d] | ||
Sune Foldager
|
r10103 | oldpaths = sys.path | ||
Augie Fackler
|
r14941 | if util.mainfrozen(): | ||
Steve Borho
|
r9332 | # binary installs require sys.path manipulation | ||
Sune Foldager
|
r10103 | modpath, modfile = os.path.split(modname) | ||
if modpath and modfile: | ||||
sys.path = sys.path[:] + [modpath] | ||||
modname = modfile | ||||
Jordi Gutiérrez Hermoso
|
r25328 | with demandimport.deactivated(): | ||
Matt Mackall
|
r4622 | try: | ||
Gregory Szorc
|
r36125 | obj = __import__(pycompat.sysstr(modname)) | ||
Siddharth Agarwal
|
r28109 | except (ImportError, SyntaxError): | ||
Siddharth Agarwal
|
r28078 | e1 = sys.exc_info() | ||
Jordi Gutiérrez Hermoso
|
r25328 | try: | ||
# extensions are loaded with hgext_ prefix | ||||
Gregory Szorc
|
r36125 | obj = __import__(r"hgext_%s" % pycompat.sysstr(modname)) | ||
Siddharth Agarwal
|
r28109 | except (ImportError, SyntaxError): | ||
Siddharth Agarwal
|
r28078 | e2 = sys.exc_info() | ||
Jordi Gutiérrez Hermoso
|
r25328 | if ui.tracebackflag: | ||
ui.warn(_('exception from first failed import ' | ||||
'attempt:\n')) | ||||
ui.traceback(e1) | ||||
if ui.tracebackflag: | ||||
ui.warn(_('exception from second failed import ' | ||||
'attempt:\n')) | ||||
ui.traceback(e2) | ||||
Siddharth Agarwal
|
r28080 | |||
if not ui.tracebackflag: | ||||
tracebackhint = _( | ||||
'run with --traceback for stack trace') | ||||
else: | ||||
tracebackhint = None | ||||
Siddharth Agarwal
|
r26692 | raise error.HookLoadError( | ||
Siddharth Agarwal
|
r28079 | _('%s hook is invalid: import of "%s" failed') % | ||
Siddharth Agarwal
|
r28080 | (hname, modname), hint=tracebackhint) | ||
Steve Borho
|
r9332 | sys.path = oldpaths | ||
Matt Mackall
|
r4622 | try: | ||
for p in funcname.split('.')[1:]: | ||||
obj = getattr(obj, p) | ||||
Benoit Boissinot
|
r7280 | except AttributeError: | ||
Siddharth Agarwal
|
r26692 | raise error.HookLoadError( | ||
Siddharth Agarwal
|
r28079 | _('%s hook is invalid: "%s" is not defined') | ||
Siddharth Agarwal
|
r26692 | % (hname, funcname)) | ||
Augie Fackler
|
r21797 | if not callable(obj): | ||
Siddharth Agarwal
|
r26692 | raise error.HookLoadError( | ||
Siddharth Agarwal
|
r28079 | _('%s hook is invalid: "%s" is not callable') | ||
Siddharth Agarwal
|
r26692 | % (hname, funcname)) | ||
Mads Kiilerich
|
r20547 | |||
ui.note(_("calling hook %s: %s\n") % (hname, funcname)) | ||||
Simon Farnsworth
|
r30975 | starttime = util.timer() | ||
Mads Kiilerich
|
r20547 | |||
Matt Mackall
|
r4622 | try: | ||
Pulkit Goyal
|
r35358 | r = obj(ui=ui, repo=repo, hooktype=htype, **pycompat.strkwargs(args)) | ||
Gregory Szorc
|
r25660 | except Exception as exc: | ||
Pierre-Yves David
|
r26587 | if isinstance(exc, error.Abort): | ||
Matt Mackall
|
r25084 | ui.warn(_('error: %s hook failed: %s\n') % | ||
(hname, exc.args[0])) | ||||
else: | ||||
ui.warn(_('error: %s hook raised an exception: ' | ||||
Pulkit Goyal
|
r32642 | '%s\n') % (hname, encoding.strtolocal(str(exc)))) | ||
Matt Mackall
|
r25084 | if throw: | ||
Matt Mackall
|
r4622 | raise | ||
Siddharth Agarwal
|
r28108 | if not ui.tracebackflag: | ||
ui.warn(_('(run with --traceback for stack trace)\n')) | ||||
Matt Mackall
|
r25084 | ui.traceback() | ||
Siddharth Agarwal
|
r26739 | return True, True | ||
Idan Kamara
|
r14889 | finally: | ||
Simon Farnsworth
|
r30975 | duration = util.timer() - starttime | ||
Durham Goode
|
r18691 | ui.log('pythonhook', 'pythonhook-%s: %s finished in %0.2f seconds\n', | ||
Pierre-Yves David
|
r31742 | htype, funcname, duration) | ||
Matt Mackall
|
r4622 | if r: | ||
if throw: | ||||
Pierre-Yves David
|
r23415 | raise error.HookAbort(_('%s hook failed') % hname) | ||
Matt Mackall
|
r4622 | ui.warn(_('warning: %s hook failed\n') % hname) | ||
Siddharth Agarwal
|
r26739 | return r, False | ||
Matt Mackall
|
r4622 | |||
Pierre-Yves David
|
r31746 | def _exthook(ui, repo, htype, name, cmd, args, throw): | ||
Matt Mackall
|
r4622 | ui.note(_("running hook %s: %s\n") % (name, cmd)) | ||
Matt Mackall
|
r7787 | |||
Simon Farnsworth
|
r30975 | starttime = util.timer() | ||
Matt Mackall
|
r7787 | env = {} | ||
FUJIWARA Katsunori
|
r26751 | |||
# make in-memory changes visible to external process | ||||
Sietse Brouwer
|
r27228 | if repo is not None: | ||
tr = repo.currenttransaction() | ||||
repo.dirstate.write(tr) | ||||
if tr and tr.writepending(): | ||||
env['HG_PENDING'] = repo.root | ||||
Pierre-Yves David
|
r31746 | env['HG_HOOKTYPE'] = htype | ||
Pierre-Yves David
|
r31747 | env['HG_HOOKNAME'] = name | ||
FUJIWARA Katsunori
|
r26751 | |||
Matt Mackall
|
r7787 | for k, v in args.iteritems(): | ||
Augie Fackler
|
r21797 | if callable(v): | ||
Matt Mackall
|
r7787 | v = v() | ||
Dan Villiom Podlaski Christiansen
|
r13207 | if isinstance(v, dict): | ||
# make the dictionary element order stable across Python | ||||
# implementations | ||||
v = ('{' + | ||||
', '.join('%r: %r' % i for i in sorted(v.iteritems())) + | ||||
'}') | ||||
Matt Mackall
|
r7787 | env['HG_' + k.upper()] = v | ||
Matt Mackall
|
r5869 | if repo: | ||
cwd = repo.root | ||||
else: | ||||
Pulkit Goyal
|
r30519 | cwd = pycompat.getcwd() | ||
Simon Farnsworth
|
r31205 | r = ui.system(cmd, environ=env, cwd=cwd, blockedtag='exthook-%s' % (name,)) | ||
Durham Goode
|
r18671 | |||
Simon Farnsworth
|
r30975 | duration = util.timer() - starttime | ||
Durham Goode
|
r18691 | ui.log('exthook', 'exthook-%s: %s finished in %0.2f seconds\n', | ||
Durham Goode
|
r18671 | name, cmd, duration) | ||
Matt Mackall
|
r4622 | if r: | ||
Adrian Buehlmann
|
r14234 | desc, r = util.explainexit(r) | ||
Matt Mackall
|
r4622 | if throw: | ||
Pierre-Yves David
|
r23415 | raise error.HookAbort(_('%s hook %s') % (name, desc)) | ||
Matt Mackall
|
r4622 | ui.warn(_('warning: %s hook %s\n') % (name, desc)) | ||
return r | ||||
Pierre-Yves David
|
r28938 | # represent an untrusted hook command | ||
_fromuntrusted = object() | ||||
Matt Zuba
|
r15896 | def _allhooks(ui): | ||
Pierre-Yves David
|
r28937 | """return a list of (hook-id, cmd) pairs sorted by priority""" | ||
hooks = _hookitems(ui) | ||||
Pierre-Yves David
|
r28938 | # Be careful in this section, propagating the real commands from untrusted | ||
# sources would create a security vulnerability, make sure anything altered | ||||
# in that section uses "_fromuntrusted" as its command. | ||||
untrustedhooks = _hookitems(ui, _untrusted=True) | ||||
for name, value in untrustedhooks.items(): | ||||
trustedvalue = hooks.get(name, (None, None, name, _fromuntrusted)) | ||||
if value != trustedvalue: | ||||
(lp, lo, lk, lv) = trustedvalue | ||||
hooks[name] = (lp, lo, lk, _fromuntrusted) | ||||
# (end of the security sensitive section) | ||||
Pierre-Yves David
|
r28937 | return [(k, v) for p, o, k, v in sorted(hooks.values())] | ||
Pierre-Yves David
|
r28938 | def _hookitems(ui, _untrusted=False): | ||
Pierre-Yves David
|
r28937 | """return all hooks items ready to be sorted""" | ||
Pierre-Yves David
|
r28936 | hooks = {} | ||
Pierre-Yves David
|
r28938 | for name, cmd in ui.configitems('hooks', untrusted=_untrusted): | ||
Matt Zuba
|
r15896 | if not name.startswith('priority'): | ||
priority = ui.configint('hooks', 'priority.%s' % name, 0) | ||||
Pierre-Yves David
|
r28936 | hooks[name] = (-priority, len(hooks), name, cmd) | ||
Pierre-Yves David
|
r28937 | return hooks | ||
Matt Zuba
|
r15896 | |||
Matt Mackall
|
r5833 | _redirect = False | ||
def redirect(state): | ||||
Alexis S. L. Carvalho
|
r6266 | global _redirect | ||
Matt Mackall
|
r5833 | _redirect = state | ||
Boris Feld
|
r34688 | def hashook(ui, htype): | ||
"""return True if a hook is configured for 'htype'""" | ||||
if not ui.callhooks: | ||||
return False | ||||
for hname, cmd in _allhooks(ui): | ||||
if hname.split('.')[0] == htype and cmd: | ||||
return True | ||||
return False | ||||
Pierre-Yves David
|
r31745 | def hook(ui, repo, htype, throw=False, **args): | ||
Idan Kamara
|
r17048 | if not ui.callhooks: | ||
return False | ||||
Siddharth Agarwal
|
r26737 | hooks = [] | ||
for hname, cmd in _allhooks(ui): | ||||
Pierre-Yves David
|
r31745 | if hname.split('.')[0] == htype and cmd: | ||
Siddharth Agarwal
|
r26737 | hooks.append((hname, cmd)) | ||
Pierre-Yves David
|
r31745 | res = runhooks(ui, repo, htype, hooks, throw=throw, **args) | ||
Siddharth Agarwal
|
r26738 | r = False | ||
for hname, cmd in hooks: | ||||
Siddharth Agarwal
|
r26739 | r = res[hname][0] or r | ||
Siddharth Agarwal
|
r26738 | return r | ||
Siddharth Agarwal
|
r26737 | |||
Pierre-Yves David
|
r31744 | def runhooks(ui, repo, htype, hooks, throw=False, **args): | ||
Pulkit Goyal
|
r32897 | args = pycompat.byteskwargs(args) | ||
Siddharth Agarwal
|
r26738 | res = {} | ||
Sune Foldager
|
r9658 | oldstdout = -1 | ||
Matt Mackall
|
r5833 | |||
Jesse Long
|
r7416 | try: | ||
Siddharth Agarwal
|
r26737 | for hname, cmd in hooks: | ||
Matt Mackall
|
r17963 | if oldstdout == -1 and _redirect: | ||
try: | ||||
Yuya Nishihara
|
r30473 | stdoutno = util.stdout.fileno() | ||
stderrno = util.stderr.fileno() | ||||
Matt Mackall
|
r17963 | # temporarily redirect stdout to stderr, if possible | ||
if stdoutno >= 0 and stderrno >= 0: | ||||
Yuya Nishihara
|
r30473 | util.stdout.flush() | ||
Matt Mackall
|
r17963 | oldstdout = os.dup(stdoutno) | ||
os.dup2(stderrno, stdoutno) | ||||
Matt Mackall
|
r17964 | except (OSError, AttributeError): | ||
# files seem to be bogus, give up on redirecting (WSGI, etc) | ||||
Matt Mackall
|
r17963 | pass | ||
Pierre-Yves David
|
r28938 | if cmd is _fromuntrusted: | ||
if throw: | ||||
raise error.HookAbort( | ||||
Pierre-Yves David
|
r31743 | _('untrusted hook %s not executed') % hname, | ||
Pierre-Yves David
|
r28938 | hint = _("see 'hg help config.trusted'")) | ||
Pierre-Yves David
|
r31743 | ui.warn(_('warning: untrusted hook %s not executed\n') % hname) | ||
Pierre-Yves David
|
r28938 | r = 1 | ||
raised = False | ||||
elif callable(cmd): | ||||
Pierre-Yves David
|
r31744 | r, raised = _pythonhook(ui, repo, htype, hname, cmd, args, | ||
throw) | ||||
Jesse Long
|
r7416 | elif cmd.startswith('python:'): | ||
Steve Borho
|
r9332 | if cmd.count(':') >= 2: | ||
path, cmd = cmd[7:].rsplit(':', 1) | ||||
Alexander Solovyov
|
r13118 | path = util.expandpath(path) | ||
Matt Mackall
|
r13119 | if repo: | ||
path = os.path.join(repo.root, path) | ||||
Simon Heimberg
|
r17217 | try: | ||
mod = extensions.loadpath(path, 'hghook.%s' % hname) | ||||
except Exception: | ||||
ui.write(_("loading %s hook failed:\n") % hname) | ||||
raise | ||||
Alexander Solovyov
|
r7916 | hookfn = getattr(mod, cmd) | ||
else: | ||||
hookfn = cmd[7:].strip() | ||||
Pierre-Yves David
|
r31744 | r, raised = _pythonhook(ui, repo, htype, hname, hookfn, args, | ||
Siddharth Agarwal
|
r26739 | throw) | ||
Jesse Long
|
r7416 | else: | ||
Pierre-Yves David
|
r31746 | r = _exthook(ui, repo, htype, hname, cmd, args, throw) | ||
Siddharth Agarwal
|
r26739 | raised = False | ||
Siddharth Agarwal
|
r26738 | |||
Siddharth Agarwal
|
r26739 | res[hname] = r, raised | ||
Matt Harbison
|
r24716 | |||
# The stderr is fully buffered on Windows when connected to a pipe. | ||||
# A forcible flush is required to make small stderr data in the | ||||
# remote side available to the client immediately. | ||||
Yuya Nishihara
|
r30473 | util.stderr.flush() | ||
Jesse Long
|
r7416 | finally: | ||
Sune Foldager
|
r9658 | if _redirect and oldstdout >= 0: | ||
Yuya Nishihara
|
r30473 | util.stdout.flush() # write hook output to stderr fd | ||
Sune Foldager
|
r9658 | os.dup2(oldstdout, stdoutno) | ||
Jesse Long
|
r7416 | os.close(oldstdout) | ||
Alexis S. L. Carvalho
|
r6266 | |||
Siddharth Agarwal
|
r26738 | return res | ||