- 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
This commit is contained in:
holger krekel 2014-10-08 11:27:14 +02:00
parent 3d6ad054c0
commit f5f924d293
3 changed files with 90 additions and 99 deletions

View File

@ -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)
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,23 +375,12 @@ class MultiCall:
return "<MultiCall %s, kwargs=%r>" %(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:
return wrapped_call(method(*args), self.execute)
res = method(*args)
if res is not None:
self.results.append(res)
@ -380,15 +388,6 @@ class MultiCall:
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")
def varnames(func, startindex=None):

View File

@ -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):

View File

@ -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
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,)
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]