Try harder to show unicode lines in SyntaxErrors

If the interpreter doesn't decode the line for us, do so ourselves
using the linecache module.
This commit is contained in:
Robert Collins 2015-03-09 17:14:26 +13:00
parent 70a91e4398
commit ffe9882743
2 changed files with 46 additions and 5 deletions

View File

@ -530,7 +530,19 @@ class TracebackException:
lineno = str(self.lineno) or u('?') lineno = str(self.lineno) or u('?')
yield u(' File "{0}", line {1}\n').format(filename, lineno) yield u(' File "{0}", line {1}\n').format(filename, lineno)
badline = self.text and u(self.text) badline = None
if self.text is not None:
if type(self.text) is bytes:
# Not decoded - get the line via linecache which will decode
# for us.
if self.lineno:
badline = linecache.getline(filename, int(lineno))
if not badline:
# But we can't for some reason, so fallback to attempting a
# u cast.
badline = u(self.text)
else:
badline = self.text
offset = self.offset offset = self.offset
if badline is not None: if badline is not None:
yield u(' {0}\n').format(badline.strip()) yield u(' {0}\n').format(badline.strip())

View File

@ -44,11 +44,13 @@ FNAME = __file__
if FNAME.endswith('.pyc'): if FNAME.endswith('.pyc'):
FNAME = FNAME[:-1] FNAME = FNAME[:-1]
class FakeLoader: class FakeLoader:
def __init__(self, lines):
self._lines = lines
def get_source(self, name): def get_source(self, name):
return ''.join(linecache.getlines(FNAME)) return self._lines
fake_module = dict( fake_module = dict(
__name__='fred', __name__='fred',
__loader__=FakeLoader() __loader__=FakeLoader(''.join(linecache.getlines(FNAME)))
) )
@ -470,7 +472,7 @@ ZeroDivisionError
def test_syntax_error_offset_at_eol(self): def test_syntax_error_offset_at_eol(self):
# See #10186. # See #10186.
def e(): def e():
raise SyntaxError('', ('', 0, 5, 'hello')) raise SyntaxError('', ('', 0, 5, u('hello')))
msg = self.get_report(e).splitlines() msg = self.get_report(e).splitlines()
self.assertEqual(msg[-2], " ^") self.assertEqual(msg[-2], " ^")
def e(): def e():
@ -637,7 +639,7 @@ class TestStack(unittest.TestCase):
traceback.walk_stack(None), capture_locals=True, limit=1) traceback.walk_stack(None), capture_locals=True, limit=1)
s = some_inner(3, 4) s = some_inner(3, 4)
self.assertEqual( self.assertEqual(
[' File "' + FNAME + '", line 637, ' [' File "' + FNAME + '", line 639, '
'in some_inner\n' 'in some_inner\n'
' traceback.walk_stack(None), capture_locals=True, limit=1)\n' ' traceback.walk_stack(None), capture_locals=True, limit=1)\n'
' a = 1\n' ' a = 1\n'
@ -792,3 +794,30 @@ class TestTracebackException(unittest.TestCase):
u(' File "<string>", line None\n'), u(' File "<string>", line None\n'),
u('SyntaxError: uh oh\n')], u('SyntaxError: uh oh\n')],
list(exc.format())) list(exc.format()))
def test_syntax_undecoded_lines(self):
# If the interpreter returns bytestrings, we have to decode ourselves.
lines = u("1\n\u5341\n3\n")
fake_module = dict(
__name__='fred',
__loader__=FakeLoader(lines)
)
linecache.updatecache('/foo.py', fake_module)
e = SyntaxError("uh oh")
e.filename = '/foo.py'
e.lineno = 2
e.text = b('something wrong')
e.offset = 1
c = test_code('/foo.py', 'method')
f = test_frame(c, fake_module, {'something': 1})
tb = test_tb(f, 2, None)
exc = traceback.TracebackException(SyntaxError, e, tb)
list(exc.format_exception_only())
self.assertEqual([
u('Traceback (most recent call last):\n'),
u(' File "/foo.py", line 2, in method\n \u5341\n'),
u(' File "/foo.py", line 2\n'),
u(' \u5341\n'),
u(' ^\n'),
u('SyntaxError: uh oh\n')],
list(exc.format()))