Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

1479 lignes
78 KiB

il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
il y a 14 ans
  1. %% -------------------------------------------------------------------
  2. %%
  3. %% Copyright (c) 2011-2015 Basho Technologies, Inc.
  4. %%
  5. %% This file is provided to you under the Apache License,
  6. %% Version 2.0 (the "License"); you may not use this file
  7. %% except in compliance with the License. You may obtain
  8. %% a copy of the License at
  9. %%
  10. %% http://www.apache.org/licenses/LICENSE-2.0
  11. %%
  12. %% Unless required by applicable law or agreed to in writing,
  13. %% software distributed under the License is distributed on an
  14. %% "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
  15. %% KIND, either express or implied. See the License for the
  16. %% specific language governing permissions and limitations
  17. %% under the License.
  18. %%
  19. %% -------------------------------------------------------------------
  20. -module(lager_test_backend).
  21. -include("lager.hrl").
  22. -behaviour(gen_event).
  23. -export([init/1, handle_call/2, handle_event/2, handle_info/2, terminate/2,
  24. code_change/3]).
  25. -define(TEST_SINK_NAME, '__lager_test_sink'). %% <-- used by parse transform
  26. -define(TEST_SINK_EVENT, '__lager_test_sink_event'). %% <-- used by lager API calls and internals for gen_event
  27. -record(state, {level :: list(), buffer :: list(), ignored :: term()}).
  28. -compile({parse_transform, lager_transform}).
  29. -ifdef(TEST).
  30. -include_lib("eunit/include/eunit.hrl").
  31. -record(test, {attrs :: list(), format :: list(), args :: list()}).
  32. -export([pop/0, count/0, count_ignored/0, flush/0, print_state/0]).
  33. -endif.
  34. init(Level) ->
  35. {ok, #state{level=lager_util:config_to_mask(Level), buffer=[], ignored=[]}}.
  36. handle_call(count, #state{buffer=Buffer} = State) ->
  37. {ok, length(Buffer), State};
  38. handle_call(count_ignored, #state{ignored=Ignored} = State) ->
  39. {ok, length(Ignored), State};
  40. handle_call(flush, State) ->
  41. {ok, ok, State#state{buffer=[], ignored=[]}};
  42. handle_call(pop, #state{buffer=Buffer} = State) ->
  43. case Buffer of
  44. [] ->
  45. {ok, undefined, State};
  46. [H|T] ->
  47. {ok, H, State#state{buffer=T}}
  48. end;
  49. handle_call(get_loglevel, #state{level=Level} = State) ->
  50. {ok, Level, State};
  51. handle_call({set_loglevel, Level}, State) ->
  52. {ok, ok, State#state{level=lager_util:config_to_mask(Level)}};
  53. handle_call(print_state, State) ->
  54. spawn(fun() -> lager:info("State ~p", [lager:pr(State, ?MODULE)]) end),
  55. timer:sleep(100),
  56. {ok, ok, State};
  57. handle_call(print_bad_state, State) ->
  58. spawn(fun() -> lager:info("State ~p", [lager:pr({state, 1}, ?MODULE)]) end),
  59. timer:sleep(100),
  60. {ok, ok, State};
  61. handle_call(_Request, State) ->
  62. {ok, ok, State}.
  63. handle_event({log, Msg},
  64. #state{level=LogLevel,buffer=Buffer,ignored=Ignored} = State) ->
  65. case lager_util:is_loggable(Msg, LogLevel, ?MODULE) of
  66. true ->
  67. {ok, State#state{buffer=Buffer ++
  68. [{lager_msg:severity_as_int(Msg),
  69. lager_msg:datetime(Msg),
  70. lager_msg:message(Msg), lager_msg:metadata(Msg)}]}};
  71. _ ->
  72. {ok, State#state{ignored=Ignored ++ [ignored]}}
  73. end;
  74. handle_event(_Event, State) ->
  75. {ok, State}.
  76. handle_info(_Info, State) ->
  77. {ok, State}.
  78. terminate(_Reason, _State) ->
  79. ok.
  80. code_change(_OldVsn, State, _Extra) ->
  81. {ok, State}.
  82. -ifdef(TEST).
  83. pop() ->
  84. pop(lager_event).
  85. count() ->
  86. count(lager_event).
  87. count_ignored() ->
  88. count_ignored(lager_event).
  89. flush() ->
  90. flush(lager_event).
  91. print_state() ->
  92. print_state(lager_event).
  93. print_bad_state() ->
  94. print_bad_state(lager_event).
  95. pop(Sink) ->
  96. gen_event:call(Sink, ?MODULE, pop).
  97. count(Sink) ->
  98. gen_event:call(Sink, ?MODULE, count).
  99. count_ignored(Sink) ->
  100. gen_event:call(Sink, ?MODULE, count_ignored).
  101. flush(Sink) ->
  102. gen_event:call(Sink, ?MODULE, flush).
  103. print_state(Sink) ->
  104. gen_event:call(Sink, ?MODULE, print_state).
  105. print_bad_state(Sink) ->
  106. gen_event:call(Sink, ?MODULE, print_bad_state).
  107. has_line_numbers() ->
  108. %% are we R15 or greater
  109. % this gets called a LOT - cache the answer
  110. case erlang:get({?MODULE, has_line_numbers}) of
  111. undefined ->
  112. R = otp_version() >= 15,
  113. erlang:put({?MODULE, has_line_numbers}, R),
  114. R;
  115. Bool ->
  116. Bool
  117. end.
  118. otp_version() ->
  119. otp_version(erlang:system_info(otp_release)).
  120. otp_version([$R | Rel]) ->
  121. {Ver, _} = string:to_integer(Rel),
  122. Ver;
  123. otp_version(Rel) ->
  124. {Ver, _} = string:to_integer(Rel),
  125. Ver.
  126. not_running_test() ->
  127. ?assertEqual({error, lager_not_running}, lager:log(info, self(), "not running")).
  128. lager_test_() ->
  129. {foreach,
  130. fun setup/0,
  131. fun cleanup/1,
  132. [
  133. {"observe that there is nothing up my sleeve",
  134. fun() ->
  135. ?assertEqual(undefined, pop()),
  136. ?assertEqual(0, count())
  137. end
  138. },
  139. {"test sink not running",
  140. fun() ->
  141. ?assertEqual({error, {sink_not_configured, test}}, lager:log(test, info, self(), "~p", "not running"))
  142. end
  143. },
  144. {"logging works",
  145. fun() ->
  146. lager:warning("test message"),
  147. ?assertEqual(1, count()),
  148. {Level, _Time, Message, _Metadata} = pop(),
  149. ?assertMatch(Level, lager_util:level_to_num(warning)),
  150. ?assertEqual("test message", Message),
  151. ok
  152. end
  153. },
  154. {"logging with arguments works",
  155. fun() ->
  156. lager:warning("test message ~p", [self()]),
  157. ?assertEqual(1, count()),
  158. {Level, _Time, Message,_Metadata} = pop(),
  159. ?assertMatch(Level, lager_util:level_to_num(warning)),
  160. ?assertEqual(lists:flatten(io_lib:format("test message ~p", [self()])), lists:flatten(Message)),
  161. ok
  162. end
  163. },
  164. {"logging works from inside a begin/end block",
  165. fun() ->
  166. ?assertEqual(0, count()),
  167. begin
  168. lager:warning("test message 2")
  169. end,
  170. ?assertEqual(1, count()),
  171. ok
  172. end
  173. },
  174. {"logging works from inside a list comprehension",
  175. fun() ->
  176. ?assertEqual(0, count()),
  177. [lager:warning("test message") || _N <- lists:seq(1, 10)],
  178. ?assertEqual(10, count()),
  179. ok
  180. end
  181. },
  182. {"logging works from a begin/end block inside a list comprehension",
  183. fun() ->
  184. ?assertEqual(0, count()),
  185. [ begin lager:warning("test message") end || _N <- lists:seq(1, 10)],
  186. ?assertEqual(10, count()),
  187. ok
  188. end
  189. },
  190. {"logging works from a nested list comprehension",
  191. fun() ->
  192. ?assertEqual(0, count()),
  193. [ [lager:warning("test message") || _N <- lists:seq(1, 10)] ||
  194. _I <- lists:seq(1, 10)],
  195. ?assertEqual(100, count()),
  196. ok
  197. end
  198. },
  199. {"variables inplace of literals in logging statements work",
  200. fun() ->
  201. ?assertEqual(0, count()),
  202. Attr = [{a, alpha}, {b, beta}],
  203. Fmt = "format ~p",
  204. Args = [world],
  205. lager:info(Attr, "hello"),
  206. lager:info(Attr, "hello ~p", [world]),
  207. lager:info(Fmt, [world]),
  208. lager:info("hello ~p", Args),
  209. lager:info(Attr, "hello ~p", Args),
  210. lager:info([{d, delta}, {g, gamma}], Fmt, Args),
  211. ?assertEqual(6, count()),
  212. {_Level, _Time, Message, Metadata} = pop(),
  213. ?assertMatch([{a, alpha}, {b, beta}|_], Metadata),
  214. ?assertEqual("hello", lists:flatten(Message)),
  215. {_Level, _Time2, Message2, _Metadata2} = pop(),
  216. ?assertEqual("hello world", lists:flatten(Message2)),
  217. {_Level, _Time3, Message3, _Metadata3} = pop(),
  218. ?assertEqual("format world", lists:flatten(Message3)),
  219. {_Level, _Time4, Message4, _Metadata4} = pop(),
  220. ?assertEqual("hello world", lists:flatten(Message4)),
  221. {_Level, _Time5, Message5, _Metadata5} = pop(),
  222. ?assertEqual("hello world", lists:flatten(Message5)),
  223. {_Level, _Time6, Message6, Metadata6} = pop(),
  224. ?assertMatch([{d, delta}, {g, gamma}|_], Metadata6),
  225. ?assertEqual("format world", lists:flatten(Message6)),
  226. ok
  227. end
  228. },
  229. {"list comprehension inplace of literals in logging statements work",
  230. fun() ->
  231. ?assertEqual(0, count()),
  232. Attr = [{a, alpha}, {b, beta}],
  233. Fmt = "format ~p",
  234. Args = [world],
  235. lager:info([{K, atom_to_list(V)} || {K, V} <- Attr], "hello"),
  236. lager:info([{K, atom_to_list(V)} || {K, V} <- Attr], "hello ~p", [{atom, X} || X <- Args]),
  237. lager:info([X || X <- Fmt], [world]),
  238. lager:info("hello ~p", [{atom, X} || X <- Args]),
  239. lager:info([{K, atom_to_list(V)} || {K, V} <- Attr], "hello ~p", [{atom, X} || X <- Args]),
  240. lager:info([{d, delta}, {g, gamma}], Fmt, [{atom, X} || X <- Args]),
  241. ?assertEqual(6, count()),
  242. {_Level, _Time, Message, Metadata} = pop(),
  243. ?assertMatch([{a, "alpha"}, {b, "beta"}|_], Metadata),
  244. ?assertEqual("hello", lists:flatten(Message)),
  245. {_Level, _Time2, Message2, _Metadata2} = pop(),
  246. ?assertEqual("hello {atom,world}", lists:flatten(Message2)),
  247. {_Level, _Time3, Message3, _Metadata3} = pop(),
  248. ?assertEqual("format world", lists:flatten(Message3)),
  249. {_Level, _Time4, Message4, _Metadata4} = pop(),
  250. ?assertEqual("hello {atom,world}", lists:flatten(Message4)),
  251. {_Level, _Time5, Message5, _Metadata5} = pop(),
  252. ?assertEqual("hello {atom,world}", lists:flatten(Message5)),
  253. {_Level, _Time6, Message6, Metadata6} = pop(),
  254. ?assertMatch([{d, delta}, {g, gamma}|_], Metadata6),
  255. ?assertEqual("format {atom,world}", lists:flatten(Message6)),
  256. ok
  257. end
  258. },
  259. {"function calls inplace of literals in logging statements work",
  260. fun() ->
  261. ?assertEqual(0, count()),
  262. put(attrs, [{a, alpha}, {b, beta}]),
  263. put(format, "format ~p"),
  264. put(args, [world]),
  265. lager:info(get(attrs), "hello"),
  266. lager:info(get(attrs), "hello ~p", get(args)),
  267. lager:info(get(format), [world]),
  268. lager:info("hello ~p", erlang:get(args)),
  269. lager:info(fun() -> get(attrs) end(), "hello ~p", get(args)),
  270. lager:info([{d, delta}, {g, gamma}], get(format), get(args)),
  271. ?assertEqual(6, count()),
  272. {_Level, _Time, Message, Metadata} = pop(),
  273. ?assertMatch([{a, alpha}, {b, beta}|_], Metadata),
  274. ?assertEqual("hello", lists:flatten(Message)),
  275. {_Level, _Time2, Message2, _Metadata2} = pop(),
  276. ?assertEqual("hello world", lists:flatten(Message2)),
  277. {_Level, _Time3, Message3, _Metadata3} = pop(),
  278. ?assertEqual("format world", lists:flatten(Message3)),
  279. {_Level, _Time4, Message4, _Metadata4} = pop(),
  280. ?assertEqual("hello world", lists:flatten(Message4)),
  281. {_Level, _Time5, Message5, _Metadata5} = pop(),
  282. ?assertEqual("hello world", lists:flatten(Message5)),
  283. {_Level, _Time6, Message6, Metadata6} = pop(),
  284. ?assertMatch([{d, delta}, {g, gamma}|_], Metadata6),
  285. ?assertEqual("format world", lists:flatten(Message6)),
  286. ok
  287. end
  288. },
  289. {"record fields inplace of literals in logging statements work",
  290. fun() ->
  291. ?assertEqual(0, count()),
  292. Test = #test{attrs=[{a, alpha}, {b, beta}], format="format ~p", args=[world]},
  293. lager:info(Test#test.attrs, "hello"),
  294. lager:info(Test#test.attrs, "hello ~p", Test#test.args),
  295. lager:info(Test#test.format, [world]),
  296. lager:info("hello ~p", Test#test.args),
  297. lager:info(Test#test.attrs, "hello ~p", Test#test.args),
  298. lager:info([{d, delta}, {g, gamma}], Test#test.format, Test#test.args),
  299. ?assertEqual(6, count()),
  300. {_Level, _Time, Message, Metadata} = pop(),
  301. ?assertMatch([{a, alpha}, {b, beta}|_], Metadata),
  302. ?assertEqual("hello", lists:flatten(Message)),
  303. {_Level, _Time2, Message2, _Metadata2} = pop(),
  304. ?assertEqual("hello world", lists:flatten(Message2)),
  305. {_Level, _Time3, Message3, _Metadata3} = pop(),
  306. ?assertEqual("format world", lists:flatten(Message3)),
  307. {_Level, _Time4, Message4, _Metadata4} = pop(),
  308. ?assertEqual("hello world", lists:flatten(Message4)),
  309. {_Level, _Time5, Message5, _Metadata5} = pop(),
  310. ?assertEqual("hello world", lists:flatten(Message5)),
  311. {_Level, _Time6, Message6, Metadata6} = pop(),
  312. ?assertMatch([{d, delta}, {g, gamma}|_], Metadata6),
  313. ?assertEqual("format world", lists:flatten(Message6)),
  314. ok
  315. end
  316. },
  317. {"log messages below the threshold are ignored",
  318. fun() ->
  319. ?assertEqual(0, count()),
  320. lager:debug("this message will be ignored"),
  321. ?assertEqual(0, count()),
  322. ?assertEqual(0, count_ignored()),
  323. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(debug)), []}),
  324. lager:debug("this message should be ignored"),
  325. ?assertEqual(0, count()),
  326. ?assertEqual(1, count_ignored()),
  327. lager:set_loglevel(?MODULE, debug),
  328. ?assertEqual({?DEBUG bor ?INFO bor ?NOTICE bor ?WARNING bor ?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  329. lager:debug("this message should be logged"),
  330. ?assertEqual(1, count()),
  331. ?assertEqual(1, count_ignored()),
  332. ?assertEqual(debug, lager:get_loglevel(?MODULE)),
  333. ok
  334. end
  335. },
  336. {"tracing works",
  337. fun() ->
  338. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(error)), []}),
  339. ok = lager:info("hello world"),
  340. ?assertEqual(0, count()),
  341. lager:trace(?MODULE, [{module, ?MODULE}], debug),
  342. ?assertMatch({?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, _}, lager_config:get(loglevel)),
  343. %% elegible for tracing
  344. ok = lager:info("hello world"),
  345. %% NOT elegible for tracing
  346. ok = lager:log(info, [{pid, self()}], "hello world"),
  347. ?assertEqual(1, count()),
  348. ok
  349. end
  350. },
  351. {"tracing works with custom attributes",
  352. fun() ->
  353. lager:set_loglevel(?MODULE, error),
  354. ?assertEqual({?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  355. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(error)), []}),
  356. lager:info([{requestid, 6}], "hello world"),
  357. ?assertEqual(0, count()),
  358. lager:trace(?MODULE, [{requestid, 6}, {foo, bar}], debug),
  359. lager:info([{requestid, 6}, {foo, bar}], "hello world"),
  360. ?assertEqual(1, count()),
  361. lager:trace(?MODULE, [{requestid, '*'}], debug),
  362. lager:info([{requestid, 6}], "hello world"),
  363. ?assertEqual(2, count()),
  364. lager:clear_all_traces(),
  365. lager:info([{requestid, 6}], "hello world"),
  366. ?assertEqual(2, count()),
  367. ok
  368. end
  369. },
  370. {"tracing works with custom attributes and event stream processing",
  371. fun() ->
  372. lager:set_loglevel(?MODULE, error),
  373. ?assertEqual({?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  374. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(error)), []}),
  375. lager:info([{requestid, 6}], "hello world"),
  376. ?assertEqual(0, count()),
  377. lager:trace(?MODULE, [{requestid, '>', 5}, {requestid, '<', 7}, {foo, bar}], debug),
  378. lager:info([{requestid, 5}, {foo, bar}], "hello world"),
  379. lager:info([{requestid, 6}, {foo, bar}], "hello world"),
  380. ?assertEqual(1, count()),
  381. lager:clear_all_traces(),
  382. lager:trace(?MODULE, [{requestid, '>', 8}, {foo, bar}]),
  383. lager:info([{foo, bar}], "hello world"),
  384. lager:info([{requestid, 6}], "hello world"),
  385. lager:info([{requestid, 7}], "hello world"),
  386. lager:info([{requestid, 8}], "hello world"),
  387. lager:info([{requestid, 9}, {foo, bar}], "hello world"),
  388. lager:info([{requestid, 10}], "hello world"),
  389. ?assertEqual(2, count()),
  390. lager:trace(?MODULE, [{requestid, '>', 8}]),
  391. lager:info([{foo, bar}], "hello world"),
  392. lager:info([{requestid, 6}], "hello world"),
  393. lager:info([{requestid, 7}], "hello world"),
  394. lager:info([{requestid, 8}], "hello world"),
  395. lager:info([{requestid, 9}, {foo, bar}], "hello world"),
  396. lager:info([{requestid, 10}], "hello world"),
  397. ?assertEqual(4, count()),
  398. lager:trace(?MODULE, [{foo, '=', bar}]),
  399. lager:info([{foo, bar}], "hello world"),
  400. lager:info([{requestid, 6}], "hello world"),
  401. lager:info([{requestid, 7}], "hello world"),
  402. lager:info([{requestid, 8}], "hello world"),
  403. lager:info([{requestid, 9}, {foo, bar}], "hello world"),
  404. lager:info([{requestid, 10}], "hello world"),
  405. lager:trace(?MODULE, [{fu, '!'}]),
  406. lager:info([{foo, bar}], "hello world"),
  407. lager:info([{ooh, car}], "hello world"),
  408. lager:info([{fu, bar}], "hello world"),
  409. lager:trace(?MODULE, [{fu, '*'}]),
  410. lager:info([{fu, bar}], "hello world"),
  411. ?assertEqual(10, count()),
  412. lager:clear_all_traces(),
  413. lager:info([{requestid, 6}], "hello world"),
  414. ?assertEqual(10, count()),
  415. ok
  416. end
  417. },
  418. {"tracing custom attributes works with event stream processing statistics and reductions",
  419. fun() ->
  420. lager:set_loglevel(?MODULE, error),
  421. ?assertEqual({?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  422. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(error)), []}),
  423. lager:info([{requestid, 6}], "hello world"),
  424. ?assertEqual(0, count()),
  425. lager:trace(?MODULE, [{beta, '*'}]),
  426. lager:trace(?MODULE, [{meta, "data"}]),
  427. lager:info([{meta, "data"}], "hello world"),
  428. lager:info([{beta, 2}], "hello world"),
  429. lager:info([{beta, 2.1}, {foo, bar}], "hello world"),
  430. lager:info([{meta, <<"data">>}], "hello world"),
  431. ?assertEqual(8, ?DEFAULT_TRACER:info(input)),
  432. ?assertEqual(6, ?DEFAULT_TRACER:info(output)),
  433. ?assertEqual(2, ?DEFAULT_TRACER:info(filter)),
  434. lager:clear_all_traces(),
  435. lager:trace(?MODULE, [{meta, "data"}]),
  436. lager:trace(?MODULE, [{beta, '>', 2}, {beta, '<', 2.12}]),
  437. lager:info([{meta, "data"}], "hello world"),
  438. lager:info([{beta, 2}], "hello world"),
  439. lager:info([{beta, 2.1}, {foo, bar}], "hello world"),
  440. lager:info([{meta, <<"data">>}], "hello world"),
  441. ?assertEqual(8, ?DEFAULT_TRACER:info(input)),
  442. ?assertEqual(4, ?DEFAULT_TRACER:info(output)),
  443. ?assertEqual(4, ?DEFAULT_TRACER:info(filter)),
  444. lager:clear_all_traces(),
  445. lager:trace_console([{beta, '>', 2}, {meta, "data"}]),
  446. lager:trace_console([{beta, '>', 2}, {beta, '<', 2.12}]),
  447. Reduced = {all,[{any,[{beta,'<',2.12},{meta,'=',"data"}]},
  448. {beta,'>',2}]},
  449. ?assertEqual(Reduced, ?DEFAULT_TRACER:info('query')),
  450. lager:clear_all_traces(),
  451. lager:info([{requestid, 6}], "hello world"),
  452. ?assertEqual(5, count()),
  453. ok
  454. end
  455. },
  456. {"persistent traces work",
  457. fun() ->
  458. ?assertEqual(0, count()),
  459. lager:debug([{foo, bar}], "hello world"),
  460. ?assertEqual(0, count()),
  461. application:stop(lager),
  462. application:set_env(lager, traces, [{lager_test_backend, [{foo, bar}], debug}]),
  463. lager:start(),
  464. lager:debug([{foo, bar}], "hello world"),
  465. ?assertEqual(1, count()),
  466. application:unset_env(lager, traces),
  467. ok
  468. end
  469. },
  470. {"tracing honors loglevel",
  471. fun() ->
  472. lager:set_loglevel(?MODULE, error),
  473. ?assertEqual({?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  474. {ok, T} = lager:trace(?MODULE, [{module, ?MODULE}], notice),
  475. ok = lager:info("hello world"),
  476. ?assertEqual(0, count()),
  477. ok = lager:notice("hello world"),
  478. ?assertEqual(1, count()),
  479. lager:stop_trace(T),
  480. ok = lager:notice("hello world"),
  481. ?assertEqual(1, count()),
  482. ok
  483. end
  484. },
  485. {"record printing works",
  486. fun() ->
  487. print_state(),
  488. {Level, _Time, Message, _Metadata} = pop(),
  489. ?assertMatch(Level, lager_util:level_to_num(info)),
  490. {mask, Mask} = lager_util:config_to_mask(info),
  491. ?assertEqual("State #state{level={mask,"++integer_to_list(Mask)++"},buffer=[],ignored=[]}", lists:flatten(Message)),
  492. ok
  493. end
  494. },
  495. {"record printing fails gracefully",
  496. fun() ->
  497. print_bad_state(),
  498. {Level, _Time, Message, _Metadata} = pop(),
  499. ?assertMatch(Level, lager_util:level_to_num(info)),
  500. ?assertEqual("State {state,1}", lists:flatten(Message)),
  501. ok
  502. end
  503. },
  504. {"record printing fails gracefully when no lager_record attribute",
  505. fun() ->
  506. spawn(fun() -> lager:info("State ~p", [lager:pr({state, 1}, lager)]) end),
  507. timer:sleep(100),
  508. {Level, _Time, Message, _Metadata} = pop(),
  509. ?assertMatch(Level, lager_util:level_to_num(info)),
  510. ?assertEqual("State {state,1}", lists:flatten(Message)),
  511. ok
  512. end
  513. },
  514. {"record printing fails gracefully when input is not a tuple",
  515. fun() ->
  516. spawn(fun() -> lager:info("State ~p", [lager:pr(ok, lager)]) end),
  517. timer:sleep(100),
  518. {Level, _Time, Message, _Metadata} = pop(),
  519. ?assertMatch(Level, lager_util:level_to_num(info)),
  520. ?assertEqual("State ok", lists:flatten(Message)),
  521. ok
  522. end
  523. },
  524. {"record printing fails gracefully when module is invalid",
  525. fun() ->
  526. spawn(fun() -> lager:info("State ~p", [lager:pr({state, 1}, not_a_module)]) end),
  527. timer:sleep(1000),
  528. {Level, _Time, Message, _Metadata} = pop(),
  529. ?assertMatch(Level, lager_util:level_to_num(info)),
  530. ?assertEqual("State {state,1}", lists:flatten(Message)),
  531. ok
  532. end
  533. },
  534. {"installing a new handler adjusts the global loglevel if necessary",
  535. fun() ->
  536. ?assertEqual({?INFO bor ?NOTICE bor ?WARNING bor ?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  537. supervisor:start_child(lager_handler_watcher_sup, [lager_event, {?MODULE, foo}, debug]),
  538. ?assertEqual({?DEBUG bor ?INFO bor ?NOTICE bor ?WARNING bor ?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  539. ok
  540. end
  541. },
  542. {"metadata in the process dictionary works",
  543. fun() ->
  544. lager:md([{platypus, gravid}, {sloth, hirsute}, {duck, erroneous}]),
  545. lager:info("I sing the animal kingdom electric!"),
  546. {_Level, _Time, _Message, Metadata} = pop(),
  547. ?assertEqual(gravid, proplists:get_value(platypus, Metadata)),
  548. ?assertEqual(hirsute, proplists:get_value(sloth, Metadata)),
  549. ?assertEqual(erroneous, proplists:get_value(duck, Metadata)),
  550. ?assertEqual(undefined, proplists:get_value(eagle, Metadata)),
  551. lager:md([{platypus, gravid}, {sloth, hirsute}, {eagle, superincumbent}]),
  552. lager:info("I sing the animal kingdom dielectric!"),
  553. {_Level2, _Time2, _Message2, Metadata2} = pop(),
  554. ?assertEqual(gravid, proplists:get_value(platypus, Metadata2)),
  555. ?assertEqual(hirsute, proplists:get_value(sloth, Metadata2)),
  556. ?assertEqual(undefined, proplists:get_value(duck, Metadata2)),
  557. ?assertEqual(superincumbent, proplists:get_value(eagle, Metadata2)),
  558. ok
  559. end
  560. },
  561. {"can't store invalid metadata",
  562. fun() ->
  563. ?assertEqual(ok, lager:md([{platypus, gravid}, {sloth, hirsute}, {duck, erroneous}])),
  564. ?assertError(badarg, lager:md({flamboyant, flamingo})),
  565. ?assertError(badarg, lager:md("zookeeper zephyr")),
  566. ok
  567. end
  568. }
  569. ]
  570. }.
  571. extra_sinks_test_() ->
  572. {foreach,
  573. fun setup_sink/0,
  574. fun cleanup/1,
  575. [
  576. {"observe that there is nothing up my sleeve",
  577. fun() ->
  578. ?assertEqual(undefined, pop(?TEST_SINK_EVENT)),
  579. ?assertEqual(0, count(?TEST_SINK_EVENT))
  580. end
  581. },
  582. {"logging works",
  583. fun() ->
  584. ?TEST_SINK_NAME:warning("test message"),
  585. ?assertEqual(1, count(?TEST_SINK_EVENT)),
  586. {Level, _Time, Message, _Metadata} = pop(?TEST_SINK_EVENT),
  587. ?assertMatch(Level, lager_util:level_to_num(warning)),
  588. ?assertEqual("test message", Message),
  589. ok
  590. end
  591. },
  592. {"logging with arguments works",
  593. fun() ->
  594. ?TEST_SINK_NAME:warning("test message ~p", [self()]),
  595. ?assertEqual(1, count(?TEST_SINK_EVENT)),
  596. {Level, _Time, Message,_Metadata} = pop(?TEST_SINK_EVENT),
  597. ?assertMatch(Level, lager_util:level_to_num(warning)),
  598. ?assertEqual(lists:flatten(io_lib:format("test message ~p", [self()])), lists:flatten(Message)),
  599. ok
  600. end
  601. },
  602. {"variables inplace of literals in logging statements work",
  603. fun() ->
  604. ?assertEqual(0, count(?TEST_SINK_EVENT)),
  605. Attr = [{a, alpha}, {b, beta}],
  606. Fmt = "format ~p",
  607. Args = [world],
  608. ?TEST_SINK_NAME:info(Attr, "hello"),
  609. ?TEST_SINK_NAME:info(Attr, "hello ~p", [world]),
  610. ?TEST_SINK_NAME:info(Fmt, [world]),
  611. ?TEST_SINK_NAME:info("hello ~p", Args),
  612. ?TEST_SINK_NAME:info(Attr, "hello ~p", Args),
  613. ?TEST_SINK_NAME:info([{d, delta}, {g, gamma}], Fmt, Args),
  614. ?assertEqual(6, count(?TEST_SINK_EVENT)),
  615. {_Level, _Time, Message, Metadata} = pop(?TEST_SINK_EVENT),
  616. ?assertMatch([{a, alpha}, {b, beta}|_], Metadata),
  617. ?assertEqual("hello", lists:flatten(Message)),
  618. {_Level, _Time2, Message2, _Metadata2} = pop(?TEST_SINK_EVENT),
  619. ?assertEqual("hello world", lists:flatten(Message2)),
  620. {_Level, _Time3, Message3, _Metadata3} = pop(?TEST_SINK_EVENT),
  621. ?assertEqual("format world", lists:flatten(Message3)),
  622. {_Level, _Time4, Message4, _Metadata4} = pop(?TEST_SINK_EVENT),
  623. ?assertEqual("hello world", lists:flatten(Message4)),
  624. {_Level, _Time5, Message5, _Metadata5} = pop(?TEST_SINK_EVENT),
  625. ?assertEqual("hello world", lists:flatten(Message5)),
  626. {_Level, _Time6, Message6, Metadata6} = pop(?TEST_SINK_EVENT),
  627. ?assertMatch([{d, delta}, {g, gamma}|_], Metadata6),
  628. ?assertEqual("format world", lists:flatten(Message6)),
  629. ok
  630. end
  631. },
  632. {"log messages below the threshold are ignored",
  633. fun() ->
  634. ?assertEqual(0, count(?TEST_SINK_EVENT)),
  635. ?TEST_SINK_NAME:debug("this message will be ignored"),
  636. ?assertEqual(0, count(?TEST_SINK_EVENT)),
  637. ?assertEqual(0, count_ignored(?TEST_SINK_EVENT)),
  638. lager_config:set({?TEST_SINK_EVENT, loglevel}, {element(2, lager_util:config_to_mask(debug)), []}),
  639. ?TEST_SINK_NAME:debug("this message should be ignored"),
  640. ?assertEqual(0, count(?TEST_SINK_EVENT)),
  641. ?assertEqual(1, count_ignored(?TEST_SINK_EVENT)),
  642. lager:set_loglevel(?TEST_SINK_EVENT, ?MODULE, undefined, debug),
  643. ?assertEqual({?DEBUG bor ?INFO bor ?NOTICE bor ?WARNING bor ?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get({?TEST_SINK_EVENT, loglevel})),
  644. ?TEST_SINK_NAME:debug("this message should be logged"),
  645. ?assertEqual(1, count(?TEST_SINK_EVENT)),
  646. ?assertEqual(1, count_ignored(?TEST_SINK_EVENT)),
  647. ?assertEqual(debug, lager:get_loglevel(?TEST_SINK_EVENT, ?MODULE)),
  648. ok
  649. end
  650. }
  651. ]
  652. }.
  653. setup_sink() ->
  654. error_logger:tty(false),
  655. application:load(lager),
  656. application:set_env(lager, handlers, []),
  657. application:set_env(lager, error_logger_redirect, false),
  658. application:set_env(lager, extra_sinks, [{?TEST_SINK_EVENT, [{handlers, [{?MODULE, info}]}]}]),
  659. lager:start(),
  660. gen_event:call(lager_event, ?MODULE, flush),
  661. gen_event:call(?TEST_SINK_EVENT, ?MODULE, flush).
  662. setup() ->
  663. error_logger:tty(false),
  664. application:load(lager),
  665. application:set_env(lager, handlers, [{?MODULE, info}]),
  666. application:set_env(lager, error_logger_redirect, false),
  667. lager:start(),
  668. gen_event:call(lager_event, ?MODULE, flush).
  669. cleanup(_) ->
  670. application:stop(lager),
  671. application:stop(goldrush),
  672. error_logger:tty(true).
  673. crash(Type) ->
  674. spawn(fun() -> gen_server:call(crash, Type) end),
  675. timer:sleep(100),
  676. _ = gen_event:which_handlers(error_logger),
  677. ok.
  678. test_body(Expected, Actual) ->
  679. case has_line_numbers() of
  680. true ->
  681. ExLen = length(Expected),
  682. {Body, Rest} = case length(Actual) > ExLen of
  683. true ->
  684. {string:substr(Actual, 1, ExLen),
  685. string:substr(Actual, (ExLen + 1))};
  686. _ ->
  687. {Actual, []}
  688. end,
  689. ?assertEqual(Expected, Body),
  690. % OTP-17 (and maybe later releases) may tack on additional info
  691. % about the failure, so if Actual starts with Expected (already
  692. % confirmed by having gotten past assertEqual above) and ends
  693. % with " line NNN" we can ignore what's in-between. By extension,
  694. % since there may not be line information appended at all, any
  695. % text we DO find is reportable, but not a test failure.
  696. case Rest of
  697. [] ->
  698. ok;
  699. _ ->
  700. % isolate the extra data and report it if it's not just
  701. % a line number indicator
  702. case re:run(Rest, "^.*( line \\d+)$", [{capture, [1]}]) of
  703. nomatch ->
  704. ?debugFmt(
  705. "Trailing data \"~s\" following \"~s\"",
  706. [Rest, Expected]);
  707. {match, [{0, _}]} ->
  708. % the whole sting is " line NNN"
  709. ok;
  710. {match, [{Off, _}]} ->
  711. ?debugFmt(
  712. "Trailing data \"~s\" following \"~s\"",
  713. [string:substr(Rest, 1, Off), Expected])
  714. end
  715. end;
  716. _ ->
  717. ?assertEqual(Expected, Actual)
  718. end.
  719. error_logger_redirect_crash_test_() ->
  720. TestBody=fun(Name,CrashReason,Expected) -> {Name,
  721. fun() ->
  722. Pid = whereis(crash),
  723. crash(CrashReason),
  724. {Level, _, Msg,Metadata} = pop(),
  725. test_body(Expected, lists:flatten(Msg)),
  726. ?assertEqual(Pid,proplists:get_value(pid,Metadata)),
  727. ?assertEqual(lager_util:level_to_num(error),Level)
  728. end
  729. }
  730. end,
  731. {foreach,
  732. fun() ->
  733. error_logger:tty(false),
  734. application:load(lager),
  735. application:set_env(lager, error_logger_redirect, true),
  736. application:set_env(lager, handlers, [{?MODULE, error}]),
  737. lager:start(),
  738. crash:start()
  739. end,
  740. fun(_) ->
  741. application:stop(lager),
  742. application:stop(goldrush),
  743. case whereis(crash) of
  744. undefined -> ok;
  745. Pid -> exit(Pid, kill)
  746. end,
  747. error_logger:tty(true)
  748. end,
  749. [
  750. {"again, there is nothing up my sleeve",
  751. fun() ->
  752. ?assertEqual(undefined, pop()),
  753. ?assertEqual(0, count())
  754. end
  755. },
  756. TestBody("bad return value",bad_return,"gen_server crash terminated with reason: bad return value: bleh"),
  757. TestBody("bad return value with string",bad_return_string,"gen_server crash terminated with reason: bad return value: {tuple,{tuple,\"string\"}}"),
  758. TestBody("bad return uncaught throw",throw,"gen_server crash terminated with reason: bad return value: a_ball"),
  759. TestBody("case clause",case_clause,"gen_server crash terminated with reason: no case clause matching {} in crash:handle_call/3"),
  760. TestBody("case clause string",case_clause_string,"gen_server crash terminated with reason: no case clause matching \"crash\" in crash:handle_call/3"),
  761. TestBody("function clause",function_clause,"gen_server crash terminated with reason: no function clause matching crash:function({})"),
  762. TestBody("if clause",if_clause,"gen_server crash terminated with reason: no true branch found while evaluating if expression in crash:handle_call/3"),
  763. TestBody("try clause",try_clause,"gen_server crash terminated with reason: no try clause matching [] in crash:handle_call/3"),
  764. TestBody("undefined function",undef,"gen_server crash terminated with reason: call to undefined function crash:booger/0 from crash:handle_call/3"),
  765. TestBody("bad math",badarith,"gen_server crash terminated with reason: bad arithmetic expression in crash:handle_call/3"),
  766. TestBody("bad match",badmatch,"gen_server crash terminated with reason: no match of right hand value {} in crash:handle_call/3"),
  767. TestBody("bad arity",badarity,"gen_server crash terminated with reason: fun called with wrong arity of 1 instead of 3 in crash:handle_call/3"),
  768. TestBody("bad arg1",badarg1,"gen_server crash terminated with reason: bad argument in crash:handle_call/3"),
  769. TestBody("bad arg2",badarg2,"gen_server crash terminated with reason: bad argument in call to erlang:iolist_to_binary([\"foo\",bar]) in crash:handle_call/3"),
  770. TestBody("bad record",badrecord,"gen_server crash terminated with reason: bad record state in crash:handle_call/3"),
  771. TestBody("noproc",noproc,"gen_server crash terminated with reason: no such process or port in call to gen_event:call(foo, bar, baz)"),
  772. TestBody("badfun",badfun,"gen_server crash terminated with reason: bad function booger in crash:handle_call/3")
  773. ]
  774. }.
  775. error_logger_redirect_test_() ->
  776. {foreach,
  777. fun() ->
  778. error_logger:tty(false),
  779. application:load(lager),
  780. application:set_env(lager, error_logger_redirect, true),
  781. application:set_env(lager, handlers, [{?MODULE, info}]),
  782. lager:start(),
  783. lager:log(error, self(), "flush flush"),
  784. timer:sleep(100),
  785. gen_event:call(lager_event, ?MODULE, flush)
  786. end,
  787. fun(_) ->
  788. application:stop(lager),
  789. application:stop(goldrush),
  790. error_logger:tty(true)
  791. end,
  792. [
  793. {"error reports are printed",
  794. fun() ->
  795. sync_error_logger:error_report([{this, is}, a, {silly, format}]),
  796. _ = gen_event:which_handlers(error_logger),
  797. {Level, _, Msg,Metadata} = pop(),
  798. ?assertEqual(lager_util:level_to_num(error),Level),
  799. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  800. Expected = "this: is, a, silly: format",
  801. ?assertEqual(Expected, lists:flatten(Msg))
  802. end
  803. },
  804. {"string error reports are printed",
  805. fun() ->
  806. sync_error_logger:error_report("this is less silly"),
  807. _ = gen_event:which_handlers(error_logger),
  808. {Level, _, Msg,Metadata} = pop(),
  809. ?assertEqual(lager_util:level_to_num(error),Level),
  810. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  811. Expected = "this is less silly",
  812. ?assertEqual(Expected, lists:flatten(Msg))
  813. end
  814. },
  815. {"error messages are printed",
  816. fun() ->
  817. sync_error_logger:error_msg("doom, doom has come upon you all"),
  818. _ = gen_event:which_handlers(error_logger),
  819. {Level, _, Msg,Metadata} = pop(),
  820. ?assertEqual(lager_util:level_to_num(error),Level),
  821. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  822. Expected = "doom, doom has come upon you all",
  823. ?assertEqual(Expected, lists:flatten(Msg))
  824. end
  825. },
  826. {"error messages with unicode characters in Args are printed",
  827. fun() ->
  828. sync_error_logger:error_msg("~ts", ["Привет!"]),
  829. _ = gen_event:which_handlers(error_logger),
  830. {Level, _, Msg,Metadata} = pop(),
  831. ?assertEqual(lager_util:level_to_num(error),Level),
  832. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  833. ?assertEqual("Привет!", lists:flatten(Msg))
  834. end
  835. },
  836. {"error messages are truncated at 4096 characters",
  837. fun() ->
  838. sync_error_logger:error_msg("doom, doom has come upon you all ~p", [string:copies("doom", 10000)]),
  839. _ = gen_event:which_handlers(error_logger),
  840. {_, _, Msg,_Metadata} = pop(),
  841. ?assert(length(lists:flatten(Msg)) < 5100)
  842. end
  843. },
  844. {"info reports are printed",
  845. fun() ->
  846. sync_error_logger:info_report([{this, is}, a, {silly, format}]),
  847. _ = gen_event:which_handlers(error_logger),
  848. {Level, _, Msg,Metadata} = pop(),
  849. ?assertEqual(lager_util:level_to_num(info),Level),
  850. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  851. Expected = "this: is, a, silly: format",
  852. ?assertEqual(Expected, lists:flatten(Msg))
  853. end
  854. },
  855. {"info reports are truncated at 4096 characters",
  856. fun() ->
  857. sync_error_logger:info_report([[{this, is}, a, {silly, format}] || _ <- lists:seq(0, 600)]),
  858. _ = gen_event:which_handlers(error_logger),
  859. {Level, _, Msg,Metadata} = pop(),
  860. ?assertEqual(lager_util:level_to_num(info),Level),
  861. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  862. ?assert(length(lists:flatten(Msg)) < 5000)
  863. end
  864. },
  865. {"single term info reports are printed",
  866. fun() ->
  867. sync_error_logger:info_report({foolish, bees}),
  868. _ = gen_event:which_handlers(error_logger),
  869. {Level, _, Msg,Metadata} = pop(),
  870. ?assertEqual(lager_util:level_to_num(info),Level),
  871. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  872. ?assertEqual("{foolish,bees}", lists:flatten(Msg))
  873. end
  874. },
  875. {"single term error reports are printed",
  876. fun() ->
  877. sync_error_logger:error_report({foolish, bees}),
  878. _ = gen_event:which_handlers(error_logger),
  879. {Level, _, Msg,Metadata} = pop(),
  880. ?assertEqual(lager_util:level_to_num(error),Level),
  881. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  882. ?assertEqual("{foolish,bees}", lists:flatten(Msg))
  883. end
  884. },
  885. {"string info reports are printed",
  886. fun() ->
  887. sync_error_logger:info_report("this is less silly"),
  888. _ = gen_event:which_handlers(error_logger),
  889. {Level, _, Msg,Metadata} = pop(),
  890. ?assertEqual(lager_util:level_to_num(info),Level),
  891. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  892. ?assertEqual("this is less silly", lists:flatten(Msg))
  893. end
  894. },
  895. {"string info reports are truncated at 4096 characters",
  896. fun() ->
  897. sync_error_logger:info_report(string:copies("this is less silly", 1000)),
  898. _ = gen_event:which_handlers(error_logger),
  899. {Level, _, Msg,Metadata} = pop(),
  900. ?assertEqual(lager_util:level_to_num(info),Level),
  901. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  902. ?assert(length(lists:flatten(Msg)) < 5100)
  903. end
  904. },
  905. {"strings in a mixed report are printed as strings",
  906. fun() ->
  907. sync_error_logger:info_report(["this is less silly", {than, "this"}]),
  908. _ = gen_event:which_handlers(error_logger),
  909. {Level, _, Msg,Metadata} = pop(),
  910. ?assertEqual(lager_util:level_to_num(info),Level),
  911. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  912. ?assertEqual("\"this is less silly\", than: \"this\"", lists:flatten(Msg))
  913. end
  914. },
  915. {"info messages are printed",
  916. fun() ->
  917. sync_error_logger:info_msg("doom, doom has come upon you all"),
  918. _ = gen_event:which_handlers(error_logger),
  919. {Level, _, Msg,Metadata} = pop(),
  920. ?assertEqual(lager_util:level_to_num(info),Level),
  921. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  922. ?assertEqual("doom, doom has come upon you all", lists:flatten(Msg))
  923. end
  924. },
  925. {"info messages are truncated at 4096 characters",
  926. fun() ->
  927. sync_error_logger:info_msg("doom, doom has come upon you all ~p", [string:copies("doom", 10000)]),
  928. _ = gen_event:which_handlers(error_logger),
  929. {Level, _, Msg,Metadata} = pop(),
  930. ?assertEqual(lager_util:level_to_num(info),Level),
  931. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  932. ?assert(length(lists:flatten(Msg)) < 5100)
  933. end
  934. },
  935. {"info messages with unicode characters in Args are printed",
  936. fun() ->
  937. sync_error_logger:info_msg("~ts", ["Привет!"]),
  938. _ = gen_event:which_handlers(error_logger),
  939. {Level, _, Msg,Metadata} = pop(),
  940. ?assertEqual(lager_util:level_to_num(info),Level),
  941. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  942. ?assertEqual("Привет!", lists:flatten(Msg))
  943. end
  944. },
  945. {"warning messages with unicode characters in Args are printed",
  946. fun() ->
  947. sync_error_logger:warning_msg("~ts", ["Привет!"]),
  948. Map = error_logger:warning_map(),
  949. _ = gen_event:which_handlers(error_logger),
  950. {Level, _, Msg,Metadata} = pop(),
  951. ?assertEqual(lager_util:level_to_num(Map),Level),
  952. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  953. ?assertEqual("Привет!", lists:flatten(Msg))
  954. end
  955. },
  956. {"warning messages are printed at the correct level",
  957. fun() ->
  958. sync_error_logger:warning_msg("doom, doom has come upon you all"),
  959. Map = error_logger:warning_map(),
  960. _ = gen_event:which_handlers(error_logger),
  961. {Level, _, Msg,Metadata} = pop(),
  962. ?assertEqual(lager_util:level_to_num(Map),Level),
  963. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  964. ?assertEqual("doom, doom has come upon you all", lists:flatten(Msg))
  965. end
  966. },
  967. {"warning reports are printed at the correct level",
  968. fun() ->
  969. sync_error_logger:warning_report([{i, like}, pie]),
  970. Map = error_logger:warning_map(),
  971. _ = gen_event:which_handlers(error_logger),
  972. {Level, _, Msg,Metadata} = pop(),
  973. ?assertEqual(lager_util:level_to_num(Map),Level),
  974. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  975. ?assertEqual("i: like, pie", lists:flatten(Msg))
  976. end
  977. },
  978. {"single term warning reports are printed at the correct level",
  979. fun() ->
  980. sync_error_logger:warning_report({foolish, bees}),
  981. Map = error_logger:warning_map(),
  982. _ = gen_event:which_handlers(error_logger),
  983. {Level, _, Msg,Metadata} = pop(),
  984. ?assertEqual(lager_util:level_to_num(Map),Level),
  985. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  986. ?assertEqual("{foolish,bees}", lists:flatten(Msg))
  987. end
  988. },
  989. {"application stop reports",
  990. fun() ->
  991. sync_error_logger:info_report([{application, foo}, {exited, quittin_time}, {type, lazy}]),
  992. _ = gen_event:which_handlers(error_logger),
  993. {Level, _, Msg,Metadata} = pop(),
  994. ?assertEqual(lager_util:level_to_num(info),Level),
  995. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  996. ?assertEqual("Application foo exited with reason: quittin_time", lists:flatten(Msg))
  997. end
  998. },
  999. {"supervisor reports",
  1000. fun() ->
  1001. sync_error_logger:error_report(supervisor_report, [{errorContext, france}, {offender, [{name, mini_steve}, {mfargs, {a, b, [c]}}, {pid, bleh}]}, {reason, fired}, {supervisor, {local, steve}}]),
  1002. _ = gen_event:which_handlers(error_logger),
  1003. {Level, _, Msg,Metadata} = pop(),
  1004. ?assertEqual(lager_util:level_to_num(error),Level),
  1005. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1006. ?assertEqual("Supervisor steve had child mini_steve started with a:b(c) at bleh exit with reason fired in context france", lists:flatten(Msg))
  1007. end
  1008. },
  1009. {"supervisor reports with real error",
  1010. fun() ->
  1011. sync_error_logger:error_report(supervisor_report, [{errorContext, france}, {offender, [{name, mini_steve}, {mfargs, {a, b, [c]}}, {pid, bleh}]}, {reason, {function_clause,[{crash,handle_info,[foo]}]}}, {supervisor, {local, steve}}]),
  1012. _ = gen_event:which_handlers(error_logger),
  1013. {Level, _, Msg,Metadata} = pop(),
  1014. ?assertEqual(lager_util:level_to_num(error),Level),
  1015. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1016. ?assertEqual("Supervisor steve had child mini_steve started with a:b(c) at bleh exit with reason no function clause matching crash:handle_info(foo) in context france", lists:flatten(Msg))
  1017. end
  1018. },
  1019. {"supervisor reports with real error and pid",
  1020. fun() ->
  1021. sync_error_logger:error_report(supervisor_report, [{errorContext, france}, {offender, [{name, mini_steve}, {mfargs, {a, b, [c]}}, {pid, bleh}]}, {reason, {function_clause,[{crash,handle_info,[foo]}]}}, {supervisor, somepid}]),
  1022. _ = gen_event:which_handlers(error_logger),
  1023. {Level, _, Msg,Metadata} = pop(),
  1024. ?assertEqual(lager_util:level_to_num(error),Level),
  1025. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1026. ?assertEqual("Supervisor somepid had child mini_steve started with a:b(c) at bleh exit with reason no function clause matching crash:handle_info(foo) in context france", lists:flatten(Msg))
  1027. end
  1028. },
  1029. {"supervisor_bridge reports",
  1030. fun() ->
  1031. sync_error_logger:error_report(supervisor_report, [{errorContext, france}, {offender, [{mod, mini_steve}, {pid, bleh}]}, {reason, fired}, {supervisor, {local, steve}}]),
  1032. _ = gen_event:which_handlers(error_logger),
  1033. {Level, _, Msg,Metadata} = pop(),
  1034. ?assertEqual(lager_util:level_to_num(error),Level),
  1035. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1036. ?assertEqual("Supervisor steve had child at module mini_steve at bleh exit with reason fired in context france", lists:flatten(Msg))
  1037. end
  1038. },
  1039. {"application progress report",
  1040. fun() ->
  1041. sync_error_logger:info_report(progress, [{application, foo}, {started_at, node()}]),
  1042. _ = gen_event:which_handlers(error_logger),
  1043. {Level, _, Msg,Metadata} = pop(),
  1044. ?assertEqual(lager_util:level_to_num(info),Level),
  1045. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1046. Expected = lists:flatten(io_lib:format("Application foo started on node ~w", [node()])),
  1047. ?assertEqual(Expected, lists:flatten(Msg))
  1048. end
  1049. },
  1050. {"supervisor progress report",
  1051. fun() ->
  1052. lager:set_loglevel(?MODULE, debug),
  1053. ?assertEqual({?DEBUG bor ?INFO bor ?NOTICE bor ?WARNING bor ?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  1054. sync_error_logger:info_report(progress, [{supervisor, {local, foo}}, {started, [{mfargs, {foo, bar, 1}}, {pid, baz}]}]),
  1055. _ = gen_event:which_handlers(error_logger),
  1056. {Level, _, Msg,Metadata} = pop(),
  1057. ?assertEqual(lager_util:level_to_num(debug),Level),
  1058. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1059. ?assertEqual("Supervisor foo started foo:bar/1 at pid baz", lists:flatten(Msg))
  1060. end
  1061. },
  1062. {"supervisor progress report with pid",
  1063. fun() ->
  1064. lager:set_loglevel(?MODULE, debug),
  1065. ?assertEqual({?DEBUG bor ?INFO bor ?NOTICE bor ?WARNING bor ?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  1066. sync_error_logger:info_report(progress, [{supervisor, somepid}, {started, [{mfargs, {foo, bar, 1}}, {pid, baz}]}]),
  1067. _ = gen_event:which_handlers(error_logger),
  1068. {Level, _, Msg,Metadata} = pop(),
  1069. ?assertEqual(lager_util:level_to_num(debug),Level),
  1070. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1071. ?assertEqual("Supervisor somepid started foo:bar/1 at pid baz", lists:flatten(Msg))
  1072. end
  1073. },
  1074. {"crash report for emfile",
  1075. fun() ->
  1076. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, emfile, [{stack, trace, 1}]}}], []]),
  1077. _ = gen_event:which_handlers(error_logger),
  1078. {Level, _, Msg,Metadata} = pop(),
  1079. ?assertEqual(lager_util:level_to_num(error),Level),
  1080. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1081. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: maximum number of file descriptors exhausted, check ulimit -n", [self()])),
  1082. ?assertEqual(Expected, lists:flatten(Msg))
  1083. end
  1084. },
  1085. {"crash report for system process limit",
  1086. fun() ->
  1087. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, system_limit, [{erlang, spawn, 1}]}}], []]),
  1088. _ = gen_event:which_handlers(error_logger),
  1089. {Level, _, Msg,Metadata} = pop(),
  1090. ?assertEqual(lager_util:level_to_num(error),Level),
  1091. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1092. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: system limit: maximum number of processes exceeded", [self()])),
  1093. ?assertEqual(Expected, lists:flatten(Msg))
  1094. end
  1095. },
  1096. {"crash report for system process limit2",
  1097. fun() ->
  1098. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, system_limit, [{erlang, spawn_opt, 1}]}}], []]),
  1099. _ = gen_event:which_handlers(error_logger),
  1100. {Level, _, Msg,Metadata} = pop(),
  1101. ?assertEqual(lager_util:level_to_num(error),Level),
  1102. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1103. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: system limit: maximum number of processes exceeded", [self()])),
  1104. ?assertEqual(Expected, lists:flatten(Msg))
  1105. end
  1106. },
  1107. {"crash report for system port limit",
  1108. fun() ->
  1109. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, system_limit, [{erlang, open_port, 1}]}}], []]),
  1110. _ = gen_event:which_handlers(error_logger),
  1111. {Level, _, Msg,Metadata} = pop(),
  1112. ?assertEqual(lager_util:level_to_num(error),Level),
  1113. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1114. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: system limit: maximum number of ports exceeded", [self()])),
  1115. ?assertEqual(Expected, lists:flatten(Msg))
  1116. end
  1117. },
  1118. {"crash report for system port limit",
  1119. fun() ->
  1120. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, system_limit, [{erlang, list_to_atom, 1}]}}], []]),
  1121. _ = gen_event:which_handlers(error_logger),
  1122. {Level, _, Msg,Metadata} = pop(),
  1123. ?assertEqual(lager_util:level_to_num(error),Level),
  1124. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1125. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: system limit: tried to create an atom larger than 255, or maximum atom count exceeded", [self()])),
  1126. ?assertEqual(Expected, lists:flatten(Msg))
  1127. end
  1128. },
  1129. {"crash report for system ets table limit",
  1130. fun() ->
  1131. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, test}, {error_info, {error, system_limit, [{ets,new,[segment_offsets,[ordered_set,public]]},{mi_segment,open_write,1},{mi_buffer_converter,handle_cast,2},{gen_server,handle_msg,5},{proc_lib,init_p_do_apply,3}]}}], []]),
  1132. _ = gen_event:which_handlers(error_logger),
  1133. {Level, _, Msg,Metadata} = pop(),
  1134. ?assertEqual(lager_util:level_to_num(error),Level),
  1135. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1136. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: system limit: maximum number of ETS tables exceeded", [test])),
  1137. ?assertEqual(Expected, lists:flatten(Msg))
  1138. end
  1139. },
  1140. {"crash report for unknown system limit should be truncated at 500 characters",
  1141. fun() ->
  1142. sync_error_logger:error_report(crash_report, [[{pid, self()}, {error_info, {error, system_limit, [{wtf,boom,[string:copies("aaaa", 4096)]}]}}], []]),
  1143. _ = gen_event:which_handlers(error_logger),
  1144. {_, _, Msg,_Metadata} = pop(),
  1145. ?assert(length(lists:flatten(Msg)) > 550),
  1146. ?assert(length(lists:flatten(Msg)) < 600)
  1147. end
  1148. },
  1149. {"crash reports for 'special processes' should be handled right - function_clause",
  1150. fun() ->
  1151. {ok, Pid} = special_process:start(),
  1152. unlink(Pid),
  1153. Pid ! function_clause,
  1154. timer:sleep(500),
  1155. _ = gen_event:which_handlers(error_logger),
  1156. {_, _, Msg, _Metadata} = pop(),
  1157. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~p with 0 neighbours crashed with reason: no function clause matching special_process:foo(bar)",
  1158. [Pid])),
  1159. test_body(Expected, lists:flatten(Msg))
  1160. end
  1161. },
  1162. {"crash reports for 'special processes' should be handled right - case_clause",
  1163. fun() ->
  1164. {ok, Pid} = special_process:start(),
  1165. unlink(Pid),
  1166. Pid ! {case_clause, wtf},
  1167. timer:sleep(500),
  1168. _ = gen_event:which_handlers(error_logger),
  1169. {_, _, Msg, _Metadata} = pop(),
  1170. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~p with 0 neighbours crashed with reason: no case clause matching wtf in special_process:loop/0",
  1171. [Pid])),
  1172. test_body(Expected, lists:flatten(Msg))
  1173. end
  1174. },
  1175. {"crash reports for 'special processes' should be handled right - exit",
  1176. fun() ->
  1177. {ok, Pid} = special_process:start(),
  1178. unlink(Pid),
  1179. Pid ! exit,
  1180. timer:sleep(500),
  1181. _ = gen_event:which_handlers(error_logger),
  1182. {_, _, Msg, _Metadata} = pop(),
  1183. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~p with 0 neighbours exited with reason: byebye in special_process:loop/0",
  1184. [Pid])),
  1185. test_body(Expected, lists:flatten(Msg))
  1186. end
  1187. },
  1188. {"crash reports for 'special processes' should be handled right - error",
  1189. fun() ->
  1190. {ok, Pid} = special_process:start(),
  1191. unlink(Pid),
  1192. Pid ! error,
  1193. timer:sleep(500),
  1194. _ = gen_event:which_handlers(error_logger),
  1195. {_, _, Msg, _Metadata} = pop(),
  1196. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~p with 0 neighbours crashed with reason: mybad in special_process:loop/0",
  1197. [Pid])),
  1198. test_body(Expected, lists:flatten(Msg))
  1199. end
  1200. },
  1201. {"webmachine error reports",
  1202. fun() ->
  1203. Path = "/cgi-bin/phpmyadmin",
  1204. Reason = {error,{error,{badmatch,{error,timeout}},
  1205. [{myapp,dostuff,2,[{file,"src/myapp.erl"},{line,123}]},
  1206. {webmachine_resource,resource_call,3,[{file,"src/webmachine_resource.erl"},{line,169}]}]}},
  1207. sync_error_logger:error_msg("webmachine error: path=~p~n~p~n", [Path, Reason]),
  1208. _ = gen_event:which_handlers(error_logger),
  1209. {Level, _, Msg,Metadata} = pop(),
  1210. ?assertEqual(lager_util:level_to_num(error),Level),
  1211. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1212. ?assertEqual("Webmachine error at path \"/cgi-bin/phpmyadmin\" : no match of right hand value {error,timeout} in myapp:dostuff/2 line 123", lists:flatten(Msg))
  1213. end
  1214. },
  1215. {"Cowboy error reports, 8 arg version",
  1216. fun() ->
  1217. Stack = [{my_handler,init, 3,[{file,"src/my_handler.erl"},{line,123}]},
  1218. {cowboy_handler,handler_init,4,[{file,"src/cowboy_handler.erl"},{line,169}]}],
  1219. sync_error_logger:error_msg(
  1220. "** Cowboy handler ~p terminating in ~p/~p~n"
  1221. " for the reason ~p:~p~n"
  1222. "** Options were ~p~n"
  1223. "** Request was ~p~n"
  1224. "** Stacktrace: ~p~n~n",
  1225. [my_handler, init, 3, error, {badmatch, {error, timeout}}, [],
  1226. "Request", Stack]),
  1227. _ = gen_event:which_handlers(error_logger),
  1228. {Level, _, Msg,Metadata} = pop(),
  1229. ?assertEqual(lager_util:level_to_num(error),Level),
  1230. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1231. ?assertEqual("Cowboy handler my_handler terminated in my_handler:init/3 with reason: no match of right hand value {error,timeout} in my_handler:init/3 line 123", lists:flatten(Msg))
  1232. end
  1233. },
  1234. {"Cowboy error reports, 10 arg version",
  1235. fun() ->
  1236. Stack = [{my_handler,somecallback, 3,[{file,"src/my_handler.erl"},{line,123}]},
  1237. {cowboy_handler,handler_init,4,[{file,"src/cowboy_handler.erl"},{line,169}]}],
  1238. sync_error_logger:error_msg(
  1239. "** Cowboy handler ~p terminating in ~p/~p~n"
  1240. " for the reason ~p:~p~n** Message was ~p~n"
  1241. "** Options were ~p~n** Handler state was ~p~n"
  1242. "** Request was ~p~n** Stacktrace: ~p~n~n",
  1243. [my_handler, somecallback, 3, error, {badmatch, {error, timeout}}, hello, [],
  1244. {}, "Request", Stack]),
  1245. _ = gen_event:which_handlers(error_logger),
  1246. {Level, _, Msg,Metadata} = pop(),
  1247. ?assertEqual(lager_util:level_to_num(error),Level),
  1248. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1249. ?assertEqual("Cowboy handler my_handler terminated in my_handler:somecallback/3 with reason: no match of right hand value {error,timeout} in my_handler:somecallback/3 line 123", lists:flatten(Msg))
  1250. end
  1251. },
  1252. {"Cowboy error reports, 5 arg version",
  1253. fun() ->
  1254. sync_error_logger:error_msg(
  1255. "** Cowboy handler ~p terminating; "
  1256. "function ~p/~p was not exported~n"
  1257. "** Request was ~p~n** State was ~p~n~n",
  1258. [my_handler, to_json, 2, "Request", {}]),
  1259. _ = gen_event:which_handlers(error_logger),
  1260. {Level, _, Msg,Metadata} = pop(),
  1261. ?assertEqual(lager_util:level_to_num(error),Level),
  1262. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  1263. ?assertEqual("Cowboy handler my_handler terminated with reason: call to undefined function my_handler:to_json/2", lists:flatten(Msg))
  1264. end
  1265. },
  1266. {"messages should not be generated if they don't satisfy the threshold",
  1267. fun() ->
  1268. lager:set_loglevel(?MODULE, error),
  1269. ?assertEqual({?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  1270. sync_error_logger:info_report([hello, world]),
  1271. _ = gen_event:which_handlers(error_logger),
  1272. ?assertEqual(0, count()),
  1273. ?assertEqual(0, count_ignored()),
  1274. lager:set_loglevel(?MODULE, info),
  1275. ?assertEqual({?INFO bor ?NOTICE bor ?WARNING bor ?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  1276. sync_error_logger:info_report([hello, world]),
  1277. _ = gen_event:which_handlers(error_logger),
  1278. ?assertEqual(1, count()),
  1279. ?assertEqual(0, count_ignored()),
  1280. lager:set_loglevel(?MODULE, error),
  1281. ?assertEqual({?ERROR bor ?CRITICAL bor ?ALERT bor ?EMERGENCY, []}, lager_config:get(loglevel)),
  1282. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(debug)), []}),
  1283. sync_error_logger:info_report([hello, world]),
  1284. _ = gen_event:which_handlers(error_logger),
  1285. ?assertEqual(1, count()),
  1286. ?assertEqual(1, count_ignored())
  1287. end
  1288. }
  1289. ]
  1290. }.
  1291. safe_format_test() ->
  1292. ?assertEqual("foo bar", lists:flatten(lager:safe_format("~p ~p", [foo, bar], 1024))),
  1293. ?assertEqual("FORMAT ERROR: \"~p ~p ~p\" [foo,bar]", lists:flatten(lager:safe_format("~p ~p ~p", [foo, bar], 1024))),
  1294. ok.
  1295. async_threshold_test_() ->
  1296. {foreach,
  1297. fun() ->
  1298. error_logger:tty(false),
  1299. ets:new(async_threshold_test, [set, named_table, public]),
  1300. ets:insert_new(async_threshold_test, {sync_toggled, 0}),
  1301. ets:insert_new(async_threshold_test, {async_toggled, 0}),
  1302. application:load(lager),
  1303. application:set_env(lager, error_logger_redirect, false),
  1304. application:set_env(lager, async_threshold, 2),
  1305. application:set_env(lager, async_threshold_window, 1),
  1306. application:set_env(lager, handlers, [{?MODULE, info}]),
  1307. lager:start()
  1308. end,
  1309. fun(_) ->
  1310. application:unset_env(lager, async_threshold),
  1311. application:stop(lager),
  1312. application:stop(goldrush),
  1313. ets:delete(async_threshold_test),
  1314. error_logger:tty(true)
  1315. end,
  1316. [
  1317. {"async threshold works",
  1318. fun() ->
  1319. %% we start out async
  1320. ?assertEqual(true, lager_config:get(async)),
  1321. %% put a ton of things in the queue
  1322. Workers = [spawn_monitor(fun() -> [lager:info("hello world") || _ <- lists:seq(1, 1000)] end) || _ <- lists:seq(1, 15)],
  1323. %% serialize on mailbox
  1324. _ = gen_event:which_handlers(lager_event),
  1325. timer:sleep(500),
  1326. %% By now the flood of messages will have
  1327. %% forced the backend throttle to turn off
  1328. %% async mode, but it's possible all
  1329. %% outstanding requests have been processed,
  1330. %% so checking the current status (sync or
  1331. %% async) is an exercise in race control.
  1332. %% Instead, we'll see whether the backend
  1333. %% throttle has toggled into sync mode at any
  1334. %% point in the past
  1335. ?assertMatch([{sync_toggled, N}] when N > 0,
  1336. ets:lookup(async_threshold_test, sync_toggled)),
  1337. %% wait for all the workers to return, meaning that all the messages have been logged (since we're definitely in sync mode at the end of the run)
  1338. collect_workers(Workers),
  1339. %% serialize on the mailbox again
  1340. _ = gen_event:which_handlers(lager_event),
  1341. %% just in case...
  1342. timer:sleep(1000),
  1343. lager:info("hello world"),
  1344. _ = gen_event:which_handlers(lager_event),
  1345. %% async is true again now that the mailbox has drained
  1346. ?assertEqual(true, lager_config:get(async)),
  1347. ok
  1348. end
  1349. }
  1350. ]
  1351. }.
  1352. collect_workers([]) ->
  1353. ok;
  1354. collect_workers(Workers) ->
  1355. receive
  1356. {'DOWN', Ref, _, _, _} ->
  1357. collect_workers(lists:keydelete(Ref, 2, Workers))
  1358. end.
  1359. produce_n_error_logger_msgs(N) ->
  1360. lists:foreach(fun (K) ->
  1361. error_logger:error_msg("Foo ~p!", [K])
  1362. end,
  1363. lists:seq(0, N-1)
  1364. ).
  1365. high_watermark_test_() ->
  1366. {foreach,
  1367. fun() ->
  1368. error_logger:tty(false),
  1369. application:load(lager),
  1370. application:set_env(lager, error_logger_redirect, true),
  1371. application:set_env(lager, handlers, [{lager_test_backend, info}]),
  1372. application:set_env(lager, async_threshold, undefined),
  1373. lager:start()
  1374. end,
  1375. fun(_) ->
  1376. application:stop(lager),
  1377. error_logger:tty(true)
  1378. end,
  1379. [
  1380. {"Nothing dropped when error_logger high watermark is undefined",
  1381. fun () ->
  1382. ok = error_logger_lager_h:set_high_water(undefined),
  1383. timer:sleep(100),
  1384. produce_n_error_logger_msgs(10),
  1385. timer:sleep(500),
  1386. ?assert(count() >= 10)
  1387. end
  1388. },
  1389. {"Mostly dropped according to error_logger high watermark",
  1390. fun () ->
  1391. ok = error_logger_lager_h:set_high_water(5),
  1392. timer:sleep(100),
  1393. produce_n_error_logger_msgs(50),
  1394. timer:sleep(1000),
  1395. ?assert(count() < 20)
  1396. end
  1397. },
  1398. {"Non-notifications are not dropped",
  1399. fun () ->
  1400. ok = error_logger_lager_h:set_high_water(2),
  1401. timer:sleep(100),
  1402. spawn(fun () -> produce_n_error_logger_msgs(300) end),
  1403. timer:sleep(50),
  1404. %% if everything were dropped, this call would be dropped
  1405. %% too, so lets hope it's not
  1406. ?assert(is_integer(count())),
  1407. timer:sleep(1000),
  1408. ?assert(count() < 10)
  1409. end
  1410. }
  1411. ]
  1412. }.
  1413. -endif.