Merge pull request #8217 from terry-xiaoyu/fix_delayed_module_disbled_after_emqx_stop

fix: the delayed module is disbled after ./bin/emqx stop
This commit is contained in:
Xinyu Liu 2022-06-15 16:42:29 +08:00 committed by GitHub
commit 17b2665a6d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 53 additions and 65 deletions

View File

@ -45,20 +45,24 @@
%% gen_server callbacks %% gen_server callbacks
-export([ -export([
enable/0, load/0,
disable/0, unload/0,
set_max_delayed_messages/1, load_or_unload/1,
get_conf/1,
update_config/1, update_config/1,
list/1, list/1,
get_delayed_message/1, get_delayed_message/1,
get_delayed_message/2, get_delayed_message/2,
delete_delayed_message/1, delete_delayed_message/1,
delete_delayed_message/2, delete_delayed_message/2,
post_config_update/5,
cluster_list/1, cluster_list/1,
cluster_query/4 cluster_query/4
]). ]).
-export([
post_config_update/5
]).
-export([format_delayed/1]). -export([format_delayed/1]).
%% exported for `emqx_telemetry' %% exported for `emqx_telemetry'
@ -75,8 +79,7 @@
publish_timer := maybe(timer:tref()), publish_timer := maybe(timer:tref()),
publish_at := non_neg_integer(), publish_at := non_neg_integer(),
stats_timer := maybe(reference()), stats_timer := maybe(reference()),
stats_fun := maybe(fun((pos_integer()) -> ok)), stats_fun := maybe(fun((pos_integer()) -> ok))
max_delayed_messages := non_neg_integer()
}. }.
%% sync ms with record change %% sync ms with record change
@ -138,21 +141,23 @@ on_message_publish(Msg) ->
-spec start_link() -> emqx_types:startlink_ret(). -spec start_link() -> emqx_types:startlink_ret().
start_link() -> start_link() ->
Opts = emqx_conf:get([delayed], #{}), gen_server:start_link({local, ?SERVER}, ?MODULE, [], []).
gen_server:start_link({local, ?SERVER}, ?MODULE, [Opts], []).
-spec store(delayed_message()) -> ok | {error, atom()}. -spec store(delayed_message()) -> ok | {error, atom()}.
store(DelayedMsg) -> store(DelayedMsg) ->
gen_server:call(?SERVER, {store, DelayedMsg}, infinity). gen_server:call(?SERVER, {store, DelayedMsg}, infinity).
enable() -> get_conf(Key) ->
enable(true). emqx_conf:get([delayed, Key]).
disable() -> load() ->
enable(false). load_or_unload(true).
set_max_delayed_messages(Max) -> unload() ->
gen_server:call(?SERVER, {set_max_delayed_messages, Max}). load_or_unload(false).
load_or_unload(Bool) ->
gen_server:call(?SERVER, {do_load_or_unload, Bool}).
list(Params) -> list(Params) ->
emqx_mgmt_api:paginate(?TAB, Params, ?FORMAT_FUN). emqx_mgmt_api:paginate(?TAB, Params, ?FORMAT_FUN).
@ -240,54 +245,46 @@ delete_delayed_message(Node, Id) ->
update_config(Config) -> update_config(Config) ->
emqx_conf:update([delayed], Config, #{rawconf_with_defaults => true, override_to => cluster}). emqx_conf:update([delayed], Config, #{rawconf_with_defaults => true, override_to => cluster}).
post_config_update(_KeyPath, Config, _NewConf, _OldConf, _AppEnvs) -> post_config_update(_KeyPath, _ConfigReq, NewConf, _OldConf, _AppEnvs) ->
gen_server:call(?SERVER, {update_config, Config}). Enable = maps:get(enable, NewConf, undefined),
load_or_unload(Enable).
%%-------------------------------------------------------------------- %%--------------------------------------------------------------------
%% gen_server callback %% gen_server callback
%%-------------------------------------------------------------------- %%--------------------------------------------------------------------
init([Opts]) -> init([]) ->
ok = mria:wait_for_tables([?TAB]),
erlang:process_flag(trap_exit, true), erlang:process_flag(trap_exit, true),
emqx_conf:add_handler([delayed], ?MODULE), emqx_conf:add_handler([delayed], ?MODULE),
MaxDelayedMessages = maps:get(max_delayed_messages, Opts, 0),
State = State =
ensure_stats_event( ensure_stats_event(
ensure_publish_timer(#{ ensure_publish_timer(#{
publish_timer => undefined, publish_timer => undefined,
publish_at => 0, publish_at => 0,
stats_timer => undefined, stats_timer => undefined,
stats_fun => undefined, stats_fun => undefined
max_delayed_messages => MaxDelayedMessages
}) })
), ),
{ok, ensure_enable(emqx:get_config([delayed, enable]), State)}. {ok, do_load_or_unload(emqx:get_config([delayed, enable]), State)}.
handle_call({set_max_delayed_messages, Max}, _From, State) -> handle_call({store, DelayedMsg = #delayed_message{key = Key}}, _From, State) ->
{reply, ok, State#{max_delayed_messages => Max}}; Size = mnesia:table_info(?TAB, size),
handle_call( case get_conf(max_delayed_messages) of
{store, DelayedMsg = #delayed_message{key = Key}}, _From, State = #{max_delayed_messages := 0} 0 ->
) ->
ok = mria:dirty_write(?TAB, DelayedMsg), ok = mria:dirty_write(?TAB, DelayedMsg),
emqx_metrics:inc('messages.delayed'), emqx_metrics:inc('messages.delayed'),
{reply, ok, ensure_publish_timer(Key, State)}; {reply, ok, ensure_publish_timer(Key, State)};
handle_call( Max when Size >= Max ->
{store, DelayedMsg = #delayed_message{key = Key}}, _From, State = #{max_delayed_messages := Max}
) ->
Size = mnesia:table_info(?TAB, size),
case Size >= Max of
true ->
{reply, {error, max_delayed_messages_full}, State}; {reply, {error, max_delayed_messages_full}, State};
false -> Max when Size < Max ->
ok = mria:dirty_write(?TAB, DelayedMsg), ok = mria:dirty_write(?TAB, DelayedMsg),
emqx_metrics:inc('messages.delayed'), emqx_metrics:inc('messages.delayed'),
{reply, ok, ensure_publish_timer(Key, State)} {reply, ok, ensure_publish_timer(Key, State)}
end; end;
handle_call({update_config, Config}, _From, #{max_delayed_messages := Max} = State) -> handle_call({do_load_or_unload, Bool}, _From, State0) ->
Max2 = maps:get(<<"max_delayed_messages">>, Config, Max), State = do_load_or_unload(Bool, State0),
State2 = State#{max_delayed_messages := Max2}, {reply, ok, State};
State3 = ensure_enable(maps:get(<<"enable">>, Config, undefined), State2),
{reply, ok, State3};
handle_call(Req, _From, State) -> handle_call(Req, _From, State) ->
?tp(error, emqx_delayed_unexpected_call, #{call => Req}), ?tp(error, emqx_delayed_unexpected_call, #{call => Req}),
{reply, ignored, State}. {reply, ignored, State}.
@ -312,7 +309,7 @@ handle_info(Info, State) ->
terminate(_Reason, #{stats_timer := StatsTimer} = State) -> terminate(_Reason, #{stats_timer := StatsTimer} = State) ->
emqx_conf:remove_handler([delayed]), emqx_conf:remove_handler([delayed]),
emqx_misc:cancel_timer(StatsTimer), emqx_misc:cancel_timer(StatsTimer),
ensure_enable(false, State). do_load_or_unload(false, State).
code_change(_Vsn, State, _Extra) -> code_change(_Vsn, State, _Extra) ->
{ok, State}. {ok, State}.
@ -381,22 +378,13 @@ do_publish(Key = {Ts, _Id}, Now, Acc) when Ts =< Now ->
-spec delayed_count() -> non_neg_integer(). -spec delayed_count() -> non_neg_integer().
delayed_count() -> mnesia:table_info(?TAB, size). delayed_count() -> mnesia:table_info(?TAB, size).
enable(Enable) -> do_load_or_unload(true, State) ->
case emqx_conf:get_raw([delayed]) of
#{<<"enable">> := Enable} ->
ok;
Cfg ->
{ok, _} = update_config(Cfg#{<<"enable">> := Enable}),
ok
end.
ensure_enable(true, State) ->
emqx_hooks:put('message.publish', {?MODULE, on_message_publish, []}), emqx_hooks:put('message.publish', {?MODULE, on_message_publish, []}),
State; State;
ensure_enable(false, #{publish_timer := PubTimer} = State) -> do_load_or_unload(false, #{publish_timer := PubTimer} = State) ->
emqx_hooks:del('message.publish', {?MODULE, on_message_publish}), emqx_hooks:del('message.publish', {?MODULE, on_message_publish}),
emqx_misc:cancel_timer(PubTimer), emqx_misc:cancel_timer(PubTimer),
ets:delete_all_objects(?TAB), ets:delete_all_objects(?TAB),
State#{publish_timer := undefined, publish_at := 0}; State#{publish_timer := undefined, publish_at := 0};
ensure_enable(_, State) -> do_load_or_unload(_, State) ->
State. State.

View File

@ -33,7 +33,7 @@ stop(_State) ->
ok. ok.
maybe_enable_modules() -> maybe_enable_modules() ->
emqx_conf:get([delayed, enable], true) andalso emqx_delayed:enable(), emqx_conf:get([delayed, enable], true) andalso emqx_delayed:load(),
emqx_modules_conf:is_telemetry_enabled() andalso emqx_telemetry:enable(), emqx_modules_conf:is_telemetry_enabled() andalso emqx_telemetry:enable(),
emqx_observer_cli:enable(), emqx_observer_cli:enable(),
emqx_conf_cli:load(), emqx_conf_cli:load(),
@ -42,7 +42,7 @@ maybe_enable_modules() ->
emqx_modules_conf:load(). emqx_modules_conf:load().
maybe_disable_modules() -> maybe_disable_modules() ->
emqx_conf:get([delayed, enable], true) andalso emqx_delayed:disable(), emqx_conf:get([delayed, enable], true) andalso emqx_delayed:unload(),
emqx_modules_conf:is_telemetry_enabled() andalso emqx_telemetry:disable(), emqx_modules_conf:is_telemetry_enabled() andalso emqx_telemetry:disable(),
emqx_conf:get([observer_cli, enable], true) andalso emqx_observer_cli:disable(), emqx_conf:get([observer_cli, enable], true) andalso emqx_observer_cli:disable(),
emqx_rewrite:disable(), emqx_rewrite:disable(),

View File

@ -52,24 +52,24 @@ init_per_testcase(t_load_case, Config) ->
Config; Config;
init_per_testcase(_Case, Config) -> init_per_testcase(_Case, Config) ->
{atomic, ok} = mria:clear_table(emqx_delayed), {atomic, ok} = mria:clear_table(emqx_delayed),
ok = emqx_delayed:enable(), ok = emqx_delayed:load(),
Config. Config.
end_per_testcase(_Case, _Config) -> end_per_testcase(_Case, _Config) ->
{atomic, ok} = mria:clear_table(emqx_delayed), {atomic, ok} = mria:clear_table(emqx_delayed),
ok = emqx_delayed:disable(). ok = emqx_delayed:unload().
%%-------------------------------------------------------------------- %%--------------------------------------------------------------------
%% Test cases %% Test cases
%%-------------------------------------------------------------------- %%--------------------------------------------------------------------
t_enable_disable_case(_) -> t_enable_disable_case(_) ->
emqx_delayed:disable(), emqx_delayed:unload(),
Hooks = emqx_hooks:lookup('message.publish'), Hooks = emqx_hooks:lookup('message.publish'),
MFA = {emqx_delayed, on_message_publish, []}, MFA = {emqx_delayed, on_message_publish, []},
?assertEqual(false, lists:keyfind(MFA, 2, Hooks)), ?assertEqual(false, lists:keyfind(MFA, 2, Hooks)),
ok = emqx_delayed:enable(), ok = emqx_delayed:load(),
Hooks1 = emqx_hooks:lookup('message.publish'), Hooks1 = emqx_hooks:lookup('message.publish'),
?assertNotEqual(false, lists:keyfind(MFA, 2, Hooks1)), ?assertNotEqual(false, lists:keyfind(MFA, 2, Hooks1)),
@ -80,7 +80,7 @@ t_enable_disable_case(_) ->
_ = on_message_publish(DelayedMsg0), _ = on_message_publish(DelayedMsg0),
?assertMatch(#{data := Datas} when Datas =/= [], emqx_delayed:list(#{})), ?assertMatch(#{data := Datas} when Datas =/= [], emqx_delayed:list(#{})),
emqx_delayed:disable(), emqx_delayed:unload(),
?assertEqual(false, lists:keyfind(MFA, 2, Hooks)), ?assertEqual(false, lists:keyfind(MFA, 2, Hooks)),
?assertMatch(#{data := []}, emqx_delayed:list(#{})), ?assertMatch(#{data := []}, emqx_delayed:list(#{})),
ok. ok.
@ -144,7 +144,7 @@ t_list(_) ->
). ).
t_max(_) -> t_max(_) ->
emqx_delayed:set_max_delayed_messages(1), emqx:update_config([delayed, max_delayed_messages], 1),
DelayedMsg0 = emqx_message:make(?MODULE, 1, <<"$delayed/10/t0">>, <<"delayed0">>), DelayedMsg0 = emqx_message:make(?MODULE, 1, <<"$delayed/10/t0">>, <<"delayed0">>),
DelayedMsg1 = emqx_message:make(?MODULE, 1, <<"$delayed/10/t1">>, <<"delayed1">>), DelayedMsg1 = emqx_message:make(?MODULE, 1, <<"$delayed/10/t1">>, <<"delayed1">>),

View File

@ -40,11 +40,11 @@ init_per_suite(Config) ->
[emqx_conf, emqx_modules, emqx_dashboard], [emqx_conf, emqx_modules, emqx_dashboard],
fun set_special_configs/1 fun set_special_configs/1
), ),
emqx_delayed:enable(), emqx_delayed:load(),
Config. Config.
end_per_suite(Config) -> end_per_suite(Config) ->
ok = emqx_delayed:disable(), ok = emqx_delayed:unload(),
emqx_common_test_helpers:stop_apps([emqx_conf, emqx_dashboard, emqx_modules]), emqx_common_test_helpers:stop_apps([emqx_conf, emqx_dashboard, emqx_modules]),
Config. Config.
@ -100,7 +100,7 @@ t_status(_Config) ->
t_messages(_) -> t_messages(_) ->
clear_all_record(), clear_all_record(),
emqx_delayed:enable(), emqx_delayed:load(),
{ok, C1} = emqtt:start_link([{clean_start, true}]), {ok, C1} = emqtt:start_link([{clean_start, true}]),
{ok, _} = emqtt:connect(C1), {ok, _} = emqtt:connect(C1),
@ -200,7 +200,7 @@ t_messages(_) ->
t_large_payload(_) -> t_large_payload(_) ->
clear_all_record(), clear_all_record(),
emqx_delayed:enable(), emqx_delayed:load(),
{ok, C1} = emqtt:start_link([{clean_start, true}]), {ok, C1} = emqtt:start_link([{clean_start, true}]),
{ok, _} = emqtt:connect(C1), {ok, _} = emqtt:connect(C1),