import sys, os import inspect import py from pytest import hookspec assert py.__version__.split(".")[:2] >= ['2', '0'], ("installation problem: " "%s is too old, remove or upgrade 'py'" % (py.__version__)) default_plugins = ( "config session terminal python runner pdb capture unittest mark skipping " "tmpdir monkeypatch recwarn pastebin helpconfig nose assertion genscript " "junitxml doctest").split() IMPORTPREFIX = "pytest_" class PluginManager(object): def __init__(self, load=False): self._name2plugin = {} self._plugins = [] self._hints = [] self.hook = HookRelay([hookspec], pm=self) self.register(self) if load: for spec in default_plugins: self.import_plugin(spec) def _getpluginname(self, plugin, name): if name is None: if hasattr(plugin, '__name__'): name = plugin.__name__.split(".")[-1] else: name = id(plugin) return name def register(self, plugin, name=None, prepend=False): assert not self.isregistered(plugin), plugin assert not self.isregistered(plugin), plugin name = self._getpluginname(plugin, name) if name in self._name2plugin: return False self._name2plugin[name] = plugin self.call_plugin(plugin, "pytest_addhooks", {'pluginmanager': self}) self.hook.pytest_plugin_registered(manager=self, plugin=plugin) if not prepend: self._plugins.append(plugin) else: self._plugins.insert(0, plugin) return True def unregister(self, plugin): self.hook.pytest_plugin_unregistered(plugin=plugin) self._plugins.remove(plugin) for name, value in list(self._name2plugin.items()): if value == plugin: del self._name2plugin[name] def isregistered(self, plugin, name=None): if self._getpluginname(plugin, name) in self._name2plugin: return True for val in self._name2plugin.values(): if plugin == val: return True def addhooks(self, spec): self.hook._addhooks(spec, prefix="pytest_") def getplugins(self): return list(self._plugins) def skipifmissing(self, name): if not self.hasplugin(name): py.test.skip("plugin %r is missing" % name) def hasplugin(self, name): try: self.getplugin(name) return True except KeyError: return False def getplugin(self, name): try: return self._name2plugin[name] except KeyError: impname = canonical_importname(name) return self._name2plugin[impname] # API for bootstrapping # def _envlist(self, varname): val = py.std.os.environ.get(varname, None) 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: from pkg_resources import iter_entry_points except ImportError: return # XXX issue a warning for ep in iter_entry_points('pytest11'): name = canonical_importname(ep.name) if name in self._name2plugin: continue plugin = ep.load() self.register(plugin, name=name) def consider_preparse(self, args): for opt1,opt2 in zip(args, args[1:]): if opt1 == "-p": self.import_plugin(opt2) def consider_conftest(self, conftestmodule): cls = getattr(conftestmodule, 'ConftestPlugin', None) if cls is not None: raise ValueError("%r: 'ConftestPlugins' only existed till 1.0.0b1, " "were removed in 1.0.0b2" % (cls,)) if self.register(conftestmodule, name=conftestmodule.__file__): 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) def import_plugin(self, spec): assert isinstance(spec, str) modname = canonical_importname(spec) if modname in self._name2plugin: return try: mod = importplugin(modname) except KeyboardInterrupt: raise except: e = py.std.sys.exc_info()[1] if not hasattr(py.test, 'skip'): raise elif not isinstance(e, py.test.skip.Exception): raise self._hints.append("skipped plugin %r: %s" %((modname, e.msg))) else: self.register(mod, modname) self.consider_module(mod) def pytest_plugin_registered(self, plugin): dic = self.call_plugin(plugin, "pytest_namespace", {}) or {} if dic: self._setns(py.test, dic) if hasattr(self, '_config'): self.call_plugin(plugin, "pytest_addoption", {'parser': self._config._parser}) self.call_plugin(plugin, "pytest_configure", {'config': self._config}) def _setns(self, obj, dic): for name, value in dic.items(): if isinstance(value, dict): mod = getattr(obj, name, None) if mod is None: mod = py.std.types.ModuleType(name) sys.modules['pytest.%s' % name] = mod sys.modules['py.test.%s' % name] = mod mod.__all__ = [] setattr(obj, name, mod) self._setns(mod, value) else: #print "setting", name, value, "on", obj setattr(obj, name, value) obj.__all__.append(name) def pytest_terminal_summary(self, terminalreporter): tw = terminalreporter._tw if terminalreporter.config.option.traceconfig: for hint in self._hints: tw.line("hint: %s" % hint) def do_addoption(self, parser): mname = "pytest_addoption" methods = reversed(self.listattr(mname)) MultiCall(methods, {'parser': parser}).execute() def do_configure(self, config): assert not hasattr(self, '_config') self._config = config config.hook.pytest_configure(config=self._config) def do_unconfigure(self, config): config = self._config del self._config config.hook.pytest_unconfigure(config=config) config.pluginmanager.unregister(self) def notify_exception(self, excinfo): excrepr = excinfo.getrepr(funcargs=True, showlocals=True) res = self.hook.pytest_internalerror(excrepr=excrepr) if not py.builtin.any(res): for line in str(excrepr).split("\n"): sys.stderr.write("INTERNALERROR> %s\n" %line) sys.stderr.flush() def listattr(self, attrname, plugins=None): if plugins is None: plugins = self._plugins l = [] for plugin in plugins: try: l.append(getattr(plugin, attrname)) except AttributeError: continue return l def call_plugin(self, plugin, methname, kwargs): return MultiCall(methods=self.listattr(methname, plugins=[plugin]), kwargs=kwargs, firstresult=True).execute() def canonical_importname(name): name = name.lower() if not name.startswith(IMPORTPREFIX): name = IMPORTPREFIX + name return name def importplugin(importspec): #print "importing", importspec try: return __import__(importspec) except ImportError: e = py.std.sys.exc_info()[1] if str(e).find(importspec) == -1: raise name = importspec try: if name.startswith("pytest_"): name = importspec[7:] return __import__("pytest.plugin.%s" %(name), None, None, '__doc__') except ImportError: e = py.std.sys.exc_info()[1] if str(e).find(name) == -1: raise # show the original exception, not the failing internal one return __import__(importspec) class MultiCall: """ execute a call into multiple python functions/methods. """ def __init__(self, methods, kwargs, firstresult=False): self.methods = list(methods) self.kwargs = kwargs self.results = [] self.firstresult = firstresult def __repr__(self): status = "%d results, %d meths" % (len(self.results), len(self.methods)) return "" %(status, self.kwargs) def execute(self): while self.methods: method = self.methods.pop() kwargs = self.getkwargs(method) res = method(**kwargs) if res is not None: self.results.append(res) if self.firstresult: return res if not self.firstresult: return self.results def getkwargs(self, method): kwargs = {} for argname in varnames(method): try: kwargs[argname] = self.kwargs[argname] except KeyError: if argname == "__multicall__": kwargs[argname] = self return kwargs def varnames(func): if not inspect.isfunction(func) and not inspect.ismethod(func): func = getattr(func, '__call__', func) ismethod = inspect.ismethod(func) rawcode = py.code.getrawcode(func) try: return rawcode.co_varnames[ismethod:rawcode.co_argcount] except AttributeError: return () class HookRelay: def __init__(self, hookspecs, pm, prefix="pytest_"): if not isinstance(hookspecs, list): hookspecs = [hookspecs] self._hookspecs = [] self._pm = pm 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): if not method.__doc__: raise ValueError("docstring required for hook %r, in %r" % (method, hookspecs)) 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,)) class HookCaller: def __init__(self, hookrelay, name, firstresult): self.hookrelay = hookrelay self.name = name self.firstresult = firstresult def __repr__(self): return "" %(self.name,) def __call__(self, **kwargs): methods = self.hookrelay._pm.listattr(self.name) mc = MultiCall(methods, kwargs, firstresult=self.firstresult) return mc.execute() def pcall(self, plugins, **kwargs): methods = self.hookrelay._pm.listattr(self.name, plugins=plugins) mc = MultiCall(methods, kwargs, firstresult=self.firstresult) return mc.execute() pluginmanager = PluginManager(load=True) # will trigger default plugin importing def main(args=None): global pluginmanager if args is None: args = sys.argv[1:] hook = pluginmanager.hook config = hook.pytest_cmdline_parse(pluginmanager=pluginmanager, args=args) try: exitstatus = hook.pytest_cmdline_main(config=config) except config.Error: e = sys.exc_info()[1] sys.stderr.write("ERROR: %s\n" %(e.args[0],)) exitstatus = 3 pluginmanager = PluginManager(load=True) return exitstatus