pooler_tests.erl 21 KB

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