introduce new py.io.saferepr for printing the 'repr' of an object safely
and without consuming too much space --HG-- branch : trunk
This commit is contained in:
parent
1c1623885f
commit
5ece3858e4
|
@ -8,6 +8,10 @@ Changes between 1.2.1 and 1.2.2 (release pending)
|
||||||
- fixes for handling of unicode exception values and unprintable objects
|
- fixes for handling of unicode exception values and unprintable objects
|
||||||
- (issue87) fix unboundlocal error in assertionold code
|
- (issue87) fix unboundlocal error in assertionold code
|
||||||
- (issue86) improve documentation for looponfailing
|
- (issue86) improve documentation for looponfailing
|
||||||
|
- expose previously internal commonly useful methods:
|
||||||
|
py.io.get_terminal_with() -> return terminal width
|
||||||
|
py.io.ansi_print(...) -> print colored/bold text on linux/win32
|
||||||
|
py.io.saferepr(obj) -> return limited representation string
|
||||||
- expose test outcome related exceptions as py.test.skip.Exception,
|
- expose test outcome related exceptions as py.test.skip.Exception,
|
||||||
py.test.raises.Exception etc., useful mostly for plugins
|
py.test.raises.Exception etc., useful mostly for plugins
|
||||||
doing special outcome interpreteration/tweaking
|
doing special outcome interpreteration/tweaking
|
||||||
|
|
|
@ -140,6 +140,7 @@ py.apipkg.initpkg(__name__, dict(
|
||||||
'TerminalWriter' : '._io.terminalwriter:TerminalWriter',
|
'TerminalWriter' : '._io.terminalwriter:TerminalWriter',
|
||||||
'ansi_print' : '._io.terminalwriter:ansi_print',
|
'ansi_print' : '._io.terminalwriter:ansi_print',
|
||||||
'get_terminal_width' : '._io.terminalwriter:get_terminal_width',
|
'get_terminal_width' : '._io.terminalwriter:get_terminal_width',
|
||||||
|
'saferepr' : '._io.saferepr:saferepr',
|
||||||
},
|
},
|
||||||
|
|
||||||
# small and mean xml/html generation
|
# small and mean xml/html generation
|
||||||
|
|
|
@ -147,7 +147,7 @@ class Frame(object):
|
||||||
def repr(self, object):
|
def repr(self, object):
|
||||||
""" return a 'safe' (non-recursive, one-line) string repr for 'object'
|
""" return a 'safe' (non-recursive, one-line) string repr for 'object'
|
||||||
"""
|
"""
|
||||||
return safe_repr(object)
|
return py.io.saferepr(object)
|
||||||
|
|
||||||
def is_true(self, object):
|
def is_true(self, object):
|
||||||
return object
|
return object
|
||||||
|
@ -457,7 +457,7 @@ class FormattedExcinfo(object):
|
||||||
return source
|
return source
|
||||||
|
|
||||||
def _saferepr(self, obj):
|
def _saferepr(self, obj):
|
||||||
return safe_repr(obj)
|
return py.io.saferepr(obj)
|
||||||
|
|
||||||
def repr_args(self, entry):
|
def repr_args(self, entry):
|
||||||
if self.funcargs:
|
if self.funcargs:
|
||||||
|
@ -605,7 +605,7 @@ def unicode_or_repr(obj):
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
raise
|
raise
|
||||||
except Exception:
|
except Exception:
|
||||||
return "<print-error: %r>" % safe_repr(obj)
|
return "<print-error: %r>" % py.io.saferepr(obj)
|
||||||
|
|
||||||
class ReprExceptionInfo(TerminalRepr):
|
class ReprExceptionInfo(TerminalRepr):
|
||||||
def __init__(self, reprtraceback, reprcrash):
|
def __init__(self, reprtraceback, reprcrash):
|
||||||
|
@ -717,48 +717,6 @@ class ReprFuncArgs(TerminalRepr):
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class SafeRepr(reprlib.Repr):
|
|
||||||
""" subclass of repr.Repr that limits the resulting size of repr()
|
|
||||||
and includes information on exceptions raised during the call.
|
|
||||||
"""
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
reprlib.Repr.__init__(self, *args, **kwargs)
|
|
||||||
self.maxstring = 240 # 3 * 80 chars
|
|
||||||
self.maxother = 160 # 2 * 80 chars
|
|
||||||
|
|
||||||
def repr(self, x):
|
|
||||||
return self._callhelper(reprlib.Repr.repr, self, x)
|
|
||||||
|
|
||||||
def repr_instance(self, x, level):
|
|
||||||
return self._callhelper(builtin_repr, x)
|
|
||||||
|
|
||||||
def _callhelper(self, call, x, *args):
|
|
||||||
try:
|
|
||||||
# Try the vanilla repr and make sure that the result is a string
|
|
||||||
s = call(x, *args)
|
|
||||||
except (KeyboardInterrupt, MemoryError, SystemExit):
|
|
||||||
raise
|
|
||||||
except:
|
|
||||||
cls, e, tb = sys.exc_info()
|
|
||||||
try:
|
|
||||||
exc_name = cls.__name__
|
|
||||||
except:
|
|
||||||
exc_name = 'unknown'
|
|
||||||
try:
|
|
||||||
exc_info = str(e)
|
|
||||||
except:
|
|
||||||
exc_info = 'unknown'
|
|
||||||
return '<[%s("%s") raised in repr()] %s object at 0x%x>' % (
|
|
||||||
exc_name, exc_info, x.__class__.__name__, id(x))
|
|
||||||
else:
|
|
||||||
if len(s) > self.maxstring:
|
|
||||||
i = max(0, (self.maxstring-3)//2)
|
|
||||||
j = max(0, self.maxstring-3-i)
|
|
||||||
s = s[:i] + '...' + s[len(s)-j:]
|
|
||||||
return s
|
|
||||||
|
|
||||||
safe_repr = SafeRepr().repr
|
|
||||||
|
|
||||||
oldbuiltins = {}
|
oldbuiltins = {}
|
||||||
|
|
||||||
def patch_builtins(assertion=True, compile=True):
|
def patch_builtins(assertion=True, compile=True):
|
||||||
|
|
|
@ -0,0 +1,55 @@
|
||||||
|
import py
|
||||||
|
import sys, os.path
|
||||||
|
|
||||||
|
builtin_repr = repr
|
||||||
|
|
||||||
|
reprlib = py.builtin._tryimport('repr', 'reprlib')
|
||||||
|
|
||||||
|
sysex = (KeyboardInterrupt, MemoryError, SystemExit)
|
||||||
|
|
||||||
|
class SafeRepr(reprlib.Repr):
|
||||||
|
""" subclass of repr.Repr that limits the resulting size of repr()
|
||||||
|
and includes information on exceptions raised during the call.
|
||||||
|
"""
|
||||||
|
def repr(self, x):
|
||||||
|
return self._callhelper(reprlib.Repr.repr, self, x)
|
||||||
|
|
||||||
|
def repr_instance(self, x, level):
|
||||||
|
return self._callhelper(builtin_repr, x)
|
||||||
|
|
||||||
|
def _callhelper(self, call, x, *args):
|
||||||
|
try:
|
||||||
|
# Try the vanilla repr and make sure that the result is a string
|
||||||
|
s = call(x, *args)
|
||||||
|
except sysex:
|
||||||
|
raise
|
||||||
|
except:
|
||||||
|
cls, e, tb = sys.exc_info()
|
||||||
|
exc_name = getattr(cls, '__name__', 'unknown')
|
||||||
|
try:
|
||||||
|
exc_info = str(e)
|
||||||
|
except sysex:
|
||||||
|
raise
|
||||||
|
except:
|
||||||
|
exc_info = 'unknown'
|
||||||
|
return '<[%s("%s") raised in repr()] %s object at 0x%x>' % (
|
||||||
|
exc_name, exc_info, x.__class__.__name__, id(x))
|
||||||
|
else:
|
||||||
|
if len(s) > self.maxsize:
|
||||||
|
i = max(0, (self.maxsize-3)//2)
|
||||||
|
j = max(0, self.maxsize-3-i)
|
||||||
|
s = s[:i] + '...' + s[len(s)-j:]
|
||||||
|
return s
|
||||||
|
|
||||||
|
def saferepr(obj, maxsize=240):
|
||||||
|
""" return a size-limited safe repr-string for the given object.
|
||||||
|
Failing __repr__ functions of user instances will be represented
|
||||||
|
with a short exception info and 'saferepr' generally takes
|
||||||
|
care to never raise exceptions itself. This function is a wrapper
|
||||||
|
around the Repr/reprlib functionality of the standard 2.6 lib.
|
||||||
|
"""
|
||||||
|
# review exception handling
|
||||||
|
srepr = SafeRepr()
|
||||||
|
srepr.maxstring = maxsize
|
||||||
|
srepr.maxsize = maxsize
|
||||||
|
return srepr.repr(obj)
|
|
@ -1,7 +1,6 @@
|
||||||
from __future__ import generators
|
from __future__ import generators
|
||||||
import py
|
import py
|
||||||
import sys
|
import sys
|
||||||
from py._code.code import safe_repr
|
|
||||||
|
|
||||||
failsonjython = py.test.mark.xfail("sys.platform.startswith('java')")
|
failsonjython = py.test.mark.xfail("sys.platform.startswith('java')")
|
||||||
|
|
||||||
|
@ -139,43 +138,6 @@ def test_code_from_func():
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class TestSafeRepr:
|
|
||||||
def test_simple_repr(self):
|
|
||||||
assert safe_repr(1) == '1'
|
|
||||||
assert safe_repr(None) == 'None'
|
|
||||||
|
|
||||||
def test_exceptions(self):
|
|
||||||
class BrokenRepr:
|
|
||||||
def __init__(self, ex):
|
|
||||||
self.ex = ex
|
|
||||||
foo = 0
|
|
||||||
def __repr__(self):
|
|
||||||
raise self.ex
|
|
||||||
class BrokenReprException(Exception):
|
|
||||||
__str__ = None
|
|
||||||
__repr__ = None
|
|
||||||
assert 'Exception' in safe_repr(BrokenRepr(Exception("broken")))
|
|
||||||
s = safe_repr(BrokenReprException("really broken"))
|
|
||||||
assert 'TypeError' in s
|
|
||||||
if py.std.sys.version_info < (2,6):
|
|
||||||
assert 'unknown' in safe_repr(BrokenRepr("string"))
|
|
||||||
else:
|
|
||||||
assert 'TypeError' in safe_repr(BrokenRepr("string"))
|
|
||||||
|
|
||||||
def test_big_repr(self):
|
|
||||||
from py._code.code import SafeRepr
|
|
||||||
assert len(safe_repr(range(1000))) <= \
|
|
||||||
len('[' + SafeRepr().maxlist * "1000" + ']')
|
|
||||||
|
|
||||||
def test_repr_on_newstyle(self):
|
|
||||||
class Function(object):
|
|
||||||
def __repr__(self):
|
|
||||||
return "<%s>" %(self.name)
|
|
||||||
try:
|
|
||||||
s = safe_repr(Function())
|
|
||||||
except Exception:
|
|
||||||
py.test.fail("saferepr failed for newstyle class")
|
|
||||||
|
|
||||||
def test_builtin_patch_unpatch(monkeypatch):
|
def test_builtin_patch_unpatch(monkeypatch):
|
||||||
cpy_builtin = py.builtin.builtins
|
cpy_builtin = py.builtin.builtins
|
||||||
comp = cpy_builtin.compile
|
comp = cpy_builtin.compile
|
||||||
|
|
|
@ -0,0 +1,99 @@
|
||||||
|
from __future__ import generators
|
||||||
|
import py
|
||||||
|
import sys
|
||||||
|
|
||||||
|
saferepr = py.io.saferepr
|
||||||
|
|
||||||
|
class TestSafeRepr:
|
||||||
|
def test_simple_repr(self):
|
||||||
|
assert saferepr(1) == '1'
|
||||||
|
assert saferepr(None) == 'None'
|
||||||
|
|
||||||
|
def test_maxsize(self):
|
||||||
|
s = saferepr('x'*50, maxsize=25)
|
||||||
|
assert len(s) == 25
|
||||||
|
expected = repr('x'*10 + '...' + 'x'*10)
|
||||||
|
assert s == expected
|
||||||
|
|
||||||
|
def test_maxsize_error_on_instance(self):
|
||||||
|
class A:
|
||||||
|
def __repr__(self):
|
||||||
|
raise ValueError('...')
|
||||||
|
|
||||||
|
s = saferepr(('*'*50, A()), maxsize=25)
|
||||||
|
assert len(s) == 25
|
||||||
|
assert s[0] == '(' and s[-1] == ')'
|
||||||
|
|
||||||
|
def test_exceptions(self):
|
||||||
|
class BrokenRepr:
|
||||||
|
def __init__(self, ex):
|
||||||
|
self.ex = ex
|
||||||
|
foo = 0
|
||||||
|
def __repr__(self):
|
||||||
|
raise self.ex
|
||||||
|
class BrokenReprException(Exception):
|
||||||
|
__str__ = None
|
||||||
|
__repr__ = None
|
||||||
|
assert 'Exception' in saferepr(BrokenRepr(Exception("broken")))
|
||||||
|
s = saferepr(BrokenReprException("really broken"))
|
||||||
|
assert 'TypeError' in s
|
||||||
|
if py.std.sys.version_info < (2,6):
|
||||||
|
assert 'unknown' in saferepr(BrokenRepr("string"))
|
||||||
|
else:
|
||||||
|
assert 'TypeError' in saferepr(BrokenRepr("string"))
|
||||||
|
|
||||||
|
def test_big_repr(self):
|
||||||
|
from py._io.saferepr import SafeRepr
|
||||||
|
assert len(saferepr(range(1000))) <= \
|
||||||
|
len('[' + SafeRepr().maxlist * "1000" + ']')
|
||||||
|
|
||||||
|
def test_repr_on_newstyle(self):
|
||||||
|
class Function(object):
|
||||||
|
def __repr__(self):
|
||||||
|
return "<%s>" %(self.name)
|
||||||
|
try:
|
||||||
|
s = saferepr(Function())
|
||||||
|
except Exception:
|
||||||
|
py.test.fail("saferepr failed for newstyle class")
|
||||||
|
|
||||||
|
def test_builtin_patch_unpatch(monkeypatch):
|
||||||
|
cpy_builtin = py.builtin.builtins
|
||||||
|
comp = cpy_builtin.compile
|
||||||
|
def mycompile(*args, **kwargs):
|
||||||
|
return comp(*args, **kwargs)
|
||||||
|
class Sub(AssertionError):
|
||||||
|
pass
|
||||||
|
monkeypatch.setattr(cpy_builtin, 'AssertionError', Sub)
|
||||||
|
monkeypatch.setattr(cpy_builtin, 'compile', mycompile)
|
||||||
|
py.code.patch_builtins()
|
||||||
|
assert cpy_builtin.AssertionError != Sub
|
||||||
|
assert cpy_builtin.compile != mycompile
|
||||||
|
py.code.unpatch_builtins()
|
||||||
|
assert cpy_builtin.AssertionError is Sub
|
||||||
|
assert cpy_builtin.compile == mycompile
|
||||||
|
|
||||||
|
|
||||||
|
def test_unicode_handling():
|
||||||
|
value = py.builtin._totext('\xc4\x85\xc4\x87\n', 'utf-8').encode('utf8')
|
||||||
|
def f():
|
||||||
|
raise Exception(value)
|
||||||
|
excinfo = py.test.raises(Exception, f)
|
||||||
|
s = str(excinfo)
|
||||||
|
if sys.version_info[0] < 3:
|
||||||
|
u = unicode(excinfo)
|
||||||
|
|
||||||
|
def test_unicode_or_repr():
|
||||||
|
from py._code.code import unicode_or_repr
|
||||||
|
assert unicode_or_repr('hello') == "hello"
|
||||||
|
if sys.version_info[0] < 3:
|
||||||
|
s = unicode_or_repr('\xf6\xc4\x85')
|
||||||
|
else:
|
||||||
|
s = eval("unicode_or_repr(b'\\f6\\xc4\\x85')")
|
||||||
|
assert 'print-error' in s
|
||||||
|
assert 'c4' in s
|
||||||
|
class A:
|
||||||
|
def __repr__(self):
|
||||||
|
raise ValueError()
|
||||||
|
s = unicode_or_repr(A())
|
||||||
|
assert 'print-error' in s
|
||||||
|
assert 'ValueError' in s
|
Loading…
Reference in New Issue