25개 이상의 토픽을 선택하실 수 없습니다. Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1287 lines
69 KiB

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