fix #4386 - restructure construction and partial state of ExceptionInfo
This commit is contained in:
parent
6e85febf20
commit
88bf01a31e
|
@ -0,0 +1 @@
|
||||||
|
Restructure ExceptionInfo object construction and ensure incomplete instances have a ``repr``/``str``.
|
|
@ -391,40 +391,85 @@ co_equal = compile(
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@attr.s(repr=False)
|
||||||
class ExceptionInfo(object):
|
class ExceptionInfo(object):
|
||||||
""" wraps sys.exc_info() objects and offers
|
""" wraps sys.exc_info() objects and offers
|
||||||
help for navigating the traceback.
|
help for navigating the traceback.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
_striptext = ""
|
|
||||||
_assert_start_repr = (
|
_assert_start_repr = (
|
||||||
"AssertionError(u'assert " if _PY2 else "AssertionError('assert "
|
"AssertionError(u'assert " if _PY2 else "AssertionError('assert "
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, tup=None, exprinfo=None):
|
_excinfo = attr.ib()
|
||||||
import _pytest._code
|
_striptext = attr.ib(default="")
|
||||||
|
_traceback = attr.ib(default=None)
|
||||||
|
|
||||||
if tup is None:
|
@classmethod
|
||||||
tup = sys.exc_info()
|
def from_current(cls, exprinfo=None):
|
||||||
if exprinfo is None and isinstance(tup[1], AssertionError):
|
"""returns a exceptioninfo matching the current traceback
|
||||||
exprinfo = getattr(tup[1], "msg", None)
|
|
||||||
if exprinfo is None:
|
.. warning::
|
||||||
exprinfo = py.io.saferepr(tup[1])
|
|
||||||
if exprinfo and exprinfo.startswith(self._assert_start_repr):
|
experimental api
|
||||||
self._striptext = "AssertionError: "
|
|
||||||
self._excinfo = tup
|
|
||||||
#: the exception class
|
:param exprinfo: an text string helping to determine if we should
|
||||||
self.type = tup[0]
|
strip assertionerror from the output, defaults
|
||||||
#: the exception instance
|
to the exception message/__str__()
|
||||||
self.value = tup[1]
|
|
||||||
#: the exception raw traceback
|
"""
|
||||||
self.tb = tup[2]
|
tup = sys.exc_info()
|
||||||
#: the exception type name
|
_striptext = ""
|
||||||
self.typename = self.type.__name__
|
if exprinfo is None and isinstance(tup[1], AssertionError):
|
||||||
#: the exception traceback (_pytest._code.Traceback instance)
|
exprinfo = getattr(tup[1], "msg", None)
|
||||||
self.traceback = _pytest._code.Traceback(self.tb, excinfo=ref(self))
|
if exprinfo is None:
|
||||||
|
exprinfo = py.io.saferepr(tup[1])
|
||||||
|
if exprinfo and exprinfo.startswith(cls._assert_start_repr):
|
||||||
|
_striptext = "AssertionError: "
|
||||||
|
|
||||||
|
return cls(tup, _striptext)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def for_later(cls):
|
||||||
|
"""return an unfilled ExceptionInfo
|
||||||
|
"""
|
||||||
|
return cls(None)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def type(self):
|
||||||
|
"""the exception class"""
|
||||||
|
return self._excinfo[0]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def value(self):
|
||||||
|
"""the exception value"""
|
||||||
|
return self._excinfo[1]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def tb(self):
|
||||||
|
"""the exception raw traceback"""
|
||||||
|
return self._excinfo[2]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def typename(self):
|
||||||
|
"""the type name of the exception"""
|
||||||
|
return self.type.__name__
|
||||||
|
|
||||||
|
@property
|
||||||
|
def traceback(self):
|
||||||
|
"""the traceback"""
|
||||||
|
if self._traceback is None:
|
||||||
|
self._traceback = Traceback(self.tb, excinfo=ref(self))
|
||||||
|
return self._traceback
|
||||||
|
|
||||||
|
@traceback.setter
|
||||||
|
def traceback(self, value):
|
||||||
|
self._traceback = value
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
if self._excinfo is None:
|
||||||
|
return "<ExceptionInfo for raises contextmanager>"
|
||||||
return "<ExceptionInfo %s tblen=%d>" % (self.typename, len(self.traceback))
|
return "<ExceptionInfo %s tblen=%d>" % (self.typename, len(self.traceback))
|
||||||
|
|
||||||
def exconly(self, tryshort=False):
|
def exconly(self, tryshort=False):
|
||||||
|
@ -513,6 +558,8 @@ class ExceptionInfo(object):
|
||||||
return fmt.repr_excinfo(self)
|
return fmt.repr_excinfo(self)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
|
if self._excinfo is None:
|
||||||
|
return repr(self)
|
||||||
entry = self.traceback[-1]
|
entry = self.traceback[-1]
|
||||||
loc = ReprFileLocation(entry.path, entry.lineno + 1, self.exconly())
|
loc = ReprFileLocation(entry.path, entry.lineno + 1, self.exconly())
|
||||||
return str(loc)
|
return str(loc)
|
||||||
|
|
|
@ -155,7 +155,7 @@ def assertrepr_compare(config, op, left, right):
|
||||||
explanation = [
|
explanation = [
|
||||||
u"(pytest_assertion plugin: representation of details failed. "
|
u"(pytest_assertion plugin: representation of details failed. "
|
||||||
u"Probably an object has a faulty __repr__.)",
|
u"Probably an object has a faulty __repr__.)",
|
||||||
six.text_type(_pytest._code.ExceptionInfo()),
|
six.text_type(_pytest._code.ExceptionInfo.from_current()),
|
||||||
]
|
]
|
||||||
|
|
||||||
if not explanation:
|
if not explanation:
|
||||||
|
|
|
@ -188,7 +188,7 @@ def wrap_session(config, doit):
|
||||||
except Failed:
|
except Failed:
|
||||||
session.exitstatus = EXIT_TESTSFAILED
|
session.exitstatus = EXIT_TESTSFAILED
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
exitstatus = EXIT_INTERRUPTED
|
exitstatus = EXIT_INTERRUPTED
|
||||||
if initstate <= 2 and isinstance(excinfo.value, exit.Exception):
|
if initstate <= 2 and isinstance(excinfo.value, exit.Exception):
|
||||||
sys.stderr.write("{}: {}\n".format(excinfo.typename, excinfo.value.msg))
|
sys.stderr.write("{}: {}\n".format(excinfo.typename, excinfo.value.msg))
|
||||||
|
@ -197,7 +197,7 @@ def wrap_session(config, doit):
|
||||||
config.hook.pytest_keyboard_interrupt(excinfo=excinfo)
|
config.hook.pytest_keyboard_interrupt(excinfo=excinfo)
|
||||||
session.exitstatus = exitstatus
|
session.exitstatus = exitstatus
|
||||||
except: # noqa
|
except: # noqa
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
config.notify_exception(excinfo, config.option)
|
config.notify_exception(excinfo, config.option)
|
||||||
session.exitstatus = EXIT_INTERNALERROR
|
session.exitstatus = EXIT_INTERNALERROR
|
||||||
if excinfo.errisinstance(SystemExit):
|
if excinfo.errisinstance(SystemExit):
|
||||||
|
|
|
@ -450,7 +450,7 @@ class Module(nodes.File, PyCollector):
|
||||||
mod = self.fspath.pyimport(ensuresyspath=importmode)
|
mod = self.fspath.pyimport(ensuresyspath=importmode)
|
||||||
except SyntaxError:
|
except SyntaxError:
|
||||||
raise self.CollectError(
|
raise self.CollectError(
|
||||||
_pytest._code.ExceptionInfo().getrepr(style="short")
|
_pytest._code.ExceptionInfo.from_current().getrepr(style="short")
|
||||||
)
|
)
|
||||||
except self.fspath.ImportMismatchError:
|
except self.fspath.ImportMismatchError:
|
||||||
e = sys.exc_info()[1]
|
e = sys.exc_info()[1]
|
||||||
|
@ -466,7 +466,7 @@ class Module(nodes.File, PyCollector):
|
||||||
except ImportError:
|
except ImportError:
|
||||||
from _pytest._code.code import ExceptionInfo
|
from _pytest._code.code import ExceptionInfo
|
||||||
|
|
||||||
exc_info = ExceptionInfo()
|
exc_info = ExceptionInfo.from_current()
|
||||||
if self.config.getoption("verbose") < 2:
|
if self.config.getoption("verbose") < 2:
|
||||||
exc_info.traceback = exc_info.traceback.filter(filter_traceback)
|
exc_info.traceback = exc_info.traceback.filter(filter_traceback)
|
||||||
exc_repr = (
|
exc_repr = (
|
||||||
|
|
|
@ -684,13 +684,13 @@ def raises(expected_exception, *args, **kwargs):
|
||||||
# XXX didn't mean f_globals == f_locals something special?
|
# XXX didn't mean f_globals == f_locals something special?
|
||||||
# this is destroyed here ...
|
# this is destroyed here ...
|
||||||
except expected_exception:
|
except expected_exception:
|
||||||
return _pytest._code.ExceptionInfo()
|
return _pytest._code.ExceptionInfo.from_current()
|
||||||
else:
|
else:
|
||||||
func = args[0]
|
func = args[0]
|
||||||
try:
|
try:
|
||||||
func(*args[1:], **kwargs)
|
func(*args[1:], **kwargs)
|
||||||
except expected_exception:
|
except expected_exception:
|
||||||
return _pytest._code.ExceptionInfo()
|
return _pytest._code.ExceptionInfo.from_current()
|
||||||
fail(message)
|
fail(message)
|
||||||
|
|
||||||
|
|
||||||
|
@ -705,7 +705,7 @@ class RaisesContext(object):
|
||||||
self.excinfo = None
|
self.excinfo = None
|
||||||
|
|
||||||
def __enter__(self):
|
def __enter__(self):
|
||||||
self.excinfo = object.__new__(_pytest._code.ExceptionInfo)
|
self.excinfo = _pytest._code.ExceptionInfo.for_later()
|
||||||
return self.excinfo
|
return self.excinfo
|
||||||
|
|
||||||
def __exit__(self, *tp):
|
def __exit__(self, *tp):
|
||||||
|
|
|
@ -211,12 +211,12 @@ class CallInfo(object):
|
||||||
self.result = func()
|
self.result = func()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
if treat_keyboard_interrupt_as_exception:
|
if treat_keyboard_interrupt_as_exception:
|
||||||
self.excinfo = ExceptionInfo()
|
self.excinfo = ExceptionInfo.from_current()
|
||||||
else:
|
else:
|
||||||
self.stop = time()
|
self.stop = time()
|
||||||
raise
|
raise
|
||||||
except: # noqa
|
except: # noqa
|
||||||
self.excinfo = ExceptionInfo()
|
self.excinfo = ExceptionInfo.from_current()
|
||||||
self.stop = time()
|
self.stop = time()
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
|
|
|
@ -115,6 +115,10 @@ class TestCaseFunction(Function):
|
||||||
rawexcinfo = getattr(rawexcinfo, "_rawexcinfo", rawexcinfo)
|
rawexcinfo = getattr(rawexcinfo, "_rawexcinfo", rawexcinfo)
|
||||||
try:
|
try:
|
||||||
excinfo = _pytest._code.ExceptionInfo(rawexcinfo)
|
excinfo = _pytest._code.ExceptionInfo(rawexcinfo)
|
||||||
|
# invoke the attributes to trigger storing the traceback
|
||||||
|
# trial causes some issue there
|
||||||
|
excinfo.value
|
||||||
|
excinfo.traceback
|
||||||
except TypeError:
|
except TypeError:
|
||||||
try:
|
try:
|
||||||
try:
|
try:
|
||||||
|
@ -136,7 +140,7 @@ class TestCaseFunction(Function):
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
raise
|
raise
|
||||||
except fail.Exception:
|
except fail.Exception:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
self.__dict__.setdefault("_excinfo", []).append(excinfo)
|
self.__dict__.setdefault("_excinfo", []).append(excinfo)
|
||||||
|
|
||||||
def addError(self, testcase, rawexcinfo):
|
def addError(self, testcase, rawexcinfo):
|
||||||
|
|
|
@ -169,7 +169,7 @@ class TestExceptionInfo(object):
|
||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
except AssertionError:
|
except AssertionError:
|
||||||
exci = _pytest._code.ExceptionInfo()
|
exci = _pytest._code.ExceptionInfo.from_current()
|
||||||
assert exci.getrepr()
|
assert exci.getrepr()
|
||||||
|
|
||||||
|
|
||||||
|
@ -181,7 +181,7 @@ class TestTracebackEntry(object):
|
||||||
else:
|
else:
|
||||||
assert False
|
assert False
|
||||||
except AssertionError:
|
except AssertionError:
|
||||||
exci = _pytest._code.ExceptionInfo()
|
exci = _pytest._code.ExceptionInfo.from_current()
|
||||||
entry = exci.traceback[0]
|
entry = exci.traceback[0]
|
||||||
source = entry.getsource()
|
source = entry.getsource()
|
||||||
assert len(source) == 6
|
assert len(source) == 6
|
||||||
|
|
|
@ -71,7 +71,7 @@ def test_excinfo_simple():
|
||||||
try:
|
try:
|
||||||
raise ValueError
|
raise ValueError
|
||||||
except ValueError:
|
except ValueError:
|
||||||
info = _pytest._code.ExceptionInfo()
|
info = _pytest._code.ExceptionInfo.from_current()
|
||||||
assert info.type == ValueError
|
assert info.type == ValueError
|
||||||
|
|
||||||
|
|
||||||
|
@ -85,7 +85,7 @@ def test_excinfo_getstatement():
|
||||||
try:
|
try:
|
||||||
f()
|
f()
|
||||||
except ValueError:
|
except ValueError:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
linenumbers = [
|
linenumbers = [
|
||||||
_pytest._code.getrawcode(f).co_firstlineno - 1 + 4,
|
_pytest._code.getrawcode(f).co_firstlineno - 1 + 4,
|
||||||
_pytest._code.getrawcode(f).co_firstlineno - 1 + 1,
|
_pytest._code.getrawcode(f).co_firstlineno - 1 + 1,
|
||||||
|
@ -126,7 +126,7 @@ class TestTraceback_f_g_h(object):
|
||||||
try:
|
try:
|
||||||
h()
|
h()
|
||||||
except ValueError:
|
except ValueError:
|
||||||
self.excinfo = _pytest._code.ExceptionInfo()
|
self.excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
|
|
||||||
def test_traceback_entries(self):
|
def test_traceback_entries(self):
|
||||||
tb = self.excinfo.traceback
|
tb = self.excinfo.traceback
|
||||||
|
@ -163,7 +163,7 @@ class TestTraceback_f_g_h(object):
|
||||||
try:
|
try:
|
||||||
exec(source.compile())
|
exec(source.compile())
|
||||||
except NameError:
|
except NameError:
|
||||||
tb = _pytest._code.ExceptionInfo().traceback
|
tb = _pytest._code.ExceptionInfo.from_current().traceback
|
||||||
print(tb[-1].getsource())
|
print(tb[-1].getsource())
|
||||||
s = str(tb[-1].getsource())
|
s = str(tb[-1].getsource())
|
||||||
assert s.startswith("def xyz():\n try:")
|
assert s.startswith("def xyz():\n try:")
|
||||||
|
@ -356,6 +356,12 @@ def test_excinfo_str():
|
||||||
assert len(s.split(":")) >= 3 # on windows it's 4
|
assert len(s.split(":")) >= 3 # on windows it's 4
|
||||||
|
|
||||||
|
|
||||||
|
def test_excinfo_for_later():
|
||||||
|
e = ExceptionInfo.for_later()
|
||||||
|
assert "for raises" in repr(e)
|
||||||
|
assert "for raises" in str(e)
|
||||||
|
|
||||||
|
|
||||||
def test_excinfo_errisinstance():
|
def test_excinfo_errisinstance():
|
||||||
excinfo = pytest.raises(ValueError, h)
|
excinfo = pytest.raises(ValueError, h)
|
||||||
assert excinfo.errisinstance(ValueError)
|
assert excinfo.errisinstance(ValueError)
|
||||||
|
@ -365,7 +371,7 @@ def test_excinfo_no_sourcecode():
|
||||||
try:
|
try:
|
||||||
exec("raise ValueError()")
|
exec("raise ValueError()")
|
||||||
except ValueError:
|
except ValueError:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
s = str(excinfo.traceback[-1])
|
s = str(excinfo.traceback[-1])
|
||||||
assert s == " File '<string>':1 in <module>\n ???\n"
|
assert s == " File '<string>':1 in <module>\n ???\n"
|
||||||
|
|
||||||
|
@ -390,7 +396,7 @@ def test_entrysource_Queue_example():
|
||||||
try:
|
try:
|
||||||
queue.Queue().get(timeout=0.001)
|
queue.Queue().get(timeout=0.001)
|
||||||
except queue.Empty:
|
except queue.Empty:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
entry = excinfo.traceback[-1]
|
entry = excinfo.traceback[-1]
|
||||||
source = entry.getsource()
|
source = entry.getsource()
|
||||||
assert source is not None
|
assert source is not None
|
||||||
|
@ -402,7 +408,7 @@ def test_codepath_Queue_example():
|
||||||
try:
|
try:
|
||||||
queue.Queue().get(timeout=0.001)
|
queue.Queue().get(timeout=0.001)
|
||||||
except queue.Empty:
|
except queue.Empty:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
entry = excinfo.traceback[-1]
|
entry = excinfo.traceback[-1]
|
||||||
path = entry.path
|
path = entry.path
|
||||||
assert isinstance(path, py.path.local)
|
assert isinstance(path, py.path.local)
|
||||||
|
@ -453,7 +459,7 @@ class TestFormattedExcinfo(object):
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
raise
|
raise
|
||||||
except: # noqa
|
except: # noqa
|
||||||
return _pytest._code.ExceptionInfo()
|
return _pytest._code.ExceptionInfo.from_current()
|
||||||
assert 0, "did not raise"
|
assert 0, "did not raise"
|
||||||
|
|
||||||
def test_repr_source(self):
|
def test_repr_source(self):
|
||||||
|
@ -491,7 +497,7 @@ class TestFormattedExcinfo(object):
|
||||||
try:
|
try:
|
||||||
exec(co)
|
exec(co)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
repr = pr.repr_excinfo(excinfo)
|
repr = pr.repr_excinfo(excinfo)
|
||||||
assert repr.reprtraceback.reprentries[1].lines[0] == "> ???"
|
assert repr.reprtraceback.reprentries[1].lines[0] == "> ???"
|
||||||
if sys.version_info[0] >= 3:
|
if sys.version_info[0] >= 3:
|
||||||
|
@ -510,7 +516,7 @@ raise ValueError()
|
||||||
try:
|
try:
|
||||||
exec(co)
|
exec(co)
|
||||||
except ValueError:
|
except ValueError:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
repr = pr.repr_excinfo(excinfo)
|
repr = pr.repr_excinfo(excinfo)
|
||||||
assert repr.reprtraceback.reprentries[1].lines[0] == "> ???"
|
assert repr.reprtraceback.reprentries[1].lines[0] == "> ???"
|
||||||
if sys.version_info[0] >= 3:
|
if sys.version_info[0] >= 3:
|
||||||
|
@ -1340,7 +1346,7 @@ def test_repr_traceback_with_unicode(style, encoding):
|
||||||
try:
|
try:
|
||||||
raise RuntimeError(msg)
|
raise RuntimeError(msg)
|
||||||
except RuntimeError:
|
except RuntimeError:
|
||||||
e_info = ExceptionInfo()
|
e_info = ExceptionInfo.from_current()
|
||||||
formatter = FormattedExcinfo(style=style)
|
formatter = FormattedExcinfo(style=style)
|
||||||
repr_traceback = formatter.repr_traceback(e_info)
|
repr_traceback = formatter.repr_traceback(e_info)
|
||||||
assert repr_traceback is not None
|
assert repr_traceback is not None
|
||||||
|
|
|
@ -151,7 +151,7 @@ class TestWithFunctionIntegration(object):
|
||||||
try:
|
try:
|
||||||
raise ValueError
|
raise ValueError
|
||||||
except ValueError:
|
except ValueError:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
reslog = ResultLog(None, py.io.TextIO())
|
reslog = ResultLog(None, py.io.TextIO())
|
||||||
reslog.pytest_internalerror(excinfo.getrepr(style=style))
|
reslog.pytest_internalerror(excinfo.getrepr(style=style))
|
||||||
entry = reslog.logfile.getvalue()
|
entry = reslog.logfile.getvalue()
|
||||||
|
|
|
@ -561,20 +561,16 @@ def test_outcomeexception_passes_except_Exception():
|
||||||
|
|
||||||
|
|
||||||
def test_pytest_exit():
|
def test_pytest_exit():
|
||||||
try:
|
with pytest.raises(pytest.exit.Exception) as excinfo:
|
||||||
pytest.exit("hello")
|
pytest.exit("hello")
|
||||||
except pytest.exit.Exception:
|
assert excinfo.errisinstance(KeyboardInterrupt)
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
|
||||||
assert excinfo.errisinstance(KeyboardInterrupt)
|
|
||||||
|
|
||||||
|
|
||||||
def test_pytest_fail():
|
def test_pytest_fail():
|
||||||
try:
|
with pytest.raises(pytest.fail.Exception) as excinfo:
|
||||||
pytest.fail("hello")
|
pytest.fail("hello")
|
||||||
except pytest.fail.Exception:
|
s = excinfo.exconly(tryshort=True)
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
assert s.startswith("Failed")
|
||||||
s = excinfo.exconly(tryshort=True)
|
|
||||||
assert s.startswith("Failed")
|
|
||||||
|
|
||||||
|
|
||||||
def test_pytest_exit_msg(testdir):
|
def test_pytest_exit_msg(testdir):
|
||||||
|
@ -683,7 +679,7 @@ def test_exception_printing_skip():
|
||||||
try:
|
try:
|
||||||
pytest.skip("hello")
|
pytest.skip("hello")
|
||||||
except pytest.skip.Exception:
|
except pytest.skip.Exception:
|
||||||
excinfo = _pytest._code.ExceptionInfo()
|
excinfo = _pytest._code.ExceptionInfo.from_current()
|
||||||
s = excinfo.exconly(tryshort=True)
|
s = excinfo.exconly(tryshort=True)
|
||||||
assert s.startswith("Skipped")
|
assert s.startswith("Skipped")
|
||||||
|
|
||||||
|
@ -718,7 +714,7 @@ def test_importorskip(monkeypatch):
|
||||||
mod2 = pytest.importorskip("hello123", minversion="1.3")
|
mod2 = pytest.importorskip("hello123", minversion="1.3")
|
||||||
assert mod2 == mod
|
assert mod2 == mod
|
||||||
except pytest.skip.Exception:
|
except pytest.skip.Exception:
|
||||||
print(_pytest._code.ExceptionInfo())
|
print(_pytest._code.ExceptionInfo.from_current())
|
||||||
pytest.fail("spurious skip")
|
pytest.fail("spurious skip")
|
||||||
|
|
||||||
|
|
||||||
|
@ -740,7 +736,7 @@ def test_importorskip_dev_module(monkeypatch):
|
||||||
pytest.importorskip('mockmodule1', minversion='0.14.0')""",
|
pytest.importorskip('mockmodule1', minversion='0.14.0')""",
|
||||||
)
|
)
|
||||||
except pytest.skip.Exception:
|
except pytest.skip.Exception:
|
||||||
print(_pytest._code.ExceptionInfo())
|
print(_pytest._code.ExceptionInfo.from_current())
|
||||||
pytest.fail("spurious skip")
|
pytest.fail("spurious skip")
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue