Merge pull request #7632 from lafirest/fix/exhook_cfg_update
fix(exhook): refine the status of the exhook callback server
This commit is contained in:
commit
1be1ecd979
|
@ -170,7 +170,7 @@ fields(node_metrics) ->
|
||||||
fields(node_status) ->
|
fields(node_status) ->
|
||||||
[
|
[
|
||||||
{node, mk(string(), #{})},
|
{node, mk(string(), #{})},
|
||||||
{status, mk(enum([running, waiting, stopped, error]), #{})}
|
{status, mk(enum([connected, connecting, unconnected, disable, error]), #{})}
|
||||||
];
|
];
|
||||||
fields(hook_info) ->
|
fields(hook_info) ->
|
||||||
[
|
[
|
||||||
|
|
|
@ -63,21 +63,25 @@
|
||||||
-export([roots/0]).
|
-export([roots/0]).
|
||||||
|
|
||||||
%% Running servers
|
%% Running servers
|
||||||
-type state() :: #{
|
-type state() :: #{servers := servers()}.
|
||||||
running := servers(),
|
|
||||||
%% Wait to reload servers
|
|
||||||
waiting := servers(),
|
|
||||||
%% Marked stopped servers
|
|
||||||
stopped := servers(),
|
|
||||||
%% Timer references
|
|
||||||
trefs := map(),
|
|
||||||
orders := orders()
|
|
||||||
}.
|
|
||||||
|
|
||||||
-type server_name() :: binary().
|
|
||||||
-type servers() :: #{server_name() => server()}.
|
|
||||||
-type server() :: server_options().
|
|
||||||
-type server_options() :: map().
|
-type server_options() :: map().
|
||||||
|
-type server_name() :: binary().
|
||||||
|
|
||||||
|
-type status() ::
|
||||||
|
connected
|
||||||
|
| connecting
|
||||||
|
| unconnected
|
||||||
|
| disable.
|
||||||
|
|
||||||
|
-type server() :: #{
|
||||||
|
status := status(),
|
||||||
|
timer := reference(),
|
||||||
|
order := integer(),
|
||||||
|
%% include the content of server_options
|
||||||
|
atom() => any()
|
||||||
|
}.
|
||||||
|
-type servers() :: #{server_name() => server()}.
|
||||||
|
|
||||||
-type position() ::
|
-type position() ::
|
||||||
front
|
front
|
||||||
|
@ -85,19 +89,10 @@
|
||||||
| {before, binary()}
|
| {before, binary()}
|
||||||
| {'after', binary()}.
|
| {'after', binary()}.
|
||||||
|
|
||||||
-type orders() :: #{server_name() => integer()}.
|
|
||||||
|
|
||||||
-type server_info() :: #{
|
|
||||||
name := server_name(),
|
|
||||||
status := running | waiting | stopped,
|
|
||||||
|
|
||||||
atom() => term()
|
|
||||||
}.
|
|
||||||
|
|
||||||
-define(DEFAULT_TIMEOUT, 60000).
|
-define(DEFAULT_TIMEOUT, 60000).
|
||||||
-define(REFRESH_INTERVAL, timer:seconds(5)).
|
-define(REFRESH_INTERVAL, timer:seconds(5)).
|
||||||
|
|
||||||
-export_type([servers/0, server/0, server_info/0]).
|
-export_type([servers/0, server/0]).
|
||||||
|
|
||||||
%%--------------------------------------------------------------------
|
%%--------------------------------------------------------------------
|
||||||
%% APIs
|
%% APIs
|
||||||
|
@ -113,7 +108,7 @@ start_link() ->
|
||||||
list() ->
|
list() ->
|
||||||
call(list).
|
call(list).
|
||||||
|
|
||||||
-spec lookup(server_name()) -> not_found | server_info().
|
-spec lookup(server_name()) -> not_found | server().
|
||||||
lookup(Name) ->
|
lookup(Name) ->
|
||||||
call({lookup, Name}).
|
call({lookup, Name}).
|
||||||
|
|
||||||
|
@ -195,104 +190,56 @@ init([]) ->
|
||||||
process_flag(trap_exit, true),
|
process_flag(trap_exit, true),
|
||||||
emqx_conf:add_handler([exhook, servers], ?MODULE),
|
emqx_conf:add_handler([exhook, servers], ?MODULE),
|
||||||
ServerL = emqx:get_config([exhook, servers]),
|
ServerL = emqx:get_config([exhook, servers]),
|
||||||
{Waiting, Running, Stopped} = load_all_servers(ServerL),
|
Servers = load_all_servers(ServerL),
|
||||||
Orders = reorder(ServerL),
|
Servers2 = reorder(ServerL, Servers),
|
||||||
refresh_tick(),
|
refresh_tick(),
|
||||||
{ok,
|
{ok, #{servers => Servers2}}.
|
||||||
ensure_reload_timer(
|
|
||||||
#{
|
|
||||||
waiting => Waiting,
|
|
||||||
running => Running,
|
|
||||||
stopped => Stopped,
|
|
||||||
trefs => #{},
|
|
||||||
orders => Orders
|
|
||||||
}
|
|
||||||
)}.
|
|
||||||
|
|
||||||
-spec load_all_servers(list(server_options())) -> {servers(), servers(), servers()}.
|
-spec load_all_servers(list(server_options())) -> servers().
|
||||||
load_all_servers(ServerL) ->
|
load_all_servers(ServerL) ->
|
||||||
load_all_servers(ServerL, #{}, #{}, #{}).
|
load_all_servers(ServerL, #{}).
|
||||||
|
|
||||||
load_all_servers([#{name := Name} = Options | More], Waiting, Running, Stopped) ->
|
load_all_servers([#{name := Name} = Options | More], Servers) ->
|
||||||
case emqx_exhook_server:load(Name, Options) of
|
{_, Server} = do_load_server(options_to_server(Options)),
|
||||||
{ok, ServerState} ->
|
load_all_servers(More, Servers#{Name => Server});
|
||||||
save(Name, ServerState),
|
load_all_servers([], Servers) ->
|
||||||
load_all_servers(More, Waiting, Running#{Name => Options}, Stopped);
|
Servers.
|
||||||
{error, _} ->
|
|
||||||
load_all_servers(More, Waiting#{Name => Options}, Running, Stopped);
|
|
||||||
disable ->
|
|
||||||
load_all_servers(More, Waiting, Running, Stopped#{Name => Options})
|
|
||||||
end;
|
|
||||||
load_all_servers([], Waiting, Running, Stopped) ->
|
|
||||||
{Waiting, Running, Stopped}.
|
|
||||||
|
|
||||||
handle_call(
|
handle_call(
|
||||||
list,
|
list,
|
||||||
_From,
|
_From,
|
||||||
State = #{
|
State = #{servers := Servers}
|
||||||
running := Running,
|
|
||||||
waiting := Waiting,
|
|
||||||
stopped := Stopped,
|
|
||||||
orders := Orders
|
|
||||||
}
|
|
||||||
) ->
|
) ->
|
||||||
R = get_servers_info(running, Running),
|
Infos = get_servers_info(Servers),
|
||||||
W = get_servers_info(waiting, Waiting),
|
OrderServers = sort_name_by_order(Infos, Servers),
|
||||||
S = get_servers_info(stopped, Stopped),
|
|
||||||
|
|
||||||
Servers = R ++ W ++ S,
|
|
||||||
OrderServers = sort_name_by_order(Servers, Orders),
|
|
||||||
|
|
||||||
{reply, OrderServers, State};
|
{reply, OrderServers, State};
|
||||||
handle_call(
|
handle_call(
|
||||||
{update_config, {move, _Name, _Position}, NewConfL},
|
{update_config, {move, _Name, _Position}, NewConfL},
|
||||||
_From,
|
_From,
|
||||||
State
|
#{servers := Servers} = State
|
||||||
) ->
|
) ->
|
||||||
Orders = reorder(NewConfL),
|
Servers2 = reorder(NewConfL, Servers),
|
||||||
{reply, ok, State#{orders := Orders}};
|
{reply, ok, State#{servers := Servers2}};
|
||||||
handle_call({update_config, {delete, ToDelete}, _}, _From, State) ->
|
handle_call({update_config, {delete, ToDelete}, _}, _From, State) ->
|
||||||
{ok,
|
|
||||||
#{
|
|
||||||
orders := Orders,
|
|
||||||
stopped := Stopped
|
|
||||||
} = State2} = do_unload_server(ToDelete, State),
|
|
||||||
|
|
||||||
State3 = State2#{
|
|
||||||
stopped := maps:remove(ToDelete, Stopped),
|
|
||||||
orders := maps:remove(ToDelete, Orders)
|
|
||||||
},
|
|
||||||
|
|
||||||
emqx_exhook_metrics:on_server_deleted(ToDelete),
|
emqx_exhook_metrics:on_server_deleted(ToDelete),
|
||||||
|
|
||||||
{reply, ok, State3};
|
#{servers := Servers} = State2 = do_unload_server(ToDelete, State),
|
||||||
|
|
||||||
|
Servers2 = maps:remove(ToDelete, Servers),
|
||||||
|
|
||||||
|
{reply, ok, update_servers(Servers2, State2)};
|
||||||
handle_call(
|
handle_call(
|
||||||
{update_config, {add, RawConf}, NewConfL},
|
{update_config, {add, RawConf}, NewConfL},
|
||||||
_From,
|
_From,
|
||||||
#{running := Running, waiting := Waitting, stopped := Stopped} = State
|
#{servers := Servers} = State
|
||||||
) ->
|
) ->
|
||||||
{_, #{name := Name} = Conf} = emqx_config:check_config(?MODULE, RawConf),
|
{_, #{name := Name} = Conf} = emqx_config:check_config(?MODULE, RawConf),
|
||||||
|
{Result, Server} = do_load_server(options_to_server(Conf)),
|
||||||
case emqx_exhook_server:load(Name, Conf) of
|
Servers2 = Servers#{Name => Server},
|
||||||
{ok, ServerState} ->
|
Servers3 = reorder(NewConfL, Servers2),
|
||||||
save(Name, ServerState),
|
{reply, Result, State#{servers := Servers3}};
|
||||||
State2 = State#{running := Running#{Name => Conf}};
|
|
||||||
{error, _} ->
|
|
||||||
StateT = State#{waiting := Waitting#{Name => Conf}},
|
|
||||||
State2 = ensure_reload_timer(StateT);
|
|
||||||
disable ->
|
|
||||||
State2 = State#{stopped := Stopped#{Name => Conf}}
|
|
||||||
end,
|
|
||||||
Orders = reorder(NewConfL),
|
|
||||||
{reply, ok, State2#{orders := Orders}};
|
|
||||||
handle_call({lookup, Name}, _From, State) ->
|
handle_call({lookup, Name}, _From, State) ->
|
||||||
case where_is_server(Name, State) of
|
{reply, where_is_server(Name, State), State};
|
||||||
not_found ->
|
|
||||||
Result = not_found;
|
|
||||||
{Where, #{Name := Conf}} ->
|
|
||||||
Result = maps:merge(Conf, #{status => Where})
|
|
||||||
end,
|
|
||||||
{reply, Result, State};
|
|
||||||
handle_call({update_config, {update, Name, _Conf}, NewConfL}, _From, State) ->
|
handle_call({update_config, {update, Name, _Conf}, NewConfL}, _From, State) ->
|
||||||
{Result, State2} = restart_server(Name, NewConfL, State),
|
{Result, State2} = restart_server(Name, NewConfL, State),
|
||||||
{reply, Result, State2};
|
{reply, Result, State2};
|
||||||
|
@ -303,7 +250,7 @@ handle_call({server_info, Name}, _From, State) ->
|
||||||
case where_is_server(Name, State) of
|
case where_is_server(Name, State) of
|
||||||
not_found ->
|
not_found ->
|
||||||
Result = not_found;
|
Result = not_found;
|
||||||
{Status, _} ->
|
#{status := Status} ->
|
||||||
HooksMetrics = emqx_exhook_metrics:server_metrics(Name),
|
HooksMetrics = emqx_exhook_metrics:server_metrics(Name),
|
||||||
Result = #{
|
Result = #{
|
||||||
status => Status,
|
status => Status,
|
||||||
|
@ -314,25 +261,9 @@ handle_call({server_info, Name}, _From, State) ->
|
||||||
handle_call(
|
handle_call(
|
||||||
all_servers_info,
|
all_servers_info,
|
||||||
_From,
|
_From,
|
||||||
#{
|
#{servers := Servers} = State
|
||||||
running := Running,
|
|
||||||
waiting := Waiting,
|
|
||||||
stopped := Stopped
|
|
||||||
} = State
|
|
||||||
) ->
|
) ->
|
||||||
MakeStatus = fun(Status, Servers, Acc) ->
|
Status = maps:map(fun(_Name, #{status := Status}) -> Status end, Servers),
|
||||||
lists:foldl(
|
|
||||||
fun(Name, IAcc) -> IAcc#{Name => Status} end,
|
|
||||||
Acc,
|
|
||||||
maps:keys(Servers)
|
|
||||||
)
|
|
||||||
end,
|
|
||||||
Status = lists:foldl(
|
|
||||||
fun({Status, Servers}, Acc) -> MakeStatus(Status, Servers, Acc) end,
|
|
||||||
#{},
|
|
||||||
[{running, Running}, {waiting, Waiting}, {stopped, Stopped}]
|
|
||||||
),
|
|
||||||
|
|
||||||
Metrics = emqx_exhook_metrics:servers_metrics(),
|
Metrics = emqx_exhook_metrics:servers_metrics(),
|
||||||
|
|
||||||
Result = #{
|
Result = #{
|
||||||
|
@ -352,23 +283,8 @@ handle_cast(_Msg, State) ->
|
||||||
{noreply, State}.
|
{noreply, State}.
|
||||||
|
|
||||||
handle_info({timeout, _Ref, {reload, Name}}, State) ->
|
handle_info({timeout, _Ref, {reload, Name}}, State) ->
|
||||||
{Result, NState} = do_load_server(Name, State),
|
{_, NState} = do_reload_server(Name, State),
|
||||||
case Result of
|
|
||||||
ok ->
|
|
||||||
{noreply, NState};
|
{noreply, NState};
|
||||||
{error, not_found} ->
|
|
||||||
{noreply, NState};
|
|
||||||
{error, Reason} ->
|
|
||||||
?SLOG(
|
|
||||||
warning,
|
|
||||||
#{
|
|
||||||
msg => "failed_to_reload_exhook_callback_server",
|
|
||||||
reason => Reason,
|
|
||||||
name => Name
|
|
||||||
}
|
|
||||||
),
|
|
||||||
{noreply, ensure_reload_timer(NState)}
|
|
||||||
end;
|
|
||||||
handle_info(refresh_tick, State) ->
|
handle_info(refresh_tick, State) ->
|
||||||
refresh_tick(),
|
refresh_tick(),
|
||||||
emqx_exhook_metrics:update(?REFRESH_INTERVAL),
|
emqx_exhook_metrics:update(?REFRESH_INTERVAL),
|
||||||
|
@ -376,14 +292,13 @@ handle_info(refresh_tick, State) ->
|
||||||
handle_info(_Info, State) ->
|
handle_info(_Info, State) ->
|
||||||
{noreply, State}.
|
{noreply, State}.
|
||||||
|
|
||||||
terminate(_Reason, State = #{running := Running}) ->
|
terminate(_Reason, State = #{servers := Servers}) ->
|
||||||
_ = maps:fold(
|
_ = maps:fold(
|
||||||
fun(Name, _, AccIn) ->
|
fun(Name, _, AccIn) ->
|
||||||
{ok, NAccIn} = do_unload_server(Name, AccIn),
|
do_unload_server(Name, AccIn)
|
||||||
NAccIn
|
|
||||||
end,
|
end,
|
||||||
State,
|
State,
|
||||||
Running
|
Servers
|
||||||
),
|
),
|
||||||
_ = unload_exhooks(),
|
_ = unload_exhooks(),
|
||||||
ok.
|
ok.
|
||||||
|
@ -401,121 +316,82 @@ unload_exhooks() ->
|
||||||
|| {Name, {M, F, _A}} <- ?ENABLED_HOOKS
|
|| {Name, {M, F, _A}} <- ?ENABLED_HOOKS
|
||||||
].
|
].
|
||||||
|
|
||||||
-spec do_load_server(server_name(), state()) ->
|
do_load_server(#{name := Name} = Server) ->
|
||||||
{{error, not_found}, state()}
|
case emqx_exhook_server:load(Name, Server) of
|
||||||
| {{error, already_started}, state()}
|
|
||||||
| {ok, state()}.
|
|
||||||
do_load_server(Name, State = #{orders := Orders}) ->
|
|
||||||
case where_is_server(Name, State) of
|
|
||||||
not_found ->
|
|
||||||
{{error, not_found}, State};
|
|
||||||
{running, _} ->
|
|
||||||
{ok, State};
|
|
||||||
{Where, Map} ->
|
|
||||||
State2 = clean_reload_timer(Name, State),
|
|
||||||
{Options, Map2} = maps:take(Name, Map),
|
|
||||||
State3 = State2#{Where := Map2},
|
|
||||||
#{
|
|
||||||
running := Running,
|
|
||||||
stopped := Stopped
|
|
||||||
} = State3,
|
|
||||||
case emqx_exhook_server:load(Name, Options) of
|
|
||||||
{ok, ServerState} ->
|
{ok, ServerState} ->
|
||||||
save(Name, ServerState),
|
save(Name, ServerState),
|
||||||
update_order(Orders),
|
{ok, Server#{status => connected}};
|
||||||
?SLOG(info, #{
|
|
||||||
msg => "load_exhook_callback_server_ok",
|
|
||||||
name => Name
|
|
||||||
}),
|
|
||||||
{ok, State3#{running := maps:put(Name, Options, Running)}};
|
|
||||||
{error, Reason} ->
|
|
||||||
{{error, Reason}, State};
|
|
||||||
disable ->
|
disable ->
|
||||||
{ok, State3#{stopped := Stopped#{Name => Options}}}
|
{ok, set_disable(Server)};
|
||||||
|
{ErrorType, Reason} = Error ->
|
||||||
|
?SLOG(
|
||||||
|
error,
|
||||||
|
#{
|
||||||
|
msg => "failed_to_load_exhook_callback_server",
|
||||||
|
reason => Reason,
|
||||||
|
name => Name
|
||||||
|
}
|
||||||
|
),
|
||||||
|
case ErrorType of
|
||||||
|
load_error ->
|
||||||
|
{ok, ensure_reload_timer(Server)};
|
||||||
|
_ ->
|
||||||
|
{Error, Server#{status => unconnected}}
|
||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
-spec do_unload_server(server_name(), state()) -> {ok, state()}.
|
do_load_server(#{name := Name} = Server, #{servers := Servers} = State) ->
|
||||||
do_unload_server(Name, #{stopped := Stopped} = State) ->
|
{Result, Server2} = do_load_server(Server),
|
||||||
|
{Result, update_servers(Servers#{Name => Server2}, State)}.
|
||||||
|
|
||||||
|
-spec do_reload_server(server_name(), state()) ->
|
||||||
|
{{error, term()}, state()}
|
||||||
|
| {ok, state()}.
|
||||||
|
do_reload_server(Name, State = #{servers := Servers}) ->
|
||||||
case where_is_server(Name, State) of
|
case where_is_server(Name, State) of
|
||||||
{stopped, _} ->
|
not_found ->
|
||||||
|
{{error, not_found}, State};
|
||||||
|
#{timer := undefined} ->
|
||||||
{ok, State};
|
{ok, State};
|
||||||
{waiting, Waiting} ->
|
Server ->
|
||||||
{Options, Waiting2} = maps:take(Name, Waiting),
|
clean_reload_timer(Server),
|
||||||
{ok,
|
do_load_server(Server, Servers)
|
||||||
clean_reload_timer(
|
end.
|
||||||
Name,
|
|
||||||
State#{
|
-spec do_unload_server(server_name(), state()) -> state().
|
||||||
waiting := Waiting2,
|
do_unload_server(Name, #{servers := Servers} = State) ->
|
||||||
stopped := maps:put(Name, Options, Stopped)
|
case where_is_server(Name, State) of
|
||||||
}
|
not_found ->
|
||||||
)};
|
State;
|
||||||
{running, Running} ->
|
#{status := disable} ->
|
||||||
Service = server(Name),
|
State;
|
||||||
|
Server ->
|
||||||
|
clean_reload_timer(Server),
|
||||||
|
case server(Name) of
|
||||||
|
undefined ->
|
||||||
|
State;
|
||||||
|
Service ->
|
||||||
ok = unsave(Name),
|
ok = unsave(Name),
|
||||||
ok = emqx_exhook_server:unload(Service),
|
ok = emqx_exhook_server:unload(Service),
|
||||||
{Options, Running2} = maps:take(Name, Running),
|
Servers2 = Servers#{Name := set_disable(Server)},
|
||||||
{ok, State#{
|
State#{servers := Servers2}
|
||||||
running := Running2,
|
end
|
||||||
stopped := maps:put(Name, Options, Stopped)
|
|
||||||
}};
|
|
||||||
not_found ->
|
|
||||||
{ok, State}
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
-spec ensure_reload_timer(state()) -> state().
|
ensure_reload_timer(#{timer := Timer} = Server) when is_reference(Timer) ->
|
||||||
ensure_reload_timer(
|
Server;
|
||||||
State = #{
|
ensure_reload_timer(#{name := Name, auto_reconnect := Intv} = Server) when is_integer(Intv) ->
|
||||||
waiting := Waiting,
|
|
||||||
stopped := Stopped,
|
|
||||||
trefs := TRefs
|
|
||||||
}
|
|
||||||
) ->
|
|
||||||
Iter = maps:iterator(Waiting),
|
|
||||||
|
|
||||||
{Waitting2, Stopped2, TRefs2} =
|
|
||||||
ensure_reload_timer(maps:next(Iter), Waiting, Stopped, TRefs),
|
|
||||||
|
|
||||||
State#{
|
|
||||||
waiting := Waitting2,
|
|
||||||
stopped := Stopped2,
|
|
||||||
trefs := TRefs2
|
|
||||||
}.
|
|
||||||
|
|
||||||
ensure_reload_timer(none, Waiting, Stopped, TimerRef) ->
|
|
||||||
{Waiting, Stopped, TimerRef};
|
|
||||||
ensure_reload_timer(
|
|
||||||
{Name, #{auto_reconnect := Intv}, Iter},
|
|
||||||
Waiting,
|
|
||||||
Stopped,
|
|
||||||
TimerRef
|
|
||||||
) when is_integer(Intv) ->
|
|
||||||
Next = maps:next(Iter),
|
|
||||||
case maps:is_key(Name, TimerRef) of
|
|
||||||
true ->
|
|
||||||
ensure_reload_timer(Next, Waiting, Stopped, TimerRef);
|
|
||||||
_ ->
|
|
||||||
Ref = erlang:start_timer(Intv, self(), {reload, Name}),
|
Ref = erlang:start_timer(Intv, self(), {reload, Name}),
|
||||||
TimerRef2 = maps:put(Name, Ref, TimerRef),
|
Server#{status := connecting, timer := Ref};
|
||||||
ensure_reload_timer(Next, Waiting, Stopped, TimerRef2)
|
ensure_reload_timer(Server) ->
|
||||||
end;
|
Server#{status := unconnected}.
|
||||||
ensure_reload_timer({Name, Opts, Iter}, Waiting, Stopped, TimerRef) ->
|
|
||||||
ensure_reload_timer(
|
|
||||||
maps:next(Iter),
|
|
||||||
maps:remove(Name, Waiting),
|
|
||||||
maps:put(Name, Opts, Stopped),
|
|
||||||
TimerRef
|
|
||||||
).
|
|
||||||
|
|
||||||
-spec clean_reload_timer(server_name(), state()) -> state().
|
-spec clean_reload_timer(server()) -> ok.
|
||||||
clean_reload_timer(Name, State = #{trefs := TRefs}) ->
|
clean_reload_timer(#{timer := undefined}) ->
|
||||||
case maps:take(Name, TRefs) of
|
ok;
|
||||||
error ->
|
clean_reload_timer(#{timer := Timer}) ->
|
||||||
State;
|
_ = erlang:cancel_timer(Timer),
|
||||||
{TRef, NTRefs} ->
|
ok.
|
||||||
_ = erlang:cancel_timer(TRef),
|
|
||||||
State#{trefs := NTRefs}
|
|
||||||
end.
|
|
||||||
|
|
||||||
-spec do_move(binary(), position(), list(server_options())) ->
|
-spec do_move(binary(), position(), list(server_options())) ->
|
||||||
not_found | list(server_options()).
|
not_found | list(server_options()).
|
||||||
|
@ -545,37 +421,32 @@ move_to([H | T], Position, Server, HeadL) ->
|
||||||
move_to([], _Position, _Server, _HeadL) ->
|
move_to([], _Position, _Server, _HeadL) ->
|
||||||
not_found.
|
not_found.
|
||||||
|
|
||||||
-spec reorder(list(server_options())) -> orders().
|
-spec reorder(list(server_options()), servers()) -> servers().
|
||||||
reorder(ServerL) ->
|
reorder(ServerL, Servers) ->
|
||||||
Orders = reorder(ServerL, 1, #{}),
|
Orders = reorder(ServerL, 1, Servers),
|
||||||
update_order(Orders),
|
update_order(Orders),
|
||||||
Orders.
|
Orders.
|
||||||
|
|
||||||
reorder([#{name := Name} | T], Order, Orders) ->
|
reorder([#{name := Name} | T], Order, Servers) ->
|
||||||
reorder(T, Order + 1, Orders#{Name => Order});
|
reorder(T, Order + 1, update_order(Name, Servers, Order));
|
||||||
reorder([], _Order, Orders) ->
|
reorder([], _Order, Servers) ->
|
||||||
Orders.
|
Servers.
|
||||||
|
|
||||||
get_servers_info(Status, Map) ->
|
update_order(Name, Servers, Order) ->
|
||||||
|
Server = maps:get(Name, Servers),
|
||||||
|
Servers#{Name := Server#{order := Order}}.
|
||||||
|
|
||||||
|
get_servers_info(Svrs) ->
|
||||||
Fold = fun(Name, Conf, Acc) ->
|
Fold = fun(Name, Conf, Acc) ->
|
||||||
[
|
[
|
||||||
maps:merge(Conf, #{
|
maps:merge(Conf, #{hooks => hooks(Name)})
|
||||||
status => Status,
|
|
||||||
hooks => hooks(Name)
|
|
||||||
})
|
|
||||||
| Acc
|
| Acc
|
||||||
]
|
]
|
||||||
end,
|
end,
|
||||||
maps:fold(Fold, [], Map).
|
maps:fold(Fold, [], Svrs).
|
||||||
|
|
||||||
where_is_server(Name, #{running := Running}) when is_map_key(Name, Running) ->
|
where_is_server(Name, #{servers := Servers}) ->
|
||||||
{running, Running};
|
maps:get(Name, Servers, not_found).
|
||||||
where_is_server(Name, #{waiting := Waiting}) when is_map_key(Name, Waiting) ->
|
|
||||||
{waiting, Waiting};
|
|
||||||
where_is_server(Name, #{stopped := Stopped}) when is_map_key(Name, Stopped) ->
|
|
||||||
{stopped, Stopped};
|
|
||||||
where_is_server(_, _) ->
|
|
||||||
not_found.
|
|
||||||
|
|
||||||
-type replace_fun() :: fun((server_options()) -> server_options()).
|
-type replace_fun() :: fun((server_options()) -> server_options()).
|
||||||
|
|
||||||
|
@ -604,15 +475,10 @@ restart_server(Name, ConfL, State) ->
|
||||||
case where_is_server(Name, State) of
|
case where_is_server(Name, State) of
|
||||||
not_found ->
|
not_found ->
|
||||||
{{error, not_found}, State};
|
{{error, not_found}, State};
|
||||||
{Where, Map} ->
|
Server ->
|
||||||
State2 = State#{Where := Map#{Name := Conf}},
|
Server2 = maps:merge(Server, Conf),
|
||||||
{ok, State3} = do_unload_server(Name, State2),
|
State2 = do_unload_server(Name, State),
|
||||||
case do_load_server(Name, State3) of
|
do_load_server(Server2, State2)
|
||||||
{ok, State4} ->
|
|
||||||
{ok, State4};
|
|
||||||
{Error, State4} ->
|
|
||||||
{Error, State4}
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
@ -620,9 +486,11 @@ sort_name_by_order(Names, Orders) ->
|
||||||
lists:sort(
|
lists:sort(
|
||||||
fun
|
fun
|
||||||
(A, B) when is_binary(A) ->
|
(A, B) when is_binary(A) ->
|
||||||
maps:get(A, Orders) < maps:get(B, Orders);
|
emqx_map_lib:deep_get([A, order], Orders) <
|
||||||
|
emqx_map_lib:deep_get([B, order], Orders);
|
||||||
(#{name := A}, #{name := B}) ->
|
(#{name := A}, #{name := B}) ->
|
||||||
maps:get(A, Orders) < maps:get(B, Orders)
|
emqx_map_lib:deep_get([A, order], Orders) <
|
||||||
|
emqx_map_lib:deep_get([B, order], Orders)
|
||||||
end,
|
end,
|
||||||
Names
|
Names
|
||||||
).
|
).
|
||||||
|
@ -630,6 +498,16 @@ sort_name_by_order(Names, Orders) ->
|
||||||
refresh_tick() ->
|
refresh_tick() ->
|
||||||
erlang:send_after(?REFRESH_INTERVAL, self(), ?FUNCTION_NAME).
|
erlang:send_after(?REFRESH_INTERVAL, self(), ?FUNCTION_NAME).
|
||||||
|
|
||||||
|
options_to_server(Options) ->
|
||||||
|
maps:merge(Options, #{status => unconnected, timer => undefined, order => 0}).
|
||||||
|
|
||||||
|
update_servers(Servers, State) ->
|
||||||
|
update_order(Servers),
|
||||||
|
State#{servers := Servers}.
|
||||||
|
|
||||||
|
set_disable(Server) ->
|
||||||
|
Server#{status := disable, timer := undefined}.
|
||||||
|
|
||||||
%%--------------------------------------------------------------------
|
%%--------------------------------------------------------------------
|
||||||
%% Server state persistent
|
%% Server state persistent
|
||||||
save(Name, ServerState) ->
|
save(Name, ServerState) ->
|
||||||
|
@ -661,8 +539,17 @@ server(Name) ->
|
||||||
Service -> Service
|
Service -> Service
|
||||||
end.
|
end.
|
||||||
|
|
||||||
update_order(Orders) ->
|
update_order(Servers) ->
|
||||||
Running = running(),
|
Running = running(),
|
||||||
|
Orders = maps:filter(
|
||||||
|
fun
|
||||||
|
(Name, #{status := connected}) ->
|
||||||
|
lists:member(Name, Running);
|
||||||
|
(_, _) ->
|
||||||
|
false
|
||||||
|
end,
|
||||||
|
Servers
|
||||||
|
),
|
||||||
Running2 = sort_name_by_order(Running, Orders),
|
Running2 = sort_name_by_order(Running, Orders),
|
||||||
persistent_term:put(?APP, Running2).
|
persistent_term:put(?APP, Running2).
|
||||||
|
|
||||||
|
|
|
@ -86,12 +86,13 @@
|
||||||
%% Load/Unload APIs
|
%% Load/Unload APIs
|
||||||
%%--------------------------------------------------------------------
|
%%--------------------------------------------------------------------
|
||||||
|
|
||||||
-spec load(binary(), map()) -> {ok, server()} | {error, term()} | disable.
|
-spec load(binary(), map()) -> {ok, server()} | {error, term()} | {load_error, term()} | disable.
|
||||||
load(_Name, #{enable := false}) ->
|
load(_Name, #{enable := false}) ->
|
||||||
disable;
|
disable;
|
||||||
load(Name, #{request_timeout := Timeout, failed_action := FailedAction} = Opts) ->
|
load(Name, #{request_timeout := Timeout, failed_action := FailedAction} = Opts) ->
|
||||||
ReqOpts = #{timeout => Timeout, failed_action => FailedAction},
|
ReqOpts = #{timeout => Timeout, failed_action => FailedAction},
|
||||||
{SvrAddr, ClientOpts} = channel_opts(Opts),
|
case channel_opts(Opts) of
|
||||||
|
{ok, {SvrAddr, ClientOpts}} ->
|
||||||
case
|
case
|
||||||
emqx_exhook_sup:start_grpc_client_channel(
|
emqx_exhook_sup:start_grpc_client_channel(
|
||||||
Name,
|
Name,
|
||||||
|
@ -114,12 +115,15 @@ load(Name, #{request_timeout := Timeout, failed_action := FailedAction} = Opts)
|
||||||
hookspec => HookSpecs,
|
hookspec => HookSpecs,
|
||||||
prefix => Prefix
|
prefix => Prefix
|
||||||
}};
|
}};
|
||||||
{error, _} = E ->
|
{error, Reason} ->
|
||||||
emqx_exhook_sup:stop_grpc_client_channel(Name),
|
emqx_exhook_sup:stop_grpc_client_channel(Name),
|
||||||
E
|
{load_error, Reason}
|
||||||
end;
|
end;
|
||||||
{error, _} = E ->
|
{error, _} = E ->
|
||||||
E
|
E
|
||||||
|
end;
|
||||||
|
Error ->
|
||||||
|
Error
|
||||||
end.
|
end.
|
||||||
|
|
||||||
%% @private
|
%% @private
|
||||||
|
@ -130,7 +134,7 @@ channel_opts(Opts = #{url := URL}) ->
|
||||||
),
|
),
|
||||||
case uri_string:parse(URL) of
|
case uri_string:parse(URL) of
|
||||||
#{scheme := <<"http">>, host := Host, port := Port} ->
|
#{scheme := <<"http">>, host := Host, port := Port} ->
|
||||||
{format_http_uri("http", Host, Port), ClientOpts};
|
{ok, {format_http_uri("http", Host, Port), ClientOpts}};
|
||||||
#{scheme := <<"https">>, host := Host, port := Port} ->
|
#{scheme := <<"https">>, host := Host, port := Port} ->
|
||||||
SslOpts =
|
SslOpts =
|
||||||
case maps:get(ssl, Opts, undefined) of
|
case maps:get(ssl, Opts, undefined) of
|
||||||
|
@ -154,9 +158,9 @@ channel_opts(Opts = #{url := URL}) ->
|
||||||
transport_opts => SslOpts
|
transport_opts => SslOpts
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{format_http_uri("https", Host, Port), NClientOpts};
|
{ok, {format_http_uri("https", Host, Port), NClientOpts}};
|
||||||
Error ->
|
Error ->
|
||||||
error({bad_server_url, URL, Error})
|
{error, {bad_server_url, URL, Error}}
|
||||||
end.
|
end.
|
||||||
|
|
||||||
format_http_uri(Scheme, Host, Port) ->
|
format_http_uri(Scheme, Host, Port) ->
|
||||||
|
|
Loading…
Reference in New Issue