Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

There's no readline module on Windows Python (cmd.Cmd) #90028

Closed
keeely mannequin opened this issue Nov 22, 2021 · 19 comments
Closed

There's no readline module on Windows Python (cmd.Cmd) #90028

keeely mannequin opened this issue Nov 22, 2021 · 19 comments
Labels
3.10 only security fixes stdlib Python modules in the Lib dir type-bug An unexpected behavior, bug, or error

Comments

@keeely
Copy link
Mannequin

keeely mannequin commented Nov 22, 2021

BPO 45870
Nosy @gvanrossum, @rhettinger, @terryjreedy, @eryksun, @zooba, @keeely, @E3V3A
Files
  • cmd_shell_example.py: Script to demo use of readline module with cmd
  • Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.

    Show more details

    GitHub fields:

    assignee = None
    closed_at = None
    created_at = <Date 2021-11-22.13:51:22.349>
    labels = ['type-bug', 'library', '3.10']
    title = "There's no readline module on Windows Python (cmd.Cmd)"
    updated_at = <Date 2022-02-02.16:23:35.559>
    user = 'https://github.com/keeely'

    bugs.python.org fields:

    activity = <Date 2022-02-02.16:23:35.559>
    actor = 'E3V3A'
    assignee = 'none'
    closed = False
    closed_date = None
    closer = None
    components = ['Library (Lib)']
    creation = <Date 2021-11-22.13:51:22.349>
    creator = 'keeely'
    dependencies = []
    files = ['50461']
    hgrepos = []
    issue_num = 45870
    keywords = []
    message_count = 16.0
    messages = ['406778', '406800', '407118', '407162', '407164', '407168', '407169', '407172', '407174', '407176', '407177', '407178', '407180', '407258', '407261', '412372']
    nosy_count = 7.0
    nosy_names = ['gvanrossum', 'rhettinger', 'terry.reedy', 'eryksun', 'steve.dower', 'keeely', 'E3V3A']
    pr_nums = []
    priority = 'normal'
    resolution = None
    stage = None
    status = 'open'
    superseder = None
    type = 'behavior'
    url = 'https://bugs.python.org/issue45870'
    versions = ['Python 3.10']

    @keeely
    Copy link
    Mannequin Author

    keeely mannequin commented Nov 22, 2021

    In the past this was worked around by installing a third-party module:
    https://github.com/pyreadline/pyreadline

    For Python 3.10, however this module doesn't work. And pyreadline doesn't seem to be maintained anymore, so it's possible it won't get fixed.
    pyreadline/pyreadline#73

    Consider the following code:

    import cmd
    import readline

    open("history.txt", "w").write("first line\nsecond line\n")

    readline.clear_history()
    readline.read_history_file("history.txt")
    
    class MyShell(cmd.Cmd):
        def __init__(self):
            super().__init__()
    
    shell = MyShell()
    shell.cmdloop()

    This works fine on MacOs Python, also on Linux and on Windows prior to 3.10 with the readline module added. It won't work going forward.

    The Windows cmd implementation clearly uses readline or some compatible lib under the hood, so Python should make it available to developers.

    @keeely keeely mannequin added 3.10 only security fixes stdlib Python modules in the Lib dir type-bug An unexpected behavior, bug, or error labels Nov 22, 2021
    @eryksun
    Copy link
    Contributor

    eryksun commented Nov 22, 2021

    The Windows cmd implementation clearly uses readline or
    some compatible lib under the hood

    The REPL shell and input() call PyOS_Readline(). If this call isn't hooked (e.g. by the readline module), and stdin is a console file, then it reads a line from the console via ReadConsoleW(). If the console's input stream is configured in line-input mode, which we assume it is, this function provides basic readline-ish support, including a line editor that supports input history and aliases. The history is stored in an in-memory buffer in the console, which doesn't get save and reused across console sessions. There is no public API to get the history contents, and there is no support at all to set it.

    The console/terminal team at Microsoft apparently don't want to do anything with the builtin readline support, which is seen as a legacy feature. To the contrary, I've even seen them suggest that they're evaluating the addition of a new client-side readline library in the native API, which I assume would be similar to PowerShell's PSReadLine (e.g. based on ReadConsoleInputW(), and supporting a classic console mode in addition to emacs and vi modes).

    For now, I suggest patching pyreadline for your own use. In the reported issue, I see it's trying to use collections.Callable. Instead it should import collections.abc and use collections.abc.Callable.

    @terryjreedy
    Copy link
    Member

    Thank you Eryk for the info.

    As a bug report, this should be closed as '3rd party'. As an enhancement request, it needs to be specified more and should perhaps be discussed on python-ideas.

    @keeely
    Copy link
    Mannequin Author

    keeely mannequin commented Nov 27, 2021

    You can take the view that it's not a bug (with some justification), but a few lines in the cmd docs would make all the difference in terms of wasted time.

    I have now abandoned my Windows port and suggested users install WSL2 instead which is the easiest way forward for me, but it'd be nice to have known from the start that portions of cmd functionality are not available for Win32 instead of the indirect references via readline. You could throw us a bone here.

    @terryjreedy
    Copy link
    Member

    What specific sentences would you like where in which doc. (Please link as 'cmd doc' is too vague.)

    @terryjreedy
    Copy link
    Member

    Sorry, you obviously mean
    https://docs.python.org/3/library/cmd.html#module-cmd
    What to add where still applies.

    @terryjreedy
    Copy link
    Member

    Guido and Raymond, you are the two active coredevs that have contributed the most lines to cmd module. What do either of you think?

    @eryksun
    Copy link
    Contributor

    eryksun commented Nov 27, 2021

    You can take the view that it's not a bug (with some justification),
    but a few lines in the cmd docs would make all the difference in
    terms of wasted time.

    If anything, I think the readline documentation should have a note explaining the situation in Windows. The documentation of the cmd module already makes the readline dependency clear:

    If the readline module is loaded, input will automatically inherit 
    bash-like history-list editing (e.g. Control-P scrolls back to the
    last command, Control-N forward to the next one, Control-F moves the 
    cursor to the right non-destructively, Control-B moves the cursor to 
    the left non-destructively, etc.).
    

    @gvanrossum
    Copy link
    Member

    AFAIK the reason command history works in cmd.py on Windows is that it's built into the terminal program. Or maybe into the operating system.

    Thus, the user can use line editing and history, but there is no API (in Python) to interact with these.

    I'm sure Steve Dower can explain the exact situation -- it may depend on which Windows version and which terminal program you use (my only recent experience is with winterm on Windows 10).

    I agree that (once we sort out what works in what versions of Windows and which terminal programs) we should clarify this in the docs.

    @eryksun
    Copy link
    Contributor

    eryksun commented Nov 27, 2021

    AFAIK the reason command history works in cmd.py on Windows is
    that it's built into the terminal program. Or maybe into the
    operating system.

    As mentioned in msg406800, input editing, history (e.g. up/down arrows, F7 popup, F8 completion), and alias support is implemented by the Windows console host (conhost.exe or openconsole.exe) for ReadFile() and ReadConsole() calls when the input stream is in line-input mode. Currently, it's the same whether a classic console session or a pseudoconsole (headless) session is hosted. When Windows Terminal is used, the overall connection of components looks like Python<->ConDrv (kernel device)<->OpenConsole<->NamedPipe (kernel device)<->Windows Terminal. The headless console's use of Windows Terminal for the user interface doesn't matter to Python's ReadConsoleW() call.

    A headless console session always starts with 4 history buffers (one for each attached process) that store up to 50 commands. For a classic console session, the initial number and size of history buffers can be configured in the session properties or defaults. It can always be set dynamically via SetConsoleHistoryInfo(). There's *undocumented* support to get the commands from a history buffer that's associated with the name of an attached process: GetConsoleCommandHistoryLengthW(executable_name) and GetConsoleCommandHistoryW(buffer, buffer_length, executable_name). However, the API provides no function to set the command history. I suppose one could loop over WriteConsoleInputW() and ReadConsoleW() to implement it as a kludge.

    @gvanrossum
    Copy link
    Member

    Thanks, Eryk, I only read the part of the issue that landed in my inbox (fhe first message and everything after Terry added me to the nosy list). Sorry.

    You wrote:

    The console/terminal team at Microsoft apparently don't want to do anything with the builtin readline support, which is seen as a legacy feature.

    What does "the builtin readline support" refer to here? Presumably not GNU Readline?

    @eryksun
    Copy link
    Contributor

    eryksun commented Nov 27, 2021

    What does "the builtin readline support" refer to here?
    Presumably not GNU Readline?

    That's referring to the readline(ish) support that's built into the console host for ReadFile() and ReadConsole() calls when the input stream is in line-input mode. I've never seen the console developers speak positively of this feature on their GitHub repo. They've suggested the addition of a native readline API on the client side, like PowerShell's PSReadLine module provides. But who knows when/if that would be released.

    Python has the third-party pyreadline module, but it's no longer actively developed. To bring pyreadline into the standard library would be a non-trivial task. OTOH, I assume if Microsoft provided an official readline API, which does all the heavy lifting, that Python could support it in the readline extension module, if the API is basically compatible with libreadline/libedit.

    @gvanrossum
    Copy link
    Member

    Okay, so that's all hypothetical. It looks like the status quo is not
    likely to change, so we should just document it. I wonder if keeely is
    interested in submitting a PR for the docs?

    @keeely
    Copy link
    Mannequin Author

    keeely mannequin commented Nov 29, 2021

    Regrettably I cannot submit a PR for the docs because I value my online anonymity and Python submissions require a real name (IIRC), but my suggestion would be pretty simple.

    Taking as an example, for termios (https://docs.python.org/3/library/termios.html), we currently have:

    This module provides an interface to the POSIX calls for tty I/O control. For a complete description of these calls, see termios(3) Unix manual page. It is only available for those Unix versions that support POSIX termios style tty I/O control configured during installation.

    For readline (https://docs.python.org/3/library/readline.html#module-readline) we have:

    The readline module defines a number of functions to facilitate completion and reading/writing of history files from the Python interpreter. This module can be used directly, or via the rlcompleter module, which supports completion of Python identifiers at the interactive prompt. Settings made using this module affect the behaviour of both the interpreter’s interactive prompt and the prompts offered by the built-in input() function.

    In similar way to the first para of the termios description I would add the following text: “It is only available on platforms that support the readline functionality, generally POSIX”.

    Then perhaps I’d also add to the cmd documentation at https://docs.python.org/3/library/cmd.html

    The Cmd class provides a simple framework for writing line-oriented command interpreters. These are often useful for test harnesses, administrative tools, and prototypes that will later be wrapped in a more sophisticated interface.

    I would add at the end of that first paragraph:
    “Some features will be unavailable on non-POSIX platforms due to the readline requirement”.

    Hope this helps, if it's not clear I can provide in diff form if you prefer.

    @keeely
    Copy link
    Mannequin Author

    keeely mannequin commented Nov 29, 2021

    I'm attaching an example usage of cmd + readline to show how you can have context-specific history for sub-shells. WARNING: WRITES FILES TO CWD! In the event that someone does implement this on Windows it would be nice if this worked. That doesn't mean less-capable readline support wouldn't also be useful.

    thanks.

    @E3V3A
    Copy link
    Mannequin

    E3V3A mannequin commented Feb 2, 2022

    I would like to make the python community aware that there has recently been a renewed interest in updating and maintaining 'pyreadline', but in a new(ish) repository 'pyreadline3'.

    https://github.com/pyreadline3/pyreadline3

    This apparently now works under Py3.10, but there are some minor issues with how pyreadline handles ANSI escape sequences for coloring auto completion etc. This is probably due to the code developed for the older/original Windows cmd/powershell use of 4-bit ANSI colors, no longer being able to handle 256 or RGB. There was a PR for that, but because that PR is aged it seem limited to older 4-bit ANSI's, since it is using a RegEx for stripping the sequences to calculate the actual line length.

    [OT. This also bring up the question where the function "rl_expand_prompt()" (handling RL_PROMPT_START_IGNORE and RL_PROMPT_END_IGNORE) was moved, as it can't be found in the cpython codebase. (https://github.com/python/cpython/blob/main/Modules/readline.c) The equivalent HEX for those _does_ work, at least for the prompt, but maybe they are handled on the windows side?]

    @ezio-melotti ezio-melotti transferred this issue from another repository Apr 10, 2022
    @keeely
    Copy link

    keeely commented Aug 19, 2022

    It took me a while to try the new pyreadline3 module, because I had settled into using WSL2 on Windows. However I can now report it ticks all the boxes for me and is viable.

    It's a bit of a shame it has to be pyreadline3 and we can't somehow give it the simpler name. It may also be nice if the Python docs gave an indication that this should be used, saving people a visit to pyreadline github, and figuring out it won't work from the issue tracker there.

    One thing I noted is that pyreadline3 is missing the append_history_file() function, giving it something in common with the libedit based readline shipped with Mac, so I needed a minor change to my readline detection code to sort that but it's not a big deal.

    From my POV this can be closed.

    @gvanrossum
    Copy link
    Member

    Are you interested in submitting a docs PR to link to pyreadline3?

    @keeely
    Copy link

    keeely commented Aug 19, 2022

    I'm afraid not, see: #90028 (comment)

    Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
    Labels
    3.10 only security fixes stdlib Python modules in the Lib dir type-bug An unexpected behavior, bug, or error
    Projects
    None yet
    Development

    No branches or pull requests

    4 participants