Skip to content
Merged
Changes from 1 commit
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
bb60653
gh-138577: Fix keyboard shortcuts in getpass with echo_char
CuriousLearner Nov 15, 2025
31e35e4
Address reviews on dispatcher pattern + handle other ctrl chars
CuriousLearner Nov 16, 2025
b8609bd
Address reviews
CuriousLearner Nov 30, 2025
2691ad9
Merge remote-tracking branch 'upstream/main' into fix-gh-138577
CuriousLearner Mar 21, 2026
6a59f3b
Address reviews
CuriousLearner Mar 22, 2026
d280ce9
fix: prevent prompt corruption during getpass echo_char line editing
CuriousLearner Mar 23, 2026
3f1a861
fix: disable IEXTEN in non-canonical mode to allow Ctrl+V (LNEXT) han…
CuriousLearner Mar 23, 2026
537392c
refactor: address review on getpass echo_char line editing
CuriousLearner Mar 23, 2026
e1e4aa3
fix: Add comments about ICANON and IEXTEN
CuriousLearner Mar 23, 2026
3e05fa3
Merge branch 'main' of github.com:python/cpython into fix-gh-138577
CuriousLearner Mar 23, 2026
90da605
Address reviews
CuriousLearner Mar 24, 2026
ef1efcb
Merge branch 'main' into fix-gh-138577
CuriousLearner Mar 24, 2026
a7c1de3
Remove prefix from private class methods
CuriousLearner Mar 24, 2026
78b2c6a
Merge branch 'fix-gh-138577' of github.com:CuriousLearner/cpython int…
CuriousLearner Mar 24, 2026
e1461a7
Merge branch 'main' of github.com:python/cpython into fix-gh-138577
CuriousLearner Mar 24, 2026
741a817
Apply suggestions from code review
vstinner Mar 24, 2026
cc5dc99
chore! update some documentation
picnixz Mar 29, 2026
0f5f5c8
fix! refresh screen on Ctrl+A/Ctrl-E
picnixz Mar 29, 2026
9d90dfa
refactor! use more handlers
picnixz Mar 29, 2026
3b2ae38
chore! reduce diff against `main`
picnixz Mar 29, 2026
919af5c
test: add cursor position tests for Ctrl+A/Ctrl+E in getpass echo_char
CuriousLearner Mar 30, 2026
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Remove prefix from private class methods
  • Loading branch information
CuriousLearner committed Mar 24, 2026
commit a7c1de380dfa102b64d373f34c083931ca6eaacf
52 changes: 26 additions & 26 deletions Lib/getpass.py
Original file line number Diff line number Diff line change
Expand Up @@ -247,17 +247,17 @@ def __init__(self, stream, echo_char, ctrl_chars, prompt=""):
self.eof_pressed = False
self.literal_next = False
self.ctrl = ctrl_chars
self._dispatch = {
ctrl_chars['SOH']: self._handle_move_start, # Ctrl+A
ctrl_chars['ENQ']: self._handle_move_end, # Ctrl+E
ctrl_chars['VT']: self._handle_kill_forward, # Ctrl+K
ctrl_chars['KILL']: self._handle_kill_line, # Ctrl+U
ctrl_chars['WERASE']: self._handle_erase_word, # Ctrl+W
ctrl_chars['ERASE']: self._handle_erase, # DEL
'\b': self._handle_erase, # Backspace
self.dispatch = {
ctrl_chars['SOH']: self.handle_move_start, # Ctrl+A
ctrl_chars['ENQ']: self.handle_move_end, # Ctrl+E
ctrl_chars['VT']: self.handle_kill_forward, # Ctrl+K
ctrl_chars['KILL']: self.handle_kill_line, # Ctrl+U
ctrl_chars['WERASE']: self.handle_erase_word, # Ctrl+W
ctrl_chars['ERASE']: self.handle_erase, # DEL
'\b': self.handle_erase, # Backspace
}

def _refresh_display(self, prev_len=None):
def refresh_display(self, prev_len=None):
"""Redraw the entire password line with *echo_char*."""
prompt_len = len(self.prompt)
# Use prev_len if given, otherwise current password length
Expand All @@ -269,48 +269,48 @@ def _refresh_display(self, prev_len=None):
self.stream.write('\b' * (len(self.password) - self.cursor_pos))
self.stream.flush()

def _insert_char(self, char):
def insert_char(self, char):
"""Insert *char* at cursor position."""
self.password.insert(self.cursor_pos, char)
self.cursor_pos += 1
# Only refresh if inserting in middle
if self.cursor_pos < len(self.password):
self._refresh_display()
self.refresh_display()
else:
self.stream.write(self.echo_char)
self.stream.flush()

def _handle_move_start(self):
def handle_move_start(self):
"""Move cursor to beginning (Ctrl+A)."""
self.cursor_pos = 0

def _handle_move_end(self):
def handle_move_end(self):
"""Move cursor to end (Ctrl+E)."""
self.cursor_pos = len(self.password)

def _handle_erase(self):
def handle_erase(self):
"""Delete character before cursor (Backspace/DEL)."""
if self.cursor_pos <= 0:
return
prev_len = len(self.password)
del self.password[self.cursor_pos - 1]
self.cursor_pos -= 1
self._refresh_display(prev_len)
self.refresh_display(prev_len)

def _handle_kill_line(self):
def handle_kill_line(self):
"""Erase entire line (Ctrl+U)."""
prev_len = len(self.password)
self.password.clear()
self.cursor_pos = 0
self._refresh_display(prev_len)
self.refresh_display(prev_len)

def _handle_kill_forward(self):
def handle_kill_forward(self):
"""Kill from cursor to end (Ctrl+K)."""
prev_len = len(self.password)
del self.password[self.cursor_pos:]
self._refresh_display(prev_len)
self.refresh_display(prev_len)

def _handle_erase_word(self):
def handle_erase_word(self):
"""Erase previous word (Ctrl+W)."""
old_cursor = self.cursor_pos
# Skip trailing spaces
Expand All @@ -322,11 +322,11 @@ def _handle_erase_word(self):
# Remove the deleted portion
prev_len = len(self.password)
del self.password[self.cursor_pos:old_cursor]
self._refresh_display(prev_len)
self.refresh_display(prev_len)

def _handle(self, char):
def handle(self, char):
"""Handle a single character input. Returns True if handled."""
handler = self._dispatch.get(char)
handler = self.dispatch.get(char)
if handler:
handler()
return True
Expand All @@ -342,7 +342,7 @@ def readline(self, input):
break
# Handle literal next mode FIRST (Ctrl+V quotes next char)
elif self.literal_next:
self._insert_char(char)
self.insert_char(char)
self.literal_next = False
# Check if it's the LNEXT character
elif char == self.ctrl['LNEXT']:
Expand All @@ -355,12 +355,12 @@ def readline(self, input):
break
elif char == '\x00':
pass
elif self._handle(char):
elif self.handle(char):
# Dispatched to handler
pass
else:
# Insert as normal character
self._insert_char(char)
self.insert_char(char)

self.eof_pressed = (char == self.ctrl['EOF'])

Comment thread
vstinner marked this conversation as resolved.
Expand Down