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

gh-85984: Add POSIX pseudo-terminal functions. #102413

Merged
merged 14 commits into from
Jan 29, 2024
Merged

Conversation

8vasu
Copy link
Contributor

@8vasu 8vasu commented Mar 4, 2023

This follows #101831. This is one in a series of PRs aimed at cleaning-up, fixing bugs in, introducing new features in, and updating the code in "Lib/pty.py".

This PR answers the following question: os.forkpty() and pty.fork() return a pair pid, fd, where fd is a file descriptor of the master end of a pseudo-terminal pair; if at a later stage one needs to make some modifications to the slave end (such as setting termios attributes), then how does one obtain access to it without reimplementing ptsname() in Python or loading it from a shared library like someone is doing here: https://stackoverflow.com/questions/52338062/calling-libc-select-from-python-from-pty-master-side?

This is a dependency of #101833, which only needs os.ptsname(). However, this PR also adds os.posix_openpt(), os.grantpt(), and os.unlockpt() since all of these POSIX functions are "companions" of each other.

Signed-off-by: Soumendra Ganguly [email protected]

@8vasu 8vasu changed the title gh-85984: Add posix pseudo-terminal functions. gh-85984: Add POSIX pseudo-terminal functions. Mar 4, 2023
@8vasu
Copy link
Contributor Author

8vasu commented Mar 4, 2023

@gpshead This PR introduces 4 new functions which are simple wrappers. The rationale is explained in the main comment, and other details including testing instructions are provided as inline comments on the diffs.

For now I have converted #101832 and #101833 to drafts since they require more work and are slightly more ambitious projects.

As always, no rush.

@arhadthedev arhadthedev added interpreter-core (Objects, Python, Grammar, and Parser dirs) topic-IO labels Mar 4, 2023
@arhadthedev
Copy link
Member

@gpshead as a reviewer of the preceeding gh-101831.

configure.ac Outdated Show resolved Hide resolved
@8vasu 8vasu requested review from erlend-aasland and removed request for corona10 March 8, 2023 07:23
@8vasu 8vasu force-pushed the posix_pt branch 4 times, most recently from 98516af to 2ab16ad Compare March 15, 2023 02:13
Signed-off-by: Soumendra Ganguly <[email protected]>
@8vasu
Copy link
Contributor Author

8vasu commented Mar 15, 2023

@erlend-aasland Thank you for being patient.

Turns out that unlike unlike openpty(), login_tty(), and forkpty(), these 4 POSIX functions are a part of the standard C library on *nix and not a part of libutil. Therefore, I just needed to add the function names in an AC_CHECK_FUNCS block in configure.ac.

However, maybe we need to make a separate PR to modify the openpty(), login_tty(), forkpty() block in configure.ac to use WITH_SAVE_ENV?

@erlend-aasland
Copy link
Contributor

[…] Therefore, I just needed to add the function names in an AC_CHECK_FUNCS block in configure.ac.

Much better, thanks! There is no need for WITH_SAVE_ENV now, AFAICS.

The configure.ac changes look good to me. I'll leave the rest of the review for @zware and @gpshead who've been active on the issue and previous PRs.

@8vasu
Copy link
Contributor Author

8vasu commented Mar 23, 2023

@erlend-aasland Thank you!

@8vasu
Copy link
Contributor Author

8vasu commented Apr 14, 2023

@gpshead Just a gentle reminder. No rush.

@gpshead gpshead added 🔨 test-with-buildbots Test PR w/ buildbots; report in status section extension-modules C modules in the Modules dir labels May 19, 2023
@bedevere-bot
Copy link

🤖 New build scheduled with the buildbot fleet by @gpshead for commit f6dcd54 🤖

If you want to schedule another build, you need to add the 🔨 test-with-buildbots label again.

