選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

1719 行
89 KiB

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