您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

1684 行
88 KiB

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