我目前正在为一个模块编写一个测试,该模块在一个以spawn_link(?MODULE, init, [self()])
.
在我的 eunit 测试中,我定义了一个 setup 和 teardown 函数以及一组测试生成器。
all_tests_test_() ->
{inorder, {
foreach,
fun setup/0,
fun teardown/1,
[
fun my_test/1
]}
}.
设置乐趣创建了被测过程:
setup() ->
{ok, Pid} = protocol:start_link(),
process_flag(trap_exit,true),
error_logger:info_msg("[~p] Setting up process ~p~n", [self(), Pid]),
Pid.
测试看起来像这样:
my_test(Pid) ->
[ fun() ->
error_logger:info_msg("[~p] Sending to ~p~n", [self(), Pid]),
Pid ! something,
receive
Msg -> ?assertMatch(expected_result, Msg)
after
500 -> ?assert(false)
end
end ].
我的大多数模块都是 gen_server 但为此我认为没有所有 gen_server 样板代码会更容易......
测试的输出如下所示:
=INFO REPORT==== 31-Mar-2014::21:20:12 ===
[<0.117.0>] Setting up process <0.122.0>
=INFO REPORT==== 31-Mar-2014::21:20:12 ===
[<0.124.0>] Sending to <0.122.0>
=INFO REPORT==== 31-Mar-2014::21:20:12 ===
[<0.122.0>] Sending expected_result to <0.117.0>
protocol_test: my_test...*failed*
in function protocol_test:'-my_test/1-fun-0-'/0 (test/protocol_test.erl, line 37)
**error:{assertion_failed,[{module,protocol_test},
{line,37},
{expression,"false"},
{expected,true},
{value,false}]}
从 Pids 中,您可以看到运行 setup (117) 的任何进程与运行测试用例 (124) 的进程不同。然而,被测试的过程是相同的(122)。这会导致测试用例失败,因为接收端永远不会收到消息并且会超时。
这是 eunit 生成一个新进程来运行测试用例的预期行为吗?
一般来说,是否有更好的方法来测试进程或其他异步行为(如强制转换)?或者你会建议总是使用 gen_server 来拥有一个同步接口?
谢谢!
[编辑]
澄清一下,协议是如何知道这个过程的,这很start_link/0
有趣:
start_link() ->
Pid = spawn_link(?MODULE, init, [self()]),
{ok, Pid}.
该协议与调用者紧密相连。如果他们中的任何一个崩溃,我希望另一个也死。我知道我可以使用 gen_server 和 supervisors 并且实际上它在应用程序的某些部分中做到了,但是对于这个模块,我认为它有点过头了。