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

1265 行
68 KiB

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