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

Support IO capture from an already-started process #13374

Merged
merged 3 commits into from
Feb 29, 2024

Conversation

voughtdq
Copy link
Contributor

capture_io/1,2,3 does not capture IO from a process that was already started. This change allows someone to pass a :pid option so that Process.group_leader/2 is called on the target process rather than the calling process.

`capture_io/1,2,3` does not capture IO from a process that was already
started. This change allows someone to pass a `:pid` option so that
`Process.group_leader/2` is called on the target process rather than
the calling process.
@josevalim
Copy link
Member

Thank you for the PR but unfortunately this is a bit trickier documentation wise.

Capturing anything other than stdio is global, you don't need :pid. So the test for capturing stderr from the GenServer is misleading. Furthermore, capture_io is documented as being safe under concurrency scenarios, but that's not true if you are changing the group leader of a process different than self.

Given this, I believe the preferred API would be to pass the pid as argument instead of :stdio or :stderr, since it does not overlap with them, and document it. I will submit changes to the docs, and if you agree, you can implement as proposed in my changes?

Thank you.

@josevalim
Copy link
Member

I pushed a commit here: 73cc6e7

I tried to submit a PR to your fork but GitHub did not allow me to. We would be able to accept a PR that implements the API above.

@voughtdq
Copy link
Contributor Author

Given this, I believe the preferred API would be to pass the pid as argument instead of :stdio or :stderr, since it does not overlap with them, and document it. I will submit changes to the docs, and if you agree, you can implement as proposed in my changes?

Sounds good, I'll implement as described.

One thing about taking pid as the first argument:

Do you anticipate that it will be useful or desirable for users to set a custom device pid? If we go the route of taking pid as the first argument, this closes the door to being able to accept a non-named device. Thoughts?

@josevalim
Copy link
Member

Do you anticipate that it will be useful or desirable for users to set a custom device pid? If we go the route of taking pid as the first argument, this closes the door to being able to accept a non-named device. Thoughts?

The whole mechanism works because we make the named device point to another PID. If there is no name, there is no way to redirect.

@spec capture_io(atom() | String.t() | keyword(), (-> any())) :: String.t()
def capture_io(device_input_or_options, fun)
@spec capture_io(atom() | pid() | String.t() | keyword(), (-> any())) :: String.t()
def capture_io(device_pid_input_or_options, fun)
Copy link
Contributor Author

Choose a reason for hiding this comment

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

I was on the fence on whether to collapse these into fewer clauses. I can do that if you think it improves readability.


original_gl = Process.group_leader()
{:ok, capture_gl} = StringIO.open(input, capture_prompt: prompt_config, encoding: encoding)
pid = if is_pid(device_or_pid), do: device_or_pid, else: self()
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Originally had a pid_for private function for this, but it seemed excessive.

defp map_dev(:stdio), do: :standard_io
defp map_dev(:stderr), do: :standard_error
defp map_dev(other), do: other

defp do_with_io(:standard_io, options, fun) do
defp do_with_io(device_or_pid, options, fun)
Copy link
Member

Choose a reason for hiding this comment

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

I would have a separate clause for this instead:

Suggested change
defp do_with_io(device_or_pid, options, fun)
defp do_with_io(:standard_io, options, fun), do: do_with_io(self(), options, fun)

WDYT?

Copy link
Member

Choose a reason for hiding this comment

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

Or maybe change this in map_dev. No worries, I will merge it and play with it locally.

@josevalim josevalim merged commit 2a430b1 into elixir-lang:main Feb 29, 2024
8 checks passed
@josevalim
Copy link
Member

💚 💙 💜 💛 ❤️

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging this pull request may close these issues.

2 participants