pooler_tests.erl 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664
  1. -module(pooler_tests).
  2. -include_lib("eunit/include/eunit.hrl").
  3. -compile([export_all]).
  4. % The `user' processes represent users of the pooler library. A user
  5. % process will take a pid, report details on the pid it has, release
  6. % and take a new pid, stop cleanly, and crash.
  7. start_user() ->
  8. spawn(fun() -> user_loop(start) end).
  9. user_id(Pid) ->
  10. Pid ! {get_tc_id, self()},
  11. receive
  12. {Type, Id} ->
  13. {Type, Id}
  14. end.
  15. user_new_tc(Pid) ->
  16. Pid ! new_tc.
  17. user_stop(Pid) ->
  18. Pid ! stop.
  19. user_crash(Pid) ->
  20. Pid ! crash.
  21. user_loop(Atom) when Atom =:= error_no_members orelse Atom =:= start ->
  22. user_loop(pooler:take_member(test_pool_1));
  23. user_loop(MyTC) ->
  24. receive
  25. {get_tc_id, From} ->
  26. From ! pooled_gs:get_id(MyTC),
  27. user_loop(MyTC);
  28. {ping_tc, From} ->
  29. From ! pooled_gs:ping(MyTC),
  30. user_loop(MyTC);
  31. {ping_count, From} ->
  32. From ! pooled_gs:ping_count(MyTC),
  33. user_loop(MyTC);
  34. new_tc ->
  35. pooler:return_member(test_pool_1, MyTC, ok),
  36. MyNewTC = pooler:take_member(test_pool_1),
  37. user_loop(MyNewTC);
  38. stop ->
  39. pooler:return_member(test_pool_1, MyTC, ok),
  40. stopped;
  41. crash ->
  42. erlang:error({user_loop, kaboom})
  43. end.
  44. % The `tc' processes represent the pids tracked by pooler for testing.
  45. % They have a type and an ID and can report their type and ID and
  46. % stop.
  47. tc_loop({Type, Id}) ->
  48. receive
  49. {get_id, From} ->
  50. From ! {ok, Type, Id},
  51. tc_loop({Type, Id});
  52. stop -> stopped;
  53. crash ->
  54. erlang:error({tc_loop, kaboom})
  55. end.
  56. get_tc_id(Pid) ->
  57. Pid ! {get_id, self()},
  58. receive
  59. {ok, Type, Id} ->
  60. {Type, Id}
  61. after 200 ->
  62. timeout
  63. end.
  64. stop_tc(Pid) ->
  65. Pid ! stop.
  66. tc_starter(Type) ->
  67. Ref = make_ref(),
  68. spawn_link(fun() -> tc_loop({Type, Ref}) end).
  69. assert_tc_valid(Pid) ->
  70. ?assertMatch({_Type, _Ref}, get_tc_id(Pid)),
  71. ok.
  72. % tc_sanity_test() ->
  73. % Pid1 = tc_starter("1"),
  74. % {"1", Id1} = get_tc_id(Pid1),
  75. % Pid2 = tc_starter("1"),
  76. % {"1", Id2} = get_tc_id(Pid2),
  77. % ?assertNot(Id1 == Id2),
  78. % stop_tc(Pid1),
  79. % stop_tc(Pid2).
  80. % user_sanity_test() ->
  81. % Pid1 = tc_starter("1"),
  82. % User = spawn(fun() -> user_loop(Pid1) end),
  83. % ?assertMatch({"1", _Ref}, user_id(User)),
  84. % user_crash(User),
  85. % stop_tc(Pid1).
  86. pooler_basics_via_config_test_() ->
  87. {setup,
  88. fun() ->
  89. application:set_env(pooler, metrics_module, fake_metrics),
  90. fake_metrics:start_link()
  91. end,
  92. fun(_X) ->
  93. fake_metrics:stop()
  94. end,
  95. {foreach,
  96. % setup
  97. fun() ->
  98. Pools = [[{name, test_pool_1},
  99. {max_count, 3},
  100. {init_count, 2},
  101. {start_mfa,
  102. {pooled_gs, start_link, [{"type-0"}]}}]],
  103. application:set_env(pooler, pools, Pools),
  104. error_logger:delete_report_handler(error_logger_tty_h),
  105. application:start(pooler)
  106. end,
  107. fun(_X) ->
  108. application:stop(pooler)
  109. end,
  110. basic_tests()}}.
  111. pooler_basics_dynamic_test_() ->
  112. {setup,
  113. fun() ->
  114. application:set_env(pooler, metrics_module, fake_metrics),
  115. fake_metrics:start_link()
  116. end,
  117. fun(_X) ->
  118. fake_metrics:stop()
  119. end,
  120. {foreach,
  121. % setup
  122. fun() ->
  123. Pool = [{name, test_pool_1},
  124. {max_count, 3},
  125. {init_count, 2},
  126. {start_mfa,
  127. {pooled_gs, start_link, [{"type-0"}]}}],
  128. application:unset_env(pooler, pools),
  129. error_logger:delete_report_handler(error_logger_tty_h),
  130. application:start(pooler),
  131. pooler:new_pool(Pool)
  132. end,
  133. fun(_X) ->
  134. application:stop(pooler)
  135. end,
  136. basic_tests()}}.
  137. basic_tests() ->
  138. [
  139. {"there are init_count members at start",
  140. fun() ->
  141. Stats = [ P || {P, {_, free, _}} <- pooler:pool_stats(test_pool_1) ],
  142. ?assertEqual(2, length(Stats))
  143. end},
  144. {"take and return one",
  145. fun() ->
  146. P = pooler:take_member(test_pool_1),
  147. ?assertMatch({"type-0", _Id}, pooled_gs:get_id(P)),
  148. ok = pooler:return_member(test_pool_1, P, ok)
  149. end},
  150. {"take and return one, named pool",
  151. fun() ->
  152. P = pooler:take_member(test_pool_1),
  153. ?assertMatch({"type-0", _Id}, pooled_gs:get_id(P)),
  154. ok, pooler:return_member(test_pool_1, P)
  155. end},
  156. {"attempt to take form unknown pool",
  157. fun() ->
  158. %% since pools are now servers, an unknown pool will timeout
  159. ?assertExit({noproc, _}, pooler:take_member(bad_pool_name))
  160. end},
  161. {"members creation is triggered after pool exhaustion until max",
  162. fun() ->
  163. %% init count is 2
  164. Pids0 = [pooler:take_member(test_pool_1), pooler:take_member(test_pool_1)],
  165. %% since new member creation is async, can only assert
  166. %% that we will get a pid, but may not be first try.
  167. Pids = get_n_pids(1, Pids0),
  168. %% pool is at max now, requests should give error
  169. ?assertEqual(error_no_members, pooler:take_member(test_pool_1)),
  170. ?assertEqual(error_no_members, pooler:take_member(test_pool_1)),
  171. PRefs = [ R || {_T, R} <- [ pooled_gs:get_id(P) || P <- Pids ] ],
  172. % no duplicates
  173. ?assertEqual(length(PRefs), length(lists:usort(PRefs)))
  174. end
  175. },
  176. {"pids are reused most recent return first",
  177. fun() ->
  178. P1 = pooler:take_member(test_pool_1),
  179. P2 = pooler:take_member(test_pool_1),
  180. ?assertNot(P1 == P2),
  181. ok = pooler:return_member(test_pool_1, P1, ok),
  182. ok = pooler:return_member(test_pool_1, P2, ok),
  183. % pids are reused most recent first
  184. ?assertEqual(P2, pooler:take_member(test_pool_1)),
  185. ?assertEqual(P1, pooler:take_member(test_pool_1))
  186. end},
  187. {"if an in-use pid crashes it is replaced",
  188. fun() ->
  189. Pids0 = get_n_pids(3, []),
  190. Ids0 = [ pooled_gs:get_id(P) || P <- Pids0 ],
  191. % crash them all
  192. [ pooled_gs:crash(P) || P <- Pids0 ],
  193. Pids1 = get_n_pids(3, []),
  194. Ids1 = [ pooled_gs:get_id(P) || P <- Pids1 ],
  195. [ ?assertNot(lists:member(I, Ids0)) || I <- Ids1 ]
  196. end
  197. },
  198. {"if a free pid crashes it is replaced",
  199. fun() ->
  200. FreePids = [ P || {P, {_, free, _}} <- pooler:pool_stats(test_pool_1) ],
  201. [ exit(P, kill) || P <- FreePids ],
  202. Pids1 = get_n_pids(3, []),
  203. ?assertEqual(3, length(Pids1))
  204. end},
  205. {"if a pid is returned with bad status it is replaced",
  206. fun() ->
  207. Pids0 = get_n_pids(3, []),
  208. Ids0 = [ pooled_gs:get_id(P) || P <- Pids0 ],
  209. % return them all marking as bad
  210. [ pooler:return_member(test_pool_1, P, fail) || P <- Pids0 ],
  211. Pids1 = get_n_pids(3, []),
  212. Ids1 = [ pooled_gs:get_id(P) || P <- Pids1 ],
  213. [ ?assertNot(lists:member(I, Ids0)) || I <- Ids1 ]
  214. end
  215. },
  216. {"if a consumer crashes, pid is replaced",
  217. fun() ->
  218. Consumer = start_user(),
  219. StartId = user_id(Consumer),
  220. user_crash(Consumer),
  221. NewPid = hd(get_n_pids(1, [])),
  222. NewId = pooled_gs:get_id(NewPid),
  223. ?assertNot(NewId == StartId)
  224. end
  225. },
  226. {"it is ok to return an unknown pid",
  227. fun() ->
  228. Bogus1 = spawn(fun() -> ok end),
  229. Bogus2 = spawn(fun() -> ok end),
  230. ?assertEqual(ok, pooler:return_member(test_pool_1, Bogus1, ok)),
  231. ?assertEqual(ok, pooler:return_member(test_pool_1, Bogus2, fail))
  232. end
  233. },
  234. {"calling return_member on error_no_members is ignored",
  235. fun() ->
  236. ?assertEqual(ok, pooler:return_member(test_pool_1, error_no_members)),
  237. ?assertEqual(ok, pooler:return_member(test_pool_1, error_no_members, ok)),
  238. ?assertEqual(ok, pooler:return_member(test_pool_1, error_no_members, fail))
  239. end
  240. },
  241. {"dynamic pool creation",
  242. fun() ->
  243. {ok, SupPid} = pooler:new_pool([{name, dyn_pool_1},
  244. {max_count, 3},
  245. {init_count, 2},
  246. {start_mfa,
  247. {pooled_gs, start_link, [{"dyn-0"}]}}]),
  248. ?assert(is_pid(SupPid)),
  249. M = pooler:take_member(dyn_pool_1),
  250. ?assertMatch({"dyn-0", _Id}, pooled_gs:get_id(M)),
  251. ?assertEqual(ok, pooler:rm_pool(dyn_pool_1)),
  252. ?assertExit({noproc, _}, pooler:take_member(dyn_pool_1)),
  253. %% remove non-existing pool
  254. ?assertEqual(ok, pooler:rm_pool(dyn_pool_X)),
  255. ?assertEqual(ok, pooler:rm_pool(dyn_pool_1))
  256. end},
  257. {"metrics have been called",
  258. fun() ->
  259. %% exercise the API to ensure we have certain keys reported as metrics
  260. fake_metrics:reset_metrics(),
  261. Pids = [ pooler:take_member(test_pool_1) || _I <- lists:seq(1, 10) ],
  262. [ pooler:return_member(test_pool_1, P) || P <- Pids ],
  263. catch pooler:take_member(bad_pool_name),
  264. %% kill and unused member
  265. exit(hd(Pids), kill),
  266. %% kill a used member
  267. KillMe = pooler:take_member(test_pool_1),
  268. exit(KillMe, kill),
  269. %% FIXME: We need to wait for pooler to process the
  270. %% exit message. This is ugly, will fix later.
  271. timer:sleep(200), % :(
  272. ExpectKeys = lists:sort([<<"pooler.test_pool_1.error_no_members_count">>,
  273. <<"pooler.test_pool_1.events">>,
  274. <<"pooler.test_pool_1.free_count">>,
  275. <<"pooler.test_pool_1.in_use_count">>,
  276. <<"pooler.test_pool_1.killed_free_count">>,
  277. <<"pooler.test_pool_1.killed_in_use_count">>,
  278. <<"pooler.test_pool_1.take_rate">>]),
  279. Metrics = fake_metrics:get_metrics(),
  280. GotKeys = lists:usort([ Name || {Name, _, _} <- Metrics ]),
  281. ?assertEqual(ExpectKeys, GotKeys)
  282. end},
  283. {"accept bad member is handled",
  284. fun() ->
  285. Bad = spawn(fun() -> ok end),
  286. Ref = erlang:make_ref(),
  287. ?assertEqual(ok, pooler:accept_member(test_pool_1, {Ref, Bad}))
  288. end}
  289. ].
  290. pooler_groups_test_() ->
  291. {setup,
  292. fun() ->
  293. application:set_env(pooler, metrics_module, fake_metrics),
  294. fake_metrics:start_link()
  295. end,
  296. fun(_X) ->
  297. fake_metrics:stop()
  298. end,
  299. {foreach,
  300. % setup
  301. fun() ->
  302. Pools = [[{name, test_pool_1},
  303. {group, group_1},
  304. {max_count, 3},
  305. {init_count, 2},
  306. {start_mfa,
  307. {pooled_gs, start_link, [{"type-1-1"}]}}],
  308. [{name, test_pool_2},
  309. {group, group_1},
  310. {max_count, 3},
  311. {init_count, 2},
  312. {start_mfa,
  313. {pooled_gs, start_link, [{"type-1-2"}]}}],
  314. %% test_pool_3 not part of the group
  315. [{name, test_pool_3},
  316. {group, undefined},
  317. {max_count, 3},
  318. {init_count, 2},
  319. {start_mfa,
  320. {pooled_gs, start_link, [{"type-3"}]}}]
  321. ],
  322. application:set_env(pooler, pools, Pools),
  323. %% error_logger:delete_report_handler(error_logger_tty_h),
  324. pg2:start(),
  325. application:start(pooler)
  326. end,
  327. fun(_X) ->
  328. application:stop(pooler),
  329. application:stop(pg2)
  330. end,
  331. [
  332. {"take and return one group member (repeated)",
  333. fun() ->
  334. Types = [ begin
  335. Pid = pooler:take_group_member(group_1),
  336. {Type, _} = pooled_gs:get_id(Pid),
  337. ?assertMatch("type-1" ++ _, Type),
  338. ok = pooler:return_group_member(group_1, Pid, ok),
  339. Type
  340. end
  341. || _I <- lists:seq(1, 50) ],
  342. Type_1_1 = [ X || "type-1-1" = X <- Types ],
  343. Type_1_2 = [ X || "type-1-2" = X <- Types ],
  344. ?assert(length(Type_1_1) > 0),
  345. ?assert(length(Type_1_2) > 0)
  346. end},
  347. {"take member from unknown group",
  348. fun() ->
  349. ?assertEqual({error_no_group, not_a_group},
  350. pooler:take_group_member(not_a_group))
  351. end},
  352. {"return member to unknown group",
  353. fun() ->
  354. Pid = pooler:take_group_member(group_1),
  355. ?assertEqual(ok, pooler:return_group_member(no_such_group, Pid))
  356. end},
  357. {"return member to wrong group",
  358. fun() ->
  359. Pid = pooler:take_member(test_pool_3),
  360. ?assertEqual(ok, pooler:return_group_member(group_1, Pid))
  361. end},
  362. {"take member from empty group",
  363. fun() ->
  364. %% artificially empty group member list
  365. [ pg2:leave(group_1, M) || M <- pg2:get_members(group_1) ],
  366. ?assertEqual(error_no_members, pooler:take_group_member(group_1))
  367. end},
  368. {"return member to group, implied ok",
  369. fun() ->
  370. Pid = pooler:take_group_member(group_1),
  371. ?assertEqual(ok, pooler:return_group_member(group_1, Pid))
  372. end},
  373. {"return error_no_member to group",
  374. fun() ->
  375. ?assertEqual(ok, pooler:return_group_member(group_1, error_no_members))
  376. end},
  377. {"exhaust pools in group",
  378. fun() ->
  379. Pids = get_n_pids_group(group_1, 6, []),
  380. %% they should all be pids
  381. [ begin
  382. {Type, _} = pooled_gs:get_id(P),
  383. ?assertMatch("type-1" ++ _, Type),
  384. ok
  385. end || P <- Pids ],
  386. %% further attempts should be error
  387. [error_no_members,
  388. error_no_members,
  389. error_no_members] = [ pooler:take_group_member(group_1)
  390. || _I <- lists:seq(1, 3) ]
  391. end}
  392. ]}}.
  393. pooler_limit_failed_adds_test_() ->
  394. %% verify that pooler crashes completely if too many failures are
  395. %% encountered while trying to add pids.
  396. {setup,
  397. fun() ->
  398. Pools = [[{name, test_pool_1},
  399. {max_count, 10},
  400. {init_count, 10},
  401. {start_mfa,
  402. {pooled_gs, start_link, [crash]}}]],
  403. application:set_env(pooler, pools, Pools)
  404. end,
  405. fun(_) ->
  406. application:stop(pooler)
  407. end,
  408. fun() ->
  409. application:start(pooler),
  410. ?assertEqual(error_no_members, pooler:take_member(test_pool_1)),
  411. ?assertEqual(error_no_members, pooler:take_member(test_pool_1))
  412. end}.
  413. pooler_scheduled_cull_test_() ->
  414. {setup,
  415. fun() ->
  416. application:set_env(pooler, metrics_module, fake_metrics),
  417. fake_metrics:start_link(),
  418. Pools = [[{name, test_pool_1},
  419. {max_count, 10},
  420. {init_count, 2},
  421. {start_mfa, {pooled_gs, start_link, [{"type-0"}]}},
  422. {cull_interval, {200, ms}}]],
  423. application:set_env(pooler, pools, Pools),
  424. %% error_logger:delete_report_handler(error_logger_tty_h),
  425. application:start(pooler)
  426. end,
  427. fun(_X) ->
  428. fake_metrics:stop(),
  429. application:stop(pooler)
  430. end,
  431. [{"excess members are culled repeatedly",
  432. fun() ->
  433. %% take all members
  434. Pids1 = get_n_pids(test_pool_1, 10, []),
  435. %% return all
  436. [ pooler:return_member(test_pool_1, P) || P <- Pids1 ],
  437. ?assertEqual(10, length(pooler:pool_stats(test_pool_1))),
  438. %% wait for longer than cull delay
  439. timer:sleep(250),
  440. ?assertEqual(2, length(pooler:pool_stats(test_pool_1))),
  441. %% repeat the test to verify that culling gets rescheduled.
  442. Pids2 = get_n_pids(test_pool_1, 10, []),
  443. %% return all
  444. [ pooler:return_member(test_pool_1, P) || P <- Pids2 ],
  445. ?assertEqual(10, length(pooler:pool_stats(test_pool_1))),
  446. %% wait for longer than cull delay
  447. timer:sleep(250),
  448. ?assertEqual(2, length(pooler:pool_stats(test_pool_1)))
  449. end
  450. },
  451. {"non-excess members are not culled",
  452. fun() ->
  453. [P1, P2] = [pooler:take_member(test_pool_1) || _X <- [1, 2] ],
  454. [pooler:return_member(test_pool_1, P) || P <- [P1, P2] ],
  455. ?assertEqual(2, length(pooler:pool_stats(test_pool_1))),
  456. timer:sleep(250),
  457. ?assertEqual(2, length(pooler:pool_stats(test_pool_1)))
  458. end
  459. },
  460. {"in-use members are not culled",
  461. fun() ->
  462. %% take all members
  463. Pids = get_n_pids(test_pool_1, 10, []),
  464. %% don't return any
  465. ?assertEqual(10, length(pooler:pool_stats(test_pool_1))),
  466. %% wait for longer than cull delay
  467. timer:sleep(250),
  468. ?assertEqual(10, length(pooler:pool_stats(test_pool_1))),
  469. [ pooler:return_member(test_pool_1, P) || P <- Pids ]
  470. end}
  471. ]}.
  472. random_message_test_() ->
  473. {setup,
  474. fun() ->
  475. Pools = [[{name, test_pool_1},
  476. {max_count, 2},
  477. {init_count, 1},
  478. {start_mfa,
  479. {pooled_gs, start_link, [{"type-0"}]}}]],
  480. application:set_env(pooler, pools, Pools),
  481. error_logger:delete_report_handler(error_logger_tty_h),
  482. application:start(pooler),
  483. %% now send some bogus messages
  484. %% do the call in a throw-away process to avoid timeout error
  485. spawn(fun() -> catch gen_server:call(test_pool_1, {unexpected_garbage_msg, 5}) end),
  486. gen_server:cast(test_pool_1, {unexpected_garbage_msg, 6}),
  487. whereis(test_pool_1) ! {unexpected_garbage_msg, 7},
  488. ok
  489. end,
  490. fun(_) ->
  491. application:stop(pooler)
  492. end,
  493. [
  494. fun() ->
  495. Pid = spawn(fun() -> ok end),
  496. MonMsg = {'DOWN', erlang:make_ref(), process, Pid, because},
  497. test_pool_1 ! MonMsg
  498. end,
  499. fun() ->
  500. Pid = pooler:take_member(test_pool_1),
  501. {Type, _} = pooled_gs:get_id(Pid),
  502. ?assertEqual("type-0", Type)
  503. end,
  504. fun() ->
  505. RawPool = gen_server:call(test_pool_1, dump_pool),
  506. ?assertEqual(pool, element(1, RawPool))
  507. end
  508. ]}.
  509. pooler_integration_test_() ->
  510. {foreach,
  511. % setup
  512. fun() ->
  513. Pools = [[{name, test_pool_1},
  514. {max_count, 10},
  515. {init_count, 10},
  516. {start_mfa,
  517. {pooled_gs, start_link, [{"type-0"}]}}]],
  518. application:set_env(pooler, pools, Pools),
  519. error_logger:delete_report_handler(error_logger_tty_h),
  520. application:start(pooler),
  521. Users = [ start_user() || _X <- lists:seq(1, 10) ],
  522. Users
  523. end,
  524. % cleanup
  525. fun(Users) ->
  526. [ user_stop(U) || U <- Users ],
  527. application:stop(pooler)
  528. end,
  529. %
  530. [
  531. fun(Users) ->
  532. fun() ->
  533. % each user has a different tc ID
  534. TcIds = lists:sort([ user_id(UPid) || UPid <- Users ]),
  535. ?assertEqual(lists:usort(TcIds), TcIds)
  536. end
  537. end
  538. ,
  539. fun(Users) ->
  540. fun() ->
  541. % users still unique after a renew cycle
  542. [ user_new_tc(UPid) || UPid <- Users ],
  543. TcIds = lists:sort([ user_id(UPid) || UPid <- Users ]),
  544. ?assertEqual(lists:usort(TcIds), TcIds)
  545. end
  546. end
  547. ,
  548. fun(Users) ->
  549. fun() ->
  550. % all users crash, pids are replaced
  551. TcIds1 = lists:sort([ user_id(UPid) || UPid <- Users ]),
  552. [ user_crash(UPid) || UPid <- Users ],
  553. Seq = lists:seq(1, 5),
  554. Users2 = [ start_user() || _X <- Seq ],
  555. TcIds2 = lists:sort([ user_id(UPid) || UPid <- Users2 ]),
  556. Both =
  557. sets:to_list(sets:intersection([sets:from_list(TcIds1),
  558. sets:from_list(TcIds2)])),
  559. ?assertEqual([], Both)
  560. end
  561. end
  562. ]
  563. }.
  564. time_as_millis_test_() ->
  565. Zeros = [ {{0, U}, 0} || U <- [min, sec, ms, mu] ],
  566. Ones = [{{1, min}, 60000},
  567. {{1, sec}, 1000},
  568. {{1, ms}, 1},
  569. {{1, mu}, 0}],
  570. Misc = [{{3000, mu}, 3}],
  571. Tests = Zeros ++ Ones ++ Misc,
  572. [ ?_assertEqual(E, pooler:time_as_millis(I)) || {I, E} <- Tests ].
  573. time_as_micros_test_() ->
  574. Zeros = [ {{0, U}, 0} || U <- [min, sec, ms, mu] ],
  575. Ones = [{{1, min}, 60000000},
  576. {{1, sec}, 1000000},
  577. {{1, ms}, 1000},
  578. {{1, mu}, 1}],
  579. Misc = [{{3000, mu}, 3000}],
  580. Tests = Zeros ++ Ones ++ Misc,
  581. [ ?_assertEqual(E, pooler:time_as_micros(I)) || {I, E} <- Tests ].
  582. % testing crash recovery means race conditions when either pids
  583. % haven't yet crashed or pooler hasn't recovered. So this helper loops
  584. % forver until N pids are obtained, ignoring error_no_members.
  585. get_n_pids(N, Acc) ->
  586. get_n_pids(test_pool_1, N, Acc).
  587. get_n_pids(_Pool, 0, Acc) ->
  588. Acc;
  589. get_n_pids(Pool, N, Acc) ->
  590. case pooler:take_member(Pool) of
  591. error_no_members ->
  592. get_n_pids(Pool, N, Acc);
  593. Pid ->
  594. get_n_pids(Pool, N - 1, [Pid|Acc])
  595. end.
  596. get_n_pids_group(_Group, 0, Acc) ->
  597. Acc;
  598. get_n_pids_group(Group, N, Acc) ->
  599. case pooler:take_group_member(Group) of
  600. error_no_members ->
  601. get_n_pids_group(Group, N, Acc);
  602. Pid ->
  603. get_n_pids_group(Group, N - 1, [Pid|Acc])
  604. end.