dispatch.py
889 lines
| 30.9 KiB
| text/x-python
|
PythonLexer
/ mercurial / dispatch.py
Matt Mackall
|
r5178 | # dispatch.py - command dispatching for mercurial | ||
# | ||||
# Copyright 2005-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
|
r5178 | |||
Gregory Szorc
|
r27615 | from __future__ import absolute_import, print_function | ||
Gregory Szorc
|
r25932 | |||
import atexit | ||||
Augie Fackler
|
r24221 | import difflib | ||
Gregory Szorc
|
r25932 | import errno | ||
Pulkit Goyal
|
r30576 | import getopt | ||
Gregory Szorc
|
r25932 | import os | ||
import pdb | ||||
import re | ||||
import signal | ||||
import sys | ||||
import time | ||||
import traceback | ||||
from .i18n import _ | ||||
from . import ( | ||||
cmdutil, | ||||
Pierre-Yves David
|
r30653 | color, | ||
Gregory Szorc
|
r25932 | commands, | ||
Gregory Szorc
|
r30401 | debugcommands, | ||
Gregory Szorc
|
r25932 | demandimport, | ||
encoding, | ||||
error, | ||||
extensions, | ||||
fancyopts, | ||||
FUJIWARA Katsunori
|
r28447 | fileset, | ||
Gregory Szorc
|
r25932 | hg, | ||
hook, | ||||
Gregory Szorc
|
r29781 | profiling, | ||
Pulkit Goyal
|
r30468 | pycompat, | ||
FUJIWARA Katsunori
|
r28394 | revset, | ||
Jun Wu
|
r30520 | scmutil, | ||
FUJIWARA Katsunori
|
r28692 | templatefilters, | ||
FUJIWARA Katsunori
|
r28538 | templatekw, | ||
FUJIWARA Katsunori
|
r28695 | templater, | ||
Gregory Szorc
|
r25932 | ui as uimod, | ||
util, | ||||
) | ||||
Matt Mackall
|
r5178 | |||
Idan Kamara
|
r14438 | class request(object): | ||
Brodie Rao
|
r16683 | def __init__(self, args, ui=None, repo=None, fin=None, fout=None, | ||
ferr=None): | ||||
Idan Kamara
|
r14438 | self.args = args | ||
Idan Kamara
|
r14439 | self.ui = ui | ||
Idan Kamara
|
r14510 | self.repo = repo | ||
Idan Kamara
|
r14438 | |||
Idan Kamara
|
r14613 | # input/output/error streams | ||
self.fin = fin | ||||
self.fout = fout | ||||
self.ferr = ferr | ||||
Matt Mackall
|
r5178 | def run(): | ||
"run the command in sys.argv" | ||||
Pulkit Goyal
|
r30468 | sys.exit((dispatch(request(pycompat.sysargv[1:])) or 0) & 255) | ||
Matt Mackall
|
r5178 | |||
Augie Fackler
|
r24221 | def _getsimilar(symbols, value): | ||
sim = lambda x: difflib.SequenceMatcher(None, value, x).ratio() | ||||
# The cutoff for similarity here is pretty arbitrary. It should | ||||
# probably be investigated and tweaked. | ||||
return [s for s in symbols if sim(s) > 0.6] | ||||
Bryan O'Sullivan
|
r27623 | def _reportsimilar(write, similar): | ||
if len(similar) == 1: | ||||
write(_("(did you mean %s?)\n") % similar[0]) | ||||
elif similar: | ||||
ss = ", ".join(sorted(similar)) | ||||
write(_("(did you mean one of %s?)\n") % ss) | ||||
Augie Fackler
|
r24039 | def _formatparse(write, inst): | ||
Augie Fackler
|
r24221 | similar = [] | ||
if isinstance(inst, error.UnknownIdentifier): | ||||
# make sure to check fileset first, as revset can invoke fileset | ||||
similar = _getsimilar(inst.symbols, inst.function) | ||||
Augie Fackler
|
r24039 | if len(inst.args) > 1: | ||
write(_("hg: parse error at %s: %s\n") % | ||||
(inst.args[1], inst.args[0])) | ||||
if (inst.args[0][0] == ' '): | ||||
write(_("unexpected leading whitespace\n")) | ||||
else: | ||||
write(_("hg: parse error: %s\n") % inst.args[0]) | ||||
Bryan O'Sullivan
|
r27623 | _reportsimilar(write, similar) | ||
Jun Wu
|
r28515 | if inst.hint: | ||
write(_("(%s)\n") % inst.hint) | ||||
Augie Fackler
|
r24039 | |||
Idan Kamara
|
r14438 | def dispatch(req): | ||
"run the command specified in req.args" | ||||
Idan Kamara
|
r14615 | if req.ferr: | ||
ferr = req.ferr | ||||
elif req.ui: | ||||
ferr = req.ui.ferr | ||||
else: | ||||
Yuya Nishihara
|
r30473 | ferr = util.stderr | ||
Idan Kamara
|
r14615 | |||
Matt Mackall
|
r5178 | try: | ||
Idan Kamara
|
r14439 | if not req.ui: | ||
Yuya Nishihara
|
r30559 | req.ui = uimod.ui.load() | ||
Idan Kamara
|
r14438 | if '--traceback' in req.args: | ||
Mads Kiilerich
|
r20788 | req.ui.setconfig('ui', 'traceback', 'on', '--traceback') | ||
Idan Kamara
|
r14615 | |||
# set ui streams from the request | ||||
if req.fin: | ||||
req.ui.fin = req.fin | ||||
if req.fout: | ||||
req.ui.fout = req.fout | ||||
if req.ferr: | ||||
req.ui.ferr = req.ferr | ||||
Pierre-Yves David
|
r26587 | except error.Abort as inst: | ||
Idan Kamara
|
r14615 | ferr.write(_("abort: %s\n") % inst) | ||
Benoit Boissinot
|
r11574 | if inst.hint: | ||
Idan Kamara
|
r14615 | ferr.write(_("(%s)\n") % inst.hint) | ||
Matt Mackall
|
r5178 | return -1 | ||
Gregory Szorc
|
r25660 | except error.ParseError as inst: | ||
Augie Fackler
|
r24039 | _formatparse(ferr.write, inst) | ||
Martin Geisler
|
r9470 | return -1 | ||
Idan Kamara
|
r14615 | |||
Durham Goode
|
r19229 | msg = ' '.join(' ' in a and repr(a) or a for a in req.args) | ||
starttime = time.time() | ||||
ret = None | ||||
try: | ||||
ret = _runcatch(req) | ||||
Yuya Nishihara
|
r28520 | except KeyboardInterrupt: | ||
try: | ||||
req.ui.warn(_("interrupted!\n")) | ||||
except IOError as inst: | ||||
if inst.errno != errno.EPIPE: | ||||
raise | ||||
ret = -1 | ||||
Durham Goode
|
r19229 | finally: | ||
duration = time.time() - starttime | ||||
Jun Wu
|
r28534 | req.ui.flush() | ||
Durham Goode
|
r19229 | req.ui.log("commandfinish", "%s exited %s after %0.2f seconds\n", | ||
msg, ret or 0, duration) | ||||
Yuya Nishihara
|
r28520 | return ret | ||
Matt Mackall
|
r5178 | |||
Idan Kamara
|
r14439 | def _runcatch(req): | ||
Matt Mackall
|
r5178 | def catchterm(*args): | ||
Matt Mackall
|
r7644 | raise error.SignalInterrupt | ||
Matt Mackall
|
r5178 | |||
Idan Kamara
|
r14439 | ui = req.ui | ||
Simon Heimberg
|
r10952 | try: | ||
for name in 'SIGBREAK', 'SIGHUP', 'SIGTERM': | ||||
num = getattr(signal, name, None) | ||||
if num: | ||||
signal.signal(num, catchterm) | ||||
except ValueError: | ||||
pass # happens if called in a thread | ||||
Matt Mackall
|
r5178 | |||
Jun Wu
|
r29761 | def _runcatchfunc(): | ||
Matt Mackall
|
r5178 | try: | ||
Sean Farley
|
r19640 | debugger = 'pdb' | ||
debugtrace = { | ||||
'pdb' : pdb.set_trace | ||||
} | ||||
debugmortem = { | ||||
'pdb' : pdb.post_mortem | ||||
} | ||||
Sean Farley
|
r19639 | |||
# read --config before doing anything else | ||||
# (e.g. to change trust settings for reading .hg/hgrc) | ||||
cfgs = _parseconfig(req.ui, _earlygetopt(['--config'], req.args)) | ||||
if req.repo: | ||||
# copy configs that were passed on the cmdline (--config) to | ||||
# the repo ui | ||||
Matt Mackall
|
r20796 | for sec, name, val in cfgs: | ||
req.repo.ui.setconfig(sec, name, val, source='--config') | ||||
Sean Farley
|
r19639 | |||
Matt Mackall
|
r25833 | # developer config: ui.debugger | ||
Sean Farley
|
r19640 | debugger = ui.config("ui", "debugger") | ||
Jordi Gutiérrez Hermoso
|
r20826 | debugmod = pdb | ||
Sean Farley
|
r20122 | if not debugger or ui.plain(): | ||
Matt Mackall
|
r25833 | # if we are in HGPLAIN mode, then disable custom debugging | ||
Sean Farley
|
r19640 | debugger = 'pdb' | ||
Jordi Gutiérrez Hermoso
|
r20826 | elif '--debugger' in req.args: | ||
# This import can be slow for fancy debuggers, so only | ||||
# do it when absolutely necessary, i.e. when actual | ||||
# debugging has been requested | ||||
Jordi Gutiérrez Hermoso
|
r25329 | with demandimport.deactivated(): | ||
try: | ||||
debugmod = __import__(debugger) | ||||
except ImportError: | ||||
pass # Leave debugmod = pdb | ||||
Sean Farley
|
r19640 | |||
debugtrace[debugger] = debugmod.set_trace | ||||
debugmortem[debugger] = debugmod.post_mortem | ||||
Matt Mackall
|
r5178 | # enter the debugger before command execution | ||
Idan Kamara
|
r14438 | if '--debugger' in req.args: | ||
Mads Kiilerich
|
r11495 | ui.warn(_("entering debugger - " | ||
"type c to continue starting hg or h for help\n")) | ||||
Sean Farley
|
r19640 | |||
if (debugger != 'pdb' and | ||||
debugtrace[debugger] == debugtrace['pdb']): | ||||
ui.warn(_("%s debugger specified " | ||||
"but its module was not found\n") % debugger) | ||||
Jordi Gutiérrez Hermoso
|
r26236 | with demandimport.deactivated(): | ||
Jordi Gutiérrez Hermoso
|
r26216 | debugtrace[debugger]() | ||
Matt Mackall
|
r5178 | try: | ||
Idan Kamara
|
r14439 | return _dispatch(req) | ||
Matt Mackall
|
r5178 | finally: | ||
ui.flush() | ||||
Brodie Rao
|
r16705 | except: # re-raises | ||
Matt Mackall
|
r5178 | # enter the debugger when we hit an exception | ||
Idan Kamara
|
r14438 | if '--debugger' in req.args: | ||
Mads Kiilerich
|
r11494 | traceback.print_exc() | ||
Sean Farley
|
r19640 | debugmortem[debugger](sys.exc_info()[2]) | ||
Matt Mackall
|
r8206 | ui.traceback() | ||
Matt Mackall
|
r5178 | raise | ||
Jun Wu
|
r29761 | return callcatch(ui, _runcatchfunc) | ||
def callcatch(ui, func): | ||||
Jun Wu
|
r30520 | """like scmutil.callcatch but handles more high-level exceptions about | ||
config parsing and commands. besides, use handlecommandexception to handle | ||||
uncaught exceptions. | ||||
Jun Wu
|
r29761 | """ | ||
try: | ||||
Jun Wu
|
r30520 | return scmutil.callcatch(ui, func) | ||
Gregory Szorc
|
r25660 | except error.AmbiguousCommand as inst: | ||
Matt Mackall
|
r5178 | ui.warn(_("hg: command '%s' is ambiguous:\n %s\n") % | ||
(inst.args[0], " ".join(inst.args[1]))) | ||||
Gregory Szorc
|
r25660 | except error.CommandError as inst: | ||
Matt Mackall
|
r7645 | if inst.args[0]: | ||
ui.warn(_("hg %s: %s\n") % (inst.args[0], inst.args[1])) | ||||
Martin Geisler
|
r14286 | commands.help_(ui, inst.args[0], full=False, command=True) | ||
Matt Mackall
|
r7645 | else: | ||
ui.warn(_("hg: %s\n") % inst.args[1]) | ||||
commands.help_(ui, 'shortlist') | ||||
Jun Wu
|
r30520 | except error.ParseError as inst: | ||
_formatparse(ui.warn, inst) | ||||
return -1 | ||||
Gregory Szorc
|
r25660 | except error.UnknownCommand as inst: | ||
Matt Mackall
|
r7645 | ui.warn(_("hg: unknown command '%s'\n") % inst.args[0]) | ||
Brodie Rao
|
r10364 | try: | ||
# check if the command is in a disabled extension | ||||
# (but don't check for extensions themselves) | ||||
commands.help_(ui, inst.args[0], unknowncmd=True) | ||||
Pierre-Yves David
|
r26587 | except (error.UnknownCommand, error.Abort): | ||
Augie Fackler
|
r24222 | suggested = False | ||
if len(inst.args) == 2: | ||||
sim = _getsimilar(inst.args[1], inst.args[0]) | ||||
if sim: | ||||
Bryan O'Sullivan
|
r27623 | _reportsimilar(ui.warn, sim) | ||
Augie Fackler
|
r24222 | suggested = True | ||
if not suggested: | ||||
commands.help_(ui, 'shortlist') | ||||
Jun Wu
|
r30520 | except IOError: | ||
raise | ||||
Matt Mackall
|
r7645 | except KeyboardInterrupt: | ||
Yuya Nishihara
|
r28520 | raise | ||
Jun Wu
|
r30520 | except: # probably re-raises | ||
Martijn Pieters
|
r28784 | if not handlecommandexception(ui): | ||
raise | ||||
Matt Mackall
|
r5178 | |||
return -1 | ||||
Alexander Solovyov
|
r14265 | def aliasargs(fn, givenargs): | ||
args = getattr(fn, 'args', []) | ||||
Matt Mackall
|
r16294 | if args: | ||
Alexander Solovyov
|
r14265 | cmd = ' '.join(map(util.shellquote, args)) | ||
nums = [] | ||||
def replacer(m): | ||||
num = int(m.group(1)) - 1 | ||||
nums.append(num) | ||||
Matt Mackall
|
r16277 | if num < len(givenargs): | ||
return givenargs[num] | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('too few arguments for command alias')) | ||
Alexander Solovyov
|
r14265 | cmd = re.sub(r'\$(\d+|\$)', replacer, cmd) | ||
givenargs = [x for i, x in enumerate(givenargs) | ||||
if i not in nums] | ||||
Pulkit Goyal
|
r30678 | args = pycompat.shlexsplit(cmd) | ||
Alexander Solovyov
|
r14265 | return args + givenargs | ||
Brendan Cully
|
r8655 | |||
Siddharth Agarwal
|
r22158 | def aliasinterpolate(name, args, cmd): | ||
'''interpolate args into cmd for shell aliases | ||||
This also handles $0, $@ and "$@". | ||||
''' | ||||
# util.interpolate can't deal with "$@" (with quotes) because it's only | ||||
# built to match prefix + patterns. | ||||
replacemap = dict(('$%d' % (i + 1), arg) for i, arg in enumerate(args)) | ||||
replacemap['$0'] = name | ||||
replacemap['$$'] = '$' | ||||
replacemap['$@'] = ' '.join(args) | ||||
# Typical Unix shells interpolate "$@" (with quotes) as all the positional | ||||
# parameters, separated out into words. Emulate the same behavior here by | ||||
# quoting the arguments individually. POSIX shells will then typically | ||||
# tokenize each argument into exactly one word. | ||||
replacemap['"$@"'] = ' '.join(util.shellquote(arg) for arg in args) | ||||
# escape '\$' for regex | ||||
regex = '|'.join(replacemap.keys()).replace('$', r'\$') | ||||
r = re.compile(regex) | ||||
return r.sub(lambda x: replacemap[x.group()], cmd) | ||||
Brendan Cully
|
r8655 | class cmdalias(object): | ||
timeless
|
r28828 | def __init__(self, name, definition, cmdtable, source): | ||
Brodie Rao
|
r12039 | self.name = self.cmd = name | ||
Brodie Rao
|
r12092 | self.cmdname = '' | ||
Brendan Cully
|
r8655 | self.definition = definition | ||
Yuya Nishihara
|
r22160 | self.fn = None | ||
Jun Wu
|
r29087 | self.givenargs = [] | ||
Brendan Cully
|
r8655 | self.opts = [] | ||
self.help = '' | ||||
Yuya Nishihara
|
r22160 | self.badalias = None | ||
Yuya Nishihara
|
r22161 | self.unknowncmd = False | ||
timeless
|
r28828 | self.source = source | ||
Brendan Cully
|
r8655 | |||
try: | ||||
Brodie Rao
|
r12039 | aliases, entry = cmdutil.findcmd(self.name, cmdtable) | ||
for alias, e in cmdtable.iteritems(): | ||||
if e is entry: | ||||
self.cmd = alias | ||||
break | ||||
Brendan Cully
|
r8655 | self.shadows = True | ||
except error.UnknownCommand: | ||||
self.shadows = False | ||||
if not self.definition: | ||||
Yuya Nishihara
|
r22160 | self.badalias = _("no definition for alias '%s'") % self.name | ||
Brendan Cully
|
r8655 | return | ||
Steve Losh
|
r11524 | if self.definition.startswith('!'): | ||
Steve Losh
|
r12536 | self.shell = True | ||
Steve Losh
|
r11524 | def fn(ui, *args): | ||
Steve Losh
|
r11989 | env = {'HG_ARGS': ' '.join((self.name,) + args)} | ||
def _checkvar(m): | ||||
Roman Sokolov
|
r13392 | if m.groups()[0] == '$': | ||
return m.group() | ||||
elif int(m.groups()[0]) <= len(args): | ||||
Steve Losh
|
r11989 | return m.group() | ||
else: | ||||
David Soria Parra
|
r14708 | ui.debug("No argument found for substitution " | ||
"of %i variable in alias '%s' definition." | ||||
Roman Sokolov
|
r13393 | % (int(m.groups()[0]), self.name)) | ||
Steve Losh
|
r11989 | return '' | ||
Roman Sokolov
|
r13392 | cmd = re.sub(r'\$(\d+|\$)', _checkvar, self.definition[1:]) | ||
Siddharth Agarwal
|
r22158 | cmd = aliasinterpolate(self.name, args, cmd) | ||
Yuya Nishihara
|
r23270 | return ui.system(cmd, environ=env) | ||
Steve Losh
|
r11524 | self.fn = fn | ||
return | ||||
Yuya Nishihara
|
r21569 | try: | ||
Pulkit Goyal
|
r30678 | args = pycompat.shlexsplit(self.definition) | ||
Gregory Szorc
|
r25660 | except ValueError as inst: | ||
Yuya Nishihara
|
r22160 | self.badalias = (_("error in definition for alias '%s': %s") | ||
% (self.name, inst)) | ||||
Yuya Nishihara
|
r21569 | return | ||
Brodie Rao
|
r12092 | self.cmdname = cmd = args.pop(0) | ||
Jun Wu
|
r29087 | self.givenargs = args | ||
Brendan Cully
|
r8655 | |||
Simon Heimberg
|
r18693 | for invalidarg in ("--cwd", "-R", "--repository", "--repo", "--config"): | ||
Dan Villiom Podlaski Christiansen
|
r11695 | if _earlygetopt([invalidarg], args): | ||
Yuya Nishihara
|
r22160 | self.badalias = (_("error in definition for alias '%s': %s may " | ||
"only be given on the command line") | ||||
% (self.name, invalidarg)) | ||||
Dan Villiom Podlaski Christiansen
|
r11695 | return | ||
Brendan Cully
|
r8655 | try: | ||
Nicolas Dumazet
|
r9993 | tableentry = cmdutil.findcmd(cmd, cmdtable, False)[1] | ||
if len(tableentry) > 2: | ||||
self.fn, self.opts, self.help = tableentry | ||||
else: | ||||
self.fn, self.opts = tableentry | ||||
Peter Arrenbrecht
|
r9876 | if self.help.startswith("hg " + cmd): | ||
# drop prefix in old-style help lines so hg shows the alias | ||||
self.help = self.help[4 + len(cmd):] | ||||
Yuya Nishihara
|
r10564 | self.__doc__ = self.fn.__doc__ | ||
Peter Arrenbrecht
|
r9876 | |||
Brendan Cully
|
r8655 | except error.UnknownCommand: | ||
Yuya Nishihara
|
r22160 | self.badalias = (_("alias '%s' resolves to unknown command '%s'") | ||
% (self.name, cmd)) | ||||
Yuya Nishihara
|
r22161 | self.unknowncmd = True | ||
Brendan Cully
|
r8655 | except error.AmbiguousCommand: | ||
Yuya Nishihara
|
r22160 | self.badalias = (_("alias '%s' resolves to ambiguous command '%s'") | ||
% (self.name, cmd)) | ||||
Brendan Cully
|
r8655 | |||
Jun Wu
|
r29087 | @property | ||
def args(self): | ||||
args = map(util.expandpath, self.givenargs) | ||||
return aliasargs(self.fn, args) | ||||
Yuya Nishihara
|
r28621 | def __getattr__(self, name): | ||
adefaults = {'norepo': True, 'optionalrepo': False, 'inferrepo': False} | ||||
if name not in adefaults: | ||||
raise AttributeError(name) | ||||
if self.badalias or util.safehasattr(self, 'shell'): | ||||
return adefaults[name] | ||||
return getattr(self.fn, name) | ||||
Brendan Cully
|
r8655 | def __call__(self, ui, *args, **opts): | ||
Yuya Nishihara
|
r22160 | if self.badalias: | ||
Yuya Nishihara
|
r22164 | hint = None | ||
Yuya Nishihara
|
r22161 | if self.unknowncmd: | ||
Brodie Rao
|
r10364 | try: | ||
# check if the command is in a disabled extension | ||||
Yuya Nishihara
|
r22163 | cmd, ext = extensions.disabledcmd(ui, self.cmdname)[:2] | ||
Yuya Nishihara
|
r22164 | hint = _("'%s' is provided by '%s' extension") % (cmd, ext) | ||
Brodie Rao
|
r10364 | except error.UnknownCommand: | ||
pass | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(self.badalias, hint=hint) | ||
Brendan Cully
|
r8655 | if self.shadows: | ||
Martin Geisler
|
r14704 | ui.debug("alias '%s' shadows command '%s'\n" % | ||
Brodie Rao
|
r12092 | (self.name, self.cmdname)) | ||
Brendan Cully
|
r8655 | |||
Augie Fackler
|
r29846 | ui.log('commandalias', "alias '%s' expands to '%s'\n", | ||
self.name, self.definition) | ||||
Augie Fackler
|
r14950 | if util.safehasattr(self, 'shell'): | ||
Steve Losh
|
r11989 | return self.fn(ui, *args, **opts) | ||
else: | ||||
Brodie Rao
|
r12093 | try: | ||
Yuya Nishihara
|
r21556 | return util.checksignature(self.fn)(ui, *args, **opts) | ||
Brodie Rao
|
r12093 | except error.SignatureError: | ||
args = ' '.join([self.cmdname] + self.args) | ||||
Martin Geisler
|
r14704 | ui.debug("alias '%s' expands to '%s'\n" % (self.name, args)) | ||
Brodie Rao
|
r12093 | raise | ||
Brendan Cully
|
r8655 | |||
def addaliases(ui, cmdtable): | ||||
# aliases are processed after extensions have been loaded, so they | ||||
# may use extension commands. Aliases can also use other alias definitions, | ||||
# but only if they have been defined prior to the current definition. | ||||
for alias, definition in ui.configitems('alias'): | ||||
timeless
|
r28828 | source = ui.configsource('alias', alias) | ||
aliasdef = cmdalias(alias, definition, cmdtable, source) | ||||
Idan Kamara
|
r15019 | |||
try: | ||||
olddef = cmdtable[aliasdef.cmd][0] | ||||
if olddef.definition == aliasdef.definition: | ||||
continue | ||||
except (KeyError, AttributeError): | ||||
# definition might not exist or it might not be a cmdalias | ||||
pass | ||||
Augie Fackler
|
r15233 | cmdtable[aliasdef.name] = (aliasdef, aliasdef.opts, aliasdef.help) | ||
Brendan Cully
|
r8655 | |||
Matt Mackall
|
r5178 | def _parse(ui, args): | ||
options = {} | ||||
cmdoptions = {} | ||||
try: | ||||
args = fancyopts.fancyopts(args, commands.globalopts, options) | ||||
Pulkit Goyal
|
r30576 | except getopt.GetoptError as inst: | ||
Matt Mackall
|
r11287 | raise error.CommandError(None, inst) | ||
Matt Mackall
|
r5178 | |||
if args: | ||||
cmd, args = args[0], args[1:] | ||||
Henri Wiechers
|
r9875 | aliases, entry = cmdutil.findcmd(cmd, commands.table, | ||
Yuya Nishihara
|
r16591 | ui.configbool("ui", "strict")) | ||
Matt Mackall
|
r5178 | cmd = aliases[0] | ||
Alexander Solovyov
|
r14265 | args = aliasargs(entry[0], args) | ||
Matt Mackall
|
r5178 | defaults = ui.config("defaults", cmd) | ||
if defaults: | ||||
Pulkit Goyal
|
r30678 | args = map(util.expandpath, pycompat.shlexsplit(defaults)) + args | ||
Henri Wiechers
|
r9875 | c = list(entry[1]) | ||
Matt Mackall
|
r5178 | else: | ||
cmd = None | ||||
c = [] | ||||
# combine global options into local | ||||
for o in commands.globalopts: | ||||
c.append((o[0], o[1], options[o[1]], o[3])) | ||||
try: | ||||
Augie Fackler
|
r29822 | args = fancyopts.fancyopts(args, c, cmdoptions, gnu=True) | ||
Pulkit Goyal
|
r30576 | except getopt.GetoptError as inst: | ||
Matt Mackall
|
r11287 | raise error.CommandError(cmd, inst) | ||
Matt Mackall
|
r5178 | |||
# separate global options back out | ||||
for o in commands.globalopts: | ||||
n = o[1] | ||||
options[n] = cmdoptions[n] | ||||
del cmdoptions[n] | ||||
Henri Wiechers
|
r9875 | return (cmd, cmd and entry[0] or None, args, options, cmdoptions) | ||
Matt Mackall
|
r5178 | |||
Matt Mackall
|
r8137 | def _parseconfig(ui, config): | ||
Matt Mackall
|
r5178 | """parse the --config options from the command line""" | ||
Idan Kamara
|
r14753 | configs = [] | ||
Matt Mackall
|
r5178 | for cfg in config: | ||
try: | ||||
Tony Tung
|
r28081 | name, value = [cfgelem.strip() | ||
for cfgelem in cfg.split('=', 1)] | ||||
Matt Mackall
|
r5178 | section, name = name.split('.', 1) | ||
if not section or not name: | ||||
raise IndexError | ||||
Mads Kiilerich
|
r20788 | ui.setconfig(section, name, value, '--config') | ||
Idan Kamara
|
r14753 | configs.append((section, name, value)) | ||
Matt Mackall
|
r5178 | except (IndexError, ValueError): | ||
Pierre-Yves David
|
r26587 | raise error.Abort(_('malformed --config option: %r ' | ||
Bill Schroeder
|
r9825 | '(use --config section.name=value)') % cfg) | ||
Matt Mackall
|
r5178 | |||
Idan Kamara
|
r14753 | return configs | ||
Matt Mackall
|
r5178 | def _earlygetopt(aliases, args): | ||
"""Return list of values for an option (or aliases). | ||||
The values are listed in the order they appear in args. | ||||
The options and values are removed from args. | ||||
Bryan O'Sullivan
|
r19098 | |||
>>> args = ['x', '--cwd', 'foo', 'y'] | ||||
>>> _earlygetopt(['--cwd'], args), args | ||||
(['foo'], ['x', 'y']) | ||||
Bryan O'Sullivan
|
r19099 | >>> args = ['x', '--cwd=bar', 'y'] | ||
>>> _earlygetopt(['--cwd'], args), args | ||||
(['bar'], ['x', 'y']) | ||||
Bryan O'Sullivan
|
r19098 | >>> args = ['x', '-R', 'foo', 'y'] | ||
>>> _earlygetopt(['-R'], args), args | ||||
(['foo'], ['x', 'y']) | ||||
>>> args = ['x', '-Rbar', 'y'] | ||||
>>> _earlygetopt(['-R'], args), args | ||||
(['bar'], ['x', 'y']) | ||||
Matt Mackall
|
r5178 | """ | ||
try: | ||||
argcount = args.index("--") | ||||
except ValueError: | ||||
argcount = len(args) | ||||
shortopts = [opt for opt in aliases if len(opt) == 2] | ||||
values = [] | ||||
pos = 0 | ||||
while pos < argcount: | ||||
Bryan O'Sullivan
|
r19099 | fullarg = arg = args[pos] | ||
equals = arg.find('=') | ||||
if equals > -1: | ||||
arg = arg[:equals] | ||||
if arg in aliases: | ||||
Matt Mackall
|
r5178 | del args[pos] | ||
Bryan O'Sullivan
|
r19099 | if equals > -1: | ||
values.append(fullarg[equals + 1:]) | ||||
argcount -= 1 | ||||
else: | ||||
if pos + 1 >= argcount: | ||||
# ignore and let getopt report an error if there is no value | ||||
break | ||||
values.append(args.pop(pos)) | ||||
argcount -= 2 | ||||
elif arg[:2] in shortopts: | ||||
Matt Mackall
|
r5178 | # short option can have no following space, e.g. hg log -Rfoo | ||
values.append(args.pop(pos)[2:]) | ||||
argcount -= 1 | ||||
else: | ||||
pos += 1 | ||||
return values | ||||
Chad Dombrova
|
r11330 | def runcommand(lui, repo, cmd, fullargs, ui, options, d, cmdpats, cmdoptions): | ||
Bill Barry
|
r7819 | # run pre-hook, and abort if it fails | ||
Siddharth Agarwal
|
r19011 | hook.hook(lui, repo, "pre-%s" % cmd, True, args=" ".join(fullargs), | ||
pats=cmdpats, opts=cmdoptions) | ||||
Jordi Gutiérrez Hermoso
|
r29129 | try: | ||
ret = _runcommand(ui, options, cmd, d) | ||||
# run post-hook, passing command result | ||||
hook.hook(lui, repo, "post-%s" % cmd, False, args=" ".join(fullargs), | ||||
result=ret, pats=cmdpats, opts=cmdoptions) | ||||
except Exception: | ||||
# run failure hook and re-raise | ||||
hook.hook(lui, repo, "fail-%s" % cmd, False, args=" ".join(fullargs), | ||||
pats=cmdpats, opts=cmdoptions) | ||||
raise | ||||
Bill Barry
|
r7819 | return ret | ||
Jun Wu
|
r28263 | def _getlocal(ui, rpath, wd=None): | ||
Steve Losh
|
r12536 | """Return (path, local ui object) for the given target path. | ||
Martin Geisler
|
r12770 | |||
Steve Losh
|
r12536 | Takes paths in [cwd]/.hg/hgrc into account." | ||
""" | ||||
Jun Wu
|
r28263 | if wd is None: | ||
try: | ||||
Pulkit Goyal
|
r30500 | wd = pycompat.getcwd() | ||
Jun Wu
|
r28263 | except OSError as e: | ||
raise error.Abort(_("error getting current working directory: %s") % | ||||
e.strerror) | ||||
Mads Kiilerich
|
r11675 | path = cmdutil.findrepo(wd) or "" | ||
Matt Mackall
|
r5178 | if not path: | ||
lui = ui | ||||
Andrey Somov
|
r9436 | else: | ||
Brodie Rao
|
r12636 | lui = ui.copy() | ||
Brodie Rao
|
r12637 | lui.readconfig(os.path.join(path, ".hg", "hgrc"), path) | ||
Matt Mackall
|
r5178 | |||
Matt Mackall
|
r14860 | if rpath and rpath[-1]: | ||
Matt Mackall
|
r5178 | path = lui.expandpath(rpath[-1]) | ||
Matt Mackall
|
r8190 | lui = ui.copy() | ||
Brodie Rao
|
r12637 | lui.readconfig(os.path.join(path, ".hg", "hgrc"), path) | ||
Matt Mackall
|
r5178 | |||
Steve Losh
|
r12536 | return path, lui | ||
Jun Wu
|
r29132 | def _checkshellalias(lui, ui, args): | ||
"""Return the function to run the shell alias, if it is required""" | ||||
Steve Losh
|
r12536 | options = {} | ||
Steve Losh
|
r12748 | |||
try: | ||||
args = fancyopts.fancyopts(args, commands.globalopts, options) | ||||
Pulkit Goyal
|
r30576 | except getopt.GetoptError: | ||
Steve Losh
|
r12748 | return | ||
Steve Losh
|
r12536 | |||
if not args: | ||||
return | ||||
Jun Wu
|
r29132 | cmdtable = commands.table | ||
Steve Losh
|
r12536 | |||
cmd = args[0] | ||||
try: | ||||
Jun Wu
|
r29132 | strict = ui.configbool("ui", "strict") | ||
FUJIWARA Katsunori
|
r22377 | aliases, entry = cmdutil.findcmd(cmd, cmdtable, strict) | ||
Steve Losh
|
r12932 | except (error.AmbiguousCommand, error.UnknownCommand): | ||
Steve Losh
|
r12536 | return | ||
cmd = aliases[0] | ||||
fn = entry[0] | ||||
Augie Fackler
|
r14950 | if cmd and util.safehasattr(fn, 'shell'): | ||
Steve Losh
|
r12536 | d = lambda: fn(ui, *args[1:]) | ||
Brodie Rao
|
r16683 | return lambda: runcommand(lui, None, cmd, args[:1], ui, options, d, | ||
[], {}) | ||||
Steve Losh
|
r12536 | |||
_loaded = set() | ||||
FUJIWARA Katsunori
|
r28391 | |||
# list of (objname, loadermod, loadername) tuple: | ||||
# - objname is the name of an object in extension module, from which | ||||
# extra information is loaded | ||||
# - loadermod is the module where loader is placed | ||||
# - loadername is the name of the function, which takes (ui, extensionname, | ||||
# extraobj) arguments | ||||
extraloaders = [ | ||||
('cmdtable', commands, 'loadcmdtable'), | ||||
Pierre-Yves David
|
r30653 | ('colortable', color, 'loadcolortable'), | ||
FUJIWARA Katsunori
|
r28447 | ('filesetpredicate', fileset, 'loadpredicate'), | ||
FUJIWARA Katsunori
|
r28394 | ('revsetpredicate', revset, 'loadpredicate'), | ||
FUJIWARA Katsunori
|
r28692 | ('templatefilter', templatefilters, 'loadfilter'), | ||
FUJIWARA Katsunori
|
r28695 | ('templatefunc', templater, 'loadfunction'), | ||
FUJIWARA Katsunori
|
r28538 | ('templatekeyword', templatekw, 'loadkeyword'), | ||
FUJIWARA Katsunori
|
r28391 | ] | ||
Idan Kamara
|
r14439 | def _dispatch(req): | ||
Idan Kamara
|
r14438 | args = req.args | ||
Idan Kamara
|
r14439 | ui = req.ui | ||
Steve Losh
|
r12536 | # check for cwd | ||
cwd = _earlygetopt(['--cwd'], args) | ||||
if cwd: | ||||
os.chdir(cwd[-1]) | ||||
rpath = _earlygetopt(["-R", "--repository", "--repo"], args) | ||||
path, lui = _getlocal(ui, rpath) | ||||
Pierre-Yves David
|
r30917 | # Side-effect of accessing is debugcommands module is guaranteed to be | ||
# imported and commands.table is populated. | ||||
debugcommands.command | ||||
Bryan O'Sullivan
|
r30933 | uis = set([ui, lui]) | ||
if req.repo: | ||||
uis.add(req.repo.ui) | ||||
if '--profile' in args: | ||||
for ui_ in uis: | ||||
ui_.setconfig('profiling', 'enabled', 'true', '--profile') | ||||
Bryan O'Sullivan
|
r30934 | with profiling.maybeprofile(lui): | ||
# Configure extensions in phases: uisetup, extsetup, cmdtable, and | ||||
# reposetup. Programs like TortoiseHg will call _dispatch several | ||||
# times so we keep track of configured extensions in _loaded. | ||||
extensions.loadall(lui) | ||||
exts = [ext for ext in extensions.extensions() if ext[0] not in _loaded] | ||||
# Propagate any changes to lui.__class__ by extensions | ||||
ui.__class__ = lui.__class__ | ||||
Kirill Smelkov
|
r5828 | |||
Bryan O'Sullivan
|
r30934 | # (uisetup and extsetup are handled in extensions.loadall) | ||
Kirill Smelkov
|
r5828 | |||
Bryan O'Sullivan
|
r30934 | for name, module in exts: | ||
for objname, loadermod, loadername in extraloaders: | ||||
extraobj = getattr(module, objname, None) | ||||
if extraobj is not None: | ||||
getattr(loadermod, loadername)(ui, name, extraobj) | ||||
_loaded.add(name) | ||||
Brendan Cully
|
r8655 | |||
Bryan O'Sullivan
|
r30934 | # (reposetup is handled in hg.repository) | ||
Martin Geisler
|
r9410 | |||
Bryan O'Sullivan
|
r30934 | addaliases(lui, commands.table) | ||
Brendan Cully
|
r8655 | |||
Bryan O'Sullivan
|
r30934 | # All aliases and commands are completely defined, now. | ||
# Check abbreviation/ambiguity of shell alias. | ||||
shellaliasfn = _checkshellalias(lui, ui, args) | ||||
if shellaliasfn: | ||||
Arun Kulshreshtha
|
r30006 | return shellaliasfn() | ||
FUJIWARA Katsunori
|
r22377 | |||
Bryan O'Sullivan
|
r30934 | # check for fallback encoding | ||
fallback = lui.config('ui', 'fallbackencoding') | ||||
if fallback: | ||||
encoding.fallbackencoding = fallback | ||||
Matt Mackall
|
r5178 | |||
Bryan O'Sullivan
|
r30934 | fullargs = args | ||
cmd, func, args, options, cmdoptions = _parse(lui, args) | ||||
Matt Mackall
|
r5178 | |||
Bryan O'Sullivan
|
r30934 | if options["config"]: | ||
raise error.Abort(_("option --config may not be abbreviated!")) | ||||
if options["cwd"]: | ||||
raise error.Abort(_("option --cwd may not be abbreviated!")) | ||||
if options["repository"]: | ||||
raise error.Abort(_( | ||||
"option -R has to be separated from other options (e.g. not " | ||||
"-qR) and --repository may only be abbreviated as --repo!")) | ||||
Matt Mackall
|
r5178 | |||
Bryan O'Sullivan
|
r30934 | if options["encoding"]: | ||
encoding.encoding = options["encoding"] | ||||
if options["encodingmode"]: | ||||
encoding.encodingmode = options["encodingmode"] | ||||
if options["time"]: | ||||
def get_times(): | ||||
t = os.times() | ||||
if t[4] == 0.0: | ||||
# Windows leaves this as zero, so use time.clock() | ||||
t = (t[0], t[1], t[2], t[3], time.clock()) | ||||
return t | ||||
s = get_times() | ||||
def print_time(): | ||||
t = get_times() | ||||
ui.warn( | ||||
_("time: real %.3f secs (user %.3f+%.3f sys %.3f+%.3f)\n") % | ||||
(t[4]-s[4], t[0]-s[0], t[2]-s[2], t[1]-s[1], t[3]-s[3])) | ||||
atexit.register(print_time) | ||||
Matt Mackall
|
r5178 | |||
Bryan O'Sullivan
|
r30934 | if options['verbose'] or options['debug'] or options['quiet']: | ||
for opt in ('verbose', 'debug', 'quiet'): | ||||
val = str(bool(options[opt])) | ||||
for ui_ in uis: | ||||
ui_.setconfig('ui', opt, val, '--' + opt) | ||||
if options['traceback']: | ||||
Idan Kamara
|
r14752 | for ui_ in uis: | ||
Bryan O'Sullivan
|
r30934 | ui_.setconfig('ui', 'traceback', 'on', '--traceback') | ||
Idan Kamara
|
r14992 | |||
Bryan O'Sullivan
|
r30934 | if options['noninteractive']: | ||
for ui_ in uis: | ||||
ui_.setconfig('ui', 'interactive', 'off', '-y') | ||||
Matt Mackall
|
r5178 | |||
Bryan O'Sullivan
|
r30934 | if cmdoptions.get('insecure', False): | ||
for ui_ in uis: | ||||
ui_.insecureconnections = True | ||||
Yuya Nishihara
|
r13328 | |||
Bryan O'Sullivan
|
r30934 | if options['version']: | ||
return commands.version_(ui) | ||||
if options['help']: | ||||
return commands.help_(ui, cmd, command=cmd is not None) | ||||
elif not cmd: | ||||
return commands.help_(ui, 'shortlist') | ||||
Matt Mackall
|
r5178 | |||
Arun Kulshreshtha
|
r30005 | repo = None | ||
cmdpats = args[:] | ||||
Augie Fackler
|
r30485 | if not func.norepo: | ||
Arun Kulshreshtha
|
r30005 | # use the repo from the request only if we don't have -R | ||
if not rpath and not cwd: | ||||
repo = req.repo | ||||
Idan Kamara
|
r14510 | |||
Arun Kulshreshtha
|
r30005 | if repo: | ||
# set the descriptors of the repo ui to those of ui | ||||
repo.ui.fin = ui.fin | ||||
repo.ui.fout = ui.fout | ||||
repo.ui.ferr = ui.ferr | ||||
else: | ||||
try: | ||||
repo = hg.repository(ui, path=path) | ||||
if not repo.local(): | ||||
raise error.Abort(_("repository '%s' is not local") | ||||
% path) | ||||
repo.ui.setconfig("bundle", "mainreporoot", repo.root, | ||||
'repo') | ||||
except error.RequirementError: | ||||
Yuya Nishihara
|
r26142 | raise | ||
Arun Kulshreshtha
|
r30005 | except error.RepoError: | ||
if rpath and rpath[-1]: # invalid -R path | ||||
raise | ||||
Augie Fackler
|
r30485 | if not func.optionalrepo: | ||
if func.inferrepo and args and not path: | ||||
Arun Kulshreshtha
|
r30005 | # try to infer -R from command args | ||
repos = map(cmdutil.findrepo, args) | ||||
guess = repos[0] | ||||
if guess and repos.count(guess) == len(repos): | ||||
req.args = ['--repository', guess] + fullargs | ||||
return _dispatch(req) | ||||
if not path: | ||||
raise error.RepoError(_("no repository found in" | ||||
" '%s' (.hg not found)") | ||||
Pulkit Goyal
|
r30519 | % pycompat.getcwd()) | ||
Arun Kulshreshtha
|
r30005 | raise | ||
if repo: | ||||
ui = repo.ui | ||||
if options['hidden']: | ||||
repo = repo.unfiltered() | ||||
args.insert(0, repo) | ||||
elif rpath: | ||||
ui.warn(_("warning: --repository ignored\n")) | ||||
Matt Mackall
|
r7388 | |||
Arun Kulshreshtha
|
r30005 | msg = ' '.join(' ' in a and repr(a) or a for a in fullargs) | ||
ui.log("command", '%s\n', msg) | ||||
Pulkit Goyal
|
r30586 | strcmdopt = pycompat.strkwargs(cmdoptions) | ||
d = lambda: util.checksignature(func)(ui, *args, **strcmdopt) | ||||
Arun Kulshreshtha
|
r30005 | try: | ||
return runcommand(lui, repo, cmd, fullargs, ui, options, d, | ||||
cmdpats, cmdoptions) | ||||
finally: | ||||
if repo and repo != req.repo: | ||||
repo.close() | ||||
Matt Mackall
|
r5178 | |||
def _runcommand(ui, options, cmd, cmdfunc): | ||||
Gregory Szorc
|
r29784 | """Run a command function, possibly with profiling enabled.""" | ||
Arun Kulshreshtha
|
r30006 | try: | ||
return cmdfunc() | ||||
except error.SignatureError: | ||||
raise error.CommandError(cmd, _('invalid arguments')) | ||||
Martijn Pieters
|
r28784 | |||
Martijn Pieters
|
r28821 | def _exceptionwarning(ui): | ||
"""Produce a warning message for the current active exception""" | ||||
Martijn Pieters
|
r28784 | |||
# For compatibility checking, we discard the portion of the hg | ||||
# version after the + on the assumption that if a "normal | ||||
# user" is running a build with a + in it the packager | ||||
# probably built from fairly close to a tag and anyone with a | ||||
# 'make local' copy of hg (where the version number can be out | ||||
# of date) will be clueful enough to notice the implausible | ||||
# version number and try updating. | ||||
ct = util.versiontuple(n=2) | ||||
worst = None, ct, '' | ||||
if ui.config('ui', 'supportcontact', None) is None: | ||||
for name, mod in extensions.extensions(): | ||||
testedwith = getattr(mod, 'testedwith', '') | ||||
report = getattr(mod, 'buglink', _('the extension author.')) | ||||
if not testedwith.strip(): | ||||
# We found an untested extension. It's likely the culprit. | ||||
worst = name, 'unknown', report | ||||
break | ||||
# Never blame on extensions bundled with Mercurial. | ||||
Yuya Nishihara
|
r29884 | if extensions.ismoduleinternal(mod): | ||
Martijn Pieters
|
r28784 | continue | ||
tested = [util.versiontuple(t, 2) for t in testedwith.split()] | ||||
if ct in tested: | ||||
continue | ||||
lower = [t for t in tested if t < ct] | ||||
nearest = max(lower or tested) | ||||
if worst[0] is None or nearest < worst[1]: | ||||
worst = name, nearest, report | ||||
if worst[0] is not None: | ||||
name, testedwith, report = worst | ||||
if not isinstance(testedwith, str): | ||||
testedwith = '.'.join([str(c) for c in testedwith]) | ||||
warning = (_('** Unknown exception encountered with ' | ||||
'possibly-broken third-party extension %s\n' | ||||
'** which supports versions %s of Mercurial.\n' | ||||
'** Please disable %s and try your action again.\n' | ||||
'** If that fixes the bug please report it to %s\n') | ||||
% (name, testedwith, name, report)) | ||||
else: | ||||
bugtracker = ui.config('ui', 'supportcontact', None) | ||||
if bugtracker is None: | ||||
bugtracker = _("https://mercurial-scm.org/wiki/BugTracker") | ||||
warning = (_("** unknown exception encountered, " | ||||
"please report by visiting\n** ") + bugtracker + '\n') | ||||
warning += ((_("** Python %s\n") % sys.version.replace('\n', '')) + | ||||
(_("** Mercurial Distributed SCM (version %s)\n") % | ||||
util.version()) + | ||||
(_("** Extensions loaded: %s\n") % | ||||
", ".join([x[0] for x in extensions.extensions()]))) | ||||
Martijn Pieters
|
r28821 | return warning | ||
def handlecommandexception(ui): | ||||
"""Produce a warning message for broken commands | ||||
Called when handling an exception; the exception is reraised if | ||||
this function returns False, ignored otherwise. | ||||
""" | ||||
warning = _exceptionwarning(ui) | ||||
Martijn Pieters
|
r28784 | ui.log("commandexception", "%s\n%s\n", warning, traceback.format_exc()) | ||
ui.warn(warning) | ||||
return False # re-raise the exception | ||||