2017-07-20 20:22:50 +02:00
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
%%% File : mod_push_mnesia.erl
|
|
|
|
%%% Author : Holger Weiss <holger@zedat.fu-berlin.de>
|
|
|
|
%%% Purpose : Mnesia backend for Push Notifications (XEP-0357)
|
|
|
|
%%% Created : 15 Jul 2017 by Holger Weiss <holger@zedat.fu-berlin.de>
|
|
|
|
%%%
|
|
|
|
%%%
|
2018-01-05 21:18:58 +01:00
|
|
|
%%% ejabberd, Copyright (C) 2017-2018 ProcessOne
|
2017-07-20 20:22:50 +02:00
|
|
|
%%%
|
|
|
|
%%% This program is free software; you can redistribute it and/or
|
|
|
|
%%% modify it under the terms of the GNU General Public License as
|
|
|
|
%%% published by the Free Software Foundation; either version 2 of the
|
|
|
|
%%% License, or (at your option) any later version.
|
|
|
|
%%%
|
|
|
|
%%% This program is distributed in the hope that it will be useful,
|
|
|
|
%%% but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
%%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
%%% General Public License for more details.
|
|
|
|
%%%
|
|
|
|
%%% You should have received a copy of the GNU General Public License along
|
|
|
|
%%% with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
%%% 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
%%%
|
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
|
|
|
|
-module(mod_push_mnesia).
|
|
|
|
-author('holger@zedat.fu-berlin.de').
|
|
|
|
|
|
|
|
-behavior(mod_push).
|
|
|
|
|
|
|
|
%% API
|
|
|
|
-export([init/2, store_session/6, lookup_session/4, lookup_session/3,
|
|
|
|
lookup_sessions/3, lookup_sessions/2, lookup_sessions/1,
|
2017-10-27 10:46:37 +02:00
|
|
|
delete_session/3, delete_old_sessions/2, transform/1]).
|
2017-07-20 20:22:50 +02:00
|
|
|
|
|
|
|
-include_lib("stdlib/include/ms_transform.hrl").
|
|
|
|
-include("logger.hrl").
|
|
|
|
-include("xmpp.hrl").
|
2017-10-26 20:05:09 +02:00
|
|
|
-include("mod_push.hrl").
|
2017-07-20 20:22:50 +02:00
|
|
|
|
|
|
|
%%%-------------------------------------------------------------------
|
|
|
|
%%% API
|
|
|
|
%%%-------------------------------------------------------------------
|
|
|
|
init(_Host, _Opts) ->
|
|
|
|
ejabberd_mnesia:create(?MODULE, push_session,
|
|
|
|
[{disc_only_copies, [node()]},
|
|
|
|
{type, bag},
|
|
|
|
{attributes, record_info(fields, push_session)}]).
|
|
|
|
|
|
|
|
store_session(LUser, LServer, TS, PushJID, Node, XData) ->
|
|
|
|
US = {LUser, LServer},
|
|
|
|
PushLJID = jid:tolower(PushJID),
|
|
|
|
MaxSessions = ejabberd_sm:get_max_user_sessions(LUser, LServer),
|
|
|
|
F = fun() ->
|
|
|
|
if is_integer(MaxSessions) ->
|
|
|
|
enforce_max_sessions(US, MaxSessions - 1);
|
|
|
|
MaxSessions == infinity ->
|
|
|
|
ok
|
|
|
|
end,
|
|
|
|
mnesia:write(#push_session{us = US,
|
|
|
|
timestamp = TS,
|
|
|
|
service = PushLJID,
|
|
|
|
node = Node,
|
2017-10-27 10:46:37 +02:00
|
|
|
xml = encode_xdata(XData)})
|
2017-07-20 20:22:50 +02:00
|
|
|
end,
|
|
|
|
case mnesia:transaction(F) of
|
|
|
|
{atomic, ok} ->
|
|
|
|
{ok, {TS, PushLJID, Node, XData}};
|
|
|
|
{aborted, E} ->
|
|
|
|
?ERROR_MSG("Cannot store push session for ~s@~s: ~p",
|
|
|
|
[LUser, LServer, E]),
|
2017-10-26 19:11:43 +02:00
|
|
|
{error, db_failure}
|
2017-07-20 20:22:50 +02:00
|
|
|
end.
|
|
|
|
|
|
|
|
lookup_session(LUser, LServer, PushJID, Node) ->
|
|
|
|
PushLJID = jid:tolower(PushJID),
|
|
|
|
MatchSpec = ets:fun2ms(
|
|
|
|
fun(#push_session{us = {U, S}, service = P, node = N} = Rec)
|
|
|
|
when U == LUser,
|
|
|
|
S == LServer,
|
|
|
|
P == PushLJID,
|
|
|
|
N == Node ->
|
|
|
|
Rec
|
|
|
|
end),
|
|
|
|
case mnesia:dirty_select(push_session, MatchSpec) of
|
2017-10-27 10:46:37 +02:00
|
|
|
[#push_session{timestamp = TS, xml = El}] ->
|
|
|
|
{ok, {TS, PushLJID, Node, decode_xdata(El)}};
|
2017-10-26 19:11:43 +02:00
|
|
|
[] ->
|
2017-07-20 20:22:50 +02:00
|
|
|
?DEBUG("No push session found for ~s@~s (~p, ~s)",
|
|
|
|
[LUser, LServer, PushJID, Node]),
|
2017-10-27 09:55:48 +02:00
|
|
|
{error, notfound}
|
2017-07-20 20:22:50 +02:00
|
|
|
end.
|
|
|
|
|
|
|
|
lookup_session(LUser, LServer, TS) ->
|
|
|
|
MatchSpec = ets:fun2ms(
|
|
|
|
fun(#push_session{us = {U, S}, timestamp = T} = Rec)
|
|
|
|
when U == LUser,
|
|
|
|
S == LServer,
|
|
|
|
T == TS ->
|
|
|
|
Rec
|
|
|
|
end),
|
|
|
|
case mnesia:dirty_select(push_session, MatchSpec) of
|
2017-10-27 10:46:37 +02:00
|
|
|
[#push_session{service = PushLJID, node = Node, xml = El}] ->
|
|
|
|
{ok, {TS, PushLJID, Node, decode_xdata(El)}};
|
2017-10-26 19:11:43 +02:00
|
|
|
[] ->
|
2017-07-20 20:22:50 +02:00
|
|
|
?DEBUG("No push session found for ~s@~s (~p)",
|
|
|
|
[LUser, LServer, TS]),
|
2017-10-27 09:55:48 +02:00
|
|
|
{error, notfound}
|
2017-07-20 20:22:50 +02:00
|
|
|
end.
|
|
|
|
|
|
|
|
lookup_sessions(LUser, LServer, PushJID) ->
|
|
|
|
PushLJID = jid:tolower(PushJID),
|
|
|
|
MatchSpec = ets:fun2ms(
|
2017-10-27 09:55:48 +02:00
|
|
|
fun(#push_session{us = {U, S}, service = P,
|
|
|
|
node = Node, timestamp = TS,
|
2017-10-27 10:46:37 +02:00
|
|
|
xml = El} = Rec)
|
2017-07-20 20:22:50 +02:00
|
|
|
when U == LUser,
|
|
|
|
S == LServer,
|
|
|
|
P == PushLJID ->
|
2017-10-27 10:46:37 +02:00
|
|
|
Rec
|
2017-07-20 20:22:50 +02:00
|
|
|
end),
|
2017-10-27 10:46:37 +02:00
|
|
|
Records = mnesia:dirty_select(push_session, MatchSpec),
|
|
|
|
{ok, records_to_sessions(Records)}.
|
2017-07-20 20:22:50 +02:00
|
|
|
|
|
|
|
lookup_sessions(LUser, LServer) ->
|
|
|
|
Records = mnesia:dirty_read(push_session, {LUser, LServer}),
|
2017-10-27 10:46:37 +02:00
|
|
|
{ok, records_to_sessions(Records)}.
|
2017-07-20 20:22:50 +02:00
|
|
|
|
|
|
|
lookup_sessions(LServer) ->
|
|
|
|
MatchSpec = ets:fun2ms(
|
|
|
|
fun(#push_session{us = {_U, S},
|
|
|
|
timestamp = TS,
|
|
|
|
service = PushLJID,
|
|
|
|
node = Node,
|
2017-10-27 10:46:37 +02:00
|
|
|
xml = El})
|
2017-07-20 20:22:50 +02:00
|
|
|
when S == LServer ->
|
2017-10-27 10:46:37 +02:00
|
|
|
{TS, PushLJID, Node, El}
|
2017-07-20 20:22:50 +02:00
|
|
|
end),
|
2017-10-27 10:46:37 +02:00
|
|
|
Records = mnesia:dirty_select(push_session, MatchSpec),
|
|
|
|
{ok, records_to_sessions(Records)}.
|
2017-07-20 20:22:50 +02:00
|
|
|
|
|
|
|
delete_session(LUser, LServer, TS) ->
|
|
|
|
MatchSpec = ets:fun2ms(
|
|
|
|
fun(#push_session{us = {U, S}, timestamp = T} = Rec)
|
|
|
|
when U == LUser,
|
|
|
|
S == LServer,
|
|
|
|
T == TS ->
|
|
|
|
Rec
|
|
|
|
end),
|
|
|
|
F = fun() ->
|
|
|
|
Recs = mnesia:select(push_session, MatchSpec),
|
|
|
|
lists:foreach(fun mnesia:delete_object/1, Recs)
|
|
|
|
end,
|
|
|
|
case mnesia:transaction(F) of
|
|
|
|
{atomic, ok} ->
|
|
|
|
ok;
|
|
|
|
{aborted, E} ->
|
2017-09-22 23:36:51 +02:00
|
|
|
?ERROR_MSG("Cannot delete push session of ~s@~s: ~p",
|
2017-07-20 20:22:50 +02:00
|
|
|
[LUser, LServer, E]),
|
2017-10-26 19:11:43 +02:00
|
|
|
{error, db_failure}
|
2017-07-20 20:22:50 +02:00
|
|
|
end.
|
|
|
|
|
|
|
|
delete_old_sessions(_LServer, Time) ->
|
|
|
|
DelIfOld = fun(#push_session{timestamp = T} = Rec, ok) when T < Time ->
|
|
|
|
mnesia:delete_object(Rec);
|
|
|
|
(_Rec, ok) ->
|
|
|
|
ok
|
|
|
|
end,
|
|
|
|
F = fun() ->
|
|
|
|
mnesia:foldl(DelIfOld, ok, push_session)
|
|
|
|
end,
|
|
|
|
case mnesia:transaction(F) of
|
|
|
|
{atomic, ok} ->
|
|
|
|
ok;
|
|
|
|
{aborted, E} ->
|
|
|
|
?ERROR_MSG("Cannot delete old push sessions: ~p", [E]),
|
2017-10-26 19:11:43 +02:00
|
|
|
{error, db_failure}
|
2017-07-20 20:22:50 +02:00
|
|
|
end.
|
|
|
|
|
2017-10-27 10:46:37 +02:00
|
|
|
transform({push_session, US, TS, Service, Node, XData}) ->
|
|
|
|
?INFO_MSG("Transforming push_session Mnesia table", []),
|
|
|
|
#push_session{us = US, timestamp = TS, service = Service,
|
|
|
|
node = Node, xml = encode_xdata(XData)}.
|
|
|
|
|
2017-07-20 20:22:50 +02:00
|
|
|
%%--------------------------------------------------------------------
|
|
|
|
%% Internal functions.
|
|
|
|
%%--------------------------------------------------------------------
|
|
|
|
-spec enforce_max_sessions({binary(), binary()}, non_neg_integer()) -> ok.
|
|
|
|
enforce_max_sessions({U, S} = US, Max) ->
|
|
|
|
Recs = mnesia:wread({push_session, US}),
|
|
|
|
NumRecs = length(Recs),
|
|
|
|
if NumRecs > Max ->
|
|
|
|
NumOldRecs = NumRecs - Max,
|
|
|
|
Recs1 = lists:keysort(#push_session.timestamp, Recs),
|
|
|
|
Recs2 = lists:reverse(Recs1),
|
|
|
|
OldRecs = lists:sublist(Recs2, Max + 1, NumOldRecs),
|
|
|
|
?INFO_MSG("Disabling ~B old push session(s) of ~s@~s",
|
|
|
|
[NumOldRecs, U, S]),
|
|
|
|
lists:foreach(fun(Rec) -> mnesia:delete_object(Rec) end, OldRecs);
|
|
|
|
true ->
|
|
|
|
ok
|
|
|
|
end.
|
2017-10-27 10:46:37 +02:00
|
|
|
|
|
|
|
decode_xdata(undefined) ->
|
|
|
|
undefined;
|
|
|
|
decode_xdata(El) ->
|
|
|
|
xmpp:decode(El).
|
|
|
|
|
|
|
|
encode_xdata(undefined) ->
|
|
|
|
undefined;
|
|
|
|
encode_xdata(XData) ->
|
|
|
|
xmpp:encode(XData).
|
|
|
|
|
|
|
|
records_to_sessions(Records) ->
|
|
|
|
[{TS, PushLJID, Node, decode_xdata(El)}
|
|
|
|
|| #push_session{timestamp = TS,
|
|
|
|
service = PushLJID,
|
|
|
|
node = Node,
|
|
|
|
xml = El} <- Records].
|