@bedevere-bot bedevere-bot removed the 🔨 test-with-buildbots Test PR w/ buildbots; report in status section label May 19, 2023
Modules/posixmodule.c Show resolved Hide resolved
Comment on lines 8103 to 8105
sig_saved = PyOS_setsig(SIGCHLD, SIG_DFL);
ret = grantpt(fd);
PyOS_setsig(SIGCHLD, sig_saved);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This relies on the GIL. IMO, that shouldn't block this PR for 3.13, but I'd like to follow up.

@colesbury, I plan to do lots of reviews this year. Is there anything I should do when I spot a case like this?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's track these in an issue with the "topic-free-threading" label

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Filed here: #114727

Doc/library/os.rst Show resolved Hide resolved
@encukou
Copy link
Member

encukou commented Jan 8, 2024

Reading the manpages further, I see that Linux has a reentrant variant of ptsname:

ptsname_r() is a Linux extension, that is proposed for inclusion in the next major revision of POSIX.1 (Issue 8).

IMO, os.ptsname should call this on Linux if it's available (with a char buffer[MAXPATHLEN+1]), and the docs should note that the function isn't thread-safe on other systems.

@encukou
Copy link
Member

encukou commented Jan 10, 2024

(If you prefer to leave ptsname_r to another PR, that's an option too.)

@8vasu
Copy link
Contributor Author

8vasu commented Jan 10, 2024

@encukou I prefer to include ptsname_r in this PR; thank you for the suggestion and sorry about the late reply. This is on my TODO list; please give me 1 more day (or maybe 2).

@encukou
Copy link
Member

encukou commented Jan 11, 2024

There's no rush, take your time :)

…is available.

Make os.grantpt() use saved errno upon failure.
Update documentation related to the POSIX pty functions.
Add a test for the POSIX pty functions.

Signed-off-by: Soumendra Ganguly <[email protected]>
@8vasu
Copy link
Contributor Author

8vasu commented Jan 20, 2024

@encukou

I have made the changes you requested and more; please take a look at them when you get time:

  1. Change os.grantpt() so that the correct errno is set before calling posix_error().
  2. Change os.ptsname() so that it uses os.ptsname_r() when available; reflect this in docs.
  3. Mention OS_CLOEXEC in the documentation for os.posix_openpt().
  4. Mention in the docs that these functions do not close the master fd upon failure.
  5. One of the tests that you added was using os.openpty() followed by os.grantpt() and os.unlockpt(). That did not make sense to me, since I think openpty() C implementations do this already? Please let me know if this is wrong. I added a new test that follows the standard sequence posix_openpt() -> grantpt() -> unlockpt() -> ptsname() that this new test subsumed test that you had written for ptsname().
  6. I used https://github.com/tiran/cpython_autoconf (autoconf version 2.71) written by @tiran to generate the configure script. The test Tests / Check if generated files are up to date (pull_request) failed; maybe I should use something else now? The default autoconf on my Debian system is also (GNU Autoconf) 2.71, so generating using that will not make a difference I think.
  7. About the pty pair terminology: I noticed that you used the naming scheme main, second. I used the alternative mother, son as a test to see if you like it. While I am not a CPython maintainer and hence do not have much say in deciding which directions such a prestigious project should take, I have severe clinical OCD/OCPD and would like to issue a warning here: using generic names like main_fd, second_fd, or parent_fd, child_fd, or server_fd, client_fd will lead to havoc very easily. Just take a look at the source of your very own https://github.com/python/cpython/blob/3.12/Lib/pty.py and try a search/replace involving terms parent/child (already used by processes) or a program like ssh(1) which uses the terms server/client in networking context and also sets up a pty pair. On the other hand one can easily have first_fd, second_fd, third_fd or even main_fd in their existing programs. The pair main, subordinate comes to mind, but again, main_fd might be taken and main, subordinate carry the same overtones as master, slave, which is what you are trying to avoid in the first place. For the most comfortable change, we must use something that is -
    • not already taken to almost absolutely eliminate any chances of collision while
    • maintaining that the initials m and s stay the same.

Edit: while this is not the focus of this PR, about point 7 above, I have generated some data using grep on the cpython repository and added it here: #85984 (comment) (commenting again for better reach).

Edit: Sorry about the obsessive spam; I cannot help myself, I have OCD. Before taking a break from this, I would like to finally point out that a pty is a pair of files and not a pair of fds. Therefore, if we call them main file/main end and second file/second end, then in code the corresponding terms might be main_fd and second_fd. That is misleading: a file can be addressed via multiple fds and upon reopening the second file/second end, we should call the new fd third_fd or second_fd_of_second_end and not reopened_second_fd (since you open files and not file descriptors).

ON THE OTHER HAND: main end and secondARY end might be better. I still think we should grep for all possible terms to avoid collision.

@encukou
Copy link
Member

encukou commented Jan 22, 2024

1-4. Thank you!
5. Thanks. I'm not an expert here -- I'm learning about ptys and their C API as I go. I'm happy that the test I added is hackable :)
6. The command is make regen-configure. If that doesn't work for you, run the ubuntu:22.04 container (that is currently what the makefile uses, via /Tools/build/regen-configure.sh). If it's a problem, let me know and I'll regenerate this.
If you found @tiran's outdated container mentioned in some current documentation, please let me know -- we should fix that. (Sadly we can't fix all the docs, like personal notes or shell histories...)
7. For the tests, local variable names don't matter much; it's OK to be a bit creative. For the API, this adds fd -- no problem there! For docs, it's master/slave for now; changing that will be a different discussion (in which the Editorial board might well decide to keep following POSIX).

