simplify plugins bookkeeping further, refine API

--HG--
branch : more_plugin
This commit is contained in:
holger krekel 2015-04-25 20:17:32 +02:00
parent 1c0582eaa7
commit 3a1374e69c
6 changed files with 144 additions and 131 deletions

View File

@ -127,13 +127,17 @@ class PytestPluginManager(PluginManager):
kwargs=dict(plugin=plugin, manager=self)) kwargs=dict(plugin=plugin, manager=self))
return ret return ret
def unregister(self, plugin): def unregister(self, plugin=None, name=None):
super(PytestPluginManager, self).unregister(plugin) plugin = super(PytestPluginManager, self).unregister(plugin, name)
try: try:
self._globalplugins.remove(plugin) self._globalplugins.remove(plugin)
except ValueError: except ValueError:
pass pass
def getplugin(self, name):
# deprecated
return self.get_plugin(name)
def pytest_configure(self, config): def pytest_configure(self, config):
config.addinivalue_line("markers", config.addinivalue_line("markers",
"tryfirst: mark a hook implementation function such that the " "tryfirst: mark a hook implementation function such that the "
@ -238,17 +242,14 @@ class PytestPluginManager(PluginManager):
except ImportError: except ImportError:
return # XXX issue a warning return # XXX issue a warning
for ep in iter_entry_points('pytest11'): for ep in iter_entry_points('pytest11'):
name = ep.name if self.get_plugin(ep.name) or ep.name in self._name2plugin:
if name.startswith("pytest_"):
name = name[7:]
if ep.name in self._name2plugin or name in self._name2plugin:
continue continue
try: try:
plugin = ep.load() plugin = ep.load()
except DistributionNotFound: except DistributionNotFound:
continue continue
self.register(plugin, name=ep.name)
self._plugin_distinfo.append((ep.dist, plugin)) self._plugin_distinfo.append((ep.dist, plugin))
self.register(plugin, name=name)
def consider_preparse(self, args): def consider_preparse(self, args):
for opt1,opt2 in zip(args, args[1:]): for opt1,opt2 in zip(args, args[1:]):
@ -257,13 +258,8 @@ class PytestPluginManager(PluginManager):
def consider_pluginarg(self, arg): def consider_pluginarg(self, arg):
if arg.startswith("no:"): if arg.startswith("no:"):
name = arg[3:] self.set_blocked(arg[3:])
plugin = self.getplugin(name)
if plugin is not None:
self.unregister(plugin)
self._name2plugin[name] = -1
else: else:
if self.getplugin(arg) is None:
self.import_plugin(arg) self.import_plugin(arg)
def consider_conftest(self, conftestmodule): def consider_conftest(self, conftestmodule):
@ -290,7 +286,7 @@ class PytestPluginManager(PluginManager):
# basename for historic purposes but must be imported with the # basename for historic purposes but must be imported with the
# _pytest prefix. # _pytest prefix.
assert isinstance(modname, str) assert isinstance(modname, str)
if self.getplugin(modname) is not None: if self.get_plugin(modname) is not None:
return return
if modname in builtin_plugins: if modname in builtin_plugins:
importspec = "_pytest." + modname importspec = "_pytest." + modname
@ -736,7 +732,7 @@ class Config(object):
fslocation=None, nodeid=None) fslocation=None, nodeid=None)
def get_terminal_writer(self): def get_terminal_writer(self):
return self.pluginmanager.getplugin("terminalreporter")._tw return self.pluginmanager.get_plugin("terminalreporter")._tw
def pytest_cmdline_parse(self, pluginmanager, args): def pytest_cmdline_parse(self, pluginmanager, args):
# REF1 assert self == pluginmanager.config, (self, pluginmanager.config) # REF1 assert self == pluginmanager.config, (self, pluginmanager.config)

View File

