logger.hrl 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. %%--------------------------------------------------------------------
  2. %% Copyright (c) 2018-2024 EMQ Technologies Co., Ltd. All Rights Reserved.
  3. %%
  4. %% Licensed under the Apache License, Version 2.0 (the "License");
  5. %% you may not use this file except in compliance with the License.
  6. %% You may obtain 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, software
  11. %% distributed under the License is distributed on an "AS IS" BASIS,
  12. %% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. %% See the License for the specific language governing permissions and
  14. %% limitations under the License.
  15. %%--------------------------------------------------------------------
  16. -ifndef(EMQX_LOGGER_HRL).
  17. -define(EMQX_LOGGER_HRL, true).
  18. %% structured logging
  19. -define(SLOG(Level, Data),
  20. ?SLOG(Level, Data, #{})
  21. ).
  22. %% structured logging, meta is for handler's filter.
  23. -define(SLOG(Level, Data, Meta),
  24. %% check 'allow' here, only evaluate Data and Meta when necessary
  25. case logger:allow(Level, ?MODULE) of
  26. true ->
  27. logger:log(
  28. Level,
  29. (Data),
  30. maps:merge(Meta, #{
  31. mfa => {?MODULE, ?FUNCTION_NAME, ?FUNCTION_ARITY},
  32. line => ?LINE
  33. })
  34. );
  35. false ->
  36. ok
  37. end
  38. ).
  39. %% NOTE: do not forget to use atom for msg and add every used msg to
  40. %% the default value of `log.throttling.msgs` list.
  41. -define(SLOG_THROTTLE(Level, Data),
  42. ?SLOG_THROTTLE(Level, Data, #{})
  43. ).
  44. -define(SLOG_THROTTLE(Level, Data, Meta),
  45. ?SLOG_THROTTLE(Level, undefined, Data, Meta)
  46. ).
  47. -define(SLOG_THROTTLE(Level, UniqueKey, Data, Meta),
  48. case logger:allow(Level, ?MODULE) of
  49. true ->
  50. (fun(#{msg := __Msg} = __Data) ->
  51. case emqx_log_throttler:allow(__Msg, UniqueKey) of
  52. true ->
  53. logger:log(Level, __Data, Meta);
  54. false ->
  55. ?_DO_TRACE(Level, __Msg, maps:merge(__Data, Meta))
  56. end
  57. end)(
  58. Data
  59. );
  60. false ->
  61. ok
  62. end
  63. ).
  64. -define(AUDIT_HANDLER, emqx_audit).
  65. -define(TRACE_FILTER, emqx_trace_filter).
  66. -define(OWN_KEYS, [level, filters, filter_default, handlers]).
  67. %% Internal macro
  68. -define(_DO_TRACE(Tag, Msg, Meta),
  69. case persistent_term:get(?TRACE_FILTER, []) of
  70. [] -> ok;
  71. %% We can't bind filter list to a variable because we pollute the calling scope with it.
  72. %% We also don't want to wrap the macro body in a fun
  73. %% because this adds overhead to the happy path.
  74. %% So evaluate `persistent_term:get` twice.
  75. _ -> emqx_trace:log(persistent_term:get(?TRACE_FILTER, []), Msg, (Meta)#{trace_tag => Tag})
  76. end
  77. ).
  78. -define(TRACE(Tag, Msg, Meta), ?TRACE(debug, Tag, Msg, Meta)).
  79. %% Only evaluate when necessary
  80. -define(TRACE(Level, Tag, Msg, Meta), begin
  81. ?_DO_TRACE(Tag, Msg, Meta),
  82. ?SLOG(
  83. Level,
  84. (Meta)#{msg => Msg, tag => Tag},
  85. #{is_trace => false}
  86. )
  87. end).
  88. -ifdef(EMQX_RELEASE_EDITION).
  89. -if(?EMQX_RELEASE_EDITION == ee).
  90. -define(AUDIT(_LevelFun_, _MetaFun_), begin
  91. case logger_config:get(logger, ?AUDIT_HANDLER) of
  92. {error, {not_found, _}} ->
  93. ok;
  94. {ok, Handler = #{level := _AllowLevel_}} ->
  95. _Level_ = _LevelFun_,
  96. case logger:compare_levels(_AllowLevel_, _Level_) of
  97. _R_ when _R_ == lt; _R_ == eq ->
  98. emqx_audit:log(_Level_, _MetaFun_, Handler);
  99. _ ->
  100. ok
  101. end
  102. end
  103. end).
  104. -else.
  105. %% Only for compile pass, ce edition will not call it
  106. -define(AUDIT(_L_, _M_), _ = {_L_, _M_}).
  107. -endif.
  108. -else.
  109. %% Only for compile pass, ce edition will not call it
  110. -define(AUDIT(_L_, _M_), _ = {_L_, _M_}).
  111. -endif.
  112. %% print to 'user' group leader
  113. -define(ULOG(Fmt, Args), io:format(user, Fmt, Args)).
  114. -define(ELOG(Fmt, Args), io:format(standard_error, Fmt, Args)).
  115. -endif.