extensions.py
538 lines
| 16.4 KiB
| text/x-python
|
PythonLexer
/ mercurial / extensions.py
Matt Mackall
|
r4544 | # extensions.py - extension handling for mercurial | ||
# | ||||
Thomas Arendsen Hein
|
r4635 | # Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||
Matt Mackall
|
r4544 | # | ||
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
|
r4544 | |||
Gregory Szorc
|
r25946 | from __future__ import absolute_import | ||
import imp | ||||
import os | ||||
from .i18n import ( | ||||
_, | ||||
gettext, | ||||
) | ||||
from . import ( | ||||
cmdutil, | ||||
error, | ||||
util, | ||||
) | ||||
Matt Mackall
|
r4544 | |||
_extensions = {} | ||||
liscju
|
r29895 | _disabledextensions = {} | ||
Gregory Szorc
|
r24065 | _aftercallbacks = {} | ||
Alexis S. L. Carvalho
|
r5192 | _order = [] | ||
Augie Fackler
|
r27116 | _builtin = set(['hbisect', 'bookmarks', 'parentrevspec', 'progress', 'interhg', | ||
Jun Wu
|
r29162 | 'inotify', 'hgcia']) | ||
Alexis S. L. Carvalho
|
r5192 | |||
FUJIWARA Katsunori
|
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
|
r5192 | for name in _order: | ||
module = _extensions[name] | ||||
FUJIWARA Katsunori
|
r19777 | if module and enabled(name): | ||
Alexis S. L. Carvalho
|
r5192 | yield name, module | ||
Matt Mackall
|
r4544 | |||
def find(name): | ||||
'''return module with given extension name''' | ||||
Idan Kamara
|
r14415 | mod = None | ||
Matt Mackall
|
r4544 | try: | ||
timeless
|
r27637 | mod = _extensions[name] | ||
Matt Mackall
|
r4544 | except KeyError: | ||
for k, v in _extensions.iteritems(): | ||||
Matt Mackall
|
r4560 | if k.endswith('.' + name) or k.endswith('/' + name): | ||
Idan Kamara
|
r14415 | mod = v | ||
break | ||||
if not mod: | ||||
Matt Mackall
|
r4544 | raise KeyError(name) | ||
Idan Kamara
|
r14415 | return mod | ||
Matt Mackall
|
r4544 | |||
Alexander Solovyov
|
r7916 | def loadpath(path, module_name): | ||
module_name = module_name.replace('.', '_') | ||||
Ed Morley
|
r20645 | path = util.normpath(util.expandpath(path)) | ||
Alexander Solovyov
|
r7916 | if os.path.isdir(path): | ||
# module/__init__.py style | ||||
Ed Morley
|
r20645 | d, f = os.path.split(path) | ||
Alexander Solovyov
|
r7916 | fd, fpath, desc = imp.find_module(f, [d]) | ||
return imp.load_module(module_name, fd, fpath, desc) | ||||
else: | ||||
Simon Heimberg
|
r17217 | try: | ||
return imp.load_source(module_name, path) | ||||
Gregory Szorc
|
r25660 | except IOError as exc: | ||
Simon Heimberg
|
r17217 | if not exc.filename: | ||
exc.filename = path # python does not fill this | ||||
raise | ||||
Alexander Solovyov
|
r7916 | |||
Pierre-Yves David
|
r28505 | def _importh(name): | ||
"""import and return the <name> module""" | ||||
mod = __import__(name) | ||||
components = name.split('.') | ||||
for comp in components[1:]: | ||||
mod = getattr(mod, comp) | ||||
return mod | ||||
Pierre-Yves David
|
r28506 | def _reportimporterror(ui, err, failed, next): | ||
ui.debug('could not import %s (%s): trying %s\n' | ||||
% (failed, err, next)) | ||||
if ui.debugflag: | ||||
ui.traceback() | ||||
Matt Mackall
|
r4544 | def load(ui, name, path): | ||
Benoit Boissinot
|
r7011 | if name.startswith('hgext.') or name.startswith('hgext/'): | ||
Bryan O'Sullivan
|
r5031 | shortname = name[6:] | ||
else: | ||||
shortname = name | ||||
Bryan O'Sullivan
|
r27111 | if shortname in _builtin: | ||
Matt Mackall
|
r13349 | return None | ||
Bryan O'Sullivan
|
r5031 | if shortname in _extensions: | ||
Erik Zielke
|
r12779 | return _extensions[shortname] | ||
Brendan Cully
|
r5087 | _extensions[shortname] = None | ||
Matt Mackall
|
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
|
r7916 | mod = loadpath(path, 'hgext.%s' % name) | ||
Matt Mackall
|
r4544 | else: | ||
try: | ||||
Pierre-Yves David
|
r28505 | mod = _importh("hgext.%s" % name) | ||
Gregory Szorc
|
r25660 | except ImportError as err: | ||
Pierre-Yves David
|
r28506 | _reportimporterror(ui, err, "hgext.%s" % name, name) | ||
Pierre-Yves David
|
r28541 | try: | ||
mod = _importh("hgext3rd.%s" % name) | ||||
except ImportError as err: | ||||
_reportimporterror(ui, err, "hgext3rd.%s" % name, name) | ||||
mod = _importh(name) | ||||
Gregory Szorc
|
r27142 | |||
# Before we do anything with the extension, check against minimum stated | ||||
# compatibility. This gives extension authors a mechanism to have their | ||||
# extensions short circuit when loaded with a known incompatible version | ||||
# of Mercurial. | ||||
minver = getattr(mod, 'minimumhgversion', None) | ||||
if minver and util.versiontuple(minver, 2) > util.versiontuple(n=2): | ||||
ui.warn(_('(third party extension %s requires version %s or newer ' | ||||
'of Mercurial; disabling)\n') % (shortname, minver)) | ||||
return | ||||
Bryan O'Sullivan
|
r5031 | _extensions[shortname] = mod | ||
Alexis S. L. Carvalho
|
r5192 | _order.append(shortname) | ||
Gregory Szorc
|
r24065 | for fn in _aftercallbacks.get(shortname, []): | ||
fn(loaded=True) | ||||
Erik Zielke
|
r12779 | return mod | ||
Matt Mackall
|
r4544 | |||
Jun Wu
|
r29461 | def _runuisetup(name, ui): | ||
uisetup = getattr(_extensions[name], 'uisetup', None) | ||||
if uisetup: | ||||
uisetup(ui) | ||||
def _runextsetup(name, ui): | ||||
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
|
r4544 | def loadall(ui): | ||
Matt Mackall
|
r4617 | result = ui.configitems("extensions") | ||
Martin Geisler
|
r9410 | newindex = len(_order) | ||
Peter Arrenbrecht
|
r7876 | for (name, path) in result: | ||
Matt Mackall
|
r4617 | if path: | ||
Steve Borho
|
r5469 | if path[0] == '!': | ||
liscju
|
r29895 | _disabledextensions[name] = path[1:] | ||
Steve Borho
|
r5469 | continue | ||
Matt Mackall
|
r4544 | try: | ||
load(ui, name, path) | ||||
Matt Mackall
|
r7644 | except KeyboardInterrupt: | ||
Matt Mackall
|
r4544 | raise | ||
Gregory Szorc
|
r25660 | except Exception as inst: | ||
Jesse Glick
|
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)) | ||||
Yuya Nishihara
|
r25364 | ui.traceback() | ||
Matt Mackall
|
r4544 | |||
Martin Geisler
|
r9410 | for name in _order[newindex:]: | ||
Jun Wu
|
r29461 | _runuisetup(name, ui) | ||
Martin Geisler
|
r9410 | |||
Yuya Nishihara
|
r9660 | for name in _order[newindex:]: | ||
Jun Wu
|
r29461 | _runextsetup(name, ui) | ||
Yuya Nishihara
|
r9660 | |||
Gregory Szorc
|
r24065 | # Call aftercallbacks that were never met. | ||
for shortname in _aftercallbacks: | ||||
if shortname in _extensions: | ||||
continue | ||||
for fn in _aftercallbacks[shortname]: | ||||
fn(loaded=False) | ||||
Gregory Szorc
|
r24950 | # loadall() is called multiple times and lingering _aftercallbacks | ||
# entries could result in double execution. See issue4646. | ||||
_aftercallbacks.clear() | ||||
Gregory Szorc
|
r24065 | def afterloaded(extension, callback): | ||
'''Run the specified function after a named extension is loaded. | ||||
If the named extension is already loaded, the callback will be called | ||||
immediately. | ||||
If the named extension never loads, the callback will be called after | ||||
all extensions have been loaded. | ||||
The callback receives the named argument ``loaded``, which is a boolean | ||||
indicating whether the dependent extension actually loaded. | ||||
''' | ||||
if extension in _extensions: | ||||
Matt Harbison
|
r24145 | callback(loaded=True) | ||
Gregory Szorc
|
r24065 | else: | ||
_aftercallbacks.setdefault(extension, []).append(callback) | ||||
Eric Sumner
|
r24734 | def bind(func, *args): | ||
'''Partial function application | ||||
Returns a new function that is the partial application of args and kwargs | ||||
to func. For example, | ||||
f(1, 2, bar=3) === bind(f, 1)(2, bar=3)''' | ||||
assert callable(func) | ||||
def closure(*a, **kw): | ||||
return func(*(args + a), **kw) | ||||
return closure | ||||
Jun Wu
|
r29763 | def _updatewrapper(wrap, origfn, unboundwrapper): | ||
'''Copy and add some useful attributes to wrapper''' | ||||
Yuya Nishihara
|
r28310 | wrap.__module__ = getattr(origfn, '__module__') | ||
wrap.__doc__ = getattr(origfn, '__doc__') | ||||
Yuya Nishihara
|
r28312 | wrap.__dict__.update(getattr(origfn, '__dict__', {})) | ||
Jun Wu
|
r29763 | wrap._origfunc = origfn | ||
wrap._unboundwrapper = unboundwrapper | ||||
Yuya Nishihara
|
r28310 | |||
Ryan McElroy
|
r24124 | def wrapcommand(table, command, wrapper, synopsis=None, docstring=None): | ||
Dan Villiom Podlaski Christiansen
|
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. | ||||
Ryan McElroy
|
r24124 | |||
Optionally append to the command synopsis and docstring, used for help. | ||||
For example, if your extension wraps the ``bookmarks`` command to add the | ||||
flags ``--remote`` and ``--all`` you might call this function like so: | ||||
synopsis = ' [-a] [--remote]' | ||||
docstring = """ | ||||
The ``remotenames`` extension adds the ``--remote`` and ``--all`` (``-a``) | ||||
flags to the bookmarks command. Either flag will show the remote bookmarks | ||||
Mads Kiilerich
|
r26781 | known to the repository; ``--remote`` will also suppress the output of the | ||
Ryan McElroy
|
r24124 | local bookmarks. | ||
""" | ||||
extensions.wrapcommand(commands.table, 'bookmarks', exbookmarks, | ||||
synopsis, docstring) | ||||
Dan Villiom Podlaski Christiansen
|
r11519 | ''' | ||
Augie Fackler
|
r21795 | assert callable(wrapper) | ||
Matt Mackall
|
r7215 | aliases, entry = cmdutil.findcmd(command, table) | ||
for alias, e in table.iteritems(): | ||||
if e is entry: | ||||
key = alias | ||||
break | ||||
origfn = entry[0] | ||||
Eric Sumner
|
r24734 | wrap = bind(util.checksignature(wrapper), util.checksignature(origfn)) | ||
Jun Wu
|
r29763 | _updatewrapper(wrap, origfn, wrapper) | ||
Ryan McElroy
|
r24124 | if docstring is not None: | ||
Yuya Nishihara
|
r28310 | wrap.__doc__ += docstring | ||
Ryan McElroy
|
r24124 | |||
Matt Mackall
|
r7215 | newentry = list(entry) | ||
newentry[0] = wrap | ||||
Ryan McElroy
|
r24124 | if synopsis is not None: | ||
newentry[2] += synopsis | ||||
Matt Mackall
|
r7215 | table[key] = tuple(newentry) | ||
return entry | ||||
def wrapfunction(container, funcname, wrapper): | ||||
Greg Ward
|
r11402 | '''Wrap the function named funcname in container | ||
Dan Villiom Podlaski Christiansen
|
r11520 | Replace the funcname member in the given container with the specified | ||
wrapper. The container is typically a module, class, or instance. | ||||
Greg Ward
|
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
|
r21795 | assert callable(wrapper) | ||
Matt Mackall
|
r7215 | |||
origfn = getattr(container, funcname) | ||||
Augie Fackler
|
r21795 | assert callable(origfn) | ||
Yuya Nishihara
|
r28311 | wrap = bind(wrapper, origfn) | ||
Jun Wu
|
r29763 | _updatewrapper(wrap, origfn, wrapper) | ||
Yuya Nishihara
|
r28311 | setattr(container, funcname, wrap) | ||
Matt Mackall
|
r7215 | return origfn | ||
Cédric Duval
|
r8871 | |||
Jun Wu
|
r29765 | def unwrapfunction(container, funcname, wrapper=None): | ||
'''undo wrapfunction | ||||
If wrappers is None, undo the last wrap. Otherwise removes the wrapper | ||||
from the chain of wrappers. | ||||
Return the removed wrapper. | ||||
Raise IndexError if wrapper is None and nothing to unwrap; ValueError if | ||||
wrapper is not None but is not found in the wrapper chain. | ||||
''' | ||||
chain = getwrapperchain(container, funcname) | ||||
origfn = chain.pop() | ||||
if wrapper is None: | ||||
wrapper = chain[0] | ||||
chain.remove(wrapper) | ||||
setattr(container, funcname, origfn) | ||||
for w in reversed(chain): | ||||
wrapfunction(container, funcname, w) | ||||
return wrapper | ||||
Jun Wu
|
r29764 | def getwrapperchain(container, funcname): | ||
'''get a chain of wrappers of a function | ||||
Return a list of functions: [newest wrapper, ..., oldest wrapper, origfunc] | ||||
The wrapper functions are the ones passed to wrapfunction, whose first | ||||
argument is origfunc. | ||||
''' | ||||
result = [] | ||||
fn = getattr(container, funcname) | ||||
while fn: | ||||
assert callable(fn) | ||||
result.append(getattr(fn, '_unboundwrapper', fn)) | ||||
fn = getattr(fn, '_origfunc', None) | ||||
return result | ||||
Brodie Rao
|
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
|
r8872 | import hgext | ||
extpath = os.path.dirname(os.path.abspath(hgext.__file__)) | ||||
Cédric Duval
|
r8964 | try: # might not be a filesystem path | ||
files = os.listdir(extpath) | ||||
except OSError: | ||||
Brodie Rao
|
r10363 | return {} | ||
Cédric Duval
|
r8964 | |||
Cédric Duval
|
r8871 | exts = {} | ||
Cédric Duval
|
r8964 | for e in files: | ||
Dirkjan Ochtman
|
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
|
r8877 | if not os.path.exists(path): | ||
continue | ||||
Brodie Rao
|
r10364 | if strip_init: | ||
path = os.path.dirname(path) | ||||
Brodie Rao
|
r10363 | if name in exts or name in _order or name == '__init__': | ||
continue | ||||
exts[name] = path | ||||
liscju
|
r29895 | exts.update(_disabledextensions) | ||
Brodie Rao
|
r10363 | return exts | ||
Dirkjan Ochtman
|
r8872 | |||
Matt Mackall
|
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
|
r10363 | def _disabledhelp(path): | ||
'''retrieve help synopsis of a disabled extension (without importing)''' | ||||
try: | ||||
file = open(path) | ||||
except IOError: | ||||
return | ||||
else: | ||||
Matt Mackall
|
r14318 | doc = _moduledoc(file) | ||
Brodie Rao
|
r10363 | file.close() | ||
if doc: # extracting localized synopsis | ||||
return gettext(doc).splitlines()[0] | ||||
else: | ||||
return _('(no help text available)') | ||||
def disabled(): | ||||
Yuya Nishihara
|
r14530 | '''find disabled extensions from hgext. returns a dict of {name: desc}''' | ||
Yuya Nishihara
|
r14539 | try: | ||
from hgext import __index__ | ||||
return dict((name, gettext(desc)) | ||||
for name, desc in __index__.docs.iteritems() | ||||
if name not in _order) | ||||
Thomas Arendsen Hein
|
r21229 | except (ImportError, AttributeError): | ||
Yuya Nishihara
|
r14539 | pass | ||
Brodie Rao
|
r10363 | paths = _disabledpaths() | ||
if not paths: | ||||
Augie Fackler
|
r16709 | return {} | ||
Brodie Rao
|
r10363 | |||
exts = {} | ||||
for name, path in paths.iteritems(): | ||||
doc = _disabledhelp(path) | ||||
Matt Mackall
|
r14316 | if doc: | ||
exts[name] = doc | ||||
Cédric Duval
|
r8871 | |||
Matt Mackall
|
r14316 | return exts | ||
Cédric Duval
|
r8871 | |||
Brodie Rao
|
r10364 | def disabledext(name): | ||
'''find a specific disabled extension from hgext. returns desc''' | ||||
Yuya Nishihara
|
r14539 | try: | ||
from hgext import __index__ | ||||
if name in _order: # enabled | ||||
return | ||||
else: | ||||
return gettext(__index__.docs.get(name)) | ||||
Thomas Arendsen Hein
|
r21229 | except (ImportError, AttributeError): | ||
Yuya Nishihara
|
r14539 | pass | ||
Brodie Rao
|
r10364 | paths = _disabledpaths() | ||
if name in paths: | ||||
return _disabledhelp(paths[name]) | ||||
Mads Kiilerich
|
r13191 | def disabledcmd(ui, cmd, strict=False): | ||
Brodie Rao
|
r10364 | '''import disabled extensions until cmd is found. | ||
Augie Fackler
|
r16666 | returns (cmdname, extname, module)''' | ||
Brodie Rao
|
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
|
r13191 | except Exception: | ||
ui.warn(_('warning: error finding commands in %s\n') % path) | ||||
ui.traceback() | ||||
return | ||||
Brodie Rao
|
r10364 | for c in aliases: | ||
if c.startswith(cmd): | ||||
cmd = c | ||||
break | ||||
else: | ||||
cmd = aliases[0] | ||||
return (cmd, name, mod) | ||||
Martin Geisler
|
r16667 | ext = None | ||
Brodie Rao
|
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
|
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
|
r10364 | |||
raise error.UnknownCommand(cmd) | ||||
FUJIWARA Katsunori
|
r19769 | def enabled(shortname=True): | ||
Yuya Nishihara
|
r14530 | '''return a dict of {name: desc} of extensions''' | ||
Cédric Duval
|
r8871 | exts = {} | ||
for ename, ext in extensions(): | ||||
doc = (gettext(ext.__doc__) or _('(no help text available)')) | ||||
FUJIWARA Katsunori
|
r19769 | if shortname: | ||
ename = ename.split('.')[-1] | ||||
Nicolas Dumazet
|
r9136 | exts[ename] = doc.splitlines()[0].strip() | ||
Cédric Duval
|
r8871 | |||
Matt Mackall
|
r14316 | return exts | ||
anatoly techtonik
|
r21848 | |||
Jun Wu
|
r28155 | def notloaded(): | ||
'''return short names of extensions that failed to load''' | ||||
return [name for name, mod in _extensions.iteritems() if mod is None] | ||||
anatoly techtonik
|
r21848 | def moduleversion(module): | ||
'''return version information from given module as a string''' | ||||
if (util.safehasattr(module, 'getversion') | ||||
and callable(module.getversion)): | ||||
version = module.getversion() | ||||
elif util.safehasattr(module, '__version__'): | ||||
version = module.__version__ | ||||
else: | ||||
version = '' | ||||
if isinstance(version, (list, tuple)): | ||||
version = '.'.join(str(o) for o in version) | ||||
return version | ||||
liscju
|
r27990 | |||
def ismoduleinternal(module): | ||||
exttestedwith = getattr(module, 'testedwith', None) | ||||
Augie Fackler
|
r29841 | return exttestedwith == "ships-with-hg-core" | ||