Skip to content

Commit 77ca92a

Browse files
committed
pythongh-118911: Trailing whitespace in a block shouldnt prevent the user from terminating the code block
1 parent f6da790 commit 77ca92a

File tree

4 files changed

+68
-13
lines changed

4 files changed

+68
-13
lines changed

Lib/_pyrepl/historical_reader.py

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -259,7 +259,7 @@ def select_item(self, i: int) -> None:
259259
self.transient_history[self.historyi] = self.get_unicode()
260260
buf = self.transient_history.get(i)
261261
if buf is None:
262-
buf = self.history[i]
262+
buf = self.history[i].rstrip()
263263
self.buffer = list(buf)
264264
self.historyi = i
265265
self.pos = len(self.buffer)

Lib/_pyrepl/readline.py

Lines changed: 19 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -219,16 +219,29 @@ def do(self) -> None:
219219
# if there are already several lines and the cursor
220220
# is not on the last one, always insert a new \n.
221221
text = r.get_unicode()
222+
222223
if "\n" in r.buffer[r.pos :] or (
223224
r.more_lines is not None and r.more_lines(text)
224225
):
226+
def _whitespace_before_pos():
227+
before_idx = r.pos - 1
228+
while before_idx > 0 and text[before_idx].isspace():
229+
before_idx -= 1
230+
return text[before_idx : r.pos].count("\n") > 0
225231
#
226-
# auto-indent the next line like the previous line
227-
prevlinestart, indent = _get_previous_line_indent(r.buffer, r.pos)
228-
r.insert("\n")
229-
if not self.reader.paste_mode and indent:
230-
for i in range(prevlinestart, prevlinestart + indent):
231-
r.insert(r.buffer[i])
232+
# if there's already a new line before the cursor then
233+
# even if the cursor is followed by whitespace, we assume
234+
# the user is trying to terminate the block
235+
if _whitespace_before_pos() and text[r.pos:].isspace():
236+
self.finish = True
237+
else:
238+
#
239+
# auto-indent the next line like the previous line
240+
prevlinestart, indent = _get_previous_line_indent(r.buffer, r.pos)
241+
r.insert("\n")
242+
if not self.reader.paste_mode and indent:
243+
for i in range(prevlinestart, prevlinestart + indent):
244+
r.insert(r.buffer[i])
232245
elif not self.reader.paste_mode:
233246
self.finish = True
234247
else:

Lib/test/test_pyrepl/test_pyrepl.py

Lines changed: 4 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -317,14 +317,13 @@ def test_multiline_edit(self):
317317
[
318318
Event(evt="key", data="up", raw=bytearray(b"\x1bOA")),
319319
Event(evt="key", data="up", raw=bytearray(b"\x1bOA")),
320-
Event(evt="key", data="up", raw=bytearray(b"\x1bOA")),
321-
Event(evt="key", data="right", raw=bytearray(b"\x1bOC")),
322-
Event(evt="key", data="right", raw=bytearray(b"\x1bOC")),
323-
Event(evt="key", data="right", raw=bytearray(b"\x1bOC")),
320+
Event(evt="key", data="left", raw=bytearray(b"\x1bOD")),
321+
Event(evt="key", data="left", raw=bytearray(b"\x1bOD")),
322+
Event(evt="key", data="left", raw=bytearray(b"\x1bOD")),
324323
Event(evt="key", data="backspace", raw=bytearray(b"\x7f")),
325324
Event(evt="key", data="g", raw=bytearray(b"g")),
326325
Event(evt="key", data="down", raw=bytearray(b"\x1bOB")),
327-
Event(evt="key", data="down", raw=bytearray(b"\x1bOB")),
326+
Event(evt="key", data="\n", raw=bytearray(b"\n")),
328327
Event(evt="key", data="\n", raw=bytearray(b"\n")),
329328
],
330329
)

Lib/test/test_pyrepl/test_reader.py

Lines changed: 44 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
11
import itertools
2+
import functools
23
from unittest import TestCase
34

4-
from .support import handle_all_events, handle_events_narrow_console, code_to_events
5+
from .support import handle_all_events, handle_events_narrow_console, code_to_events, prepare_reader
56
from _pyrepl.console import Event
67

78

@@ -133,3 +134,45 @@ def test_up_arrow_after_ctrl_r(self):
133134

134135
reader, _ = handle_all_events(events)
135136
self.assert_screen_equals(reader, "")
137+
138+
def test_newline_within_block_trailing_whitespace(self):
139+
# fmt: off
140+
code = (
141+
"def foo():\n"
142+
" a = 1\n"
143+
)
144+
# fmt: on
145+
146+
events = itertools.chain(
147+
code_to_events(code),
148+
[
149+
# go to the end of the first line
150+
Event(evt="key", data="up", raw=bytearray(b"\x1bOA")),
151+
Event(evt="key", data="up", raw=bytearray(b"\x1bOA")),
152+
Event(evt="key", data="\x05", raw=bytearray(b"\x1bO5")),
153+
# new lines in-block shouldn't terminate the block
154+
Event(evt="key", data="\n", raw=bytearray(b"\n")),
155+
Event(evt="key", data="\n", raw=bytearray(b"\n")),
156+
# end of line 2
157+
Event(evt="key", data="down", raw=bytearray(b"\x1bOB")),
158+
Event(evt="key", data="\x05", raw=bytearray(b"\x1bO5")),
159+
# a double new line in-block should terminate the block
160+
# even if its followed by whitespace
161+
Event(evt="key", data="\n", raw=bytearray(b"\n")),
162+
Event(evt="key", data="\n", raw=bytearray(b"\n")),
163+
],
164+
)
165+
166+
no_paste_reader = functools.partial(prepare_reader, paste_mode=False)
167+
reader, _ = handle_all_events(events, prepare_reader=no_paste_reader)
168+
169+
expected = (
170+
"def foo():\n"
171+
"\n"
172+
"\n"
173+
" a = 1\n"
174+
" \n"
175+
" " # HistoricalReader will trim trailing whitespace
176+
)
177+
self.assert_screen_equals(reader, expected)
178+
self.assertEqual(reader.finished, True)

0 commit comments

Comments
 (0)