Skip to content

Commit 2a430b1

Browse files
authored
Support IO capture of group leader from external process (#13374)
1 parent d9005eb commit 2a430b1

File tree

2 files changed

+142
-30
lines changed

2 files changed

+142
-30
lines changed

lib/ex_unit/lib/ex_unit/capture_io.ex

+84-30
Original file line numberDiff line numberDiff line change
@@ -29,15 +29,15 @@ defmodule ExUnit.CaptureIO do
2929
3030
Returns the binary which is the captured output.
3131
32-
By default, `capture_io` replaces the `group_leader` (`:stdio`)
33-
for the current process. Capturing the group leader is done per
34-
process and therefore can be done concurrently.
32+
By default, `capture_io` replaces the `Process.group_leader/0` of the current
33+
process, which is the process used by default for all IO operations. Capturing
34+
the group leader of the current process is safe to run concurrently, under
35+
`async: true` tests. You may also explicitly capture the group leader of
36+
another process, however that is not safe to do concurrently.
3537
36-
However, the capturing of any other named device, such as `:stderr`,
37-
happens globally and persists until the function has ended. While this means
38-
it is safe to run your tests with `async: true` in many cases, captured output
39-
may include output from a different test and care must be taken when using
40-
`capture_io` with a named process asynchronously.
38+
You may also capture any other named IO device, such as `:stderr`. This is
39+
also safe to run concurrently but, if several tests are writting to the same
40+
device at once, captured output may include output from a different test.
4141
4242
A developer can set a string as an input. The default input is an empty
4343
string. If capturing a named device asynchronously, an input can only be given
@@ -51,15 +51,28 @@ defmodule ExUnit.CaptureIO do
5151
5252
## IO devices
5353
54-
You may capture the IO from any registered IO device. The device name given
55-
must be an atom representing the name of a registered process. In addition,
56-
Elixir provides two shortcuts:
54+
You may capture the IO of the group leader of any process, by passing a `pid`
55+
as argument, or from any registered IO device given as an `atom`. Here are
56+
some example values:
5757
58-
* `:stdio` - a shortcut for `:standard_io`, which maps to
59-
the current `Process.group_leader/0` in Erlang
58+
* `:stdio`, `:standard_io` - a shortcut for capturing the group leader
59+
of the current process. It is equivalent to passing `self()` as the
60+
first argument. This is safe to run concurrently and captures only
61+
the of the current process or any child process spawned inside the
62+
given function
6063
61-
* `:stderr` - a shortcut for the named process `:standard_error`
62-
provided in Erlang
64+
* `:stderr`, `:standard_error` - captures all IO to standard error
65+
(represented internally by an Erlang process named `:standard_error`).
66+
This is safe to run concurrently but it will capture the output
67+
of any other test writing to the same named device
68+
69+
* any other atom - captures all IO to the given device given by the
70+
atom. This is safe to run concurrently but it will capture the output
71+
of any other test writing to the same named device
72+
73+
* any other pid (since v1.17.0) - captures all IO to the group leader
74+
of the given process. This option is not safe to run concurrently
75+
if the pid is not `self()`. Tests using this value must set `async: true`
6376
6477
## Options
6578
@@ -91,10 +104,10 @@ defmodule ExUnit.CaptureIO do
91104
...> end) == "this is input"
92105
true
93106
94-
Note it is fine to use `==` with standard IO, because the content is captured
95-
per test process. However, `:stderr` is shared across all tests, so you will
96-
want to use `=~` instead of `==` for assertions on `:stderr` if your tests
97-
are async:
107+
Note it is fine to use `==` with `:stdio` (the default IO device), because
108+
the content is captured per test process. However, `:stderr` is shared
109+
across all tests, so you will want to use `=~` instead of `==` for assertions
110+
on `:stderr` if your tests are async:
98111
99112
iex> capture_io(:stderr, fn -> IO.write(:stderr, "john") end) =~ "john"
100113
true
@@ -110,6 +123,14 @@ defmodule ExUnit.CaptureIO do
110123
Otherwise, if the standard error of any other test is captured, the test will
111124
fail.
112125
126+
To capture the IO from another process, you can pass a `pid`:
127+
128+
capture_io(GenServer.whereis(MyServer), fn ->
129+
GenServer.call(MyServer, :do_something)
130+
end)
131+
132+
Tests that directly capture a PID cannot run concurrently.
133+
113134
## Returning values
114135
115136
As seen in the examples above, `capture_io` returns the captured output.
@@ -127,14 +148,19 @@ defmodule ExUnit.CaptureIO do
127148
128149
See `capture_io/1` for more information.
129150
"""
130-
@spec capture_io(atom() | String.t() | keyword(), (-> any())) :: String.t()
131-
def capture_io(device_input_or_options, fun)
151+
@spec capture_io(atom() | pid() | String.t() | keyword(), (-> any())) :: String.t()
152+
def capture_io(device_pid_input_or_options, fun)
132153

133154
def capture_io(device, fun) when is_atom(device) and is_function(fun, 0) do
134155
{_result, capture} = with_io(device, fun)
135156
capture
136157
end
137158

159+
def capture_io(pid, fun) when is_pid(pid) and is_function(fun, 0) do
160+
{_result, capture} = with_io(pid, fun)
161+
capture
162+
end
163+
138164
def capture_io(input, fun) when is_binary(input) and is_function(fun, 0) do
139165
{_result, capture} = with_io(input, fun)
140166
capture
@@ -150,8 +176,8 @@ defmodule ExUnit.CaptureIO do
150176
151177
See `capture_io/1` for more information.
152178
"""
153-
@spec capture_io(atom(), String.t() | keyword(), (-> any())) :: String.t()
154-
def capture_io(device, input_or_options, fun)
179+
@spec capture_io(atom() | pid(), String.t() | keyword(), (-> any())) :: String.t()
180+
def capture_io(device_or_pid, input_or_options, fun)
155181

