Show More
hook.py
362 lines
| 11.5 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 | ||
Gregory Szorc
|
r45140 | import contextlib | ||
Mitchell Plamann
|
r46329 | import errno | ||
Gregory Szorc
|
r25953 | import os | ||
import sys | ||||
from .i18n import _ | ||||
Gregory Szorc
|
r43359 | from .pycompat import getattr | ||
Gregory Szorc
|
r25953 | from . import ( | ||
demandimport, | ||||
Pulkit Goyal
|
r32642 | encoding, | ||
Gregory Szorc
|
r25953 | error, | ||
extensions, | ||||
Pulkit Goyal
|
r30519 | pycompat, | ||
Gregory Szorc
|
r25953 | util, | ||
) | ||||
Yuya Nishihara
|
r37137 | from .utils import ( | ||
procutil, | ||||
Martin von Zweigbergk
|
r44067 | resourceutil, | ||
Augie Fackler
|
r37769 | stringutil, | ||
Yuya Nishihara
|
r37137 | ) | ||
Matt Mackall
|
r4622 | |||
Augie Fackler
|
r43346 | |||
hindlemail
|
r38052 | def pythonhook(ui, repo, htype, hname, funcname, args, throw): | ||
Augie Fackler
|
r46554 | """call python hook. hook is callable object, looked up as | ||
Matt Mackall
|
r4622 | 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 | ||||
Augie Fackler
|
r46554 | be run as hooks without wrappers to convert return values.""" | ||
Matt Mackall
|
r4622 | |||
Augie Fackler
|
r21797 | if callable(funcname): | ||
Mads Kiilerich
|
r20548 | obj = funcname | ||
Augie Fackler
|
r43809 | funcname = pycompat.sysbytes(obj.__module__ + "." + obj.__name__) | ||
Mads Kiilerich
|
r20548 | else: | ||
Augie Fackler
|
r43347 | d = funcname.rfind(b'.') | ||
Matt Mackall
|
r4622 | if d == -1: | ||
Siddharth Agarwal
|
r26692 | raise error.HookLoadError( | ||
Augie Fackler
|
r43347 | _(b'%s hook is invalid: "%s" not in a module') | ||
Augie Fackler
|
r43346 | % (hname, funcname) | ||
) | ||||
Matt Mackall
|
r4622 | modname = funcname[:d] | ||
Sune Foldager
|
r10103 | oldpaths = sys.path | ||
Martin von Zweigbergk
|
r44067 | if resourceutil.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 | ||||
Augie Fackler
|
r43809 | obj = __import__("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: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b'exception from first failed import ' | ||
b'attempt:\n' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Jordi Gutiérrez Hermoso
|
r25328 | ui.traceback(e1) | ||
if ui.tracebackflag: | ||||
Augie Fackler
|
r43346 | ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b'exception from second failed import ' | ||
b'attempt:\n' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Jordi Gutiérrez Hermoso
|
r25328 | ui.traceback(e2) | ||
Siddharth Agarwal
|
r28080 | |||
if not ui.tracebackflag: | ||||
tracebackhint = _( | ||||
Augie Fackler
|
r43347 | b'run with --traceback for stack trace' | ||
Augie Fackler
|
r43346 | ) | ||
Siddharth Agarwal
|
r28080 | else: | ||
tracebackhint = None | ||||
Siddharth Agarwal
|
r26692 | raise error.HookLoadError( | ||
Augie Fackler
|
r43347 | _(b'%s hook is invalid: import of "%s" failed') | ||
Augie Fackler
|
r43346 | % (hname, modname), | ||
hint=tracebackhint, | ||||
) | ||||
Steve Borho
|
r9332 | sys.path = oldpaths | ||
Matt Mackall
|
r4622 | try: | ||
Augie Fackler
|
r43347 | for p in funcname.split(b'.')[1:]: | ||
Matt Mackall
|
r4622 | obj = getattr(obj, p) | ||
Benoit Boissinot
|
r7280 | except AttributeError: | ||
Siddharth Agarwal
|
r26692 | raise error.HookLoadError( | ||
Augie Fackler
|
r43347 | _(b'%s hook is invalid: "%s" is not defined') | ||
% (hname, funcname) | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r21797 | if not callable(obj): | ||
Siddharth Agarwal
|
r26692 | raise error.HookLoadError( | ||
Augie Fackler
|
r43347 | _(b'%s hook is invalid: "%s" is not callable') | ||
Augie Fackler
|
r43346 | % (hname, funcname) | ||
) | ||||
Mads Kiilerich
|
r20547 | |||
Augie Fackler
|
r43347 | ui.note(_(b"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): | ||
Augie Fackler
|
r43347 | ui.warn(_(b'error: %s hook failed: %s\n') % (hname, exc.args[0])) | ||
Matt Mackall
|
r25084 | else: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
Martin von Zweigbergk
|
r43387 | _(b'error: %s hook raised an exception: %s\n') | ||
Augie Fackler
|
r43346 | % (hname, stringutil.forcebytestr(exc)) | ||
) | ||||
Matt Mackall
|
r25084 | if throw: | ||
Matt Mackall
|
r4622 | raise | ||
Siddharth Agarwal
|
r28108 | if not ui.tracebackflag: | ||
Augie Fackler
|
r43347 | ui.warn(_(b'(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 | ||
Augie Fackler
|
r43346 | ui.log( | ||
Augie Fackler
|
r43347 | b'pythonhook', | ||
b'pythonhook-%s: %s finished in %0.2f seconds\n', | ||||
Augie Fackler
|
r43346 | htype, | ||
funcname, | ||||
duration, | ||||
) | ||||
Matt Mackall
|
r4622 | if r: | ||
if throw: | ||||
Augie Fackler
|
r43347 | raise error.HookAbort(_(b'%s hook failed') % hname) | ||
ui.warn(_(b'warning: %s hook failed\n') % hname) | ||||
Siddharth Agarwal
|
r26739 | return r, False | ||
Matt Mackall
|
r4622 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r31746 | def _exthook(ui, repo, htype, name, cmd, args, throw): | ||
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(): | ||||
Augie Fackler
|
r43347 | env[b'HG_PENDING'] = repo.root | ||
env[b'HG_HOOKTYPE'] = htype | ||||
env[b'HG_HOOKNAME'] = name | ||||
r47242 | ||||
r47243 | if ui.config(b'hooks', b'%s:run-with-plain' % name) == b'auto': | |||
plain = ui.plain() | ||||
else: | ||||
plain = ui.configbool(b'hooks', b'%s:run-with-plain' % name) | ||||
r47242 | if plain: | |||
env[b'HGPLAIN'] = b'1' | ||||
else: | ||||
env[b'HGPLAIN'] = b'' | ||||
FUJIWARA Katsunori
|
r26751 | |||
Gregory Szorc
|
r43376 | for k, v in pycompat.iteritems(args): | ||
Joerg Sonnenberger
|
r45350 | # transaction changes can accumulate MBs of data, so skip it | ||
# for external hooks | ||||
if k == b'changes': | ||||
continue | ||||
Augie Fackler
|
r21797 | if callable(v): | ||
Matt Mackall
|
r7787 | v = v() | ||
Augie Fackler
|
r37770 | if isinstance(v, (dict, list)): | ||
Yuya Nishihara
|
r37961 | v = stringutil.pprint(v) | ||
Augie Fackler
|
r43347 | env[b'HG_' + k.upper()] = v | ||
Matt Mackall
|
r7787 | |||
Augie Fackler
|
r43347 | if ui.configbool(b'hooks', b'tonative.%s' % name, False): | ||
Matt Harbison
|
r38746 | oldcmd = cmd | ||
Matt Harbison
|
r38648 | cmd = procutil.shelltonative(cmd, env) | ||
Matt Harbison
|
r38746 | if cmd != oldcmd: | ||
Augie Fackler
|
r43347 | ui.note(_(b'converting hook "%s" to native\n') % name) | ||
Matt Harbison
|
r38503 | |||
Augie Fackler
|
r43347 | ui.note(_(b"running hook %s: %s\n") % (name, cmd)) | ||
Matt Harbison
|
r38503 | |||
Matt Mackall
|
r5869 | if repo: | ||
cwd = repo.root | ||||
else: | ||||
Matt Harbison
|
r39843 | cwd = encoding.getcwd() | ||
Augie Fackler
|
r43347 | r = ui.system(cmd, environ=env, cwd=cwd, blockedtag=b'exthook-%s' % (name,)) | ||
Durham Goode
|
r18671 | |||
Simon Farnsworth
|
r30975 | duration = util.timer() - starttime | ||
Augie Fackler
|
r43346 | ui.log( | ||
Augie Fackler
|
r43347 | b'exthook', | ||
b'exthook-%s: %s finished in %0.2f seconds\n', | ||||
Augie Fackler
|
r43346 | name, | ||
cmd, | ||||
duration, | ||||
) | ||||
Matt Mackall
|
r4622 | if r: | ||
Yuya Nishihara
|
r37481 | desc = procutil.explainexit(r) | ||
Matt Mackall
|
r4622 | if throw: | ||
Augie Fackler
|
r43347 | raise error.HookAbort(_(b'%s hook %s') % (name, desc)) | ||
ui.warn(_(b'warning: %s hook %s\n') % (name, desc)) | ||||
Matt Mackall
|
r4622 | return r | ||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r28938 | # represent an untrusted hook command | ||
_fromuntrusted = object() | ||||
Augie Fackler
|
r43346 | |||
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(): | ||||
Charles Chamberlain
|
r45371 | trustedvalue = hooks.get(name, ((), (), name, _fromuntrusted)) | ||
Pierre-Yves David
|
r28938 | 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())] | ||
Augie Fackler
|
r43346 | |||
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 = {} | ||
Augie Fackler
|
r43347 | for name, cmd in ui.configitems(b'hooks', untrusted=_untrusted): | ||
r47240 | if ( | |||
name.startswith(b'priority.') | ||||
or name.startswith(b'tonative.') | ||||
or b':' in name | ||||
): | ||||
Matt Harbison
|
r38648 | continue | ||
Augie Fackler
|
r43347 | priority = ui.configint(b'hooks', b'priority.%s' % name, 0) | ||
Charles Chamberlain
|
r45371 | hooks[name] = ((-priority,), (len(hooks),), name, cmd) | ||
Pierre-Yves David
|
r28937 | return hooks | ||
Matt Zuba
|
r15896 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r5833 | _redirect = False | ||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r5833 | def redirect(state): | ||
Alexis S. L. Carvalho
|
r6266 | global _redirect | ||
Matt Mackall
|
r5833 | _redirect = state | ||
Augie Fackler
|
r43346 | |||
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): | ||||
Augie Fackler
|
r43347 | if hname.split(b'.')[0] == htype and cmd: | ||
Boris Feld
|
r34688 | return True | ||
return False | ||||
Augie Fackler
|
r43346 | |||
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): | ||||
Augie Fackler
|
r43347 | if hname.split(b'.')[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 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r45140 | @contextlib.contextmanager | ||
def redirect_stdio(): | ||||
"""Redirects stdout to stderr, if possible.""" | ||||
oldstdout = -1 | ||||
try: | ||||
if _redirect: | ||||
try: | ||||
stdoutno = procutil.stdout.fileno() | ||||
stderrno = procutil.stderr.fileno() | ||||
# temporarily redirect stdout to stderr, if possible | ||||
if stdoutno >= 0 and stderrno >= 0: | ||||
procutil.stdout.flush() | ||||
oldstdout = os.dup(stdoutno) | ||||
os.dup2(stderrno, stdoutno) | ||||
except (OSError, AttributeError): | ||||
# files seem to be bogus, give up on redirecting (WSGI, etc) | ||||
pass | ||||
yield | ||||
finally: | ||||
# 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. | ||||
Mitchell Plamann
|
r46329 | try: | ||
procutil.stderr.flush() | ||||
except IOError as err: | ||||
if err.errno not in (errno.EPIPE, errno.EIO, errno.EBADF): | ||||
raise error.StdioError(err) | ||||
Gregory Szorc
|
r45140 | |||
if _redirect and oldstdout >= 0: | ||||
Mitchell Plamann
|
r46329 | try: | ||
procutil.stdout.flush() # write hook output to stderr fd | ||||
except IOError as err: | ||||
if err.errno not in (errno.EPIPE, errno.EIO, errno.EBADF): | ||||
raise error.StdioError(err) | ||||
Gregory Szorc
|
r45140 | os.dup2(oldstdout, stdoutno) | ||
os.close(oldstdout) | ||||
Pierre-Yves David
|
r31744 | def runhooks(ui, repo, htype, hooks, throw=False, **args): | ||
Pulkit Goyal
|
r32897 | args = pycompat.byteskwargs(args) | ||
Siddharth Agarwal
|
r26738 | res = {} | ||
Matt Mackall
|
r5833 | |||
Gregory Szorc
|
r45140 | with redirect_stdio(): | ||
Siddharth Agarwal
|
r26737 | for hname, cmd in hooks: | ||
Pierre-Yves David
|
r28938 | if cmd is _fromuntrusted: | ||
if throw: | ||||
raise error.HookAbort( | ||||
Augie Fackler
|
r43347 | _(b'untrusted hook %s not executed') % hname, | ||
hint=_(b"see 'hg help config.trusted'"), | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | ui.warn(_(b'warning: untrusted hook %s not executed\n') % hname) | ||
Pierre-Yves David
|
r28938 | r = 1 | ||
raised = False | ||||
elif callable(cmd): | ||||
Augie Fackler
|
r43346 | r, raised = pythonhook(ui, repo, htype, hname, cmd, args, throw) | ||
Augie Fackler
|
r43347 | elif cmd.startswith(b'python:'): | ||
if cmd.count(b':') >= 2: | ||||
path, cmd = cmd[7:].rsplit(b':', 1) | ||||
Alexander Solovyov
|
r13118 | path = util.expandpath(path) | ||
Matt Mackall
|
r13119 | if repo: | ||
path = os.path.join(repo.root, path) | ||||
Simon Heimberg
|
r17217 | try: | ||
Augie Fackler
|
r43347 | mod = extensions.loadpath(path, b'hghook.%s' % hname) | ||
Simon Heimberg
|
r17217 | except Exception: | ||
Augie Fackler
|
r43347 | ui.write(_(b"loading %s hook failed:\n") % hname) | ||
Simon Heimberg
|
r17217 | raise | ||
Alexander Solovyov
|
r7916 | hookfn = getattr(mod, cmd) | ||
else: | ||||
hookfn = cmd[7:].strip() | ||||
Augie Fackler
|
r43346 | r, raised = pythonhook( | ||
ui, repo, htype, hname, hookfn, args, 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 | ||
Alexis S. L. Carvalho
|
r6266 | |||
Siddharth Agarwal
|
r26738 | return res | ||