From f5f924d293ed65ea62a66ad39d2c8074e575f2c4 Mon Sep 17 00:00:00 2001 From: holger krekel Date: Wed, 8 Oct 2014 11:27:14 +0200 Subject: [PATCH] - refactor wrapped call support to also accomodate pytest.mark.hookwrapper - introduce a CallOutcome class to hold the result/excinfo status of calling a function. - rename add_method_controller to add_method_wrapper --- _pytest/core.py | 135 +++++++++++++++++++++---------------------- _pytest/pytester.py | 4 +- testing/test_core.py | 50 +++++++--------- 3 files changed, 90 insertions(+), 99 deletions(-) diff --git a/_pytest/core.py b/_pytest/core.py index f2c120cf3..4921e97c7 100644 --- a/_pytest/core.py +++ b/_pytest/core.py @@ -10,6 +10,8 @@ import py assert py.__version__.split(".")[:2] >= ['1', '4'], ("installation problem: " "%s is too old, remove or upgrade 'py'" % (py.__version__)) +py3 = sys.version_info > (3,0) + class TagTracer: def __init__(self): self._tag2proc = {} @@ -68,42 +70,62 @@ class TagTracerSub: return self.__class__(self.root, self.tags + (name,)) -def add_method_controller(cls, func): - """ Use func as the method controler for the method found - at the class named func.__name__. +def add_method_wrapper(cls, wrapper_func): + """ Substitute the function named "wrapperfunc.__name__" at class + "cls" with a function that wraps the call to the original function. + Return an undo function which can be called to reset the class to use + the old method again. - A method controler is invoked with the same arguments - as the function it substitutes and is required to yield once - which will trigger calling the controlled method. - If it yields a second value, the value will be returned - as the result of the invocation. Errors in the controlled function - are re-raised to the controller during the first yield. + wrapper_func is called with the same arguments as the method + it wraps and its result is used as a wrap_controller for + calling the original function. """ - name = func.__name__ + name = wrapper_func.__name__ oldcall = getattr(cls, name) def wrap_exec(*args, **kwargs): - gen = func(*args, **kwargs) - next(gen) # first yield - try: - res = oldcall(*args, **kwargs) - except Exception: - excinfo = sys.exc_info() - try: - # reraise exception to controller - res = gen.throw(*excinfo) - except StopIteration: - py.builtin._reraise(*excinfo) - else: - try: - res = gen.send(res) - except StopIteration: - pass - return res + gen = wrapper_func(*args, **kwargs) + return wrapped_call(gen, lambda: oldcall(*args, **kwargs)) setattr(cls, name, wrap_exec) return lambda: setattr(cls, name, oldcall) +def wrapped_call(wrap_controller, func): + """ Wrap calling to a function with a generator. The first yield + will trigger calling the function and receive an according CallOutcome + object representing an exception or a result. + """ + next(wrap_controller) # first yield + call_outcome = CallOutcome(func) + try: + wrap_controller.send(call_outcome) + co = wrap_controller.gi_frame.f_code + raise RuntimeError("wrap_controller for %r %s:%d has second yield" % + (co.co_name, co.co_filename, co.co_firstlineno)) + except StopIteration: + pass + if call_outcome.excinfo is None: + return call_outcome.result + else: + ex = call_outcome.excinfo + if py3: + raise ex[1].with_traceback(ex[2]) + py.builtin._reraise(*ex) + + +class CallOutcome: + excinfo = None + def __init__(self, func): + try: + self.result = func() + except Exception: + self.excinfo = sys.exc_info() + + def force_result(self, result): + self.result = result + self.excinfo = None + + class PluginManager(object): def __init__(self, hookspecs=None, prefix="pytest_"): self._name2plugin = {} @@ -125,15 +147,12 @@ class PluginManager(object): trace = self.hookrelay.trace trace.root.indent += 1 trace(self.name, kwargs) - res = None - try: - res = yield - finally: - if res: - trace("finish", self.name, "-->", res) - trace.root.indent -= 1 + box = yield + if box.excinfo is None: + trace("finish", self.name, "-->", box.result) + trace.root.indent -= 1 - undo = add_method_controller(HookCaller, _docall) + undo = add_method_wrapper(HookCaller, _docall) self.add_shutdown(undo) def do_configure(self, config): @@ -356,39 +375,19 @@ class MultiCall: return "" %(status, self.kwargs) def execute(self): - next_finalizers = [] - try: - all_kwargs = self.kwargs - while self.methods: - method = self.methods.pop() - args = [all_kwargs[argname] for argname in varnames(method)] - if hasattr(method, "hookwrapper"): - it = method(*args) - 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(*args) - 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") + all_kwargs = self.kwargs + while self.methods: + method = self.methods.pop() + args = [all_kwargs[argname] for argname in varnames(method)] + if hasattr(method, "hookwrapper"): + return wrapped_call(method(*args), self.execute) + res = method(*args) + if res is not None: + self.results.append(res) + if self.firstresult: + return res + if not self.firstresult: + return self.results def varnames(func, startindex=None): diff --git a/_pytest/pytester.py b/_pytest/pytester.py index ed4580f4a..e732417ed 100644 --- a/_pytest/pytester.py +++ b/_pytest/pytester.py @@ -11,7 +11,7 @@ import subprocess import py import pytest from py.builtin import print_ -from _pytest.core import HookCaller, add_method_controller +from _pytest.core import HookCaller, add_method_wrapper from _pytest.main import Session, EXIT_OK @@ -57,7 +57,7 @@ class HookRecorder: def _docall(hookcaller, methods, kwargs): self.calls.append(ParsedCall(hookcaller.name, kwargs)) yield - self._undo_wrapping = add_method_controller(HookCaller, _docall) + self._undo_wrapping = add_method_wrapper(HookCaller, _docall) pluginmanager.add_shutdown(self._undo_wrapping) def finish_recording(self): diff --git a/testing/test_core.py b/testing/test_core.py index 03bc4813e..76b8fc5b3 100644 --- a/testing/test_core.py +++ b/testing/test_core.py @@ -280,8 +280,9 @@ class TestBootstrapping: pm.register(p) assert pm.trace.root.indent == indent - assert len(l) == 1 + assert len(l) == 2 assert 'pytest_plugin_registered' in l[0] + assert 'finish' in l[1] pytest.raises(ValueError, lambda: pm.register(api1())) assert pm.trace.root.indent == indent assert saveindent[0] > indent @@ -555,7 +556,7 @@ class TestMultiCall: l.append("m2 finish") m2.hookwrapper = True res = MultiCall([m2, m1], {}).execute() - assert res == [1, 2] + assert res == [] assert l == ["m1 init", "m2 init", "m2 finish", "m1 finish"] def test_listattr_hookwrapper_ordering(self): @@ -593,10 +594,8 @@ class TestMultiCall: m1.hookwrapper = True mc = MultiCall([m1], {}) - with pytest.raises(mc.WrongHookWrapper) as ex: + with pytest.raises(TypeError): mc.execute() - assert ex.value.func == m1 - assert ex.value.message def test_hookwrapper_too_many_yield(self): def m1(): @@ -605,10 +604,10 @@ class TestMultiCall: m1.hookwrapper = True mc = MultiCall([m1], {}) - with pytest.raises(mc.WrongHookWrapper) as ex: + with pytest.raises(RuntimeError) as ex: mc.execute() - assert ex.value.func == m1 - assert ex.value.message + assert "m1" in str(ex.value) + assert "test_core.py:" in str(ex.value) class TestHookRelay: @@ -774,9 +773,10 @@ class TestWrapMethod: l = [] def f(self): l.append(1) - yield + box = yield + assert box.result == "A.f" l.append(2) - undo = add_method_controller(A, f) + undo = add_method_wrapper(A, f) assert A().f() == "A.f" assert l == [1,2] @@ -793,14 +793,10 @@ class TestWrapMethod: l = [] def error(self, val): l.append(val) - try: - yield - except ValueError: - l.append(None) - raise + yield + l.append(None) - - undo = add_method_controller(A, error) + undo = add_method_wrapper(A, error) with pytest.raises(ValueError): A().error(42) @@ -817,12 +813,10 @@ class TestWrapMethod: raise ValueError(val) def error(self, val): - try: - yield - except ValueError: - yield 2 + box = yield + box.force_result(2) - add_method_controller(A, error) + add_method_wrapper(A, error) assert A().error(42) == 2 def test_reraise_on_controller_StopIteration(self): @@ -836,7 +830,7 @@ class TestWrapMethod: except ValueError: pass - add_method_controller(A, error) + add_method_wrapper(A, error) with pytest.raises(ValueError): A().error(42) @@ -848,13 +842,11 @@ class TestWrapMethod: l = [] def error(self): - try: - yield (1,), {'val2': 2} - except ValueError as ex: - assert ex.args == (3,) - l.append(1) + box = yield (1,), {'val2': 2} + assert box.excinfo[1].args == (3,) + l.append(1) - add_method_controller(A, error) + add_method_wrapper(A, error) with pytest.raises(ValueError): A().error() assert l == [1]