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

1284 lines
68 KiB

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