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

922 行
48 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:timestamp(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. lager:debug("this message should be logged"),
  292. ?assertEqual(1, count()),
  293. ?assertEqual(1, count_ignored()),
  294. ?assertEqual(debug, lager:get_loglevel(?MODULE)),
  295. ok
  296. end
  297. },
  298. {"tracing works",
  299. fun() ->
  300. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(error)), []}),
  301. ok = lager:info("hello world"),
  302. ?assertEqual(0, count()),
  303. lager:trace(?MODULE, [{module, ?MODULE}], debug),
  304. %% elegible for tracing
  305. ok = lager:info("hello world"),
  306. %% NOT elegible for tracing
  307. ok = lager:log(info, [{pid, self()}], "hello world"),
  308. ?assertEqual(1, count()),
  309. ok
  310. end
  311. },
  312. {"tracing works with custom attributes",
  313. fun() ->
  314. lager:set_loglevel(?MODULE, error),
  315. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(error)), []}),
  316. lager:info([{requestid, 6}], "hello world"),
  317. ?assertEqual(0, count()),
  318. lager:trace(?MODULE, [{requestid, 6}, {foo, bar}], debug),
  319. lager:info([{requestid, 6}, {foo, bar}], "hello world"),
  320. ?assertEqual(1, count()),
  321. lager:trace(?MODULE, [{requestid, '*'}], debug),
  322. lager:info([{requestid, 6}], "hello world"),
  323. ?assertEqual(2, count()),
  324. lager:clear_all_traces(),
  325. lager:info([{requestid, 6}], "hello world"),
  326. ?assertEqual(2, count()),
  327. ok
  328. end
  329. },
  330. {"tracing honors loglevel",
  331. fun() ->
  332. lager:set_loglevel(?MODULE, error),
  333. {ok, T} = lager:trace(?MODULE, [{module, ?MODULE}], notice),
  334. ok = lager:info("hello world"),
  335. ?assertEqual(0, count()),
  336. ok = lager:notice("hello world"),
  337. ?assertEqual(1, count()),
  338. lager:stop_trace(T),
  339. ok = lager:notice("hello world"),
  340. ?assertEqual(1, count()),
  341. ok
  342. end
  343. },
  344. {"record printing works",
  345. fun() ->
  346. print_state(),
  347. {Level, _Time, Message, _Metadata} = pop(),
  348. ?assertMatch(Level, lager_util:level_to_num(info)),
  349. {mask, Mask} = lager_util:config_to_mask(info),
  350. ?assertEqual("State #state{level={mask,"++integer_to_list(Mask)++"},buffer=[],ignored=[]}", lists:flatten(Message)),
  351. ok
  352. end
  353. },
  354. {"record printing fails gracefully",
  355. fun() ->
  356. print_bad_state(),
  357. {Level, _Time, Message, _Metadata} = pop(),
  358. ?assertMatch(Level, lager_util:level_to_num(info)),
  359. ?assertEqual("State {state,1}", lists:flatten(Message)),
  360. ok
  361. end
  362. },
  363. {"record printing fails gracefully when no lager_record attribute",
  364. fun() ->
  365. spawn(fun() -> lager:info("State ~p", [lager:pr({state, 1}, lager)]) end),
  366. timer:sleep(100),
  367. {Level, _Time, Message, _Metadata} = pop(),
  368. ?assertMatch(Level, lager_util:level_to_num(info)),
  369. ?assertEqual("State {state,1}", lists:flatten(Message)),
  370. ok
  371. end
  372. },
  373. {"record printing fails gracefully when input is not a tuple",
  374. fun() ->
  375. spawn(fun() -> lager:info("State ~p", [lager:pr(ok, lager)]) end),
  376. timer:sleep(100),
  377. {Level, _Time, Message, _Metadata} = pop(),
  378. ?assertMatch(Level, lager_util:level_to_num(info)),
  379. ?assertEqual("State ok", lists:flatten(Message)),
  380. ok
  381. end
  382. },
  383. {"record printing fails gracefully when module is invalid",
  384. fun() ->
  385. spawn(fun() -> lager:info("State ~p", [lager:pr({state, 1}, not_a_module)]) end),
  386. timer:sleep(100),
  387. {Level, _Time, Message, _Metadata} = pop(),
  388. ?assertMatch(Level, lager_util:level_to_num(info)),
  389. ?assertEqual("State {state,1}", lists:flatten(Message)),
  390. ok
  391. end
  392. }
  393. ]
  394. }.
  395. setup() ->
  396. error_logger:tty(false),
  397. application:load(lager),
  398. application:set_env(lager, handlers, [{?MODULE, info}]),
  399. application:set_env(lager, error_logger_redirect, false),
  400. application:start(lager),
  401. gen_event:call(lager_event, ?MODULE, flush).
  402. cleanup(_) ->
  403. application:stop(lager),
  404. error_logger:tty(true).
  405. crash(Type) ->
  406. spawn(fun() -> gen_server:call(crash, Type) end),
  407. timer:sleep(100),
  408. _ = gen_event:which_handlers(error_logger),
  409. ok.
  410. test_body(Expected, Actual) ->
  411. case has_line_numbers() of
  412. true ->
  413. FileLine = string:substr(Actual, length(Expected)+1),
  414. Body = string:substr(Actual, 1, length(Expected)),
  415. ?assertEqual(Expected, Body),
  416. case string:substr(FileLine, 1, 6) of
  417. [] ->
  418. %% sometimes there's no line information...
  419. ?assert(true);
  420. " line " ->
  421. ?assert(true);
  422. Other ->
  423. ?debugFmt("unexpected trailing data ~p", [Other]),
  424. ?assert(false)
  425. end;
  426. false ->
  427. ?assertEqual(Expected, Actual)
  428. end.
  429. error_logger_redirect_crash_test_() ->
  430. TestBody=fun(Name,CrashReason,Expected) -> {Name,
  431. fun() ->
  432. Pid = whereis(crash),
  433. crash(CrashReason),
  434. {Level, _, Msg,Metadata} = pop(),
  435. test_body(Expected, lists:flatten(Msg)),
  436. ?assertEqual(Pid,proplists:get_value(pid,Metadata)),
  437. ?assertEqual(lager_util:level_to_num(error),Level)
  438. end
  439. }
  440. end,
  441. {foreach,
  442. fun() ->
  443. error_logger:tty(false),
  444. application:load(lager),
  445. application:set_env(lager, error_logger_redirect, true),
  446. application:set_env(lager, handlers, [{?MODULE, error}]),
  447. application:start(lager),
  448. crash:start()
  449. end,
  450. fun(_) ->
  451. application:stop(lager),
  452. case whereis(crash) of
  453. undefined -> ok;
  454. Pid -> exit(Pid, kill)
  455. end,
  456. error_logger:tty(true)
  457. end,
  458. [
  459. {"again, there is nothing up my sleeve",
  460. fun() ->
  461. ?assertEqual(undefined, pop()),
  462. ?assertEqual(0, count())
  463. end
  464. },
  465. TestBody("bad return value",bad_return,"gen_server crash terminated with reason: bad return value: bleh"),
  466. TestBody("bad return value with string",bad_return_string,"gen_server crash terminated with reason: bad return value: {tuple,{tuple,\"string\"}}"),
  467. TestBody("bad return uncaught throw",throw,"gen_server crash terminated with reason: bad return value: a_ball"),
  468. TestBody("case clause",case_clause,"gen_server crash terminated with reason: no case clause matching {} in crash:handle_call/3"),
  469. TestBody("case clause string",case_clause_string,"gen_server crash terminated with reason: no case clause matching \"crash\" in crash:handle_call/3"),
  470. TestBody("function clause",function_clause,"gen_server crash terminated with reason: no function clause matching crash:function({})"),
  471. TestBody("if clause",if_clause,"gen_server crash terminated with reason: no true branch found while evaluating if expression in crash:handle_call/3"),
  472. TestBody("try clause",try_clause,"gen_server crash terminated with reason: no try clause matching [] in crash:handle_call/3"),
  473. TestBody("undefined function",undef,"gen_server crash terminated with reason: call to undefined function crash:booger/0 from crash:handle_call/3"),
  474. TestBody("bad math",badarith,"gen_server crash terminated with reason: bad arithmetic expression in crash:handle_call/3"),
  475. TestBody("bad match",badmatch,"gen_server crash terminated with reason: no match of right hand value {} in crash:handle_call/3"),
  476. 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"),
  477. TestBody("bad arg1",badarg1,"gen_server crash terminated with reason: bad argument in crash:handle_call/3"),
  478. 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"),
  479. TestBody("bad record",badrecord,"gen_server crash terminated with reason: bad record state in crash:handle_call/3"),
  480. TestBody("noproc",noproc,"gen_server crash terminated with reason: no such process or port in call to gen_event:call(foo, bar, baz)"),
  481. TestBody("badfun",badfun,"gen_server crash terminated with reason: bad function booger in crash:handle_call/3")
  482. ]
  483. }.
  484. error_logger_redirect_test_() ->
  485. {foreach,
  486. fun() ->
  487. error_logger:tty(false),
  488. application:load(lager),
  489. application:set_env(lager, error_logger_redirect, true),
  490. application:set_env(lager, handlers, [{?MODULE, info}]),
  491. application:start(lager),
  492. lager:log(error, self(), "flush flush"),
  493. timer:sleep(100),
  494. gen_event:call(lager_event, ?MODULE, flush)
  495. end,
  496. fun(_) ->
  497. application:stop(lager),
  498. error_logger:tty(true)
  499. end,
  500. [
  501. {"error reports are printed",
  502. fun() ->
  503. sync_error_logger:error_report([{this, is}, a, {silly, format}]),
  504. _ = gen_event:which_handlers(error_logger),
  505. {Level, _, Msg,Metadata} = pop(),
  506. ?assertEqual(lager_util:level_to_num(error),Level),
  507. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  508. Expected = "this: is, a, silly: format",
  509. ?assertEqual(Expected, lists:flatten(Msg))
  510. end
  511. },
  512. {"string error reports are printed",
  513. fun() ->
  514. sync_error_logger:error_report("this is less silly"),
  515. _ = gen_event:which_handlers(error_logger),
  516. {Level, _, Msg,Metadata} = pop(),
  517. ?assertEqual(lager_util:level_to_num(error),Level),
  518. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  519. Expected = "this is less silly",
  520. ?assertEqual(Expected, lists:flatten(Msg))
  521. end
  522. },
  523. {"error messages are printed",
  524. fun() ->
  525. sync_error_logger:error_msg("doom, doom has come upon you all"),
  526. _ = gen_event:which_handlers(error_logger),
  527. {Level, _, Msg,Metadata} = pop(),
  528. ?assertEqual(lager_util:level_to_num(error),Level),
  529. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  530. Expected = "doom, doom has come upon you all",
  531. ?assertEqual(Expected, lists:flatten(Msg))
  532. end
  533. },
  534. {"error messages are truncated at 4096 characters",
  535. fun() ->
  536. sync_error_logger:error_msg("doom, doom has come upon you all ~p", [string:copies("doom", 10000)]),
  537. _ = gen_event:which_handlers(error_logger),
  538. {_, _, Msg,_Metadata} = pop(),
  539. ?assert(length(lists:flatten(Msg)) < 5100)
  540. end
  541. },
  542. {"info reports are printed",
  543. fun() ->
  544. sync_error_logger:info_report([{this, is}, a, {silly, format}]),
  545. _ = gen_event:which_handlers(error_logger),
  546. {Level, _, Msg,Metadata} = pop(),
  547. ?assertEqual(lager_util:level_to_num(info),Level),
  548. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  549. Expected = "this: is, a, silly: format",
  550. ?assertEqual(Expected, lists:flatten(Msg))
  551. end
  552. },
  553. {"info reports are truncated at 4096 characters",
  554. fun() ->
  555. sync_error_logger:info_report([[{this, is}, a, {silly, format}] || _ <- lists:seq(0, 600)]),
  556. _ = gen_event:which_handlers(error_logger),
  557. {Level, _, Msg,Metadata} = pop(),
  558. ?assertEqual(lager_util:level_to_num(info),Level),
  559. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  560. ?assert(length(lists:flatten(Msg)) < 5000)
  561. end
  562. },
  563. {"single term info reports are printed",
  564. fun() ->
  565. sync_error_logger:info_report({foolish, bees}),
  566. _ = gen_event:which_handlers(error_logger),
  567. {Level, _, Msg,Metadata} = pop(),
  568. ?assertEqual(lager_util:level_to_num(info),Level),
  569. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  570. ?assertEqual("{foolish,bees}", lists:flatten(Msg))
  571. end
  572. },
  573. {"single term error reports are printed",
  574. fun() ->
  575. sync_error_logger:error_report({foolish, bees}),
  576. _ = gen_event:which_handlers(error_logger),
  577. {Level, _, Msg,Metadata} = pop(),
  578. ?assertEqual(lager_util:level_to_num(error),Level),
  579. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  580. ?assertEqual("{foolish,bees}", lists:flatten(Msg))
  581. end
  582. },
  583. {"string info reports are printed",
  584. fun() ->
  585. sync_error_logger:info_report("this is less silly"),
  586. _ = gen_event:which_handlers(error_logger),
  587. {Level, _, Msg,Metadata} = pop(),
  588. ?assertEqual(lager_util:level_to_num(info),Level),
  589. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  590. ?assertEqual("this is less silly", lists:flatten(Msg))
  591. end
  592. },
  593. {"string info reports are truncated at 4096 characters",
  594. fun() ->
  595. sync_error_logger:info_report(string:copies("this is less silly", 1000)),
  596. _ = gen_event:which_handlers(error_logger),
  597. {Level, _, Msg,Metadata} = pop(),
  598. ?assertEqual(lager_util:level_to_num(info),Level),
  599. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  600. ?assert(length(lists:flatten(Msg)) < 5100)
  601. end
  602. },
  603. {"strings in a mixed report are printed as strings",
  604. fun() ->
  605. sync_error_logger:info_report(["this is less silly", {than, "this"}]),
  606. _ = gen_event:which_handlers(error_logger),
  607. {Level, _, Msg,Metadata} = pop(),
  608. ?assertEqual(lager_util:level_to_num(info),Level),
  609. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  610. ?assertEqual("\"this is less silly\", than: \"this\"", lists:flatten(Msg))
  611. end
  612. },
  613. {"info messages are printed",
  614. fun() ->
  615. sync_error_logger:info_msg("doom, doom has come upon you all"),
  616. _ = gen_event:which_handlers(error_logger),
  617. {Level, _, Msg,Metadata} = pop(),
  618. ?assertEqual(lager_util:level_to_num(info),Level),
  619. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  620. ?assertEqual("doom, doom has come upon you all", lists:flatten(Msg))
  621. end
  622. },
  623. {"info messages are truncated at 4096 characters",
  624. fun() ->
  625. sync_error_logger:info_msg("doom, doom has come upon you all ~p", [string:copies("doom", 10000)]),
  626. _ = gen_event:which_handlers(error_logger),
  627. {Level, _, Msg,Metadata} = pop(),
  628. ?assertEqual(lager_util:level_to_num(info),Level),
  629. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  630. ?assert(length(lists:flatten(Msg)) < 5100)
  631. end
  632. },
  633. {"warning messages are printed at the correct level",
  634. fun() ->
  635. sync_error_logger:warning_msg("doom, doom has come upon you all"),
  636. Map = error_logger:warning_map(),
  637. _ = gen_event:which_handlers(error_logger),
  638. {Level, _, Msg,Metadata} = pop(),
  639. ?assertEqual(lager_util:level_to_num(Map),Level),
  640. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  641. ?assertEqual("doom, doom has come upon you all", lists:flatten(Msg))
  642. end
  643. },
  644. {"warning reports are printed at the correct level",
  645. fun() ->
  646. sync_error_logger:warning_report([{i, like}, pie]),
  647. Map = error_logger:warning_map(),
  648. _ = gen_event:which_handlers(error_logger),
  649. {Level, _, Msg,Metadata} = pop(),
  650. ?assertEqual(lager_util:level_to_num(Map),Level),
  651. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  652. ?assertEqual("i: like, pie", lists:flatten(Msg))
  653. end
  654. },
  655. {"single term warning reports are printed at the correct level",
  656. fun() ->
  657. sync_error_logger:warning_report({foolish, bees}),
  658. Map = error_logger:warning_map(),
  659. _ = gen_event:which_handlers(error_logger),
  660. {Level, _, Msg,Metadata} = pop(),
  661. ?assertEqual(lager_util:level_to_num(Map),Level),
  662. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  663. ?assertEqual("{foolish,bees}", lists:flatten(Msg))
  664. end
  665. },
  666. {"application stop reports",
  667. fun() ->
  668. sync_error_logger:info_report([{application, foo}, {exited, quittin_time}, {type, lazy}]),
  669. _ = gen_event:which_handlers(error_logger),
  670. {Level, _, Msg,Metadata} = pop(),
  671. ?assertEqual(lager_util:level_to_num(info),Level),
  672. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  673. ?assertEqual("Application foo exited with reason: quittin_time", lists:flatten(Msg))
  674. end
  675. },
  676. {"supervisor reports",
  677. fun() ->
  678. sync_error_logger:error_report(supervisor_report, [{errorContext, france}, {offender, [{name, mini_steve}, {mfargs, {a, b, [c]}}, {pid, bleh}]}, {reason, fired}, {supervisor, {local, steve}}]),
  679. _ = gen_event:which_handlers(error_logger),
  680. {Level, _, Msg,Metadata} = pop(),
  681. ?assertEqual(lager_util:level_to_num(error),Level),
  682. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  683. ?assertEqual("Supervisor steve had child mini_steve started with a:b(c) at bleh exit with reason fired in context france", lists:flatten(Msg))
  684. end
  685. },
  686. {"supervisor reports with real error",
  687. fun() ->
  688. 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}}]),
  689. _ = gen_event:which_handlers(error_logger),
  690. {Level, _, Msg,Metadata} = pop(),
  691. ?assertEqual(lager_util:level_to_num(error),Level),
  692. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  693. ?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))
  694. end
  695. },
  696. {"supervisor_bridge reports",
  697. fun() ->
  698. sync_error_logger:error_report(supervisor_report, [{errorContext, france}, {offender, [{mod, mini_steve}, {pid, bleh}]}, {reason, fired}, {supervisor, {local, steve}}]),
  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. ?assertEqual("Supervisor steve had child at module mini_steve at bleh exit with reason fired in context france", lists:flatten(Msg))
  704. end
  705. },
  706. {"application progress report",
  707. fun() ->
  708. sync_error_logger:info_report(progress, [{application, foo}, {started_at, node()}]),
  709. _ = gen_event:which_handlers(error_logger),
  710. {Level, _, Msg,Metadata} = pop(),
  711. ?assertEqual(lager_util:level_to_num(info),Level),
  712. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  713. Expected = lists:flatten(io_lib:format("Application foo started on node ~w", [node()])),
  714. ?assertEqual(Expected, lists:flatten(Msg))
  715. end
  716. },
  717. {"supervisor progress report",
  718. fun() ->
  719. lager:set_loglevel(?MODULE, debug),
  720. sync_error_logger:info_report(progress, [{supervisor, {local, foo}}, {started, [{mfargs, {foo, bar, 1}}, {pid, baz}]}]),
  721. _ = gen_event:which_handlers(error_logger),
  722. {Level, _, Msg,Metadata} = pop(),
  723. ?assertEqual(lager_util:level_to_num(debug),Level),
  724. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  725. ?assertEqual("Supervisor foo started foo:bar/1 at pid baz", lists:flatten(Msg))
  726. end
  727. },
  728. {"crash report for emfile",
  729. fun() ->
  730. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, emfile, [{stack, trace, 1}]}}], []]),
  731. _ = gen_event:which_handlers(error_logger),
  732. {Level, _, Msg,Metadata} = pop(),
  733. ?assertEqual(lager_util:level_to_num(error),Level),
  734. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  735. 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()])),
  736. ?assertEqual(Expected, lists:flatten(Msg))
  737. end
  738. },
  739. {"crash report for system process limit",
  740. fun() ->
  741. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, system_limit, [{erlang, spawn, 1}]}}], []]),
  742. _ = gen_event:which_handlers(error_logger),
  743. {Level, _, Msg,Metadata} = pop(),
  744. ?assertEqual(lager_util:level_to_num(error),Level),
  745. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  746. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: system limit: maximum number of processes exceeded", [self()])),
  747. ?assertEqual(Expected, lists:flatten(Msg))
  748. end
  749. },
  750. {"crash report for system process limit2",
  751. fun() ->
  752. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, system_limit, [{erlang, spawn_opt, 1}]}}], []]),
  753. _ = gen_event:which_handlers(error_logger),
  754. {Level, _, Msg,Metadata} = pop(),
  755. ?assertEqual(lager_util:level_to_num(error),Level),
  756. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  757. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: system limit: maximum number of processes exceeded", [self()])),
  758. ?assertEqual(Expected, lists:flatten(Msg))
  759. end
  760. },
  761. {"crash report for system port limit",
  762. fun() ->
  763. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, system_limit, [{erlang, open_port, 1}]}}], []]),
  764. _ = gen_event:which_handlers(error_logger),
  765. {Level, _, Msg,Metadata} = pop(),
  766. ?assertEqual(lager_util:level_to_num(error),Level),
  767. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  768. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~w with 0 neighbours crashed with reason: system limit: maximum number of ports exceeded", [self()])),
  769. ?assertEqual(Expected, lists:flatten(Msg))
  770. end
  771. },
  772. {"crash report for system port limit",
  773. fun() ->
  774. sync_error_logger:error_report(crash_report, [[{pid, self()}, {registered_name, []}, {error_info, {error, system_limit, [{erlang, list_to_atom, 1}]}}], []]),
  775. _ = gen_event:which_handlers(error_logger),
  776. {Level, _, Msg,Metadata} = pop(),
  777. ?assertEqual(lager_util:level_to_num(error),Level),
  778. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  779. 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()])),
  780. ?assertEqual(Expected, lists:flatten(Msg))
  781. end
  782. },
  783. {"crash report for system ets table limit",
  784. fun() ->
  785. 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}]}}], []]),
  786. _ = gen_event:which_handlers(error_logger),
  787. {Level, _, Msg,Metadata} = pop(),
  788. ?assertEqual(lager_util:level_to_num(error),Level),
  789. ?assertEqual(self(),proplists:get_value(pid,Metadata)),
  790. 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])),
  791. ?assertEqual(Expected, lists:flatten(Msg))
  792. end
  793. },
  794. {"crash report for unknown system limit should be truncated at 500 characters",
  795. fun() ->
  796. sync_error_logger:error_report(crash_report, [[{pid, self()}, {error_info, {error, system_limit, [{wtf,boom,[string:copies("aaaa", 4096)]}]}}], []]),
  797. _ = gen_event:which_handlers(error_logger),
  798. {_, _, Msg,_Metadata} = pop(),
  799. ?assert(length(lists:flatten(Msg)) > 550),
  800. ?assert(length(lists:flatten(Msg)) < 600)
  801. end
  802. },
  803. {"crash reports for 'special processes' should be handled right - function_clause",
  804. fun() ->
  805. {ok, Pid} = special_process:start(),
  806. unlink(Pid),
  807. Pid ! function_clause,
  808. timer:sleep(500),
  809. _ = gen_event:which_handlers(error_logger),
  810. {_, _, Msg, _Metadata} = pop(),
  811. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~p with 0 neighbours crashed with reason: no function clause matching special_process:foo(bar)",
  812. [Pid])),
  813. test_body(Expected, lists:flatten(Msg))
  814. end
  815. },
  816. {"crash reports for 'special processes' should be handled right - case_clause",
  817. fun() ->
  818. {ok, Pid} = special_process:start(),
  819. unlink(Pid),
  820. Pid ! {case_clause, wtf},
  821. timer:sleep(500),
  822. _ = gen_event:which_handlers(error_logger),
  823. {_, _, Msg, _Metadata} = pop(),
  824. 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",
  825. [Pid])),
  826. test_body(Expected, lists:flatten(Msg))
  827. end
  828. },
  829. {"crash reports for 'special processes' should be handled right - exit",
  830. fun() ->
  831. {ok, Pid} = special_process:start(),
  832. unlink(Pid),
  833. Pid ! exit,
  834. timer:sleep(500),
  835. _ = gen_event:which_handlers(error_logger),
  836. {_, _, Msg, _Metadata} = pop(),
  837. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~p with 0 neighbours exited with reason: byebye in special_process:loop/0",
  838. [Pid])),
  839. test_body(Expected, lists:flatten(Msg))
  840. end
  841. },
  842. {"crash reports for 'special processes' should be handled right - error",
  843. fun() ->
  844. {ok, Pid} = special_process:start(),
  845. unlink(Pid),
  846. Pid ! error,
  847. timer:sleep(500),
  848. _ = gen_event:which_handlers(error_logger),
  849. {_, _, Msg, _Metadata} = pop(),
  850. Expected = lists:flatten(io_lib:format("CRASH REPORT Process ~p with 0 neighbours crashed with reason: mybad in special_process:loop/0",
  851. [Pid])),
  852. test_body(Expected, lists:flatten(Msg))
  853. end
  854. },
  855. {"messages should not be generated if they don't satisfy the threshold",
  856. fun() ->
  857. lager:set_loglevel(?MODULE, error),
  858. sync_error_logger:info_report([hello, world]),
  859. _ = gen_event:which_handlers(error_logger),
  860. ?assertEqual(0, count()),
  861. ?assertEqual(0, count_ignored()),
  862. lager:set_loglevel(?MODULE, info),
  863. sync_error_logger:info_report([hello, world]),
  864. _ = gen_event:which_handlers(error_logger),
  865. ?assertEqual(1, count()),
  866. ?assertEqual(0, count_ignored()),
  867. lager:set_loglevel(?MODULE, error),
  868. lager_config:set(loglevel, {element(2, lager_util:config_to_mask(debug)), []}),
  869. sync_error_logger:info_report([hello, world]),
  870. _ = gen_event:which_handlers(error_logger),
  871. ?assertEqual(1, count()),
  872. ?assertEqual(1, count_ignored())
  873. end
  874. }
  875. ]
  876. }.
  877. safe_format_test() ->
  878. ?assertEqual("foo bar", lists:flatten(lager:safe_format("~p ~p", [foo, bar], 1024))),
  879. ?assertEqual("FORMAT ERROR: \"~p ~p ~p\" [foo,bar]", lists:flatten(lager:safe_format("~p ~p ~p", [foo, bar], 1024))),
  880. ok.
  881. -endif.