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:
commit
17b2665a6d
|
@ -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}};
|
|
||||||
handle_call(
|
|
||||||
{store, DelayedMsg = #delayed_message{key = Key}}, _From, State = #{max_delayed_messages := 0}
|
|
||||||
) ->
|
|
||||||
ok = mria:dirty_write(?TAB, DelayedMsg),
|
|
||||||
emqx_metrics:inc('messages.delayed'),
|
|
||||||
{reply, ok, ensure_publish_timer(Key, State)};
|
|
||||||
handle_call(
|
|
||||||
{store, DelayedMsg = #delayed_message{key = Key}}, _From, State = #{max_delayed_messages := Max}
|
|
||||||
) ->
|
|
||||||
Size = mnesia:table_info(?TAB, size),
|
Size = mnesia:table_info(?TAB, size),
|
||||||
case Size >= Max of
|
case get_conf(max_delayed_messages) of
|
||||||
true ->
|
0 ->
|
||||||
|
ok = mria:dirty_write(?TAB, DelayedMsg),
|
||||||
|
emqx_metrics:inc('messages.delayed'),
|
||||||
|
{reply, ok, ensure_publish_timer(Key, State)};
|
||||||
|
Max when Size >= Max ->
|
||||||
{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.
|
||||||
|
|
|
@ -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(),
|
||||||
|
|
|
@ -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">>),
|
||||||
|
|
|
@ -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),
|
||||||
|
|
Loading…
Reference in New Issue