Let's pretend second_fd stands for fd of the second file, and reopened_second_fd stands for fd for the reopened second file, or reopened fd for the second file. Does that work better? If not, change this to whatever feels best :)

@8vasu
Copy link
Contributor Author

8vasu commented Jan 23, 2024

@encukou you are very kind. Let me work on the reconfigure (and everything else you suggested).

Signed-off-by: Soumendra Ganguly <[email protected]>
@8vasu
Copy link
Contributor Author

8vasu commented Jan 24, 2024

@encukou Looks like all checks have passed! Your /Tools/build/regen-configure.sh suggestion worked like a charm... after I moved myself away from a directory that had colons (:) in its name as a time (hr, min, sec) separator; docker does not seem to like that.

Copy link
Member

@encukou encukou left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

+1 from me!
@gpshead, do you want to take another look?

Doc/library/os.rst Show resolved Hide resolved
@encukou encukou enabled auto-merge (squash) January 29, 2024 15:40
@encukou encukou merged commit e351ca3 into python:main Jan 29, 2024
32 checks passed
@8vasu
Copy link
Contributor Author

8vasu commented Jan 29, 2024

@encukou Thank you for merging this! Looks like finally after years I have all the prerequisites for #101833. That one now needs to be updated and maybe broken up into a few parts in case it becomes too difficult to review: one for signal handling, one for winsize handling, one for adjusting the tests, etc. I will do this sometime during the next few months.

@encukou
Copy link
Member

encukou commented Jan 30, 2024

Thank you for perservering!
Looks like the main thing missing in 101833 is tests, so, a good strategy might be to send a PR whenever you get a test working :)

If I happen to miss a notification (= I don't reply for about a week), please ping me again.

@8vasu
Copy link
Contributor Author

8vasu commented Jan 30, 2024

@encukou

Looks like the main thing missing in 101833 is tests

The tests for that were my first contribution to CPython in 2020. However, I still need to update the tests to use the various robust functions that I have added to os, termios, and tty over the years instead of the ad-hoc ones that I wrote in Lib/test/test_pty.py. If additional tests are required, I will definitely add them if you wish :)

aisk pushed a commit to aisk/cpython that referenced this pull request Feb 11, 2024
Signed-off-by: Soumendra Ganguly <[email protected]>
Co-authored-by: Gregory P. Smith <[email protected]>
Co-authored-by: Petr Viktorin <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
extension-modules C modules in the Modules dir topic-IO type-feature A feature request or enhancement
Projects
None yet
Development

Successfully merging this pull request may close these issues.

7 participants