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

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