%% -------------------------------------------------------------------
%%
%% Copyright (c) 2018 Christopher S. Meiklejohn. 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(partisan_acknowledgement_backend).
-author("Christopher S. Meiklejohn <christopher.meiklejohn@gmail.com>").
-behaviour(gen_server).
-include("partisan_logger.hrl").
%% API
-export([start_link/0,
store/2,
ack/1,
outstanding/0]).
%% gen_server callbacks
-export([init/1,
handle_call/3,
handle_cast/2,
handle_info/2,
terminate/2,
code_change/3]).
-record(state, {storage}).
%%%===================================================================
%%% API
%%%===================================================================
start_link() ->
gen_server:start_link({local, ?MODULE}, ?MODULE, [], []).
store(MessageClock, Message) ->
gen_server:call(?MODULE, {store, MessageClock, Message}, infinity).
ack(MessageClock) ->
gen_server:call(?MODULE, {ack, MessageClock}, infinity).
outstanding() ->
gen_server:call(?MODULE, outstanding, infinity).
%%%===================================================================
%%% gen_server callbacks
%%%===================================================================
%% @private
init([]) ->
Storage = ets:new(?MODULE, [named_table]),
logger:set_process_metadata(#{node => partisan:node()}),
{ok, #state{storage=Storage}}.
%% @private
handle_call({ack, MessageClock}, _From, #state{storage=Storage}=State) ->
?LOG_DEBUG(#{
description => "Acknowledgement received",
clock => MessageClock
}),
true = ets:delete(Storage, MessageClock),
{reply, ok, State};
handle_call({store, MessageClock, Message}, _From, #state{storage=Storage}=State) ->
?LOG_DEBUG(#{
description => "storing message in acknowledgement backend",
clock => MessageClock,
message => Message
}),
true = ets:insert(Storage, {MessageClock, Message}),
{reply, ok, State};
handle_call(outstanding, _From, #state{storage=Storage}=State) ->
Objects = ets:foldl(fun(X, Acc) -> Acc ++ [X] end, [], Storage),
{reply, {ok, Objects}, State};
handle_call(_Msg, _From, State) ->
{reply, ok, State}.
%% @private
handle_cast(_Msg, State) ->
{noreply, State}.
%% @private
handle_info(_Msg, State) ->
{noreply, State}.
%% @private
terminate(_Reason, _State) ->
ok.
%% @private
code_change(_OldVsn, State, _Extra) ->
{ok, State}.
%%%===================================================================
%%% Internal functions
%%%===================================================================