У меня есть простое приложение target_interceptor
, которое после получения сообщений о регистрации и отмене регистрации запускает или завершает работу под simple_one_for_one
rpc_server_supervisor
.
Код супервайзера rpc_server_supervisor
:
init([]) ->
MaxRestart = 5,
MaxTime = 3600,
{ok, {{simple_one_for_one, MaxRestart, MaxTime},
[{rbmq_rpc_server,
{rbmq_rpc_server, start_link, []},
temporary,
10000,
worker,
[rbmq_rpc_server]}]}}.
зарегистрировать сообщение target_interceptor
:
handle_cast({register, Args}, S = #state{channel = Channel, supervisor=Sup, refs=R, qs_link = QSLinks}) ->
{Connection, QueueName, Node} = Args,
{Ok, Pid} = supervisor:start_child(Sup, [Connection, QueueName, Node]),
Ref = erlang:monitor(process, Pid),
{noreply, S#state{refs=gb_sets:add(Ref,R), qs_link=orddict:append(binary_to_list(QueueName),Pid,QSLinks)}};
сообщение об отмене регистрации target_interceptor
:
handle_cast({unregister,{QueueName}}, S = #state{supervisor = Sup, qs_link = QSLinks}) ->
Pid = orddict:fetch(QueueName,QSLinks)
case supervisor:terminate_child(Sup,Pid) of
ok -> Success = true;
Error -> io:format("Error ~p~n",[Error]),
Success = false
end,
{noreply, S#state{qs_link=orddict:erase(QueueName,QSLinks)}}
моя версия Erlang: R15B01
Первая проблема заключается в том, что при обработке действия регистрации кортеж {OK, Pid} = {error, ‹0.57.0>}; хотя это указывает на то, что что-то пошло не так, gen_server rbmq_rpc_server на Pid 57 работает правильно и отвечает на сообщения. Почему возвращаемое значение функции start_child является ошибкой? Что пошло не так?
Вторая проблема связана с обработкой действия отмены регистрации, когда supervisor: terminate_child (Sup, Pid) возвращает {error, simple_one_for_one), хотя я ссылаюсь на Pid, а не на ChildID. Почему он ведет себя так и как я могу динамически и индивидуально завершить дочерние элементы моего супервизора?
РЕДАКТИРОВАТЬ: И target_interceptor
, и rpc_server_supervisor
контролируются rbmq_sup_sup
супервайзером:
init({Nodeid, Node}) ->
MaxRestart = 1,
MaxTime = 3600,
{ok, {{rest_for_one, MaxRestart, MaxTime},
[{server,
{target_interceptor, start_link, [target, self(), {Nodeid, Node}]},
permanent,
5000,
worker,
[target_interceptor]}]}}.
РЕДАКТИРОВАТЬ: rpc_server_supervisor вызывается в функции target_interceptor init () (Sup здесь - супервизор rbmq_sup_sup):
handle_info({start_worker_supervisor, Sup}, S = #state{}) ->
{ok, Pid} = supervisor:start_child(Sup, ?SPEC),
link(Pid),
{noreply, S#state{sup=Pid}};
-define(SPEC,
{rpc_server_sup,
{rpc_server_sup, start_link, []},
temporary,
10000,
supervisor,
[rpc_server_sup]}).