26
1
mirror of https://github.com/processone/ejabberd.git synced 2025-01-01 17:53:00 +01:00

Disable LRU caching algorithm for LDAP shared rosters

This commit is contained in:
Evgeniy Khramtsov 2010-11-05 17:01:18 +09:00
parent c4289095e0
commit 60b36beda8
2 changed files with 97 additions and 54 deletions

View File

@ -53,6 +53,7 @@
miss = 0, miss = 0,
procs_num, procs_num,
cache_missed, cache_missed,
lru,
shrink_size}). shrink_size}).
-define(PROCNAME, ?MODULE). -define(PROCNAME, ?MODULE).
@ -62,6 +63,7 @@
-define(MAX_SIZE, 1000). -define(MAX_SIZE, 1000).
-define(WARN, true). -define(WARN, true).
-define(CACHE_MISSED, true). -define(CACHE_MISSED, true).
-define(LRU, true).
-define(LIFETIME, 600). %% 10 minutes -define(LIFETIME, 600). %% 10 minutes
%%==================================================================== %%====================================================================
@ -109,7 +111,7 @@ delete(Tab, Key, F) ->
dirty_delete(Tab, Key, F) -> dirty_delete(Tab, Key, F) ->
F(), F(),
?GEN_SERVER:call( ?GEN_SERVER:call(
get_proc_by_hash(Tab, Key), {dirty_delete, Key}, ?CALL_TIMEOUT). get_proc_by_hash(Tab, Key), {cache_delete, Key}, ?CALL_TIMEOUT).
lookup(Tab, Key, F) -> lookup(Tab, Key, F) ->
?GEN_SERVER:call( ?GEN_SERVER:call(
@ -117,20 +119,21 @@ lookup(Tab, Key, F) ->
dirty_lookup(Tab, Key, F) -> dirty_lookup(Tab, Key, F) ->
Proc = get_proc_by_hash(Tab, Key), Proc = get_proc_by_hash(Tab, Key),
case ?GEN_SERVER:call(Proc, {dirty_lookup, Key}, ?CALL_TIMEOUT) of case ?GEN_SERVER:call(Proc, {cache_lookup, Key}, ?CALL_TIMEOUT) of
{ok, '$cached_mismatch'} -> {ok, '$cached_mismatch'} ->
error; error;
{ok, Val} -> {ok, Val} ->
{ok, Val}; {ok, Val};
_ -> _ ->
case F() of {Result, NewVal} = case F() of
{ok, Val} -> {ok, Val} ->
?GEN_SERVER:call( {{ok, Val}, Val};
Proc, {dirty_insert, Key, Val}, ?CALL_TIMEOUT),
{ok, Val};
_ -> _ ->
error {error, '$cached_mismatch'}
end end,
?GEN_SERVER:call(
Proc, {cache_insert, Key, NewVal}, ?CALL_TIMEOUT),
Result
end. end.
insert(Tab, Key, Val, F) -> insert(Tab, Key, Val, F) ->
@ -140,7 +143,7 @@ insert(Tab, Key, Val, F) ->
dirty_insert(Tab, Key, Val, F) -> dirty_insert(Tab, Key, Val, F) ->
F(), F(),
?GEN_SERVER:call( ?GEN_SERVER:call(
get_proc_by_hash(Tab, Key), {dirty_insert, Key, Val}, ?CALL_TIMEOUT). get_proc_by_hash(Tab, Key), {cache_insert, Key, Val}, ?CALL_TIMEOUT).
info(Tab, Info) -> info(Tab, Info) ->
case lists:map( case lists:map(
@ -187,8 +190,9 @@ init([Tab, Opts, N, Pid]) ->
{ok, do_setopts(State, Opts)}. {ok, do_setopts(State, Opts)}.
handle_call({lookup, Key, F}, _From, #state{tab = T} = State) -> handle_call({lookup, Key, F}, _From, #state{tab = T} = State) ->
CleanPrio = clean_priority(State#state.life_time),
case treap:lookup(Key, T) of case treap:lookup(Key, T) of
{ok, _Prio, Val} -> {ok, Prio, Val} when (State#state.lru == true) or (Prio =< CleanPrio) ->
Hits = State#state.hits, Hits = State#state.hits,
NewState = treap_update(Key, Val, State#state{hits = Hits + 1}), NewState = treap_update(Key, Val, State#state{hits = Hits + 1}),
case Val of case Val of
@ -217,50 +221,36 @@ handle_call({lookup, Key, F}, _From, #state{tab = T} = State) ->
end end
end end
end; end;
handle_call({dirty_lookup, Key}, _From, #state{tab = T} = State) -> handle_call({cache_lookup, Key}, _From, #state{tab = T} = State) ->
CleanPrio = clean_priority(State#state.life_time),
case treap:lookup(Key, T) of case treap:lookup(Key, T) of
{ok, _Prio, Val} -> {ok, Prio, Val} when (State#state.lru == true) or (Prio =< CleanPrio) ->
Hits = State#state.hits, Hits = State#state.hits,
NewState = treap_update(Key, Val, State#state{hits = Hits + 1}), NewState = treap_update(Key, Val, State#state{hits = Hits + 1}),
{reply, {ok, Val}, NewState}; {reply, {ok, Val}, NewState};
_ -> _ ->
Miss = State#state.miss, Miss = State#state.miss,
NewState = State#state{miss = Miss + 1}, NewState = State#state{miss = Miss + 1},
if State#state.cache_missed ->
{reply, error,
treap_insert(Key, '$cached_mismatch', NewState)};
true ->
{reply, error, NewState} {reply, error, NewState}
end
end; end;
handle_call({insert, Key, Val, F}, _From, #state{tab = T} = State) -> handle_call({insert, Key, Val, F}, _From, #state{tab = T} = State) ->
case treap:lookup(Key, T) of case treap:lookup(Key, T) of
{ok, _Prio, Val} -> {ok, _Prio, Val} ->
{reply, ok, State}; {reply, ok, treap_update(Key, Val, State)};
Res -> _ ->
case catch F() of case catch F() of
{'EXIT', Reason} -> {'EXIT', Reason} ->
print_error(insert, [Key, Val], Reason, State), print_error(insert, [Key, Val], Reason, State),
{reply, ok, State}; {reply, ok, State};
_ -> _ ->
NewState = case Res of {reply, ok, treap_insert(Key, Val, State)}
{ok, _, _} ->
treap_update(Key, Val, State);
_ ->
treap_insert(Key, Val, State)
end,
{reply, ok, NewState}
end end
end; end;
handle_call({dirty_insert, Key, Val}, _From, #state{tab = T} = State) -> handle_call({cache_insert, _, '$cached_mismatch'}, _From,
case treap:lookup(Key, T) of #state{cache_missed = false} = State) ->
{ok, _Prio, Val} ->
{reply, ok, State}; {reply, ok, State};
{ok, _, _} -> handle_call({cache_insert, Key, Val}, _From, State) ->
{reply, ok, treap_update(Key, Val, State)}; {reply, ok, treap_insert(Key, Val, State)};
_ ->
{reply, ok, treap_insert(Key, Val, State)}
end;
handle_call({delete, Key, F}, _From, State) -> handle_call({delete, Key, F}, _From, State) ->
NewState = treap_delete(Key, State), NewState = treap_delete(Key, State),
case catch F() of case catch F() of
@ -270,7 +260,7 @@ handle_call({delete, Key, F}, _From, State) ->
ok ok
end, end,
{reply, ok, NewState}; {reply, ok, NewState};
handle_call({dirty_delete, Key}, _From, State) -> handle_call({cache_delete, Key}, _From, State) ->
NewState = treap_delete(Key, State), NewState = treap_delete(Key, State),
{reply, ok, NewState}; {reply, ok, NewState};
handle_call({info, Info}, _From, State) -> handle_call({info, Info}, _From, State) ->
@ -288,6 +278,7 @@ handle_call({info, Info}, _From, State) ->
{hits, State#state.hits}, {hits, State#state.hits},
{miss, State#state.miss}, {miss, State#state.miss},
{cache_missed, State#state.cache_missed}, {cache_missed, State#state.cache_missed},
{lru, State#state.lru},
{warn, State#state.warn}]; {warn, State#state.warn}];
_ -> _ ->
badarg badarg
@ -372,10 +363,20 @@ do_setopts(#state{procs_num = N} = State, Opts) ->
_ -> _ ->
?CACHE_MISSED ?CACHE_MISSED
end, end,
LRU = case proplists:get_value(
lru, Opts, State#state.lru) of
false ->
false;
true ->
true;
_ ->
?LRU
end,
State#state{max_size = MaxSize, State#state{max_size = MaxSize,
warn = Warn, warn = Warn,
life_time = LifeTime, life_time = LifeTime,
cache_missed = CacheMissed, cache_missed = CacheMissed,
lru = LRU,
shrink_size = ShrinkSize}. shrink_size = ShrinkSize}.
get_proc_num() -> get_proc_num() ->
@ -396,15 +397,36 @@ now_priority() ->
{MSec, Sec, USec} = now(), {MSec, Sec, USec} = now(),
-((MSec*1000000 + Sec)*1000000 + USec). -((MSec*1000000 + Sec)*1000000 + USec).
treap_update(Key, Val, #state{tab = T} = State) -> clean_priority(LifeTime) ->
if is_integer(LifeTime) ->
now_priority() + LifeTime;
true ->
unlimited
end.
treap_update(Key, Val, #state{tab = T, lru = LRU} = State) ->
if LRU ->
Priority = now_priority(), Priority = now_priority(),
NewT = treap:insert(Key, Priority, Val, T), NewT = treap:insert(Key, Priority, Val, T),
State#state{tab = NewT}. State#state{tab = NewT};
true ->
State
end.
treap_insert(Key, Val, State) -> treap_insert(Key, Val, State) ->
State1 = clean_treap(State), State1 = clean_treap(State),
#state{size = Size} = State2 = shrink_treap(State1), #state{size = Size} = State2 = shrink_treap(State1),
treap_update(Key, Val, State2#state{size = Size+1}). T = State2#state.tab,
case treap:lookup(Key, T) of
{ok, _, Val} ->
treap_update(Key, Val, State2);
{ok, _, _} ->
NewT = treap:insert(Key, now_priority(), Val, T),
State2#state{tab = NewT};
_ ->
NewT = treap:insert(Key, now_priority(), Val, T),
State2#state{tab = NewT, size = Size+1}
end.
treap_delete(Key, #state{tab = T, size = Size} = State) -> treap_delete(Key, #state{tab = T, size = Size} = State) ->
case treap:lookup(Key, T) of case treap:lookup(Key, T) of
@ -504,7 +526,7 @@ test() ->
check([{"key", "value"}]), check([{"key", "value"}]),
{ok, "value"} = ?lookup(test_tbl, "key", fun() -> error end), {ok, "value"} = ?lookup(test_tbl, "key", fun() -> error end),
check([{"key", "value"}]), check([{"key", "value"}]),
io:format("** waiting for ~p seconds to check if cleaner works fine...~n", io:format("** waiting for ~p seconds to check if LRU works fine...~n",
[LifeTime+1]), [LifeTime+1]),
timer:sleep(timer:seconds(LifeTime+1)), timer:sleep(timer:seconds(LifeTime+1)),
ok = ?insert(test_tbl, "key1", "value1", fun() -> ok end), ok = ?insert(test_tbl, "key1", "value1", fun() -> ok end),
@ -531,13 +553,32 @@ test1() ->
end, lists:seq(1, MaxSize*get_proc_num())), end, lists:seq(1, MaxSize*get_proc_num())),
{ok, MaxSize} = info(test_tbl, size), {ok, MaxSize} = info(test_tbl, size),
delete(test_tbl), delete(test_tbl),
io:format("** testing speed, this may take a while...~n"), test2().
test2(1000),
test2(10000),
test2(100000),
test2(1000000).
test2(Iter) -> test2() ->
LifeTime = 2,
ok = new(test_tbl, [{life_time, LifeTime},
{max_size, unlimited},
{lru, false}]),
check([]),
ok = ?insert(test_tbl, "key", "value", fun() -> ok end),
{ok, "value"} = ?lookup(test_tbl, "key", fun() -> error end),
check([{"key", "value"}]),
io:format("** waiting for ~p seconds to check if non-LRU works fine...~n",
[LifeTime+1]),
timer:sleep(timer:seconds(LifeTime+1)),
error = ?lookup(test_tbl, "key", fun() -> error end),
check([{"key", '$cached_mismatch'}]),
ok = ?insert(test_tbl, "key", "value1", fun() -> ok end),
check([{"key", "value1"}]),
delete(test_tbl),
io:format("** testing speed, this may take a while...~n"),
test3(1000),
test3(10000),
test3(100000),
test3(1000000).
test3(Iter) ->
ok = new(test_tbl, [{max_size, unlimited}, {life_time, unlimited}]), ok = new(test_tbl, [{max_size, unlimited}, {life_time, unlimited}]),
L = lists:seq(1, Iter), L = lists:seq(1, Iter),
T1 = now(), T1 = now(),

View File

@ -223,9 +223,11 @@ init([Host, Opts]) ->
State = parse_options(Host, Opts), State = parse_options(Host, Opts),
cache_tab:new(shared_roster_ldap_user, cache_tab:new(shared_roster_ldap_user,
[{max_size, State#state.user_cache_size}, [{max_size, State#state.user_cache_size},
{lru, false}, % We don't need LRU algorithm
{life_time, State#state.user_cache_validity}]), {life_time, State#state.user_cache_validity}]),
cache_tab:new(shared_roster_ldap_group, cache_tab:new(shared_roster_ldap_group,
[{max_size, State#state.group_cache_size}, [{max_size, State#state.group_cache_size},
{lru, false}, % We don't need LRU algorithm
{life_time, State#state.group_cache_validity}]), {life_time, State#state.group_cache_validity}]),
ejabberd_hooks:add(roster_get, Host, ejabberd_hooks:add(roster_get, Host,
?MODULE, get_user_roster, 70), ?MODULE, get_user_roster, 70),