diff options
| author | Sergey B Kirpichev <skirpichev@gmail.com> | 2024-08-19 19:01:58 (GMT) |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2024-08-19 19:01:58 (GMT) |
| commit | c8f4069ab1602a1f67239fef0e11cc3e72c0045d (patch) | |
| tree | 958e6704cd0102ea848a511fec9f7d8f50cc9dd7 | |
| parent | 21399a096302ea577efd9a12c2f08b4458d095bd (diff) | |
| download | cpython-c8f4069ab1602a1f67239fef0e11cc3e72c0045d.zip cpython-c8f4069ab1602a1f67239fef0e11cc3e72c0045d.tar.gz cpython-c8f4069ab1602a1f67239fef0e11cc3e72c0045d.tar.bz2 | |
[3.13] gh-121804: Always show error location for SyntaxError's in new repl (GH-121886) (#123148)
(cherry picked from commit 354d55eb1fa40f272419aa6459ee5d2c4804c8ea)
| -rw-r--r-- | Lib/_pyrepl/console.py | 10 | ||||
| -rw-r--r-- | Lib/code.py | 8 | ||||
| -rwxr-xr-x | Lib/idlelib/pyshell.py | 2 | ||||
| -rw-r--r-- | Lib/test/test_pyrepl/test_interact.py | 14 | ||||
| -rw-r--r-- | Misc/NEWS.d/next/Library/2024-07-16-20-49-07.gh-issue-121804.gYN-In.rst | 2 |
5 files changed, 29 insertions, 7 deletions
diff --git a/Lib/_pyrepl/console.py b/Lib/_pyrepl/console.py index a8d3f52..4319343 100644 --- a/Lib/_pyrepl/console.py +++ b/Lib/_pyrepl/console.py @@ -161,8 +161,8 @@ class InteractiveColoredConsole(code.InteractiveConsole): super().__init__(locals=locals, filename=filename, local_exit=local_exit) # type: ignore[call-arg] self.can_colorize = _colorize.can_colorize() - def showsyntaxerror(self, filename=None): - super().showsyntaxerror(colorize=self.can_colorize) + def showsyntaxerror(self, filename=None, **kwargs): + super().showsyntaxerror(colorize=self.can_colorize, **kwargs) def showtraceback(self): super().showtraceback(colorize=self.can_colorize) @@ -171,7 +171,7 @@ class InteractiveColoredConsole(code.InteractiveConsole): try: tree = ast.parse(source) except (SyntaxError, OverflowError, ValueError): - self.showsyntaxerror(filename) + self.showsyntaxerror(filename, source=source) return False if tree.body: *_, last_stmt = tree.body @@ -188,10 +188,10 @@ class InteractiveColoredConsole(code.InteractiveConsole): f"Try the asyncio REPL ({python} -m asyncio) to use" f" top-level 'await' and run background asyncio tasks." ) - self.showsyntaxerror(filename) + self.showsyntaxerror(filename, source=source) return False except (OverflowError, ValueError): - self.showsyntaxerror(filename) + self.showsyntaxerror(filename, source=source) return False if code is None: diff --git a/Lib/code.py b/Lib/code.py index 7d68517..aec7d61 100644 --- a/Lib/code.py +++ b/Lib/code.py @@ -64,7 +64,7 @@ class InteractiveInterpreter: code = self.compile(source, filename, symbol) except (OverflowError, SyntaxError, ValueError): # Case 1 - self.showsyntaxerror(filename) + self.showsyntaxerror(filename, source=source) return False if code is None: @@ -123,6 +123,12 @@ class InteractiveInterpreter: # Stuff in the right filename value = SyntaxError(msg, (filename, lineno, offset, line)) sys.last_exc = sys.last_value = value + # Set the line of text that the exception refers to + source = kwargs.pop('source', '') + lines = source.splitlines() + if (source and type is SyntaxError + and not value.text and len(lines) >= value.lineno): + value.text = lines[value.lineno - 1] if sys.excepthook is sys.__excepthook__: lines = traceback.format_exception_only(type, value, colorize=colorize) self.write(''.join(lines)) diff --git a/Lib/idlelib/pyshell.py b/Lib/idlelib/pyshell.py index d8b2652..e882c6c 100755 --- a/Lib/idlelib/pyshell.py +++ b/Lib/idlelib/pyshell.py @@ -706,7 +706,7 @@ class ModifiedInterpreter(InteractiveInterpreter): del _filename, _sys, _dirname, _dir \n""".format(filename)) - def showsyntaxerror(self, filename=None): + def showsyntaxerror(self, filename=None, **kwargs): """Override Interactive Interpreter method: Use Colorizing Color the offending position instead of printing it and pointing at it diff --git a/Lib/test/test_pyrepl/test_interact.py b/Lib/test/test_pyrepl/test_interact.py index 538dfd3..e71ab41 100644 --- a/Lib/test/test_pyrepl/test_interact.py +++ b/Lib/test/test_pyrepl/test_interact.py @@ -88,6 +88,20 @@ class TestSimpleInteract(unittest.TestCase): self.assertFalse(result) self.assertIn('SyntaxError', f.getvalue()) + @force_not_colorized + def test_runsource_show_syntax_error_location(self): + console = InteractiveColoredConsole() + source = "def f(x, x): ..." + f = io.StringIO() + with contextlib.redirect_stderr(f): + result = console.runsource(source) + self.assertFalse(result) + r = """ + def f(x, x): ... + ^ +SyntaxError: duplicate argument 'x' in function definition""" + self.assertIn(r, f.getvalue()) + def test_runsource_shows_syntax_error_for_failed_compilation(self): console = InteractiveColoredConsole() source = "print('Hello, world!'" diff --git a/Misc/NEWS.d/next/Library/2024-07-16-20-49-07.gh-issue-121804.gYN-In.rst b/Misc/NEWS.d/next/Library/2024-07-16-20-49-07.gh-issue-121804.gYN-In.rst new file mode 100644 index 0000000..1cc1cde --- /dev/null +++ b/Misc/NEWS.d/next/Library/2024-07-16-20-49-07.gh-issue-121804.gYN-In.rst @@ -0,0 +1,2 @@ +Correctly show error locations, when :exc:`SyntaxError` raised in new repl. +Patch by Sergey B Kirpichev. |
