dispatch.py
1064 lines
| 38.1 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 | |||
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, | ||
demandimport, | ||||
encoding, | ||||
error, | ||||
extensions, | ||||
fancyopts, | ||||
Augie Fackler
|
r31060 | help, | ||
Gregory Szorc
|
r25932 | hg, | ||
hook, | ||||
Gregory Szorc
|
r29781 | profiling, | ||
Pulkit Goyal
|
r30468 | pycompat, | ||
Jun Wu
|
r30520 | scmutil, | ||
Gregory Szorc
|
r25932 | ui as uimod, | ||
util, | ||||
) | ||||
Matt Mackall
|
r5178 | |||
Yuya Nishihara
|
r37102 | from .utils import ( | ||
Yuya Nishihara
|
r37137 | procutil, | ||
Yuya Nishihara
|
r37102 | stringutil, | ||
) | ||||
Idan Kamara
|
r14438 | class request(object): | ||
Brodie Rao
|
r16683 | def __init__(self, args, ui=None, repo=None, fin=None, fout=None, | ||
Jun Wu
|
r32379 | ferr=None, prereposetups=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 | ||||
Yuya Nishihara
|
r35224 | # remember options pre-parsed by _earlyparseopts() | ||
Yuya Nishihara
|
r35059 | self.earlyoptions = {} | ||
Jun Wu
|
r32379 | # reposetups which run before extensions, useful for chg to pre-fill | ||
# low-level repo state (for example, changelog) before extensions. | ||||
self.prereposetups = prereposetups or [] | ||||
Bryan O'Sullivan
|
r31956 | def _runexithandlers(self): | ||
exc = None | ||||
handlers = self.ui._exithandlers | ||||
try: | ||||
while handlers: | ||||
func, args, kwargs = handlers.pop() | ||||
try: | ||||
func(*args, **kwargs) | ||||
except: # re-raises below | ||||
if exc is None: | ||||
exc = sys.exc_info()[1] | ||||
self.ui.warn(('error in exit handlers:\n')) | ||||
self.ui.traceback(force=True) | ||||
finally: | ||||
if exc is not None: | ||||
raise exc | ||||
Matt Mackall
|
r5178 | def run(): | ||
"run the command in sys.argv" | ||||
Yuya Nishihara
|
r37963 | initstdio() | ||
Bryan O'Sullivan
|
r31960 | req = request(pycompat.sysargv[1:]) | ||
err = None | ||||
try: | ||||
Yuya Nishihara
|
r38015 | status = dispatch(req) | ||
Yuya Nishihara
|
r34533 | except error.StdioError as e: | ||
err = e | ||||
Bryan O'Sullivan
|
r31960 | status = -1 | ||
Gregory Szorc
|
r38014 | |||
# In all cases we try to flush stdio streams. | ||||
Bryan O'Sullivan
|
r31960 | if util.safehasattr(req.ui, 'fout'): | ||
try: | ||||
Yuya Nishihara
|
r32687 | req.ui.fout.flush() | ||
Yuya Nishihara
|
r34533 | except IOError as e: | ||
err = e | ||||
Bryan O'Sullivan
|
r31960 | status = -1 | ||
Gregory Szorc
|
r38014 | |||
Bryan O'Sullivan
|
r31960 | if util.safehasattr(req.ui, 'ferr'): | ||
Gregory Szorc
|
r35671 | try: | ||
if err is not None and err.errno != errno.EPIPE: | ||||
req.ui.ferr.write('abort: %s\n' % | ||||
encoding.strtolocal(err.strerror)) | ||||
req.ui.ferr.flush() | ||||
# There's not much we can do about an I/O error here. So (possibly) | ||||
# change the status code and move on. | ||||
except IOError: | ||||
status = -1 | ||||
Yuya Nishihara
|
r36655 | |||
_silencestdio() | ||||
Bryan O'Sullivan
|
r31960 | sys.exit(status & 255) | ||
Matt Mackall
|
r5178 | |||
Yuya Nishihara
|
r36654 | if pycompat.ispy3: | ||
Yuya Nishihara
|
r37963 | def initstdio(): | ||
Yuya Nishihara
|
r36654 | pass | ||
Yuya Nishihara
|
r36655 | |||
def _silencestdio(): | ||||
for fp in (sys.stdout, sys.stderr): | ||||
# Check if the file is okay | ||||
try: | ||||
fp.flush() | ||||
continue | ||||
except IOError: | ||||
pass | ||||
# Otherwise mark it as closed to silence "Exception ignored in" | ||||
# message emitted by the interpreter finalizer. Be careful to | ||||
Yuya Nishihara
|
r37137 | # not close procutil.stdout, which may be a fdopen-ed file object | ||
# and its close() actually closes the underlying file descriptor. | ||||
Yuya Nishihara
|
r36655 | try: | ||
fp.close() | ||||
except IOError: | ||||
pass | ||||
Yuya Nishihara
|
r36654 | else: | ||
Yuya Nishihara
|
r37963 | def initstdio(): | ||
Yuya Nishihara
|
r36654 | for fp in (sys.stdin, sys.stdout, sys.stderr): | ||
Yuya Nishihara
|
r37138 | procutil.setbinary(fp) | ||
Yuya Nishihara
|
r34534 | |||
Yuya Nishihara
|
r36655 | def _silencestdio(): | ||
pass | ||||
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") % | ||||
Yuya Nishihara
|
r36521 | (pycompat.bytestr(inst.args[1]), inst.args[0])) | ||
Yuya Nishihara
|
r36747 | if inst.args[0].startswith(' '): | ||
Augie Fackler
|
r24039 | 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 | |||
Augie Fackler
|
r31492 | def _formatargs(args): | ||
Yuya Nishihara
|
r37138 | return ' '.join(procutil.shellquote(a) for a in args) | ||
Augie Fackler
|
r31492 | |||
Idan Kamara
|
r14438 | def dispatch(req): | ||
Yuya Nishihara
|
r38015 | """run the command specified in req.args; returns an integer status code""" | ||
Idan Kamara
|
r14615 | if req.ferr: | ||
ferr = req.ferr | ||||
elif req.ui: | ||||
ferr = req.ui.ferr | ||||
else: | ||||
Yuya Nishihara
|
r37137 | ferr = procutil.stderr | ||
Idan Kamara
|
r14615 | |||
Matt Mackall
|
r5178 | try: | ||
Idan Kamara
|
r14439 | if not req.ui: | ||
Yuya Nishihara
|
r30559 | req.ui = uimod.ui.load() | ||
Yuya Nishihara
|
r35224 | req.earlyoptions.update(_earlyparseopts(req.ui, req.args)) | ||
if req.earlyoptions['traceback']: | ||||
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 | |||
Augie Fackler
|
r31492 | msg = _formatargs(req.args) | ||
Simon Farnsworth
|
r30975 | starttime = util.timer() | ||
Yuya Nishihara
|
r38042 | ret = 1 # default of Python exit code on unhandled exception | ||
Durham Goode
|
r19229 | try: | ||
Yuya Nishihara
|
r38015 | ret = _runcatch(req) or 0 | ||
Yuya Nishihara
|
r32340 | except error.ProgrammingError as inst: | ||
Rodrigo Damazio Bovendorp
|
r38791 | req.ui.error(_('** ProgrammingError: %s\n') % inst) | ||
Yuya Nishihara
|
r32340 | if inst.hint: | ||
Rodrigo Damazio Bovendorp
|
r38791 | req.ui.error(_('** (%s)\n') % inst.hint) | ||
Yuya Nishihara
|
r32340 | raise | ||
Jun Wu
|
r32111 | except KeyboardInterrupt as inst: | ||
Yuya Nishihara
|
r28520 | try: | ||
Jun Wu
|
r32111 | if isinstance(inst, error.SignalInterrupt): | ||
msg = _("killed!\n") | ||||
else: | ||||
msg = _("interrupted!\n") | ||||
Rodrigo Damazio Bovendorp
|
r38791 | req.ui.error(msg) | ||
Yuya Nishihara
|
r32044 | except error.SignalInterrupt: | ||
# maybe pager would quit without consuming all the output, and | ||||
# SIGPIPE was raised. we cannot print anything in this case. | ||||
pass | ||||
Yuya Nishihara
|
r28520 | except IOError as inst: | ||
if inst.errno != errno.EPIPE: | ||||
raise | ||||
ret = -1 | ||||
Durham Goode
|
r19229 | finally: | ||
Simon Farnsworth
|
r30975 | duration = util.timer() - starttime | ||
Jun Wu
|
r28534 | req.ui.flush() | ||
Simon Farnsworth
|
r30976 | if req.ui.logblockedtimes: | ||
req.ui._blockedtimes['command_duration'] = duration * 1000 | ||||
Pulkit Goyal
|
r35355 | req.ui.log('uiblocked', 'ui blocked ms', | ||
**pycompat.strkwargs(req.ui._blockedtimes)) | ||||
Pulkit Goyal
|
r32130 | req.ui.log("commandfinish", "%s exited %d after %0.2f seconds\n", | ||
Yuya Nishihara
|
r38043 | msg, ret & 255, duration) | ||
Bryan O'Sullivan
|
r31956 | try: | ||
req._runexithandlers() | ||||
except: # exiting, so no re-raises | ||||
ret = ret or -1 | ||||
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(): | ||
Augie Fackler
|
r32050 | realcmd = None | ||
try: | ||||
cmdargs = fancyopts.fancyopts(req.args[:], commands.globalopts, {}) | ||||
cmd = cmdargs[0] | ||||
aliases, entry = cmdutil.findcmd(cmd, commands.table, False) | ||||
realcmd = aliases[0] | ||||
except (error.UnknownCommand, error.AmbiguousCommand, | ||||
IndexError, getopt.GetoptError): | ||||
# Don't handle this here. We know the command is | ||||
# invalid, but all we're worried about for now is that | ||||
# it's not a command that server operators expect to | ||||
# be safe to offer to users in a sandbox. | ||||
pass | ||||
if realcmd == 'serve' and '--stdio' in cmdargs: | ||||
# We want to constrain 'hg serve --stdio' instances pretty | ||||
# closely, as many shared-ssh access tools want to grant | ||||
# access to run *only* 'hg -R $repo serve --stdio'. We | ||||
# restrict to exactly that set of arguments, and prohibit | ||||
# any repo name that starts with '--' to prevent | ||||
# shenanigans wherein a user does something like pass | ||||
# --debugger or --config=ui.debugger=1 as a repo | ||||
# name. This used to actually run the debugger. | ||||
if (len(req.args) != 4 or | ||||
req.args[0] != '-R' or | ||||
req.args[1].startswith('--') or | ||||
req.args[2] != 'serve' or | ||||
req.args[3] != '--stdio'): | ||||
raise error.Abort( | ||||
Pulkit Goyal
|
r38122 | _('potentially unsafe serve --stdio invocation: %s') % | ||
(stringutil.pprint(req.args),)) | ||||
Augie Fackler
|
r32050 | |||
Matt Mackall
|
r5178 | try: | ||
Sean Farley
|
r19640 | debugger = 'pdb' | ||
debugtrace = { | ||||
Alex Gaynor
|
r34487 | 'pdb': pdb.set_trace | ||
Sean Farley
|
r19640 | } | ||
debugmortem = { | ||||
Alex Gaynor
|
r34487 | 'pdb': pdb.post_mortem | ||
Sean Farley
|
r19640 | } | ||
Sean Farley
|
r19639 | |||
# read --config before doing anything else | ||||
# (e.g. to change trust settings for reading .hg/hgrc) | ||||
Yuya Nishihara
|
r35224 | cfgs = _parseconfig(req.ui, req.earlyoptions['config']) | ||
Sean Farley
|
r19639 | |||
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' | ||
Yuya Nishihara
|
r35224 | elif req.earlyoptions['debugger']: | ||
Jordi Gutiérrez Hermoso
|
r20826 | # 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 | ||
Yuya Nishihara
|
r35224 | if req.earlyoptions['debugger']: | ||
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 | ||
Yuya Nishihara
|
r35224 | if req.earlyoptions['debugger']: | ||
Mads Kiilerich
|
r11494 | traceback.print_exc() | ||
Sean Farley
|
r19640 | debugmortem[debugger](sys.exc_info()[2]) | ||
Matt Mackall
|
r5178 | raise | ||
Yuya Nishihara
|
r32040 | return _callcatch(ui, _runcatchfunc) | ||
Jun Wu
|
r29761 | |||
Yuya Nishihara
|
r32040 | 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]: | ||
Augie Fackler
|
r31266 | ui.pager('help') | ||
Augie Fackler
|
r32620 | msgbytes = pycompat.bytestr(inst.args[1]) | ||
ui.warn(_("hg %s: %s\n") % (inst.args[0], msgbytes)) | ||||
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]) | ||||
Martin von Zweigbergk
|
r38811 | ui.warn(_("(use 'hg help -v' for a list of global options)\n")) | ||
Jun Wu
|
r30520 | except error.ParseError as inst: | ||
_formatparse(ui.warn, inst) | ||||
return -1 | ||||
Gregory Szorc
|
r25660 | except error.UnknownCommand as inst: | ||
Augie Fackler
|
r31060 | nocmdmsg = _("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) | ||||
Yuya Nishihara
|
r32567 | formatted = help.formattedhelp(ui, commands, inst.args[0], | ||
unknowncmd=True) | ||||
Augie Fackler
|
r31060 | ui.warn(nocmdmsg) | ||
ui.write(formatted) | ||||
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: | ||||
Augie Fackler
|
r31060 | ui.warn(nocmdmsg) | ||
Bryan O'Sullivan
|
r27623 | _reportsimilar(ui.warn, sim) | ||
Augie Fackler
|
r24222 | suggested = True | ||
if not suggested: | ||||
Augie Fackler
|
r31060 | ui.warn(nocmdmsg) | ||
Martin von Zweigbergk
|
r38810 | ui.warn(_("(use 'hg help' for a list of commands)\n")) | ||
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): | ||
Jun Wu
|
r34089 | args = [] | ||
# only care about alias 'args', ignore 'args' set by extensions.wrapfunction | ||||
if not util.safehasattr(fn, '_origfunc'): | ||||
args = getattr(fn, 'args', args) | ||||
Matt Mackall
|
r16294 | if args: | ||
Yuya Nishihara
|
r37138 | cmd = ' '.join(map(procutil.shellquote, args)) | ||
Alexander Solovyov
|
r14265 | |||
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')) | ||
Pulkit Goyal
|
r31491 | cmd = re.sub(br'\$(\d+|\$)', replacer, cmd) | ||
Alexander Solovyov
|
r14265 | 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. | ||||
Yuya Nishihara
|
r37138 | replacemap['"$@"'] = ' '.join(procutil.shellquote(arg) for arg in args) | ||
Siddharth Agarwal
|
r22158 | # escape '\$' for regex | ||
Pulkit Goyal
|
r35145 | regex = '|'.join(replacemap.keys()).replace('$', br'\$') | ||
Siddharth Agarwal
|
r22158 | r = re.compile(regex) | ||
return r.sub(lambda x: replacemap[x.group()], cmd) | ||||
Brendan Cully
|
r8655 | class cmdalias(object): | ||
Rodrigo Damazio
|
r37152 | def __init__(self, ui, 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('!'): | ||
Rodrigo Damazio
|
r37152 | shdef = self.definition[1:] | ||
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 " | ||
Kyle Lippincott
|
r35482 | "of %i variable in alias '%s' definition.\n" | ||
Roman Sokolov
|
r13393 | % (int(m.groups()[0]), self.name)) | ||
Steve Losh
|
r11989 | return '' | ||
Rodrigo Damazio
|
r37152 | cmd = re.sub(br'\$(\d+|\$)', _checkvar, shdef) | ||
Siddharth Agarwal
|
r22158 | cmd = aliasinterpolate(self.name, args, cmd) | ||
Simon Farnsworth
|
r31199 | return ui.system(cmd, environ=env, | ||
blockedtag='alias_%s' % self.name) | ||||
Steve Losh
|
r11524 | self.fn = fn | ||
Rodrigo Damazio
|
r37152 | self._populatehelp(ui, name, shdef, self.fn) | ||
Steve Losh
|
r11524 | 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") | ||
Yuya Nishihara
|
r37102 | % (self.name, stringutil.forcebytestr(inst))) | ||
Yuya Nishihara
|
r21569 | return | ||
Yuya Nishihara
|
r35225 | earlyopts, args = _earlysplitopts(args) | ||
if earlyopts: | ||||
self.badalias = (_("error in definition for alias '%s': %s may " | ||||
"only be given on the command line") | ||||
Pulkit Goyal
|
r35960 | % (self.name, '/'.join(pycompat.ziplist(*earlyopts) | ||
[0]))) | ||||
Yuya Nishihara
|
r35225 | return | ||
Brodie Rao
|
r12092 | self.cmdname = cmd = args.pop(0) | ||
Jun Wu
|
r29087 | self.givenargs = args | ||
Brendan Cully
|
r8655 | |||
try: | ||||
Nicolas Dumazet
|
r9993 | tableentry = cmdutil.findcmd(cmd, cmdtable, False)[1] | ||
if len(tableentry) > 2: | ||||
Rodrigo Damazio
|
r37152 | self.fn, self.opts, cmdhelp = tableentry | ||
Nicolas Dumazet
|
r9993 | else: | ||
self.fn, self.opts = tableentry | ||||
Rodrigo Damazio
|
r37152 | cmdhelp = None | ||
Nicolas Dumazet
|
r9993 | |||
Rodrigo Damazio
|
r37152 | self._populatehelp(ui, name, cmd, self.fn, cmdhelp) | ||
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 | |||
Rodrigo Damazio
|
r37152 | def _populatehelp(self, ui, name, cmd, fn, defaulthelp=None): | ||
Yuya Nishihara
|
r37157 | # confine strings to be passed to i18n.gettext() | ||
cfg = {} | ||||
for k in ('doc', 'help'): | ||||
v = ui.config('alias', '%s:%s' % (name, k), None) | ||||
if v is None: | ||||
continue | ||||
if not encoding.isasciistr(v): | ||||
self.badalias = (_("non-ASCII character in alias definition " | ||||
"'%s:%s'") % (name, k)) | ||||
return | ||||
cfg[k] = v | ||||
self.help = cfg.get('help', defaulthelp or '') | ||||
Rodrigo Damazio
|
r37152 | if self.help and 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
|
r37158 | doc = cfg.get('doc', pycompat.getdoc(fn)) | ||
if doc is not None: | ||||
doc = pycompat.sysstr(doc) | ||||
self.__doc__ = doc | ||||
Rodrigo Damazio
|
r37152 | |||
Jun Wu
|
r29087 | @property | ||
def args(self): | ||||
Pulkit Goyal
|
r31629 | args = pycompat.maplist(util.expandpath, self.givenargs) | ||
Jun Wu
|
r29087 | return aliasargs(self.fn, args) | ||
Yuya Nishihara
|
r28621 | def __getattr__(self, name): | ||
Gregory Szorc
|
r37734 | adefaults = {r'norepo': True, r'intents': set(), | ||
Pulkit Goyal
|
r32158 | r'optionalrepo': False, r'inferrepo': False} | ||
Yuya Nishihara
|
r28621 | 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 | |||
Jun Wu
|
r34307 | class lazyaliasentry(object): | ||
"""like a typical command entry (func, opts, help), but is lazy""" | ||||
Rodrigo Damazio
|
r37152 | def __init__(self, ui, name, definition, cmdtable, source): | ||
self.ui = ui | ||||
Jun Wu
|
r34307 | self.name = name | ||
self.definition = definition | ||||
self.cmdtable = cmdtable.copy() | ||||
self.source = source | ||||
@util.propertycache | ||||
def _aliasdef(self): | ||||
Rodrigo Damazio
|
r37152 | return cmdalias(self.ui, self.name, self.definition, self.cmdtable, | ||
self.source) | ||||
Jun Wu
|
r34307 | |||
def __getitem__(self, n): | ||||
aliasdef = self._aliasdef | ||||
if n == 0: | ||||
return aliasdef | ||||
elif n == 1: | ||||
return aliasdef.opts | ||||
elif n == 2: | ||||
return aliasdef.help | ||||
else: | ||||
raise IndexError | ||||
def __iter__(self): | ||||
for i in range(3): | ||||
yield self[i] | ||||
def __len__(self): | ||||
return 3 | ||||
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. | ||||
Rodrigo Damazio
|
r37152 | for alias, definition in ui.configitems('alias', ignoresub=True): | ||
Idan Kamara
|
r15019 | try: | ||
Jun Wu
|
r34307 | if cmdtable[alias].definition == definition: | ||
Idan Kamara
|
r15019 | continue | ||
except (KeyError, AttributeError): | ||||
# definition might not exist or it might not be a cmdalias | ||||
pass | ||||
Jun Wu
|
r34306 | source = ui.configsource('alias', alias) | ||
Rodrigo Damazio
|
r37152 | entry = lazyaliasentry(ui, alias, definition, cmdtable, source) | ||
Jun Wu
|
r34307 | cmdtable[alias] = entry | ||
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: | ||
Yuya Nishihara
|
r37102 | raise error.CommandError(None, stringutil.forcebytestr(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: | ||||
Augie Fackler
|
r31502 | args = pycompat.maplist( | ||
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: | ||
Yuya Nishihara
|
r37102 | raise error.CommandError(cmd, stringutil.forcebytestr(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 ' | ||
Yuya Nishihara
|
r36659 | '(use --config section.name=value)') | ||
% pycompat.bytestr(cfg)) | ||||
Matt Mackall
|
r5178 | |||
Idan Kamara
|
r14753 | return configs | ||
Yuya Nishihara
|
r35224 | def _earlyparseopts(ui, args): | ||
Yuya Nishihara
|
r35180 | options = {} | ||
fancyopts.fancyopts(args, commands.globalopts, options, | ||||
Yuya Nishihara
|
r35224 | gnu=not ui.plain('strictflags'), early=True, | ||
Yuya Nishihara
|
r35223 | optaliases={'repository': ['repo']}) | ||
Yuya Nishihara
|
r35180 | return options | ||
Yuya Nishihara
|
r35225 | def _earlysplitopts(args): | ||
"""Split args into a list of possible early options and remainder args""" | ||||
shortoptions = 'R:' | ||||
# TODO: perhaps 'debugger' should be included | ||||
longoptions = ['cwd=', 'repository=', 'repo=', 'config='] | ||||
return fancyopts.earlygetopt(args, shortoptions, longoptions, | ||||
gnu=True, keepsep=True) | ||||
Matt Mackall
|
r5178 | |||
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") % | ||||
Augie Fackler
|
r34024 | encoding.strtolocal(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 | |||
Yuya Nishihara
|
r35062 | if rpath: | ||
path = lui.expandpath(rpath) | ||||
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'): | ||
Yuya Nishihara
|
r35063 | # shell alias shouldn't receive early options which are consumed by hg | ||
Yuya Nishihara
|
r35225 | _earlyopts, args = _earlysplitopts(args) | ||
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 | |||
Idan Kamara
|
r14439 | def _dispatch(req): | ||
Idan Kamara
|
r14438 | args = req.args | ||
Idan Kamara
|
r14439 | ui = req.ui | ||
Steve Losh
|
r12536 | # check for cwd | ||
Yuya Nishihara
|
r35224 | cwd = req.earlyoptions['cwd'] | ||
Steve Losh
|
r12536 | if cwd: | ||
Yuya Nishihara
|
r35062 | os.chdir(cwd) | ||
Steve Losh
|
r12536 | |||
Yuya Nishihara
|
r35224 | rpath = req.earlyoptions['repository'] | ||
Steve Losh
|
r12536 | path, lui = _getlocal(ui, rpath) | ||
Martin von Zweigbergk
|
r32291 | uis = {ui, lui} | ||
Bryan O'Sullivan
|
r30933 | |||
if req.repo: | ||||
uis.add(req.repo.ui) | ||||
Boris Feld
|
r38552 | if (req.earlyoptions['verbose'] or req.earlyoptions['debug'] | ||
or req.earlyoptions['quiet']): | ||||
for opt in ('verbose', 'debug', 'quiet'): | ||||
val = pycompat.bytestr(bool(req.earlyoptions[opt])) | ||||
for ui_ in uis: | ||||
ui_.setconfig('ui', opt, val, '--' + opt) | ||||
Yuya Nishihara
|
r35224 | if req.earlyoptions['profile']: | ||
Bryan O'Sullivan
|
r30933 | for ui_ in uis: | ||
ui_.setconfig('profiling', 'enabled', 'true', '--profile') | ||||
r32788 | profile = lui.configbool('profiling', 'enabled') | |||
with profiling.profile(lui, enabled=profile) as profiler: | ||||
Bryan O'Sullivan
|
r30934 | # Configure extensions in phases: uisetup, extsetup, cmdtable, and | ||
FUJIWARA Katsunori
|
r33053 | # reposetup | ||
Bryan O'Sullivan
|
r30934 | extensions.loadall(lui) | ||
# 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 | # (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 | |||
Yuya Nishihara
|
r35063 | if options["config"] != req.earlyoptions["config"]: | ||
Bryan O'Sullivan
|
r30934 | raise error.Abort(_("option --config may not be abbreviated!")) | ||
Yuya Nishihara
|
r35063 | if options["cwd"] != req.earlyoptions["cwd"]: | ||
Bryan O'Sullivan
|
r30934 | raise error.Abort(_("option --cwd may not be abbreviated!")) | ||
Yuya Nishihara
|
r35063 | if options["repository"] != req.earlyoptions["repository"]: | ||
Bryan O'Sullivan
|
r30934 | raise error.Abort(_( | ||
"option -R has to be separated from other options (e.g. not " | ||||
"-qR) and --repository may only be abbreviated as --repo!")) | ||||
Yuya Nishihara
|
r35059 | if options["debugger"] != req.earlyoptions["debugger"]: | ||
raise error.Abort(_("option --debugger may not be abbreviated!")) | ||||
# don't validate --profile/--traceback, which can be enabled from now | ||||
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])) | ||||
Bryan O'Sullivan
|
r31958 | ui.atexit(print_time) | ||
r32787 | if options["profile"]: | |||
profiler.start() | ||||
Matt Mackall
|
r5178 | |||
Boris Feld
|
r38552 | # if abbreviated version of this were used, take them in account, now | ||
Bryan O'Sullivan
|
r30934 | if options['verbose'] or options['debug'] or options['quiet']: | ||
for opt in ('verbose', 'debug', 'quiet'): | ||||
Boris Feld
|
r38552 | if options[opt] == req.earlyoptions[opt]: | ||
continue | ||||
Yuya Nishihara
|
r35916 | val = pycompat.bytestr(bool(options[opt])) | ||
Bryan O'Sullivan
|
r30934 | 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 | |||
FUJIWARA Katsunori
|
r32404 | # setup color handling before pager, because setting up pager | ||
# might cause incorrect console information | ||||
Pierre-Yves David
|
r31110 | coloropt = options['color'] | ||
Pierre-Yves David
|
r31105 | for ui_ in uis: | ||
Pierre-Yves David
|
r31110 | if coloropt: | ||
ui_.setconfig('ui', 'color', coloropt, '--color') | ||||
color.setup(ui_) | ||||
Pierre-Yves David
|
r31105 | |||
Yuya Nishihara
|
r37102 | if stringutil.parsebool(options['pager']): | ||
FUJIWARA Katsunori
|
r33622 | # ui.pager() expects 'internal-always-' prefix in this case | ||
FUJIWARA Katsunori
|
r32404 | ui.pager('internal-always-' + cmd) | ||
elif options['pager'] != 'auto': | ||||
Jun Wu
|
r34639 | for ui_ in uis: | ||
ui_.disablepager() | ||||
FUJIWARA Katsunori
|
r32404 | |||
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: | ||||
Jun Wu
|
r32379 | repo = hg.repository(ui, path=path, | ||
Gregory Szorc
|
r37735 | presetupfuncs=req.prereposetups, | ||
intents=func.intents) | ||||
Arun Kulshreshtha
|
r30005 | 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: | ||
Yuya Nishihara
|
r35062 | if rpath: # invalid -R path | ||
Arun Kulshreshtha
|
r30005 | 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 | ||
Pulkit Goyal
|
r35150 | repos = pycompat.maplist(cmdutil.findrepo, args) | ||
Arun Kulshreshtha
|
r30005 | guess = repos[0] | ||
if guess and repos.count(guess) == len(repos): | ||||
req.args = ['--repository', guess] + fullargs | ||||
Yuya Nishihara
|
r35224 | req.earlyoptions['repository'] = guess | ||
Arun Kulshreshtha
|
r30005 | 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 | |||
Augie Fackler
|
r31492 | msg = _formatargs(fullargs) | ||
Arun Kulshreshtha
|
r30005 | 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, '' | ||||
Jun Wu
|
r33499 | if ui.config('ui', 'supportcontact') is None: | ||
Martijn Pieters
|
r28784 | for name, mod in extensions.extensions(): | ||
Yuya Nishihara
|
r35917 | # 'testedwith' should be bytes, but not all extensions are ported | ||
# to py3 and we don't want UnicodeException because of that. | ||||
Yuya Nishihara
|
r37102 | testedwith = stringutil.forcebytestr(getattr(mod, 'testedwith', '')) | ||
Martijn Pieters
|
r28784 | 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 | ||||
Augie Fackler
|
r31179 | if not isinstance(testedwith, (bytes, str)): | ||
Yuya Nishihara
|
r37102 | testedwith = '.'.join([stringutil.forcebytestr(c) | ||
for c in testedwith]) | ||||
Martijn Pieters
|
r28784 | 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') | ||||
Pulkit Goyal
|
r38045 | % (name, testedwith, name, stringutil.forcebytestr(report))) | ||
Martijn Pieters
|
r28784 | else: | ||
Jun Wu
|
r33499 | bugtracker = ui.config('ui', 'supportcontact') | ||
Martijn Pieters
|
r28784 | if bugtracker is None: | ||
bugtracker = _("https://mercurial-scm.org/wiki/BugTracker") | ||||
warning = (_("** unknown exception encountered, " | ||||
"please report by visiting\n** ") + bugtracker + '\n') | ||||
Yuya Nishihara
|
r35917 | sysversion = pycompat.sysbytes(sys.version).replace('\n', '') | ||
Augie Fackler
|
r31180 | warning += ((_("** Python %s\n") % sysversion) + | ||
Martijn Pieters
|
r28784 | (_("** 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) | ||||
Gregory Szorc
|
r36141 | ui.log("commandexception", "%s\n%s\n", warning, | ||
pycompat.sysbytes(traceback.format_exc())) | ||||
Martijn Pieters
|
r28784 | ui.warn(warning) | ||
return False # re-raise the exception | ||||