156182
def capture_io(device, input, fun)
157183
when is_atom(device) and is_binary(input) and is_function(fun, 0) do
@@ -165,6 +191,18 @@ defmodule ExUnit.CaptureIO do
165191
capture
166192
end
167193

194+
def capture_io(pid, input, fun)
195+
when is_pid(pid) and is_binary(input) and is_function(fun, 0) do
196+
{_result, capture} = with_io(pid, input, fun)
197+
capture
198+
end
199+
200+
def capture_io(pid, options, fun)
201+
when is_pid(pid) and is_list(options) and is_function(fun, 0) do
202+
{_result, capture} = with_io(pid, options, fun)
203+
capture
204+
end
205+
168206
@doc ~S"""
169207
Invokes the given `fun` and returns the result and captured output.
170208
@@ -194,13 +232,17 @@ defmodule ExUnit.CaptureIO do
194232
See `with_io/1` for more information.
195233
"""
196234
@doc since: "1.13.0"
197-
@spec with_io(atom() | String.t() | keyword(), (-> any())) :: {any(), String.t()}
198-
def with_io(device_input_or_options, fun)
235+
@spec with_io(atom() | pid() | String.t() | keyword(), (-> any())) :: {any(), String.t()}
236+
def with_io(device_pid_input_or_options, fun)
199237

200238
def with_io(device, fun) when is_atom(device) and is_function(fun, 0) do
201239
with_io(device, [], fun)
202240
end
203241

