diff --git a/testing/test_hookcaller.py b/testing/test_hookcaller.py index 88ed1316..ed1c7a86 100644 --- a/testing/test_hookcaller.py +++ b/testing/test_hookcaller.py @@ -448,3 +448,59 @@ def conflict(self) -> None: "Hook 'conflict' is already registered within namespace " ".Api1'>" ) + + +def test_hook_multi_impl(pm: PluginManager) -> None: + """Since plugins' impls are able to (optionally) specify a spec name, it is possible for a plugin to implement + the same spec multiple times.""" + + class Api: + @hookspec + def hello(self, arg: object) -> None: + "api hook 1" + + pm.add_hookspecs(Api) + hook = pm.hook + test_hc = hook.hello + + class Plugin: + @hookimpl + def hello(self, arg): + return arg + 1 + + @hookimpl(specname="hello") + def hello_again(self, arg): + return arg + 100 + + plugin = Plugin() + + # Confirm that registration puts the impls into all the right registries + pm.register(plugin) + out = test_hc(arg=3) + assert out == [103, 4] + + assert len(pm.get_plugins()) == 1 + + hookimpls = test_hc.get_hookimpls() + hook_plugins = [item.plugin for item in hookimpls] + assert hook_plugins == [plugin, plugin] + for impl in hookimpls: + pm._verify_hook(test_hc, impl) + + hook_callers = pm.get_hookcallers(plugin) + assert ( + len(hook_callers) == 2 + ), "This should return two callers. Same spec but different impls." + assert hook_callers[0].spec == hook_callers[1].spec + + subset_caller = pm.subset_hook_caller("hello", [plugin]) + out = subset_caller(arg=3) + assert out == [] + + # Confirm that 'unregistering' does the converse + pm.unregister(plugin) + assert test_hc(arg=3) == [] + hookimpls = test_hc.get_hookimpls() + assert hookimpls == [] + hook_callers = pm.get_hookcallers(plugin) + assert hook_callers is None diff --git a/testing/test_tracer.py b/testing/test_tracer.py index 5e538369..5c2ad9be 100644 --- a/testing/test_tracer.py +++ b/testing/test_tracer.py @@ -2,8 +2,14 @@ import pytest +from pluggy import HookimplMarker +from pluggy import HookspecMarker +from pluggy import PluginManager from pluggy._tracing import TagTracer +hookspec = HookspecMarker("example") +hookimpl = HookimplMarker("example") + @pytest.fixture def rootlogger() -> TagTracer: @@ -77,3 +83,79 @@ def test_setprocessor(rootlogger: TagTracer) -> None: log2("seen") tags, args = l2[0] assert args == ("seen",) + + +def test_plugin_tracing(pm: PluginManager) -> None: + class Api: + @hookspec + def hello(self, arg: object) -> None: + "api hook 1" + + pm.add_hookspecs(Api) + hook = pm.hook + test_hc = hook.hello + + class Plugin: + @hookimpl + def hello(self, arg): + return arg + 1 + + plugin = Plugin() + + trace_out: List[str] = [] + pm.trace.root.setwriter(trace_out.append) + pm.register(plugin) + pm.enable_tracing() + + out = test_hc(arg=3) + assert out == [4] + + assert trace_out == [ + " hello [hook]\n arg: 3\n", + " finish hello --> [4] [hook]\n", + ] + + +def test_dbl_plugin_tracing(pm: PluginManager) -> None: + class Api: + @hookspec + def hello(self, arg: object) -> None: + "api hook 1" + + pm.add_hookspecs(Api) + hook = pm.hook + test_hc = hook.hello + + class Plugin: + @hookimpl + def hello(self, arg): + return arg + 1 + + @hookimpl(specname="hello") + def hello_again(self, arg): + return arg + 100 + + plugin = Plugin() + + trace_out: List[str] = [] + pm.trace.root.setwriter(trace_out.append) + pm.register(plugin) + pm.enable_tracing() + + out = test_hc(arg=3) + assert out == [103, 4] + + assert trace_out == [ + " hello [hook]\n arg: 3\n", + " finish hello --> [103, 4] [hook]\n", + ] + + trace_out.clear() + pm.unregister(plugin) + out = test_hc(arg=3) + assert out == [] + + assert trace_out == [ + " hello [hook]\n arg: 3\n", + " finish hello --> [] [hook]\n", + ]