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

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