2010-11-06 06:37:31 +08:00
|
|
|
"""
|
|
|
|
pytest PluginManager, basic initialization and tracing.
|
|
|
|
"""
|
2014-08-01 06:13:40 +08:00
|
|
|
import os
|
2013-10-02 20:32:40 +08:00
|
|
|
import sys
|
2010-10-13 17:12:27 +08:00
|
|
|
import inspect
|
2010-10-12 18:54:32 +08:00
|
|
|
import py
|
2014-01-22 18:17:25 +08:00
|
|
|
# don't import pytest to avoid circular imports
|
2010-10-13 03:59:15 +08:00
|
|
|
|
2010-11-24 02:11:21 +08:00
|
|
|
assert py.__version__.split(".")[:2] >= ['1', '4'], ("installation problem: "
|
2010-10-13 03:59:15 +08:00
|
|
|
"%s is too old, remove or upgrade 'py'" % (py.__version__))
|
|
|
|
|
2010-11-06 06:37:31 +08:00
|
|
|
class TagTracer:
|
2011-07-15 01:11:50 +08:00
|
|
|
def __init__(self):
|
2010-11-06 06:37:31 +08:00
|
|
|
self._tag2proc = {}
|
|
|
|
self.writer = None
|
2010-11-06 16:05:17 +08:00
|
|
|
self.indent = 0
|
2010-11-06 06:37:31 +08:00
|
|
|
|
|
|
|
def get(self, name):
|
|
|
|
return TagTracerSub(self, (name,))
|
|
|
|
|
2012-11-29 17:04:39 +08:00
|
|
|
def format_message(self, tags, args):
|
|
|
|
if isinstance(args[-1], dict):
|
|
|
|
extra = args[-1]
|
|
|
|
args = args[:-1]
|
|
|
|
else:
|
|
|
|
extra = {}
|
|
|
|
|
|
|
|
content = " ".join(map(str, args))
|
|
|
|
indent = " " * self.indent
|
2013-02-04 23:07:51 +08:00
|
|
|
|
2012-11-29 17:04:39 +08:00
|
|
|
lines = [
|
|
|
|
"%s%s [%s]\n" %(indent, content, ":".join(tags))
|
|
|
|
]
|
|
|
|
|
|
|
|
for name, value in extra.items():
|
|
|
|
lines.append("%s %s: %s\n" % (indent, name, value))
|
|
|
|
return lines
|
|
|
|
|
2010-11-06 06:37:31 +08:00
|
|
|
def processmessage(self, tags, args):
|
2012-11-29 17:04:39 +08:00
|
|
|
if self.writer is not None and args:
|
|
|
|
lines = self.format_message(tags, args)
|
|
|
|
self.writer(''.join(lines))
|
2010-11-06 06:37:31 +08:00
|
|
|
try:
|
|
|
|
self._tag2proc[tags](tags, args)
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
def setwriter(self, writer):
|
|
|
|
self.writer = writer
|
|
|
|
|
|
|
|
def setprocessor(self, tags, processor):
|
|
|
|
if isinstance(tags, str):
|
|
|
|
tags = tuple(tags.split(":"))
|
|
|
|
else:
|
|
|
|
assert isinstance(tags, tuple)
|
|
|
|
self._tag2proc[tags] = processor
|
|
|
|
|
|
|
|
class TagTracerSub:
|
|
|
|
def __init__(self, root, tags):
|
|
|
|
self.root = root
|
|
|
|
self.tags = tags
|
|
|
|
def __call__(self, *args):
|
|
|
|
self.root.processmessage(self.tags, args)
|
|
|
|
def setmyprocessor(self, processor):
|
|
|
|
self.root.setprocessor(self.tags, processor)
|
|
|
|
def get(self, name):
|
|
|
|
return self.__class__(self.root, self.tags + (name,))
|
|
|
|
|
2010-10-12 18:54:32 +08:00
|
|
|
class PluginManager(object):
|
2013-09-30 19:14:14 +08:00
|
|
|
def __init__(self, hookspecs=None):
|
2010-10-12 18:54:32 +08:00
|
|
|
self._name2plugin = {}
|
2011-03-08 01:28:45 +08:00
|
|
|
self._listattrcache = {}
|
2010-10-13 17:12:27 +08:00
|
|
|
self._plugins = []
|
2014-10-01 18:19:11 +08:00
|
|
|
self._conftestplugins = []
|
2014-03-12 05:10:51 +08:00
|
|
|
self._warnings = []
|
2010-11-06 16:05:17 +08:00
|
|
|
self.trace = TagTracer().get("pluginmanage")
|
2011-01-13 02:39:36 +08:00
|
|
|
self._plugin_distinfo = []
|
2013-09-29 04:23:00 +08:00
|
|
|
self._shutdown = []
|
2013-09-30 19:14:14 +08:00
|
|
|
self.hook = HookRelay(hookspecs or [], pm=self)
|
|
|
|
|
2013-10-04 17:36:45 +08:00
|
|
|
def do_configure(self, config):
|
|
|
|
# backward compatibility
|
|
|
|
config.do_configure()
|
|
|
|
|
2013-09-30 19:14:14 +08:00
|
|
|
def set_register_callback(self, callback):
|
|
|
|
assert not hasattr(self, "_registercallback")
|
|
|
|
self._registercallback = callback
|
2010-10-12 18:54:32 +08:00
|
|
|
|
2014-10-01 18:19:11 +08:00
|
|
|
def register(self, plugin, name=None, prepend=False, conftest=False):
|
2012-06-17 03:29:04 +08:00
|
|
|
if self._name2plugin.get(name, None) == -1:
|
|
|
|
return
|
2010-12-06 23:54:42 +08:00
|
|
|
name = name or getattr(plugin, '__name__', str(id(plugin)))
|
2012-06-17 03:29:04 +08:00
|
|
|
if self.isregistered(plugin, name):
|
2013-09-30 19:14:14 +08:00
|
|
|
raise ValueError("Plugin already registered: %s=%s\n%s" %(
|
|
|
|
name, plugin, self._name2plugin))
|
2010-12-06 23:54:42 +08:00
|
|
|
#self.trace("registering", name, plugin)
|
2013-09-30 19:14:14 +08:00
|
|
|
reg = getattr(self, "_registercallback", None)
|
|
|
|
if reg is not None:
|
2014-10-01 18:20:11 +08:00
|
|
|
reg(plugin, name) # may call addhooks
|
|
|
|
self.hook._scan_plugin(plugin)
|
|
|
|
self._name2plugin[name] = plugin
|
2014-10-01 18:19:11 +08:00
|
|
|
if conftest:
|
|
|
|
self._conftestplugins.append(plugin)
|
2010-10-13 17:12:27 +08:00
|
|
|
else:
|
2014-10-01 18:19:11 +08:00
|
|
|
if not prepend:
|
|
|
|
self._plugins.append(plugin)
|
|
|
|
else:
|
|
|
|
self._plugins.insert(0, plugin)
|
2010-10-12 18:54:32 +08:00
|
|
|
return True
|
|
|
|
|
2010-11-01 16:20:58 +08:00
|
|
|
def unregister(self, plugin=None, name=None):
|
|
|
|
if plugin is None:
|
|
|
|
plugin = self.getplugin(name=name)
|
2014-10-01 18:19:11 +08:00
|
|
|
try:
|
|
|
|
self._plugins.remove(plugin)
|
|
|
|
except KeyError:
|
|
|
|
self._conftestplugins.remove(plugin)
|
2010-10-12 18:54:32 +08:00
|
|
|
for name, value in list(self._name2plugin.items()):
|
|
|
|
if value == plugin:
|
|
|
|
del self._name2plugin[name]
|
|
|
|
|
2013-09-29 04:23:00 +08:00
|
|
|
def add_shutdown(self, func):
|
|
|
|
self._shutdown.append(func)
|
|
|
|
|
|
|
|
def ensure_shutdown(self):
|
|
|
|
while self._shutdown:
|
|
|
|
func = self._shutdown.pop()
|
|
|
|
func()
|
2014-10-01 18:19:11 +08:00
|
|
|
self._plugins = self._conftestplugins = []
|
2013-09-29 04:23:00 +08:00
|
|
|
self._name2plugin.clear()
|
|
|
|
self._listattrcache.clear()
|
|
|
|
|
2010-10-12 18:54:32 +08:00
|
|
|
def isregistered(self, plugin, name=None):
|
2010-12-06 23:54:42 +08:00
|
|
|
if self.getplugin(name) is not None:
|
2010-10-12 18:54:32 +08:00
|
|
|
return True
|
|
|
|
for val in self._name2plugin.values():
|
|
|
|
if plugin == val:
|
|
|
|
return True
|
|
|
|
|
2013-09-30 19:14:14 +08:00
|
|
|
def addhooks(self, spec, prefix="pytest_"):
|
|
|
|
self.hook._addhooks(spec, prefix=prefix)
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
def getplugins(self):
|
2014-10-01 18:19:11 +08:00
|
|
|
return self._plugins + self._conftestplugins
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
def skipifmissing(self, name):
|
|
|
|
if not self.hasplugin(name):
|
2014-07-03 18:58:12 +08:00
|
|
|
import pytest
|
|
|
|
pytest.skip("plugin %r is missing" % name)
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
def hasplugin(self, name):
|
2010-12-07 01:32:04 +08:00
|
|
|
return bool(self.getplugin(name))
|
2008-08-16 23:26:59 +08:00
|
|
|
|
2010-10-12 18:54:32 +08:00
|
|
|
def getplugin(self, name):
|
2010-12-06 23:54:42 +08:00
|
|
|
if name is None:
|
|
|
|
return None
|
2010-10-12 18:54:32 +08:00
|
|
|
try:
|
|
|
|
return self._name2plugin[name]
|
|
|
|
except KeyError:
|
2010-12-06 23:54:42 +08:00
|
|
|
return self._name2plugin.get("_pytest." + name, None)
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
# API for bootstrapping
|
|
|
|
#
|
|
|
|
def _envlist(self, varname):
|
2014-08-01 06:13:40 +08:00
|
|
|
val = os.environ.get(varname, None)
|
2010-10-12 18:54:32 +08:00
|
|
|
if val is not None:
|
|
|
|
return val.split(',')
|
|
|
|
return ()
|
|
|
|
|
|
|
|
def consider_env(self):
|
|
|
|
for spec in self._envlist("PYTEST_PLUGINS"):
|
|
|
|
self.import_plugin(spec)
|
|
|
|
|
|
|
|
def consider_setuptools_entrypoints(self):
|
|
|
|
try:
|
2010-11-14 02:46:28 +08:00
|
|
|
from pkg_resources import iter_entry_points, DistributionNotFound
|
2010-10-12 18:54:32 +08:00
|
|
|
except ImportError:
|
|
|
|
return # XXX issue a warning
|
|
|
|
for ep in iter_entry_points('pytest11'):
|
2010-12-06 23:56:12 +08:00
|
|
|
name = ep.name
|
|
|
|
if name.startswith("pytest_"):
|
|
|
|
name = name[7:]
|
|
|
|
if ep.name in self._name2plugin or name in self._name2plugin:
|
2010-10-12 18:54:32 +08:00
|
|
|
continue
|
2010-11-14 02:46:28 +08:00
|
|
|
try:
|
|
|
|
plugin = ep.load()
|
|
|
|
except DistributionNotFound:
|
|
|
|
continue
|
2011-01-13 02:39:36 +08:00
|
|
|
self._plugin_distinfo.append((ep.dist, plugin))
|
2010-10-12 18:54:32 +08:00
|
|
|
self.register(plugin, name=name)
|
|
|
|
|
|
|
|
def consider_preparse(self, args):
|
|
|
|
for opt1,opt2 in zip(args, args[1:]):
|
|
|
|
if opt1 == "-p":
|
2011-03-17 01:00:52 +08:00
|
|
|
self.consider_pluginarg(opt2)
|
|
|
|
|
|
|
|
def consider_pluginarg(self, arg):
|
|
|
|
if arg.startswith("no:"):
|
|
|
|
name = arg[3:]
|
|
|
|
if self.getplugin(name) is not None:
|
|
|
|
self.unregister(None, name=name)
|
|
|
|
self._name2plugin[name] = -1
|
|
|
|
else:
|
|
|
|
if self.getplugin(arg) is None:
|
|
|
|
self.import_plugin(arg)
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
def consider_conftest(self, conftestmodule):
|
2014-10-01 18:19:11 +08:00
|
|
|
if self.register(conftestmodule, name=conftestmodule.__file__,
|
|
|
|
conftest=True):
|
2010-10-12 18:54:32 +08:00
|
|
|
self.consider_module(conftestmodule)
|
|
|
|
|
|
|
|
def consider_module(self, mod):
|
|
|
|
attr = getattr(mod, "pytest_plugins", ())
|
|
|
|
if attr:
|
|
|
|
if not isinstance(attr, (list, tuple)):
|
|
|
|
attr = (attr,)
|
|
|
|
for spec in attr:
|
|
|
|
self.import_plugin(spec)
|
|
|
|
|
2010-12-06 23:54:42 +08:00
|
|
|
def import_plugin(self, modname):
|
|
|
|
assert isinstance(modname, str)
|
|
|
|
if self.getplugin(modname) is not None:
|
2010-10-12 18:54:32 +08:00
|
|
|
return
|
|
|
|
try:
|
|
|
|
mod = importplugin(modname)
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
raise
|
2010-12-06 23:54:42 +08:00
|
|
|
except ImportError:
|
|
|
|
if modname.startswith("pytest_"):
|
|
|
|
return self.import_plugin(modname[7:])
|
|
|
|
raise
|
2010-10-13 17:12:27 +08:00
|
|
|
except:
|
2014-08-01 06:13:40 +08:00
|
|
|
e = sys.exc_info()[1]
|
2014-07-03 19:20:51 +08:00
|
|
|
import pytest
|
2014-07-03 18:58:12 +08:00
|
|
|
if not hasattr(pytest, 'skip') or not isinstance(e, pytest.skip.Exception):
|
2010-10-12 21:34:32 +08:00
|
|
|
raise
|
2014-03-12 05:10:51 +08:00
|
|
|
self._warnings.append("skipped plugin %r: %s" %((modname, e.msg)))
|
2010-10-12 18:54:32 +08:00
|
|
|
else:
|
|
|
|
self.register(mod, modname)
|
|
|
|
self.consider_module(mod)
|
|
|
|
|
2010-10-13 17:12:27 +08:00
|
|
|
def listattr(self, attrname, plugins=None):
|
|
|
|
if plugins is None:
|
2014-10-01 18:19:11 +08:00
|
|
|
plugins = self._plugins + self._conftestplugins
|
2010-10-13 17:12:27 +08:00
|
|
|
l = []
|
2010-11-22 06:17:59 +08:00
|
|
|
last = []
|
2014-03-14 19:49:35 +08:00
|
|
|
wrappers = []
|
2010-10-13 17:12:27 +08:00
|
|
|
for plugin in plugins:
|
|
|
|
try:
|
2010-11-22 06:17:59 +08:00
|
|
|
meth = getattr(plugin, attrname)
|
2010-10-13 17:12:27 +08:00
|
|
|
except AttributeError:
|
|
|
|
continue
|
2014-03-14 19:49:35 +08:00
|
|
|
if hasattr(meth, 'hookwrapper'):
|
|
|
|
wrappers.append(meth)
|
|
|
|
elif hasattr(meth, 'tryfirst'):
|
|
|
|
last.append(meth)
|
|
|
|
elif hasattr(meth, 'trylast'):
|
|
|
|
l.insert(0, meth)
|
|
|
|
else:
|
|
|
|
l.append(meth)
|
2010-11-22 06:17:59 +08:00
|
|
|
l.extend(last)
|
2014-03-14 19:49:35 +08:00
|
|
|
l.extend(wrappers)
|
2014-10-01 18:19:11 +08:00
|
|
|
#self._listattrcache[key] = list(l)
|
2010-10-13 17:12:27 +08:00
|
|
|
return l
|
|
|
|
|
|
|
|
def call_plugin(self, plugin, methname, kwargs):
|
|
|
|
return MultiCall(methods=self.listattr(methname, plugins=[plugin]),
|
|
|
|
kwargs=kwargs, firstresult=True).execute()
|
|
|
|
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
def importplugin(importspec):
|
2010-12-06 23:54:42 +08:00
|
|
|
name = importspec
|
2010-10-12 18:54:32 +08:00
|
|
|
try:
|
2010-12-06 23:54:42 +08:00
|
|
|
mod = "_pytest." + name
|
2012-04-28 05:51:50 +08:00
|
|
|
__import__(mod)
|
|
|
|
return sys.modules[mod]
|
2010-10-12 18:54:32 +08:00
|
|
|
except ImportError:
|
2013-02-04 23:07:51 +08:00
|
|
|
__import__(importspec)
|
2013-11-19 21:45:51 +08:00
|
|
|
return sys.modules[importspec]
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
class MultiCall:
|
2010-10-13 17:12:27 +08:00
|
|
|
""" execute a call into multiple python functions/methods. """
|
2014-03-14 19:49:35 +08:00
|
|
|
|
|
|
|
class WrongHookWrapper(Exception):
|
|
|
|
""" a hook wrapper does not behave correctly. """
|
|
|
|
def __init__(self, func, message):
|
|
|
|
Exception.__init__(self, func, message)
|
|
|
|
self.func = func
|
|
|
|
self.message = message
|
|
|
|
|
2010-10-12 18:54:32 +08:00
|
|
|
def __init__(self, methods, kwargs, firstresult=False):
|
2010-10-13 17:12:27 +08:00
|
|
|
self.methods = list(methods)
|
|
|
|
self.kwargs = kwargs
|
2010-10-12 18:54:32 +08:00
|
|
|
self.results = []
|
|
|
|
self.firstresult = firstresult
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
status = "%d results, %d meths" % (len(self.results), len(self.methods))
|
|
|
|
return "<MultiCall %s, kwargs=%r>" %(status, self.kwargs)
|
|
|
|
|
|
|
|
def execute(self):
|
2014-03-14 19:49:35 +08:00
|
|
|
next_finalizers = []
|
|
|
|
try:
|
|
|
|
while self.methods:
|
|
|
|
method = self.methods.pop()
|
|
|
|
kwargs = self.getkwargs(method)
|
|
|
|
if hasattr(method, "hookwrapper"):
|
|
|
|
it = method(**kwargs)
|
|
|
|
next = getattr(it, "next", None)
|
|
|
|
if next is None:
|
|
|
|
next = getattr(it, "__next__", None)
|
|
|
|
if next is None:
|
|
|
|
raise self.WrongHookWrapper(method,
|
|
|
|
"wrapper does not contain a yield")
|
|
|
|
res = next()
|
|
|
|
next_finalizers.append((method, next))
|
|
|
|
else:
|
|
|
|
res = method(**kwargs)
|
|
|
|
if res is not None:
|
|
|
|
self.results.append(res)
|
|
|
|
if self.firstresult:
|
|
|
|
return res
|
|
|
|
if not self.firstresult:
|
|
|
|
return self.results
|
|
|
|
finally:
|
|
|
|
for method, fin in reversed(next_finalizers):
|
|
|
|
try:
|
|
|
|
fin()
|
|
|
|
except StopIteration:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
raise self.WrongHookWrapper(method,
|
|
|
|
"wrapper contain more than one yield")
|
|
|
|
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
def getkwargs(self, method):
|
|
|
|
kwargs = {}
|
|
|
|
for argname in varnames(method):
|
|
|
|
try:
|
|
|
|
kwargs[argname] = self.kwargs[argname]
|
|
|
|
except KeyError:
|
2010-10-13 17:12:27 +08:00
|
|
|
if argname == "__multicall__":
|
|
|
|
kwargs[argname] = self
|
2010-10-12 18:54:32 +08:00
|
|
|
return kwargs
|
|
|
|
|
2010-10-26 05:08:56 +08:00
|
|
|
def varnames(func):
|
2013-11-19 22:33:52 +08:00
|
|
|
""" return argument name tuple for a function, method, class or callable.
|
|
|
|
|
|
|
|
In case of a class, its "__init__" method is considered.
|
|
|
|
For methods the "self" parameter is not included unless you are passing
|
|
|
|
an unbound method with Python3 (which has no supports for unbound methods)
|
|
|
|
"""
|
|
|
|
cache = getattr(func, "__dict__", {})
|
2011-03-08 01:28:45 +08:00
|
|
|
try:
|
2013-11-19 22:33:52 +08:00
|
|
|
return cache["_varnames"]
|
|
|
|
except KeyError:
|
2011-03-08 01:28:45 +08:00
|
|
|
pass
|
2013-11-19 22:33:52 +08:00
|
|
|
if inspect.isclass(func):
|
|
|
|
try:
|
|
|
|
func = func.__init__
|
|
|
|
except AttributeError:
|
|
|
|
return ()
|
|
|
|
ismethod = True
|
|
|
|
else:
|
|
|
|
if not inspect.isfunction(func) and not inspect.ismethod(func):
|
|
|
|
func = getattr(func, '__call__', func)
|
|
|
|
ismethod = inspect.ismethod(func)
|
2010-10-12 18:54:32 +08:00
|
|
|
rawcode = py.code.getrawcode(func)
|
|
|
|
try:
|
2011-03-08 01:28:45 +08:00
|
|
|
x = rawcode.co_varnames[ismethod:rawcode.co_argcount]
|
2010-10-12 18:54:32 +08:00
|
|
|
except AttributeError:
|
2011-03-08 01:28:45 +08:00
|
|
|
x = ()
|
2013-11-19 22:33:52 +08:00
|
|
|
try:
|
|
|
|
cache["_varnames"] = x
|
|
|
|
except TypeError:
|
|
|
|
pass
|
2011-03-08 01:28:45 +08:00
|
|
|
return x
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
class HookRelay:
|
2010-10-13 17:12:27 +08:00
|
|
|
def __init__(self, hookspecs, pm, prefix="pytest_"):
|
2010-10-12 18:54:32 +08:00
|
|
|
if not isinstance(hookspecs, list):
|
|
|
|
hookspecs = [hookspecs]
|
|
|
|
self._hookspecs = []
|
2010-10-13 17:12:27 +08:00
|
|
|
self._pm = pm
|
2010-11-06 16:05:17 +08:00
|
|
|
self.trace = pm.trace.root.get("hook")
|
2010-10-12 18:54:32 +08:00
|
|
|
for hookspec in hookspecs:
|
|
|
|
self._addhooks(hookspec, prefix)
|
|
|
|
|
|
|
|
def _addhooks(self, hookspecs, prefix):
|
|
|
|
self._hookspecs.append(hookspecs)
|
|
|
|
added = False
|
|
|
|
for name, method in vars(hookspecs).items():
|
|
|
|
if name.startswith(prefix):
|
|
|
|
firstresult = getattr(method, 'firstresult', False)
|
|
|
|
hc = HookCaller(self, name, firstresult=firstresult)
|
|
|
|
setattr(self, name, hc)
|
|
|
|
added = True
|
|
|
|
#print ("setting new hook", name)
|
|
|
|
if not added:
|
|
|
|
raise ValueError("did not find new %r hooks in %r" %(
|
|
|
|
prefix, hookspecs,))
|
|
|
|
|
2014-10-01 18:19:11 +08:00
|
|
|
def _getcaller(self, name, plugins):
|
|
|
|
caller = getattr(self, name)
|
|
|
|
methods = self._pm.listattr(name, plugins=plugins)
|
|
|
|
if methods:
|
2014-10-01 18:20:11 +08:00
|
|
|
return caller.new_cached_caller(methods)
|
|
|
|
return caller
|
|
|
|
|
|
|
|
def _scan_plugin(self, plugin):
|
|
|
|
methods = collectattr(plugin)
|
|
|
|
hooks = {}
|
|
|
|
for hookspec in self._hookspecs:
|
|
|
|
hooks.update(collectattr(hookspec))
|
|
|
|
|
|
|
|
stringio = py.io.TextIO()
|
|
|
|
def Print(*args):
|
|
|
|
if args:
|
|
|
|
stringio.write(" ".join(map(str, args)))
|
|
|
|
stringio.write("\n")
|
|
|
|
|
|
|
|
fail = False
|
|
|
|
while methods:
|
|
|
|
name, method = methods.popitem()
|
|
|
|
#print "checking", name
|
|
|
|
if isgenerichook(name):
|
|
|
|
continue
|
|
|
|
if name not in hooks:
|
|
|
|
if not getattr(method, 'optionalhook', False):
|
|
|
|
Print("found unknown hook:", name)
|
|
|
|
fail = True
|
|
|
|
else:
|
|
|
|
#print "checking", method
|
|
|
|
method_args = list(varnames(method))
|
|
|
|
if '__multicall__' in method_args:
|
|
|
|
method_args.remove('__multicall__')
|
|
|
|
hook = hooks[name]
|
|
|
|
hookargs = varnames(hook)
|
|
|
|
for arg in method_args:
|
|
|
|
if arg not in hookargs:
|
|
|
|
Print("argument %r not available" %(arg, ))
|
|
|
|
Print("actual definition: %s" %(formatdef(method)))
|
|
|
|
Print("available hook arguments: %s" %
|
|
|
|
", ".join(hookargs))
|
|
|
|
fail = True
|
|
|
|
break
|
|
|
|
#if not fail:
|
|
|
|
# print "matching hook:", formatdef(method)
|
|
|
|
getattr(self, name).clear_method_cache()
|
|
|
|
|
|
|
|
if fail:
|
|
|
|
name = getattr(plugin, '__name__', plugin)
|
|
|
|
raise PluginValidationError("%s:\n%s" % (name, stringio.getvalue()))
|
2014-10-01 18:19:11 +08:00
|
|
|
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
class HookCaller:
|
2014-10-01 18:20:11 +08:00
|
|
|
def __init__(self, hookrelay, name, firstresult, methods=None):
|
2010-10-12 18:54:32 +08:00
|
|
|
self.hookrelay = hookrelay
|
|
|
|
self.name = name
|
|
|
|
self.firstresult = firstresult
|
2010-11-06 16:58:04 +08:00
|
|
|
self.trace = self.hookrelay.trace
|
2014-10-01 18:20:11 +08:00
|
|
|
self.methods = methods
|
|
|
|
|
|
|
|
def new_cached_caller(self, methods):
|
|
|
|
return HookCaller(self.hookrelay, self.name, self.firstresult,
|
|
|
|
methods=methods)
|
2010-10-12 18:54:32 +08:00
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return "<HookCaller %r>" %(self.name,)
|
2009-03-06 05:10:18 +08:00
|
|
|
|
2014-10-01 18:20:11 +08:00
|
|
|
def clear_method_cache(self):
|
|
|
|
self.methods = None
|
|
|
|
|
2010-10-12 18:54:32 +08:00
|
|
|
def __call__(self, **kwargs):
|
2014-10-01 18:20:11 +08:00
|
|
|
methods = self.methods
|
|
|
|
if self.methods is None:
|
|
|
|
self.methods = methods = self.hookrelay._pm.listattr(self.name)
|
|
|
|
methods = self.methods
|
2010-11-07 02:46:24 +08:00
|
|
|
return self._docall(methods, kwargs)
|
2010-10-07 17:51:58 +08:00
|
|
|
|
2014-10-01 18:20:11 +08:00
|
|
|
def callextra(self, methods, **kwargs):
|
|
|
|
if self.methods is None:
|
|
|
|
self.reload_methods()
|
|
|
|
return self._docall(self.methods + methods, kwargs)
|
|
|
|
|
2010-11-07 02:46:24 +08:00
|
|
|
def _docall(self, methods, kwargs):
|
2010-11-06 16:58:04 +08:00
|
|
|
self.trace(self.name, kwargs)
|
|
|
|
self.trace.root.indent += 1
|
2010-10-12 18:54:32 +08:00
|
|
|
mc = MultiCall(methods, kwargs, firstresult=self.firstresult)
|
2010-11-07 03:06:32 +08:00
|
|
|
try:
|
|
|
|
res = mc.execute()
|
|
|
|
if res:
|
2010-11-07 03:12:45 +08:00
|
|
|
self.trace("finish", self.name, "-->", res)
|
2010-11-07 03:06:32 +08:00
|
|
|
finally:
|
|
|
|
self.trace.root.indent -= 1
|
2010-11-06 16:58:04 +08:00
|
|
|
return res
|
2010-10-07 17:51:58 +08:00
|
|
|
|
2014-10-01 18:20:11 +08:00
|
|
|
|
|
|
|
|
|
|
|
class PluginValidationError(Exception):
|
|
|
|
""" plugin failed validation. """
|
|
|
|
|
|
|
|
def isgenerichook(name):
|
|
|
|
return name == "pytest_plugins" or \
|
|
|
|
name.startswith("pytest_funcarg__")
|
|
|
|
|
|
|
|
def collectattr(obj):
|
|
|
|
methods = {}
|
|
|
|
for apiname in dir(obj):
|
|
|
|
if apiname.startswith("pytest_"):
|
|
|
|
methods[apiname] = getattr(obj, apiname)
|
|
|
|
return methods
|
|
|
|
|
|
|
|
def formatdef(func):
|
|
|
|
return "%s%s" % (
|
|
|
|
func.__name__,
|
|
|
|
inspect.formatargspec(*inspect.getargspec(func))
|
|
|
|
)
|
|
|
|
|