|
%% ------------------------------------------------------------------- |
|
%% |
|
%% supmon-i: periodically exorcise ghost children of supervisor processes |
|
%% |
|
%% Copyright (c) 2016 Basho Technologies, Inc. All Rights Reserved. |
|
%% |
|
%% This file is provided to you under the Apache License, |
|
%% Version 2.0 (the "License"); you may not use this file |
|
%% except in compliance with the License. You may obtain |
|
%% a copy of the License at |
|
%% |
|
%% http://www.apache.org/licenses/LICENSE-2.0 |
|
%% |
|
%% Unless required by applicable law or agreed to in writing, |
|
%% software distributed under the License is distributed on an |
|
%% "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
%% KIND, either express or implied. See the License for the |
|
%% specific language governing permissions and limitations |
|
%% under the License. |
|
%% |
|
%% ------------------------------------------------------------------- |
|
-module(supmon_i). |
|
-compile(export_all). |
|
-behaviour(gen_server). |
|
|
|
-export([start/0, start/2, stop/0]). |
|
-export([init/1, handle_call/3, handle_cast/2, handle_info/2, |
|
terminate/2, code_change/3]). |
|
-export([exorcise_ghost_children/1]). |
|
|
|
-define(SAMPLE_RATE, 5*1000). % in milliseconds |
|
|
|
|
|
-record(state, { |
|
sample_rate :: integer(), |
|
supervisor :: atom() |
|
}). |
|
|
|
start() -> |
|
start(?SAMPLE_RATE, riak_kv_get_fsm_sup). |
|
|
|
start(SampleRate, Supervisor) -> |
|
gen_server:start({local, ?MODULE}, ?MODULE, [SampleRate, Supervisor], []). |
|
|
|
stop() -> |
|
gen_server:cast(?MODULE, stop). |
|
|
|
init([SampleRate, Supervisor]) -> |
|
error_logger:info_msg("Starting supmon_i: Checking ~p every ~p ms",[Supervisor, SampleRate]), |
|
erlang:send_after(SampleRate, self(), check), |
|
{ok, #state{sample_rate=SampleRate, supervisor=Supervisor}}. |
|
|
|
handle_call(_Request, _From, State) -> |
|
{reply, ok, State}. |
|
|
|
handle_cast(stop, State) -> |
|
error_logger:info_msg("Stopping supmon_i",[]), |
|
{stop, normal, State}; |
|
handle_cast(_Msg, State) -> |
|
{noreply, State}. |
|
|
|
handle_info(check, State) -> |
|
erlang:spawn(?MODULE, exorcise_ghost_children,[State#state.supervisor]), |
|
erlang:send_after(State#state.sample_rate, self(), check), |
|
{noreply, State}; |
|
handle_info(_Info, State) -> |
|
{noreply, State}. |
|
|
|
terminate(_Reason, _State) -> |
|
ok. |
|
|
|
code_change(_OldVsn, State, _Extra) -> |
|
{ok, State}. |
|
|
|
|
|
|
|
%% internal functions |
|
exorcise_ghost_children(Supervisor) -> |
|
error_logger:info_msg("Checking and cleaning out ~p's children.", [Supervisor]), |
|
MaybeKill = fun({_,Pid,_,_}) -> |
|
case erlang:is_process_alive(Pid) of |
|
false -> |
|
?MODULE:mock_exit(Supervisor, Pid); |
|
_ -> |
|
ok |
|
end |
|
end, |
|
lists:foreach(MaybeKill, supervisor:which_children(Supervisor)), |
|
ok. |
|
|
|
|
|
mock_exit(Supervisor, Pid) -> |
|
%% error_logger:info_msg("Sending EXIT to ~p on behalf of dead PID ~p",[Supervisor, Pid]), |
|
Supervisor ! {'EXIT', Pid, normal}. |