2016-04-15 12:44:33 +02:00
|
|
|
%%%-------------------------------------------------------------------
|
2016-12-27 10:44:07 +01:00
|
|
|
%%% File : mod_offline_mnesia.erl
|
|
|
|
%%% Author : Evgeny Khramtsov <ekhramtsov@process-one.net>
|
2016-04-15 12:44:33 +02:00
|
|
|
%%% Created : 15 Apr 2016 by Evgeny Khramtsov <ekhramtsov@process-one.net>
|
2016-12-27 10:44:07 +01:00
|
|
|
%%%
|
|
|
|
%%%
|
2024-01-22 16:40:01 +01:00
|
|
|
%%% ejabberd, Copyright (C) 2002-2024 ProcessOne
|
2016-12-27 10:44:07 +01: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.
|
|
|
|
%%%
|
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
|
2016-04-15 12:44:33 +02:00
|
|
|
-module(mod_offline_mnesia).
|
|
|
|
|
|
|
|
-behaviour(mod_offline).
|
|
|
|
|
2017-05-21 22:21:13 +02:00
|
|
|
-export([init/2, store_message/1, pop_messages/2, remove_expired_messages/1,
|
2016-04-15 12:44:33 +02:00
|
|
|
remove_old_messages/2, remove_user/2, read_message_headers/2,
|
|
|
|
read_message/3, remove_message/3, read_all_messages/2,
|
2022-05-02 15:05:55 +02:00
|
|
|
remove_all_messages/2, count_messages/2, import/1,
|
|
|
|
remove_old_messages_batch/4]).
|
2017-04-23 15:37:58 +02:00
|
|
|
-export([need_transform/1, transform/1]).
|
2016-04-15 12:44:33 +02:00
|
|
|
|
2020-09-03 13:45:57 +02:00
|
|
|
-include_lib("xmpp/include/xmpp.hrl").
|
2016-04-15 12:44:33 +02:00
|
|
|
-include("mod_offline.hrl").
|
|
|
|
-include("logger.hrl").
|
|
|
|
|
|
|
|
%%%===================================================================
|
|
|
|
%%% API
|
|
|
|
%%%===================================================================
|
|
|
|
init(_Host, _Opts) ->
|
2016-11-30 11:09:17 +01:00
|
|
|
ejabberd_mnesia:create(?MODULE, offline_msg,
|
2017-04-23 15:37:58 +02:00
|
|
|
[{disc_only_copies, [node()]}, {type, bag},
|
|
|
|
{attributes, record_info(fields, offline_msg)}]).
|
2016-04-15 12:44:33 +02:00
|
|
|
|
2017-05-21 22:21:13 +02:00
|
|
|
store_message(#offline_msg{packet = Pkt} = OffMsg) ->
|
|
|
|
El = xmpp:encode(Pkt),
|
|
|
|
mnesia:dirty_write(OffMsg#offline_msg{packet = El}).
|
2016-04-15 12:44:33 +02:00
|
|
|
|
|
|
|
pop_messages(LUser, LServer) ->
|
|
|
|
US = {LUser, LServer},
|
|
|
|
F = fun () ->
|
|
|
|
Rs = mnesia:wread({offline_msg, US}),
|
|
|
|
mnesia:delete({offline_msg, US}),
|
|
|
|
Rs
|
|
|
|
end,
|
|
|
|
case mnesia:transaction(F) of
|
|
|
|
{atomic, L} ->
|
|
|
|
{ok, lists:keysort(#offline_msg.timestamp, L)};
|
|
|
|
{aborted, Reason} ->
|
|
|
|
{error, Reason}
|
|
|
|
end.
|
|
|
|
|
|
|
|
remove_expired_messages(_LServer) ->
|
2019-02-27 09:56:20 +01:00
|
|
|
TimeStamp = erlang:timestamp(),
|
2016-04-15 12:44:33 +02:00
|
|
|
F = fun () ->
|
|
|
|
mnesia:write_lock_table(offline_msg),
|
|
|
|
mnesia:foldl(fun (Rec, _Acc) ->
|
|
|
|
case Rec#offline_msg.expire of
|
|
|
|
never -> ok;
|
|
|
|
TS ->
|
|
|
|
if TS < TimeStamp ->
|
|
|
|
mnesia:delete_object(Rec);
|
|
|
|
true -> ok
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end,
|
|
|
|
ok, offline_msg)
|
|
|
|
end,
|
|
|
|
mnesia:transaction(F).
|
|
|
|
|
|
|
|
remove_old_messages(Days, _LServer) ->
|
2019-02-27 09:56:20 +01:00
|
|
|
S = erlang:system_time(second) - 60 * 60 * 24 * Days,
|
2016-04-15 12:44:33 +02:00
|
|
|
MegaSecs1 = S div 1000000,
|
|
|
|
Secs1 = S rem 1000000,
|
|
|
|
TimeStamp = {MegaSecs1, Secs1, 0},
|
|
|
|
F = fun () ->
|
|
|
|
mnesia:write_lock_table(offline_msg),
|
|
|
|
mnesia:foldl(fun (#offline_msg{timestamp = TS} = Rec,
|
|
|
|
_Acc)
|
|
|
|
when TS < TimeStamp ->
|
|
|
|
mnesia:delete_object(Rec);
|
|
|
|
(_Rec, _Acc) -> ok
|
|
|
|
end,
|
|
|
|
ok, offline_msg)
|
|
|
|
end,
|
|
|
|
mnesia:transaction(F).
|
|
|
|
|
2022-05-02 15:05:55 +02:00
|
|
|
delete_batch('$end_of_table', _LServer, _TS, Num) ->
|
|
|
|
{Num, '$end_of_table'};
|
|
|
|
delete_batch(LastUS, _LServer, _TS, 0) ->
|
|
|
|
{0, LastUS};
|
|
|
|
delete_batch(none, LServer, TS, Num) ->
|
|
|
|
delete_batch(mnesia:first(offline_msg), LServer, TS, Num);
|
|
|
|
delete_batch({_, LServer2} = LastUS, LServer, TS, Num) when LServer /= LServer2 ->
|
|
|
|
delete_batch(mnesia:next(offline_msg, LastUS), LServer, TS, Num);
|
|
|
|
delete_batch(LastUS, LServer, TS, Num) ->
|
|
|
|
Left =
|
|
|
|
lists:foldl(
|
|
|
|
fun(_, 0) ->
|
|
|
|
0;
|
|
|
|
(#offline_msg{timestamp = TS2} = O, Num2) when TS2 < TS ->
|
|
|
|
mnesia:delete_object(O),
|
|
|
|
Num2 - 1;
|
|
|
|
(_, Num2) ->
|
|
|
|
Num2
|
|
|
|
end, Num, mnesia:wread({offline_msg, LastUS})),
|
|
|
|
case Left of
|
|
|
|
0 -> {0, LastUS};
|
|
|
|
_ -> delete_batch(mnesia:next(offline_msg, LastUS), LServer, TS, Left)
|
|
|
|
end.
|
|
|
|
|
|
|
|
remove_old_messages_batch(LServer, Days, Batch, LastUS) ->
|
|
|
|
S = erlang:system_time(second) - 60 * 60 * 24 * Days,
|
|
|
|
MegaSecs1 = S div 1000000,
|
|
|
|
Secs1 = S rem 1000000,
|
|
|
|
TimeStamp = {MegaSecs1, Secs1, 0},
|
|
|
|
R = mnesia:transaction(
|
|
|
|
fun() ->
|
|
|
|
{Num, NextUS} = delete_batch(LastUS, LServer, TimeStamp, Batch),
|
|
|
|
{Batch - Num, NextUS}
|
|
|
|
end),
|
|
|
|
case R of
|
|
|
|
{atomic, {Num, State}} ->
|
|
|
|
{ok, State, Num};
|
|
|
|
{aborted, Err} ->
|
|
|
|
{error, Err}
|
|
|
|
end.
|
|
|
|
|
2016-04-15 12:44:33 +02:00
|
|
|
remove_user(LUser, LServer) ->
|
|
|
|
US = {LUser, LServer},
|
|
|
|
F = fun () -> mnesia:delete({offline_msg, US}) end,
|
|
|
|
mnesia:transaction(F).
|
|
|
|
|
|
|
|
read_message_headers(LUser, LServer) ->
|
|
|
|
Msgs = mnesia:dirty_read({offline_msg, {LUser, LServer}}),
|
|
|
|
Hdrs = lists:map(
|
|
|
|
fun(#offline_msg{from = From, to = To, packet = Pkt,
|
|
|
|
timestamp = TS}) ->
|
|
|
|
Seq = now_to_integer(TS),
|
2016-11-13 08:44:53 +01:00
|
|
|
{Seq, From, To, TS, Pkt}
|
2016-04-15 12:44:33 +02:00
|
|
|
end, Msgs),
|
|
|
|
lists:keysort(1, Hdrs).
|
|
|
|
|
|
|
|
read_message(LUser, LServer, I) ->
|
|
|
|
US = {LUser, LServer},
|
|
|
|
TS = integer_to_now(I),
|
|
|
|
case mnesia:dirty_match_object(
|
|
|
|
offline_msg, #offline_msg{us = US, timestamp = TS, _ = '_'}) of
|
|
|
|
[Msg|_] ->
|
|
|
|
{ok, Msg};
|
|
|
|
_ ->
|
|
|
|
error
|
|
|
|
end.
|
|
|
|
|
|
|
|
remove_message(LUser, LServer, I) ->
|
|
|
|
US = {LUser, LServer},
|
|
|
|
TS = integer_to_now(I),
|
2016-11-08 13:15:19 +01:00
|
|
|
case mnesia:dirty_match_object(
|
|
|
|
offline_msg, #offline_msg{us = US, timestamp = TS, _ = '_'}) of
|
|
|
|
[] ->
|
|
|
|
{error, notfound};
|
|
|
|
Msgs ->
|
|
|
|
lists:foreach(
|
|
|
|
fun(Msg) ->
|
|
|
|
mnesia:dirty_delete_object(Msg)
|
|
|
|
end, Msgs)
|
|
|
|
end.
|
2016-04-15 12:44:33 +02:00
|
|
|
|
|
|
|
read_all_messages(LUser, LServer) ->
|
|
|
|
US = {LUser, LServer},
|
|
|
|
lists:keysort(#offline_msg.timestamp,
|
|
|
|
mnesia:dirty_read({offline_msg, US})).
|
|
|
|
|
|
|
|
remove_all_messages(LUser, LServer) ->
|
|
|
|
US = {LUser, LServer},
|
|
|
|
F = fun () ->
|
|
|
|
mnesia:write_lock_table(offline_msg),
|
|
|
|
lists:foreach(fun (Msg) -> mnesia:delete_object(Msg) end,
|
|
|
|
mnesia:dirty_read({offline_msg, US}))
|
|
|
|
end,
|
|
|
|
mnesia:transaction(F).
|
|
|
|
|
|
|
|
count_messages(LUser, LServer) ->
|
|
|
|
US = {LUser, LServer},
|
|
|
|
F = fun () ->
|
|
|
|
count_mnesia_records(US)
|
|
|
|
end,
|
2019-06-30 20:14:37 +02:00
|
|
|
{cache, case mnesia:async_dirty(F) of
|
|
|
|
I when is_integer(I) -> I;
|
|
|
|
_ -> 0
|
|
|
|
end}.
|
2016-04-15 12:44:33 +02:00
|
|
|
|
2016-11-22 14:48:01 +01:00
|
|
|
import(#offline_msg{} = Msg) ->
|
2016-04-15 12:44:33 +02:00
|
|
|
mnesia:dirty_write(Msg).
|
|
|
|
|
2019-06-14 11:33:26 +02:00
|
|
|
need_transform({offline_msg, {U, S}, _, _, _, _, _})
|
|
|
|
when is_list(U) orelse is_list(S) ->
|
2017-04-23 15:37:58 +02:00
|
|
|
?INFO_MSG("Mnesia table 'offline_msg' will be converted to binary", []),
|
|
|
|
true;
|
2019-08-16 18:34:42 +02:00
|
|
|
need_transform({offline_msg, _, _, _, _, _, _, _}) ->
|
|
|
|
true;
|
2017-04-23 15:37:58 +02:00
|
|
|
need_transform(_) ->
|
|
|
|
false.
|
|
|
|
|
2019-08-16 18:34:42 +02:00
|
|
|
transform({offline_msg, {U, S}, Timestamp, Expire, From, To, _, Packet}) ->
|
|
|
|
#offline_msg{us = {U, S}, timestamp = Timestamp, expire = Expire,
|
|
|
|
from = From ,to = To, packet = Packet};
|
2017-04-23 15:37:58 +02:00
|
|
|
transform(#offline_msg{us = {U, S}, from = From, to = To,
|
|
|
|
packet = El} = R) ->
|
|
|
|
R#offline_msg{us = {iolist_to_binary(U), iolist_to_binary(S)},
|
|
|
|
from = jid_to_binary(From),
|
|
|
|
to = jid_to_binary(To),
|
|
|
|
packet = fxml:to_xmlel(El)}.
|
|
|
|
|
2016-04-15 12:44:33 +02:00
|
|
|
%%%===================================================================
|
|
|
|
%%% Internal functions
|
|
|
|
%%%===================================================================
|
|
|
|
%% Return the number of records matching a given match expression.
|
|
|
|
%% This function is intended to be used inside a Mnesia transaction.
|
|
|
|
%% The count has been written to use the fewest possible memory by
|
|
|
|
%% getting the record by small increment and by using continuation.
|
|
|
|
-define(BATCHSIZE, 100).
|
|
|
|
|
|
|
|
count_mnesia_records(US) ->
|
|
|
|
MatchExpression = #offline_msg{us = US, _ = '_'},
|
|
|
|
case mnesia:select(offline_msg, [{MatchExpression, [], [[]]}],
|
|
|
|
?BATCHSIZE, read) of
|
|
|
|
{Result, Cont} ->
|
|
|
|
Count = length(Result),
|
|
|
|
count_records_cont(Cont, Count);
|
|
|
|
'$end_of_table' ->
|
|
|
|
0
|
|
|
|
end.
|
|
|
|
|
|
|
|
count_records_cont(Cont, Count) ->
|
|
|
|
case mnesia:select(Cont) of
|
|
|
|
{Result, Cont} ->
|
|
|
|
NewCount = Count + length(Result),
|
|
|
|
count_records_cont(Cont, NewCount);
|
|
|
|
'$end_of_table' ->
|
|
|
|
Count
|
|
|
|
end.
|
|
|
|
|
|
|
|
jid_to_binary(#jid{user = U, server = S, resource = R,
|
|
|
|
luser = LU, lserver = LS, lresource = LR}) ->
|
|
|
|
#jid{user = iolist_to_binary(U),
|
|
|
|
server = iolist_to_binary(S),
|
|
|
|
resource = iolist_to_binary(R),
|
|
|
|
luser = iolist_to_binary(LU),
|
|
|
|
lserver = iolist_to_binary(LS),
|
|
|
|
lresource = iolist_to_binary(LR)}.
|
|
|
|
|
|
|
|
now_to_integer({MS, S, US}) ->
|
|
|
|
(MS * 1000000 + S) * 1000000 + US.
|
|
|
|
|
|
|
|
integer_to_now(Int) ->
|
|
|
|
Secs = Int div 1000000,
|
|
|
|
USec = Int rem 1000000,
|
|
|
|
MSec = Secs div 1000000,
|
|
|
|
Sec = Secs rem 1000000,
|
|
|
|
{MSec, Sec, USec}.
|