##// END OF EJS Templates
Added tag 2.8.2 for changeset ca387377df7a
Added tag 2.8.2 for changeset ca387377df7a

File last commit:

r19777:6f72e7d2 default
r20212:8cec4cf9 stable
Show More
extensions.py
369 lines | 11.0 KiB | text/x-python | PythonLexer
Matt Mackall
Create a separate module for managing extensions
r4544 # extensions.py - extension handling for mercurial
#
Thomas Arendsen Hein
Updated copyright notices and add "and others" to "hg version"
r4635 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
Matt Mackall
Create a separate module for managing extensions
r4544 #
Martin Geisler
updated license to be explicit about GPL version 2
r8225 # This software may be used and distributed according to the terms of the
Matt Mackall
Update license to GPLv2+
r10263 # GNU General Public License version 2 or any later version.
Matt Mackall
Create a separate module for managing extensions
r4544
Cédric Duval
extensions: remove import rendered unnecessary by e4e22a310b62
r8896 import imp, os
Matt Mackall
extensions: move moduledoc to break import loop with help
r14317 import util, cmdutil, error
Cédric Duval
extensions: move extensions listing functions from mercurial.help...
r8871 from i18n import _, gettext
Matt Mackall
Create a separate module for managing extensions
r4544
_extensions = {}
Alexis S. L. Carvalho
Move cmdtable and reposetup handling out of extensions.py...
r5192 _order = []
Angel Ezquerra
extensions: obsolete and remove interhg extension...
r18629 _ignore = ['hbisect', 'bookmarks', 'parentrevspec', 'interhg']
Alexis S. L. Carvalho
Move cmdtable and reposetup handling out of extensions.py...
r5192
FUJIWARA Katsunori
extensions: list up only enabled extensions, if "ui" is specified...
r19777 def extensions(ui=None):
if ui:
def enabled(name):
for format in ['%s', 'hgext.%s']:
conf = ui.config('extensions', format % name)
if conf is not None and not conf.startswith('!'):
return True
else:
enabled = lambda name: True
Alexis S. L. Carvalho
Move cmdtable and reposetup handling out of extensions.py...
r5192 for name in _order:
module = _extensions[name]
FUJIWARA Katsunori
extensions: list up only enabled extensions, if "ui" is specified...
r19777 if module and enabled(name):
Alexis S. L. Carvalho
Move cmdtable and reposetup handling out of extensions.py...
r5192 yield name, module
Matt Mackall
Create a separate module for managing extensions
r4544
def find(name):
'''return module with given extension name'''
Idan Kamara
extensions: raise when trying to find an extension that failed to load...
r14415 mod = None
Matt Mackall
Create a separate module for managing extensions
r4544 try:
Idan Kamara
extensions: raise when trying to find an extension that failed to load...
r14415 mod = _extensions[name]
Matt Mackall
Create a separate module for managing extensions
r4544 except KeyError:
for k, v in _extensions.iteritems():
Matt Mackall
extensions: fix lookup of hgext.foo modules
r4560 if k.endswith('.' + name) or k.endswith('/' + name):
Idan Kamara
extensions: raise when trying to find an extension that failed to load...
r14415 mod = v
break
if not mod:
Matt Mackall
Create a separate module for managing extensions
r4544 raise KeyError(name)
Idan Kamara
extensions: raise when trying to find an extension that failed to load...
r14415 return mod
Matt Mackall
Create a separate module for managing extensions
r4544
Alexander Solovyov
ability to load hooks from arbitrary python module
r7916 def loadpath(path, module_name):
module_name = module_name.replace('.', '_')
Alexander Solovyov
make path expanding more consistent...
r9610 path = util.expandpath(path)
Alexander Solovyov
ability to load hooks from arbitrary python module
r7916 if os.path.isdir(path):
# module/__init__.py style
Alexander Solovyov
extensions loading: don't fail if path to extension ends with a '/'
r7960 d, f = os.path.split(path.rstrip('/'))
Alexander Solovyov
ability to load hooks from arbitrary python module
r7916 fd, fpath, desc = imp.find_module(f, [d])
return imp.load_module(module_name, fd, fpath, desc)
else:
Simon Heimberg
hooks: print out more information when loading a python hook fails...
r17217 try:
return imp.load_source(module_name, path)
except IOError, exc:
if not exc.filename:
exc.filename = path # python does not fill this
raise
Alexander Solovyov
ability to load hooks from arbitrary python module
r7916
Matt Mackall
Create a separate module for managing extensions
r4544 def load(ui, name, path):
Benoit Boissinot
Do not try to load extensions twice (issue811)
r7011 if name.startswith('hgext.') or name.startswith('hgext/'):
Bryan O'Sullivan
extensions: don't get confused by aliasing between "foo" and "hgext.foo"
r5031 shortname = name[6:]
else:
shortname = name
Matt Mackall
extensions: add an ignore list for old extensions
r13349 if shortname in _ignore:
return None
Bryan O'Sullivan
extensions: don't get confused by aliasing between "foo" and "hgext.foo"
r5031 if shortname in _extensions:
Erik Zielke
extensions.load: return module...
r12779 return _extensions[shortname]
Brendan Cully
Cache extension load failures....
r5087 _extensions[shortname] = None
Matt Mackall
Create a separate module for managing extensions
r4544 if path:
# the module will be loaded in sys.modules
# choose an unique name so that it doesn't
# conflicts with other modules
Alexander Solovyov
ability to load hooks from arbitrary python module
r7916 mod = loadpath(path, 'hgext.%s' % name)
Matt Mackall
Create a separate module for managing extensions
r4544 else:
def importh(name):
mod = __import__(name)
components = name.split('.')
for comp in components[1:]:
mod = getattr(mod, comp)
return mod
try:
mod = importh("hgext.%s" % name)
Greg Ward
extensions: print some debug info on import failure...
r15199 except ImportError, err:
ui.debug('could not import hgext.%s (%s): trying %s\n'
% (name, err, name))
Matt Mackall
Create a separate module for managing extensions
r4544 mod = importh(name)
Bryan O'Sullivan
extensions: don't get confused by aliasing between "foo" and "hgext.foo"
r5031 _extensions[shortname] = mod
Alexis S. L. Carvalho
Move cmdtable and reposetup handling out of extensions.py...
r5192 _order.append(shortname)
Erik Zielke
extensions.load: return module...
r12779 return mod
Matt Mackall
Create a separate module for managing extensions
r4544
def loadall(ui):
Matt Mackall
extensions: pull extension-aware bits out of ui
r4617 result = ui.configitems("extensions")
Martin Geisler
extensions: load and configure extensions in well-defined phases...
r9410 newindex = len(_order)
Peter Arrenbrecht
cleanup: drop enumerate() when index is not used
r7876 for (name, path) in result:
Matt Mackall
extensions: pull extension-aware bits out of ui
r4617 if path:
Steve Borho
Allow explicit disabling of extensions...
r5469 if path[0] == '!':
continue
Matt Mackall
Create a separate module for managing extensions
r4544 try:
load(ui, name, path)
Matt Mackall
error: move SignalInterrupt...
r7644 except KeyboardInterrupt:
Matt Mackall
Create a separate module for managing extensions
r4544 raise
except Exception, inst:
Jesse Glick
When failing to load an extension, show where Hg tried to load it from.
r6204 if path:
ui.warn(_("*** failed to import extension %s from %s: %s\n")
% (name, path, inst))
else:
ui.warn(_("*** failed to import extension %s: %s\n")
% (name, inst))
Matt Mackall
ui: print_exc() -> traceback()
r8206 if ui.traceback():
Matt Mackall
Create a separate module for managing extensions
r4544 return 1
Martin Geisler
extensions: load and configure extensions in well-defined phases...
r9410 for name in _order[newindex:]:
uisetup = getattr(_extensions[name], 'uisetup', None)
if uisetup:
uisetup(ui)
Yuya Nishihara
extensions: changed to call extsetup() from extensions.loadall()...
r9660 for name in _order[newindex:]:
extsetup = getattr(_extensions[name], 'extsetup', None)
if extsetup:
try:
extsetup(ui)
except TypeError:
if extsetup.func_code.co_argcount != 0:
raise
extsetup() # old extsetup with no ui argument
Matt Mackall
extensions: add wrapping functions
r7215 def wrapcommand(table, command, wrapper):
Dan Villiom Podlaski Christiansen
extensions: add docstring for wrapcommand().
r11519 '''Wrap the command named `command' in table
Replace command in the command table with wrapper. The wrapped command will
be inserted into the command table specified by the table argument.
The wrapper will be called like
wrapper(orig, *args, **kwargs)
where orig is the original (wrapped) function, and *args, **kwargs
are the arguments passed to it.
'''
Augie Fackler
globally: use safehasattr(x, '__call__') instead of hasattr(x, '__call__')
r14943 assert util.safehasattr(wrapper, '__call__')
Matt Mackall
extensions: add wrapping functions
r7215 aliases, entry = cmdutil.findcmd(command, table)
for alias, e in table.iteritems():
if e is entry:
key = alias
break
origfn = entry[0]
def wrap(*args, **kwargs):
Matt Mackall
dispatch: generalize signature checking for extension command wrapping
r7388 return util.checksignature(wrapper)(
util.checksignature(origfn), *args, **kwargs)
Matt Mackall
extensions: add wrapping functions
r7215
wrap.__doc__ = getattr(origfn, '__doc__')
Dirkjan Ochtman
extensions: copy __module__ for wrapped commands
r7373 wrap.__module__ = getattr(origfn, '__module__')
Matt Mackall
extensions: add wrapping functions
r7215
newentry = list(entry)
newentry[0] = wrap
table[key] = tuple(newentry)
return entry
def wrapfunction(container, funcname, wrapper):
Greg Ward
extensions: recommend against using wrapfunction for repo methods...
r11402 '''Wrap the function named funcname in container
Dan Villiom Podlaski Christiansen
extensions: improve language for wrapfunction() docstring.
r11520 Replace the funcname member in the given container with the specified
wrapper. The container is typically a module, class, or instance.
Greg Ward
extensions: recommend against using wrapfunction for repo methods...
r11402
The wrapper will be called like
wrapper(orig, *args, **kwargs)
where orig is the original (wrapped) function, and *args, **kwargs
are the arguments passed to it.
Wrapping methods of the repository object is not recommended since
it conflicts with extensions that extend the repository by
subclassing. All extensions that need to extend methods of
localrepository should use this subclassing trick: namely,
reposetup() should look like
def reposetup(ui, repo):
class myrepo(repo.__class__):
def whatever(self, *args, **kwargs):
[...extension stuff...]
super(myrepo, self).whatever(*args, **kwargs)
[...extension stuff...]
repo.__class__ = myrepo
In general, combining wrapfunction() with subclassing does not
work. Since you cannot control what other extensions are loaded by
your end users, you should play nicely with others by using the
subclass trick.
'''
Augie Fackler
globally: use safehasattr(x, '__call__') instead of hasattr(x, '__call__')
r14943 assert util.safehasattr(wrapper, '__call__')
Matt Mackall
extensions: add wrapping functions
r7215 def wrap(*args, **kwargs):
return wrapper(origfn, *args, **kwargs)
origfn = getattr(container, funcname)
Augie Fackler
globally: use safehasattr(x, '__call__') instead of hasattr(x, '__call__')
r14943 assert util.safehasattr(origfn, '__call__')
Matt Mackall
extensions: add wrapping functions
r7215 setattr(container, funcname, wrap)
return origfn
Cédric Duval
extensions: move extensions listing functions from mercurial.help...
r8871
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364 def _disabledpaths(strip_init=False):
'''find paths of disabled extensions. returns a dict of {name: path}
removes /__init__.py from packages if strip_init is True'''
Dirkjan Ochtman
extensions: simplify by selecting primary hgext
r8872 import hgext
extpath = os.path.dirname(os.path.abspath(hgext.__file__))
Cédric Duval
extensions: catch OSError when hgext is not accessible (issue1708)...
r8964 try: # might not be a filesystem path
files = os.listdir(extpath)
except OSError:
Brodie Rao
extensions: refactor disabled()
r10363 return {}
Cédric Duval
extensions: catch OSError when hgext is not accessible (issue1708)...
r8964
Cédric Duval
extensions: move extensions listing functions from mercurial.help...
r8871 exts = {}
Cédric Duval
extensions: catch OSError when hgext is not accessible (issue1708)...
r8964 for e in files:
Dirkjan Ochtman
extensions: simplify by selecting primary hgext
r8872 if e.endswith('.py'):
name = e.rsplit('.', 1)[0]
path = os.path.join(extpath, e)
else:
name = e
path = os.path.join(extpath, e, '__init__.py')
Cédric Duval
extensions: check for path existence only when necessary
r8877 if not os.path.exists(path):
continue
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364 if strip_init:
path = os.path.dirname(path)
Brodie Rao
extensions: refactor disabled()
r10363 if name in exts or name in _order or name == '__init__':
continue
exts[name] = path
return exts
Dirkjan Ochtman
extensions: simplify by selecting primary hgext
r8872
Matt Mackall
extensions: move moduledoc to break import loop with help
r14317 def _moduledoc(file):
'''return the top-level python documentation for the given file
Loosely inspired by pydoc.source_synopsis(), but rewritten to
handle triple quotes and to return the whole text instead of just
the synopsis'''
result = []
line = file.readline()
while line[:1] == '#' or not line.strip():
line = file.readline()
if not line:
break
start = line[:3]
if start == '"""' or start == "'''":
line = line[3:]
while line:
if line.rstrip().endswith(start):
line = line.split(start)[0]
if line:
result.append(line)
break
elif not line:
return None # unmatched delimiter
result.append(line)
line = file.readline()
else:
return None
return ''.join(result)
Brodie Rao
extensions: refactor disabled()
r10363 def _disabledhelp(path):
'''retrieve help synopsis of a disabled extension (without importing)'''
try:
file = open(path)
except IOError:
return
else:
Matt Mackall
help: consolidate topic hooks in help.py...
r14318 doc = _moduledoc(file)
Brodie Rao
extensions: refactor disabled()
r10363 file.close()
if doc: # extracting localized synopsis
return gettext(doc).splitlines()[0]
else:
return _('(no help text available)')
def disabled():
Yuya Nishihara
extensions: update doc of enabled() and disabled() according to d5b525697ddb
r14530 '''find disabled extensions from hgext. returns a dict of {name: desc}'''
Yuya Nishihara
extensions: make disabled()/disabledext() load prebuilt index if available...
r14539 try:
from hgext import __index__
return dict((name, gettext(desc))
for name, desc in __index__.docs.iteritems()
if name not in _order)
except ImportError:
pass
Brodie Rao
extensions: refactor disabled()
r10363 paths = _disabledpaths()
if not paths:
Augie Fackler
extensions.disabled: return {} instead of None no extensions are disabled
r16709 return {}
Brodie Rao
extensions: refactor disabled()
r10363
exts = {}
for name, path in paths.iteritems():
doc = _disabledhelp(path)
Matt Mackall
extensions: drop maxlength from enabled and disabled...
r14316 if doc:
exts[name] = doc
Cédric Duval
extensions: move extensions listing functions from mercurial.help...
r8871
Matt Mackall
extensions: drop maxlength from enabled and disabled...
r14316 return exts
Cédric Duval
extensions: move extensions listing functions from mercurial.help...
r8871
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364 def disabledext(name):
'''find a specific disabled extension from hgext. returns desc'''
Yuya Nishihara
extensions: make disabled()/disabledext() load prebuilt index if available...
r14539 try:
from hgext import __index__
if name in _order: # enabled
return
else:
return gettext(__index__.docs.get(name))
except ImportError:
pass
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364 paths = _disabledpaths()
if name in paths:
return _disabledhelp(paths[name])
Mads Kiilerich
extensions: warn about invalid extensions when listing disabled commands...
r13191 def disabledcmd(ui, cmd, strict=False):
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364 '''import disabled extensions until cmd is found.
Augie Fackler
extensions: fix documentation of disabledcmd return value
r16666 returns (cmdname, extname, module)'''
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364
paths = _disabledpaths(strip_init=True)
if not paths:
raise error.UnknownCommand(cmd)
def findcmd(cmd, name, path):
try:
mod = loadpath(path, 'hgext.%s' % name)
except Exception:
return
try:
aliases, entry = cmdutil.findcmd(cmd,
getattr(mod, 'cmdtable', {}), strict)
except (error.AmbiguousCommand, error.UnknownCommand):
return
Mads Kiilerich
extensions: warn about invalid extensions when listing disabled commands...
r13191 except Exception:
ui.warn(_('warning: error finding commands in %s\n') % path)
ui.traceback()
return
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364 for c in aliases:
if c.startswith(cmd):
cmd = c
break
else:
cmd = aliases[0]
return (cmd, name, mod)
Martin Geisler
extensions: don't suggest commands from deprecated extensions
r16667 ext = None
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364 # first, search for an extension with the same name as the command
path = paths.pop(cmd, None)
if path:
ext = findcmd(cmd, cmd, path)
Martin Geisler
extensions: don't suggest commands from deprecated extensions
r16667 if not ext:
# otherwise, interrogate each extension until there's a match
for name, path in paths.iteritems():
ext = findcmd(cmd, name, path)
if ext:
break
if ext and 'DEPRECATED' not in ext.__doc__:
return ext
Brodie Rao
dispatch: provide help for disabled extensions and commands...
r10364
raise error.UnknownCommand(cmd)
FUJIWARA Katsunori
help: use full name of extensions to look up them for keyword search...
r19769 def enabled(shortname=True):
Yuya Nishihara
extensions: update doc of enabled() and disabled() according to d5b525697ddb
r14530 '''return a dict of {name: desc} of extensions'''
Cédric Duval
extensions: move extensions listing functions from mercurial.help...
r8871 exts = {}
for ename, ext in extensions():
doc = (gettext(ext.__doc__) or _('(no help text available)'))
FUJIWARA Katsunori
help: use full name of extensions to look up them for keyword search...
r19769 if shortname:
ename = ename.split('.')[-1]
Nicolas Dumazet
for calls expecting bool args, pass bool instead of int...
r9136 exts[ename] = doc.splitlines()[0].strip()
Cédric Duval
extensions: move extensions listing functions from mercurial.help...
r8871
Matt Mackall
extensions: drop maxlength from enabled and disabled...
r14316 return exts