pooler_tests.erl 24 KB

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