hook.py
204 lines
| 7.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 | |||
from i18n import _ | ||||
Mads Kiilerich
|
r20548 | import os, sys, time | ||
Pierre-Yves David
|
r23415 | import extensions, util, demandimport, error | ||
Matt Mackall
|
r4622 | |||
def _pythonhook(ui, repo, name, hname, funcname, args, throw): | ||||
'''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 | ||
funcname = obj.__module__ + "." + obj.__name__ | ||||
else: | ||||
Matt Mackall
|
r4622 | d = funcname.rfind('.') | ||
if d == -1: | ||||
raise util.Abort(_('%s hook is invalid ("%s" not in ' | ||||
'a module)') % (hname, funcname)) | ||||
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: | ||
Jordi Gutiérrez Hermoso
|
r25328 | obj = __import__(modname) | ||
Matt Mackall
|
r4622 | except ImportError: | ||
Jordi Gutiérrez Hermoso
|
r25328 | e1 = sys.exc_type, sys.exc_value, sys.exc_traceback | ||
try: | ||||
# extensions are loaded with hgext_ prefix | ||||
obj = __import__("hgext_%s" % modname) | ||||
except ImportError: | ||||
e2 = sys.exc_type, sys.exc_value, sys.exc_traceback | ||||
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) | ||||
raise util.Abort(_('%s hook is invalid ' | ||||
'(import of "%s" failed)') % | ||||
(hname, modname)) | ||||
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: | ||
Matt Mackall
|
r4622 | raise util.Abort(_('%s hook is invalid ' | ||
'("%s" is not defined)') % | ||||
(hname, funcname)) | ||||
Augie Fackler
|
r21797 | if not callable(obj): | ||
Matt Mackall
|
r4622 | raise util.Abort(_('%s hook is invalid ' | ||
'("%s" is not callable)') % | ||||
(hname, funcname)) | ||||
Mads Kiilerich
|
r20547 | |||
ui.note(_("calling hook %s: %s\n") % (hname, funcname)) | ||||
starttime = time.time() | ||||
Matt Mackall
|
r4622 | try: | ||
Matt Mackall
|
r25084 | # redirect IO descriptors to the ui descriptors so hooks | ||
# that write directly to these don't mess up the command | ||||
# protocol when running through the command server | ||||
old = sys.stdout, sys.stderr, sys.stdin | ||||
sys.stdout, sys.stderr, sys.stdin = ui.fout, ui.ferr, ui.fin | ||||
Idan Kamara
|
r14889 | |||
Matt Mackall
|
r25084 | r = obj(ui=ui, repo=repo, hooktype=name, **args) | ||
Gregory Szorc
|
r25660 | except Exception as exc: | ||
Matt Mackall
|
r25084 | if isinstance(exc, util.Abort): | ||
ui.warn(_('error: %s hook failed: %s\n') % | ||||
(hname, exc.args[0])) | ||||
else: | ||||
ui.warn(_('error: %s hook raised an exception: ' | ||||
'%s\n') % (hname, exc)) | ||||
if throw: | ||||
Matt Mackall
|
r4622 | raise | ||
Matt Mackall
|
r25084 | ui.traceback() | ||
return True | ||||
Idan Kamara
|
r14889 | finally: | ||
sys.stdout, sys.stderr, sys.stdin = old | ||||
Durham Goode
|
r18671 | duration = time.time() - starttime | ||
Durham Goode
|
r18691 | ui.log('pythonhook', 'pythonhook-%s: %s finished in %0.2f seconds\n', | ||
Mads Kiilerich
|
r20548 | name, 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) | ||
return r | ||||
def _exthook(ui, repo, name, cmd, args, throw): | ||||
ui.note(_("running hook %s: %s\n") % (name, cmd)) | ||||
Matt Mackall
|
r7787 | |||
Durham Goode
|
r18671 | starttime = time.time() | ||
Matt Mackall
|
r7787 | env = {} | ||
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: | ||||
cwd = os.getcwd() | ||||
Yuya Nishihara
|
r23270 | r = ui.system(cmd, environ=env, cwd=cwd) | ||
Durham Goode
|
r18671 | |||
duration = time.time() - 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 | ||||
Matt Zuba
|
r15896 | def _allhooks(ui): | ||
hooks = [] | ||||
for name, cmd in ui.configitems('hooks'): | ||||
if not name.startswith('priority'): | ||||
priority = ui.configint('hooks', 'priority.%s' % name, 0) | ||||
hooks.append((-priority, len(hooks), name, cmd)) | ||||
return [(k, v) for p, o, k, v in sorted(hooks)] | ||||
Matt Mackall
|
r5833 | _redirect = False | ||
def redirect(state): | ||||
Alexis S. L. Carvalho
|
r6266 | global _redirect | ||
Matt Mackall
|
r5833 | _redirect = state | ||
Matt Mackall
|
r4622 | def hook(ui, repo, name, throw=False, **args): | ||
Idan Kamara
|
r17048 | if not ui.callhooks: | ||
return False | ||||
Matt Mackall
|
r4622 | r = False | ||
Sune Foldager
|
r9658 | oldstdout = -1 | ||
Matt Mackall
|
r5833 | |||
Jesse Long
|
r7416 | try: | ||
Matt Zuba
|
r15896 | for hname, cmd in _allhooks(ui): | ||
Jesse Long
|
r7416 | if hname.split('.')[0] != name or not cmd: | ||
continue | ||||
Matt Mackall
|
r17963 | |||
if oldstdout == -1 and _redirect: | ||||
try: | ||||
stdoutno = sys.__stdout__.fileno() | ||||
stderrno = sys.__stderr__.fileno() | ||||
# temporarily redirect stdout to stderr, if possible | ||||
if stdoutno >= 0 and stderrno >= 0: | ||||
sys.__stdout__.flush() | ||||
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 | ||
Augie Fackler
|
r21797 | if callable(cmd): | ||
Jesse Long
|
r7416 | r = _pythonhook(ui, repo, name, hname, cmd, args, throw) or r | ||
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() | ||||
r = _pythonhook(ui, repo, name, hname, hookfn, args, throw) or r | ||||
Jesse Long
|
r7416 | else: | ||
r = _exthook(ui, repo, hname, cmd, args, throw) or r | ||||
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. | ||||
sys.stderr.flush() | ||||
Jesse Long
|
r7416 | finally: | ||
Sune Foldager
|
r9658 | if _redirect and oldstdout >= 0: | ||
os.dup2(oldstdout, stdoutno) | ||||
Jesse Long
|
r7416 | os.close(oldstdout) | ||
Alexis S. L. Carvalho
|
r6266 | |||
return r | ||||