Skip to content

Commit 7e5ccce

Browse files
committed
Simplify capture clauses and ensure group leader is reversed
1 parent 2a430b1 commit 7e5ccce

File tree

2 files changed

+30
-49
lines changed

2 files changed

+30
-49
lines changed

lib/ex_unit/lib/ex_unit/capture_io.ex

+18-44
Original file line numberDiff line numberDiff line change
@@ -151,23 +151,15 @@ defmodule ExUnit.CaptureIO do
151151
@spec capture_io(atom() | pid() | String.t() | keyword(), (-> any())) :: String.t()
152152
def capture_io(device_pid_input_or_options, fun)
153153

154-
def capture_io(device, fun) when is_atom(device) and is_function(fun, 0) do
155-
{_result, capture} = with_io(device, fun)
154+
def capture_io(device_or_pid, fun)
155+
when (is_atom(device_or_pid) or is_pid(device_or_pid)) and is_function(fun, 0) do
156+
{_result, capture} = with_io(device_or_pid, fun)
156157
capture
157158
end
158159

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-
164-
def capture_io(input, fun) when is_binary(input) and is_function(fun, 0) do
165-
{_result, capture} = with_io(input, fun)
166-
capture
167-
end
168-
169-
def capture_io(options, fun) when is_list(options) and is_function(fun, 0) do
170-
{_result, capture} = with_io(options, fun)
160+
def capture_io(input_or_options, fun)
161+
when (is_binary(input_or_options) or is_list(input_or_options)) and is_function(fun, 0) do
162+
{_result, capture} = with_io(input_or_options, fun)
171163
capture
172164
end
173165

@@ -178,28 +170,9 @@ defmodule ExUnit.CaptureIO do
178170
"""
179171
@spec capture_io(atom() | pid(), String.t() | keyword(), (-> any())) :: String.t()
180172
def capture_io(device_or_pid, input_or_options, fun)
181-
182-
def capture_io(device, input, fun)
183-
when is_atom(device) and is_binary(input) and is_function(fun, 0) do
184-
{_result, capture} = with_io(device, input, fun)
185-
capture
186-
end
187-
188-
def capture_io(device, options, fun)
189-
when is_atom(device) and is_list(options) and is_function(fun, 0) do
190-
{_result, capture} = with_io(device, options, fun)
191-
capture
192-
end
193-
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)
173+
when (is_atom(device_or_pid) or is_pid(device_or_pid)) and
174+
(is_binary(input_or_options) or is_list(input_or_options)) and is_function(fun, 0) do
175+
{_result, capture} = with_io(device_or_pid, input_or_options, fun)
203176
capture
204177
end
205178

@@ -262,7 +235,7 @@ defmodule ExUnit.CaptureIO do
262235

263236
def with_io(device, input, fun)
264237
when is_atom(device) and is_binary(input) and is_function(fun, 0) do
265-
with_io(device, [input: input], fun)
238+
do_with_io(map_dev(device), [input: input], fun)
266239
end
267240

268241
def with_io(device, options, fun)
@@ -272,27 +245,28 @@ defmodule ExUnit.CaptureIO do
272245

273246
def with_io(pid, input, fun)
274247
when is_pid(pid) and is_binary(input) and is_function(fun, 0) do
275-
with_io(pid, [input: input], fun)
248+
do_with_io(pid, [input: input], fun)
276249
end
277250

278251
def with_io(pid, options, fun)
279252
when is_pid(pid) and is_list(options) and is_function(fun, 0) do
280253
do_with_io(pid, options, fun)
281254
end
282255

283-
defp map_dev(:stdio), do: :standard_io
256+
defp map_dev(:standard_io), do: self()
257+
defp map_dev(:stdio), do: self()
284258
defp map_dev(:stderr), do: :standard_error
285259
defp map_dev(other), do: other
286260

287-
defp do_with_io(device_or_pid, options, fun)
288-
when device_or_pid == :standard_io or is_pid(device_or_pid) do
261+
defp do_with_io(pid, options, fun) when is_pid(pid) do
289262
prompt_config = Keyword.get(options, :capture_prompt, true)
290263
encoding = Keyword.get(options, :encoding, :unicode)
291264
input = Keyword.get(options, :input, "")
292265

293-
original_gl = Process.group_leader()
266+
{:group_leader, original_gl} =
267+
Process.info(pid, :group_leader) || {:group_leader, Process.group_leader()}
268+
294269
{: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()
296270

297271
try do
298272
Process.group_leader(pid, capture_gl)
@@ -302,7 +276,7 @@ defmodule ExUnit.CaptureIO do
302276
end
303277
end
304278

305-
defp do_with_io(device, options, fun) do
279+
defp do_with_io(device, options, fun) when is_atom(device) do
306280
input = Keyword.get(options, :input, "")
307281
encoding = Keyword.get(options, :encoding, :unicode)
308282

lib/ex_unit/test/ex_unit/capture_io_test.exs

+12-5
Original file line numberDiff line numberDiff line change
@@ -31,12 +31,15 @@ defmodule ExUnit.CaptureIOTest do
3131
defmodule MockProc do
3232
use GenServer
3333

34-
def start_link do
35-
GenServer.start_link(__MODULE__, [])
34+
def start_link(gl) do
35+
GenServer.start_link(__MODULE__, gl)
3636
end
3737

3838
@impl GenServer
39-
def init(_), do: {:ok, nil}
39+
def init(gl) do
40+
Process.group_leader(self(), gl)
41+
{:ok, nil}
42+
end
4043

4144
@impl GenServer
4245
def handle_call({:stdio, message}, _from, state) do
@@ -502,7 +505,10 @@ defmodule ExUnit.CaptureIOTest do
502505
end
503506

504507
test "capture_io with a separate process" do
505-
{:ok, pid} = MockProc.start_link()
508+
{:ok, gl} = StringIO.open("")
509+
pid = start_supervised!({MockProc, gl})
510+
511+
assert Process.info(pid, :group_leader) == {:group_leader, gl}
506512

507513
assert capture_io(pid, fn ->
508514
GenServer.call(pid, {:stdio, "a"})
@@ -524,7 +530,8 @@ defmodule ExUnit.CaptureIOTest do
524530
GenServer.call(pid, {:stderr, "uhoh"})
525531
end) == "uhoh\n"
526532

527-
GenServer.stop(pid)
533+
assert Process.info(pid, :group_leader) == {:group_leader, gl}
534+
assert StringIO.contents(gl) == {"", ""}
528535
end
529536

530537
test "with_io" do

0 commit comments

Comments
 (0)