@ -112,10 +112,13 @@ class TagTracerSub:
def __init__(self, root, tags): def __init__(self, root, tags):
self.root = root self.root = root
self.tags = tags self.tags = tags
def __call__(self, *args): def __call__(self, *args):
self.root.processmessage(self.tags, args) self.root.processmessage(self.tags, args)
def setmyprocessor(self, processor): def setmyprocessor(self, processor):
self.root.setprocessor(self.tags, processor) self.root.setprocessor(self.tags, processor)
def get(self, name): def get(self, name):
return self.__class__(self.root, self.tags + (name,)) return self.__class__(self.root, self.tags + (name,))
@ -125,6 +128,7 @@ def raise_wrapfail(wrap_controller, msg):
raise RuntimeError("wrap_controller at %r %s:%d %s" % raise RuntimeError("wrap_controller at %r %s:%d %s" %
(co.co_name, co.co_filename, co.co_firstlineno, msg)) (co.co_name, co.co_filename, co.co_firstlineno, msg))
def wrapped_call(wrap_controller, func): def wrapped_call(wrap_controller, func):
""" Wrap calling to a function with a generator which needs to yield """ Wrap calling to a function with a generator which needs to yield
exactly once. The yield point will trigger calling the wrapped function exactly once. The yield point will trigger calling the wrapped function
@ -208,7 +212,6 @@ class PluginManager(object):
self._prefix = prefix self._prefix = prefix
self._excludefunc = excludefunc self._excludefunc = excludefunc
self._name2plugin = {} self._name2plugin = {}
self._plugins = []
self._plugin2hookcallers = {} self._plugin2hookcallers = {}
self.trace = TagTracer().get("pluginmanage") self.trace = TagTracer().get("pluginmanage")
self.hook = HookRelay(self.trace.root.get("hook")) self.hook = HookRelay(self.trace.root.get("hook"))
@ -244,22 +247,25 @@ class PluginManager(object):
self._plugin2hookcallers.setdefault(plugin, []).append(hc) self._plugin2hookcallers.setdefault(plugin, []).append(hc)
return hc return hc
def register(self, plugin, name=None): def get_canonical_name(self, plugin):
""" Register a plugin with the given name and ensure that all its """ Return canonical name for the plugin object. """
hook implementations are integrated. If the name is not specified return getattr(plugin, "__name__", None) or str(id(plugin))
we use the ``__name__`` attribute of the plugin object or, if that
doesn't exist, the id of the plugin. This method will raise a
ValueError if the eventual name is already registered. """
name = name or self._get_canonical_name(plugin)
if self._name2plugin.get(name, None) == -1:
return
if self.hasplugin(name):
raise ValueError("Plugin already registered: %s=%s\n%s" %(
name, plugin, self._name2plugin))
self._name2plugin[name] = plugin
self._plugins.append(plugin)
# register prefix-matching hooks of the plugin def register(self, plugin, name=None):
""" Register a plugin and return its canonical name or None if it was
blocked from registering. Raise a ValueError if the plugin is already
registered. """
plugin_name = name or self.get_canonical_name(plugin)
if plugin_name in self._name2plugin or plugin in self._plugin2hookcallers:
if self._name2plugin.get(plugin_name, -1) is None:
return # blocked plugin, return None to indicate no registration
raise ValueError("Plugin already registered: %s=%s\n%s" %(
plugin_name, plugin, self._name2plugin))
self._name2plugin[plugin_name] = plugin
# register prefix-matching hook specs of the plugin
self._plugin2hookcallers[plugin] = hookcallers = [] self._plugin2hookcallers[plugin] = hookcallers = []
for name in dir(plugin): for name in dir(plugin):
if name.startswith(self._prefix): if name.startswith(self._prefix):
@ -274,18 +280,33 @@ class PluginManager(object):
hook._maybe_apply_history(getattr(plugin, name)) hook._maybe_apply_history(getattr(plugin, name))
hookcallers.append(hook) hookcallers.append(hook)
hook._add_plugin(plugin) hook._add_plugin(plugin)
return True return plugin_name
def unregister(self, plugin): def unregister(self, plugin=None, name=None):
""" unregister the plugin object and all its contained hook implementations """ unregister a plugin object and all its contained hook implementations
from internal data structures. """ from internal data structures. One of ``plugin`` or ``name`` needs to
self._plugins.remove(plugin) be specified. """
for name, value in list(self._name2plugin.items()): if name is None:
if value == plugin: assert plugin is not None
name = self.get_canonical_name(plugin)
if plugin is None:
plugin = self.get_plugin(name)
# None signals blocked registrations, don't delete it
if self._name2plugin.get(name):
del self._name2plugin[name] del self._name2plugin[name]
for hookcaller in self._plugin2hookcallers.pop(plugin):
for hookcaller in self._plugin2hookcallers.pop(plugin, []):
hookcaller._remove_plugin(plugin) hookcaller._remove_plugin(plugin)
return plugin
def set_blocked(self, name):
""" block registrations of the given name, unregister if already registered. """
self.unregister(name=name)
self._name2plugin[name] = None
def addhooks(self, module_or_class): def addhooks(self, module_or_class):
""" add new hook definitions from the given module_or_class using """ add new hook definitions from the given module_or_class using
the prefix/excludefunc with which the PluginManager was initialized. """ the prefix/excludefunc with which the PluginManager was initialized. """
@ -302,33 +323,27 @@ class PluginManager(object):
for plugin in hc._plugins: for plugin in hc._plugins:
self._verify_hook(hc, plugin) self._verify_hook(hc, plugin)
names.append(name) names.append(name)
if not names: if not names:
raise ValueError("did not find new %r hooks in %r" raise ValueError("did not find new %r hooks in %r"
%(self._prefix, module_or_class)) %(self._prefix, module_or_class))
def getplugins(self): def get_plugins(self):
""" return the complete list of registered plugins. NOTE that """ return the set of registered plugins. """
you will get the internal list and need to make a copy if you return set(self._plugin2hookcallers)
modify the list."""
return self._plugins
def isregistered(self, plugin): def is_registered(self, plugin):
""" Return True if the plugin is already registered under its """ Return True if the plugin is already registered. """
canonical name. """ return plugin in self._plugin2hookcallers
return self.hasplugin(self._get_canonical_name(plugin)) or \
plugin in self._plugins
def hasplugin(self, name): def get_plugin(self, name):
""" Return True if there is a registered with the given name. """
return name in self._name2plugin
def getplugin(self, name):
""" Return a plugin or None for the given name. """ """ Return a plugin or None for the given name. """
return self._name2plugin.get(name) return self._name2plugin.get(name)
def _verify_hook(self, hook, plugin): def _verify_hook(self, hook, plugin):
method = getattr(plugin, hook.name) method = getattr(plugin, hook.name)
pluginname = self._get_canonical_name(plugin) pluginname = self.get_canonical_name(plugin)
if hook.is_historic() and hasattr(method, "hookwrapper"): if hook.is_historic() and hasattr(method, "hookwrapper"):
raise PluginValidationError( raise PluginValidationError(
"Plugin %r\nhook %r\nhistoric incompatible to hookwrapper" %( "Plugin %r\nhook %r\nhistoric incompatible to hookwrapper" %(
@ -344,6 +359,8 @@ class PluginManager(object):
", ".join(hook.argnames))) ", ".join(hook.argnames)))
def check_pending(self): def check_pending(self):
""" Verify that all hooks which have not been verified against
a hook specification are optional, otherwise raise PluginValidationError"""
for name in self.hook.__dict__: for name in self.hook.__dict__:
if name.startswith(self._prefix): if name.startswith(self._prefix):
hook = getattr(self.hook, name) hook = getattr(self.hook, name)
@ -354,10 +371,6 @@ class PluginManager(object):
raise PluginValidationError( raise PluginValidationError(
"unknown hook %r in plugin %r" %(name, plugin)) "unknown hook %r in plugin %r" %(name, plugin))
def _get_canonical_name(self, plugin):
return getattr(plugin, "__name__", None) or str(id(plugin))
class MultiCall: class MultiCall:
""" execute a call into multiple python functions/methods. """ """ execute a call into multiple python functions/methods. """