242+
def with_io(pid, fun) when is_pid(pid) and is_function(fun, 0) do
243+
with_io(pid, [], fun)
244+
end
245+
204246
def with_io(input, fun) when is_binary(input) and is_function(fun, 0) do
205247
with_io(:stdio, [input: input], fun)
206248
end
@@ -215,8 +257,8 @@ defmodule ExUnit.CaptureIO do
215257
See `with_io/1` for more information.
216258
"""
217259
@doc since: "1.13.0"
218-
@spec with_io(atom(), String.t() | keyword(), (-> any())) :: {any(), String.t()}
219-
def with_io(device, input_or_options, fun)
260+
@spec with_io(atom() | pid(), String.t() | keyword(), (-> any())) :: {any(), String.t()}
261+
def with_io(device_or_pid, input_or_options, fun)
220262

221263
def with_io(device, input, fun)
222264
when is_atom(device) and is_binary(input) and is_function(fun, 0) do
@@ -228,23 +270,35 @@ defmodule ExUnit.CaptureIO do
228270
do_with_io(map_dev(device), options, fun)
229271
end
230272

273+
def with_io(pid, input, fun)
274+
when is_pid(pid) and is_binary(input) and is_function(fun, 0) do
275+
with_io(pid, [input: input], fun)
276+
end
277+
278+
def with_io(pid, options, fun)
279+
when is_pid(pid) and is_list(options) and is_function(fun, 0) do
280+
do_with_io(pid, options, fun)
281+
end
282+
231283
defp map_dev(:stdio), do: :standard_io
232284
defp map_dev(:stderr), do: :standard_error
233285
defp map_dev(other), do: other
234286

235-
defp do_with_io(:standard_io, options, fun) do
287+
defp do_with_io(device_or_pid, options, fun)
288+
when device_or_pid == :standard_io or is_pid(device_or_pid) do
236289
prompt_config = Keyword.get(options, :capture_prompt, true)
237290
encoding = Keyword.get(options, :encoding, :unicode)
238291
input = Keyword.get(options, :input, "")
239292

240293
original_gl = Process.group_leader()
241294
{:ok, capture_gl} = StringIO.open(input, capture_prompt: prompt_config, encoding: encoding)
295+
pid = if is_pid(device_or_pid), do: device_or_pid, else: self()
242296

243297
try do
244-
Process.group_leader(self(), capture_gl)
298+
Process.group_leader(pid, capture_gl)
245299
do_capture_gl(capture_gl, fun)
246300
after
247-
Process.group_leader(self(), original_gl)
301+
Process.group_leader(pid, original_gl)
248302
end
249303
end
250304

lib/ex_unit/test/ex_unit/capture_io_test.exs

+58
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,38 @@ defmodule ExUnit.CaptureIOTest do
2828
end
2929
end
3030

31+
defmodule MockProc do
32+
use GenServer
33+
34+
def start_link do
35+
GenServer.start_link(__MODULE__, [])
36+
end
37+
38+
@impl GenServer
39+
def init(_), do: {:ok, nil}
40+
41+
@impl GenServer
42+
def handle_call({:stdio, message}, _from, state) do
43+
IO.puts(message)
44+
{:reply, :ok, state}
45+
end
46+
47+
@impl GenServer
48+
def handle_call({:prompt, prompt}, _from, state) do
49+
prompt
50+
|> IO.gets()
51+
|> IO.puts()
52+
53+
{:reply, :ok, state}
54+
end
55+
56+
@impl GenServer
57+
def handle_call({:stderr, message}, _from, state) do
58+
IO.puts(:stderr, message)
59+
{:reply, :ok, state}
60+
end
61+
end
62+
3163
import ExUnit.CaptureIO
3264
doctest ExUnit.CaptureIO, import: true
3365

@@ -469,6 +501,32 @@ defmodule ExUnit.CaptureIOTest do
469501
end
470502
end
471503

504+
test "capture_io with a separate process" do
505+
{:ok, pid} = MockProc.start_link()
506+
507+
assert capture_io(pid, fn ->
508+
GenServer.call(pid, {:stdio, "a"})
509+
end) == "a\n"
510+
511+
assert capture_io(pid, [input: "b"], fn ->
512+
GenServer.call(pid, {:prompt, "> "})
513+
end) == "> b\n"
514+
515+
assert capture_io(pid, "c", fn ->
516+
GenServer.call(pid, {:prompt, "> "})
517+
end) == "> c\n"
518+
519+
assert capture_io(pid, [input: "d", capture_prompt: false], fn ->
520+
GenServer.call(pid, {:prompt, "> "})
521+
end) == "d\n"
522+
523+
assert capture_io(:stderr, fn ->
524+
GenServer.call(pid, {:stderr, "uhoh"})
525+
end) == "uhoh\n"
526+
527+
GenServer.stop(pid)
528+
end
529+
472530
test "with_io" do
473531
assert with_io(fn ->
474532
:io.put_chars("xyz")

0 commit comments

Comments
 (0)