View File

@ -11,7 +11,7 @@ import subprocess
import py import py
import pytest import pytest
from py.builtin import print_ from py.builtin import print_
from _pytest.core import HookCaller, TracedHookExecution from _pytest.core import TracedHookExecution
from _pytest.main import Session, EXIT_OK from _pytest.main import Session, EXIT_OK

View File

@ -313,7 +313,7 @@ def test_plugin_preparse_prevents_setuptools_loading(testdir, monkeypatch):
monkeypatch.setattr(pkg_resources, 'iter_entry_points', my_iter) monkeypatch.setattr(pkg_resources, 'iter_entry_points', my_iter)
config = testdir.parseconfig("-p", "no:mytestplugin") config = testdir.parseconfig("-p", "no:mytestplugin")
plugin = config.pluginmanager.getplugin("mytestplugin") plugin = config.pluginmanager.getplugin("mytestplugin")
assert plugin == -1 assert plugin is None
def test_cmdline_processargs_simple(testdir): def test_cmdline_processargs_simple(testdir):
testdir.makeconftest(""" testdir.makeconftest("""

View File

@ -17,20 +17,34 @@ class TestPluginManager:
pm.register(42, name="abc") pm.register(42, name="abc")
with pytest.raises(ValueError): with pytest.raises(ValueError):
pm.register(42, name="abc") pm.register(42, name="abc")
with pytest.raises(ValueError):
pm.register(42, name="def")
def test_pm(self, pm): def test_pm(self, pm):
class A: pass class A: pass
a1, a2 = A(), A() a1, a2 = A(), A()
pm.register(a1) pm.register(a1)
assert pm.isregistered(a1) assert pm.is_registered(a1)
pm.register(a2, "hello") pm.register(a2, "hello")
assert pm.isregistered(a2) assert pm.is_registered(a2)
l = pm.getplugins() l = pm.get_plugins()
assert a1 in l assert a1 in l
assert a2 in l assert a2 in l
assert pm.getplugin('hello') == a2 assert pm.get_plugin('hello') == a2
pm.unregister(a1) assert pm.unregister(a1) == a1
assert not pm.isregistered(a1) assert not pm.is_registered(a1)
def test_set_blocked(self, pm):
class A: pass
a1 = A()
name = pm.register(a1)
assert pm.is_registered(a1)
pm.set_blocked(name)
assert not pm.is_registered(a1)
pm.set_blocked("somename")
assert not pm.register(A(), "somename")
pm.unregister(name="somename")
def test_register_mismatch_method(self, pytestpm): def test_register_mismatch_method(self, pytestpm):
class hello: class hello:
@ -53,29 +67,16 @@ class TestPluginManager:
pass pass
my = MyPlugin() my = MyPlugin()
pm.register(my) pm.register(my)
assert pm.getplugins() assert pm.get_plugins()
my2 = MyPlugin() my2 = MyPlugin()
pm.register(my2) pm.register(my2)
assert pm.getplugins()[-2:] == [my, my2] assert set([my,my2]).issubset(pm.get_plugins())
assert pm.isregistered(my) assert pm.is_registered(my)
assert pm.isregistered(my2) assert pm.is_registered(my2)
pm.unregister(my) pm.unregister(my)
assert not pm.isregistered(my) assert not pm.is_registered(my)
assert pm.getplugins()[-1:] == [my2] assert my not in pm.get_plugins()
def test_register_unknown_hooks(self, pm):
class Plugin1:
def he_method1(self, arg):
return arg + 1
pm.register(Plugin1())
class Hooks:
def he_method1(self, arg):
pass
pm.addhooks(Hooks)
#assert not pm._unverified_hooks
assert pm.hook.he_method1(arg=1) == [2]
def test_register_unknown_hooks(self, pm): def test_register_unknown_hooks(self, pm):
class Plugin1: class Plugin1:
@ -231,6 +232,26 @@ class TestAddMethodOrdering:
pass pass
assert hc._nonwrappers == [he_method1, he_method1_middle, he_method1_b] assert hc._nonwrappers == [he_method1, he_method1_middle, he_method1_b]
def test_adding_nonwrappers_trylast3(self, hc, addmeth):
@addmeth()
def he_method1_a():
pass
@addmeth(trylast=True)
def he_method1_b():
pass
@addmeth()
def he_method1_c():
pass
@addmeth(trylast=True)
def he_method1_d():
pass
assert hc._nonwrappers == [he_method1_d, he_method1_b,
he_method1_a, he_method1_c]
def test_adding_nonwrappers_trylast2(self, hc, addmeth): def test_adding_nonwrappers_trylast2(self, hc, addmeth):
@addmeth() @addmeth()
def he_method1_middle(): def he_method1_middle():
@ -259,24 +280,6 @@ class TestAddMethodOrdering:
pass pass
assert hc._nonwrappers == [he_method1_middle, he_method1_b, he_method1] assert hc._nonwrappers == [he_method1_middle, he_method1_b, he_method1]
def test_adding_nonwrappers_trylast(self, hc, addmeth):
@addmeth()
def he_method1_a():
pass
@addmeth(trylast=True)
def he_method1_b():
pass
@addmeth()
def he_method1_c():
pass
@addmeth(trylast=True)
def he_method1_d():
pass
assert hc._nonwrappers == [he_method1_d, he_method1_b, he_method1_a, he_method1_c]
def test_adding_wrappers_ordering(self, hc, addmeth): def test_adding_wrappers_ordering(self, hc, addmeth):
@addmeth(hookwrapper=True) @addmeth(hookwrapper=True)
def he_method1(): def he_method1():
@ -361,7 +364,7 @@ class TestPytestPluginInteractions:
pm.hook.pytest_addhooks.call_historic( pm.hook.pytest_addhooks.call_historic(
kwargs=dict(pluginmanager=config.pluginmanager)) kwargs=dict(pluginmanager=config.pluginmanager))
config.pluginmanager._importconftest(conf) config.pluginmanager._importconftest(conf)
#print(config.pluginmanager.getplugins()) #print(config.pluginmanager.get_plugins())
res = config.hook.pytest_myhook(xyz=10) res = config.hook.pytest_myhook(xyz=10)
assert res == [11] assert res == [11]
@ -814,21 +817,21 @@ class TestPytestPluginManager:
pm = PytestPluginManager() pm = PytestPluginManager()
mod = py.std.types.ModuleType("x.y.pytest_hello") mod = py.std.types.ModuleType("x.y.pytest_hello")
pm.register(mod) pm.register(mod)
assert pm.isregistered(mod) assert pm.is_registered(mod)
l = pm.getplugins() l = pm.get_plugins()
assert mod in l assert mod in l
pytest.raises(ValueError, "pm.register(mod)") pytest.raises(ValueError, "pm.register(mod)")
pytest.raises(ValueError, lambda: pm.register(mod)) pytest.raises(ValueError, lambda: pm.register(mod))
#assert not pm.isregistered(mod2) #assert not pm.is_registered(mod2)
assert pm.getplugins() == l assert pm.get_plugins() == l
def test_canonical_import(self, monkeypatch): def test_canonical_import(self, monkeypatch):
mod = py.std.types.ModuleType("pytest_xyz") mod = py.std.types.ModuleType("pytest_xyz")
monkeypatch.setitem(py.std.sys.modules, 'pytest_xyz', mod) monkeypatch.setitem(py.std.sys.modules, 'pytest_xyz', mod)
pm = PytestPluginManager() pm = PytestPluginManager()
pm.import_plugin('pytest_xyz') pm.import_plugin('pytest_xyz')
assert pm.getplugin('pytest_xyz') == mod assert pm.get_plugin('pytest_xyz') == mod
assert pm.isregistered(mod) assert pm.is_registered(mod)
def test_consider_module(self, testdir, pytestpm): def test_consider_module(self, testdir, pytestpm):
testdir.syspathinsert() testdir.syspathinsert()
@ -837,8 +840,8 @@ class TestPytestPluginManager:
mod = py.std.types.ModuleType("temp") mod = py.std.types.ModuleType("temp")
mod.pytest_plugins = ["pytest_p1", "pytest_p2"] mod.pytest_plugins = ["pytest_p1", "pytest_p2"]
pytestpm.consider_module(mod) pytestpm.consider_module(mod)
assert pytestpm.getplugin("pytest_p1").__name__ == "pytest_p1" assert pytestpm.get_plugin("pytest_p1").__name__ == "pytest_p1"
assert pytestpm.getplugin("pytest_p2").__name__ == "pytest_p2" assert pytestpm.get_plugin("pytest_p2").__name__ == "pytest_p2"
def test_consider_module_import_module(self, testdir): def test_consider_module_import_module(self, testdir):
pytestpm = get_config().pluginmanager pytestpm = get_config().pluginmanager
@ -880,13 +883,13 @@ class TestPytestPluginManager:
testdir.syspathinsert() testdir.syspathinsert()
testdir.makepyfile(xy123="#") testdir.makepyfile(xy123="#")
monkeypatch.setitem(os.environ, 'PYTEST_PLUGINS', 'xy123') monkeypatch.setitem(os.environ, 'PYTEST_PLUGINS', 'xy123')
l1 = len(pytestpm.getplugins()) l1 = len(pytestpm.get_plugins())
pytestpm.consider_env() pytestpm.consider_env()
l2 = len(pytestpm.getplugins()) l2 = len(pytestpm.get_plugins())
assert l2 == l1 + 1 assert l2 == l1 + 1
assert pytestpm.getplugin('xy123') assert pytestpm.get_plugin('xy123')
pytestpm.consider_env() pytestpm.consider_env()
l3 = len(pytestpm.getplugins()) l3 = len(pytestpm.get_plugins())
assert l2 == l3 assert l2 == l3
def test_consider_setuptools_instantiation(self, monkeypatch, pytestpm): def test_consider_setuptools_instantiation(self, monkeypatch, pytestpm):
@ -904,7 +907,7 @@ class TestPytestPluginManager:
monkeypatch.setattr(pkg_resources, 'iter_entry_points', my_iter) monkeypatch.setattr(pkg_resources, 'iter_entry_points', my_iter)
pytestpm.consider_setuptools_entrypoints() pytestpm.consider_setuptools_entrypoints()
plugin = pytestpm.getplugin("mytestplugin") plugin = pytestpm.get_plugin("pytest_mytestplugin")
assert plugin.x == 42 assert plugin.x == 42
def test_consider_setuptools_not_installed(self, monkeypatch, pytestpm): def test_consider_setuptools_not_installed(self, monkeypatch, pytestpm):
@ -918,7 +921,7 @@ class TestPytestPluginManager:
p = testdir.makepyfile(""" p = testdir.makepyfile("""
import pytest import pytest
def test_hello(pytestconfig): def test_hello(pytestconfig):
plugin = pytestconfig.pluginmanager.getplugin('pytest_x500') plugin = pytestconfig.pluginmanager.get_plugin('pytest_x500')
assert plugin is not None assert plugin is not None
""") """)
monkeypatch.setenv('PYTEST_PLUGINS', 'pytest_x500', prepend=",") monkeypatch.setenv('PYTEST_PLUGINS', 'pytest_x500', prepend=",")
@ -934,13 +937,13 @@ class TestPytestPluginManager:
pluginname = "pytest_hello" pluginname = "pytest_hello"
testdir.makepyfile(**{pluginname: ""}) testdir.makepyfile(**{pluginname: ""})
pytestpm.import_plugin("pytest_hello") pytestpm.import_plugin("pytest_hello")
len1 = len(pytestpm.getplugins()) len1 = len(pytestpm.get_plugins())
pytestpm.import_plugin("pytest_hello") pytestpm.import_plugin("pytest_hello")
len2 = len(pytestpm.getplugins()) len2 = len(pytestpm.get_plugins())
assert len1 == len2 assert len1 == len2
plugin1 = pytestpm.getplugin("pytest_hello") plugin1 = pytestpm.get_plugin("pytest_hello")
assert plugin1.__name__.endswith('pytest_hello') assert plugin1.__name__.endswith('pytest_hello')
plugin2 = pytestpm.getplugin("pytest_hello") plugin2 = pytestpm.get_plugin("pytest_hello")
assert plugin2 is plugin1 assert plugin2 is plugin1
def test_import_plugin_dotted_name(self, testdir, pytestpm): def test_import_plugin_dotted_name(self, testdir, pytestpm):
@ -951,7 +954,7 @@ class TestPytestPluginManager:
testdir.mkpydir("pkg").join("plug.py").write("x=3") testdir.mkpydir("pkg").join("plug.py").write("x=3")
pluginname = "pkg.plug" pluginname = "pkg.plug"
pytestpm.import_plugin(pluginname) pytestpm.import_plugin(pluginname)
mod = pytestpm.getplugin("pkg.plug") mod = pytestpm.get_plugin("pkg.plug")
assert mod.x == 3 assert mod.x == 3
def test_consider_conftest_deps(self, testdir, pytestpm): def test_consider_conftest_deps(self, testdir, pytestpm):
@ -967,15 +970,16 @@ class TestPytestPluginManagerBootstrapming:
def test_plugin_prevent_register(self, pytestpm): def test_plugin_prevent_register(self, pytestpm):
pytestpm.consider_preparse(["xyz", "-p", "no:abc"]) pytestpm.consider_preparse(["xyz", "-p", "no:abc"])
l1 = pytestpm.getplugins() l1 = pytestpm.get_plugins()
pytestpm.register(42, name="abc") pytestpm.register(42, name="abc")
l2 = pytestpm.getplugins() l2 = pytestpm.get_plugins()
assert len(l2) == len(l1) assert len(l2) == len(l1)
assert 42 not in l2
def test_plugin_prevent_register_unregistered_alredy_registered(self, pytestpm): def test_plugin_prevent_register_unregistered_alredy_registered(self, pytestpm):
pytestpm.register(42, name="abc") pytestpm.register(42, name="abc")
l1 = pytestpm.getplugins() l1 = pytestpm.get_plugins()
assert 42 in l1 assert 42 in l1
pytestpm.consider_preparse(["xyz", "-p", "no:abc"]) pytestpm.consider_preparse(["xyz", "-p", "no:abc"])
l2 = pytestpm.getplugins() l2 = pytestpm.get_plugins()
assert 42 not in l2 assert 42 not in l2

View File

@ -457,7 +457,7 @@ class TestTerminalFunctional:
]) ])
assert result.ret == 1 assert result.ret == 1
if not pytestconfig.pluginmanager.hasplugin("xdist"): if not pytestconfig.pluginmanager.get_plugin("xdist"):
pytest.skip("xdist plugin not installed") pytest.skip("xdist plugin not installed")
result = testdir.runpytest(p1, '-v', '-n 1') result = testdir.runpytest(p1, '-v', '-n 1')