2010-04-27 12:33:56 +02:00
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
%%% File : mod_vcard_xupdate.erl
|
|
|
|
%%% Author : Igor Goryachev <igor@goryachev.org>
|
|
|
|
%%% Purpose : Add avatar hash in presence on behalf of client (XEP-0153)
|
|
|
|
%%% Created : 9 Mar 2007 by Igor Goryachev <igor@goryachev.org>
|
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
|
|
|
|
-module(mod_vcard_xupdate).
|
|
|
|
|
|
|
|
-behaviour(gen_mod).
|
|
|
|
|
|
|
|
%% gen_mod callbacks
|
2013-03-14 10:33:02 +01:00
|
|
|
-export([start/2, stop/1]).
|
2010-04-27 12:33:56 +02:00
|
|
|
|
|
|
|
%% hooks
|
2013-07-21 12:24:36 +02:00
|
|
|
-export([update_presence/3, vcard_set/3, export/1, import/1, import/3]).
|
2010-04-27 12:33:56 +02:00
|
|
|
|
|
|
|
-include("ejabberd.hrl").
|
2013-04-08 11:12:54 +02:00
|
|
|
-include("logger.hrl").
|
2013-03-14 10:33:02 +01:00
|
|
|
|
2010-04-27 12:33:56 +02:00
|
|
|
-include("jlib.hrl").
|
|
|
|
|
2013-03-14 10:33:02 +01:00
|
|
|
-record(vcard_xupdate, {us = {<<>>, <<>>} :: {binary(), binary()},
|
|
|
|
hash = <<>> :: binary()}).
|
2010-04-27 12:33:56 +02:00
|
|
|
|
|
|
|
%%====================================================================
|
|
|
|
%% gen_mod callbacks
|
|
|
|
%%====================================================================
|
|
|
|
|
2012-04-27 11:52:05 +02:00
|
|
|
start(Host, Opts) ->
|
|
|
|
case gen_mod:db_type(Opts) of
|
2013-03-14 10:33:02 +01:00
|
|
|
mnesia ->
|
|
|
|
mnesia:create_table(vcard_xupdate,
|
|
|
|
[{disc_copies, [node()]},
|
|
|
|
{attributes,
|
|
|
|
record_info(fields, vcard_xupdate)}]),
|
|
|
|
update_table();
|
|
|
|
_ -> ok
|
2012-04-27 11:52:05 +02:00
|
|
|
end,
|
2013-03-14 10:33:02 +01:00
|
|
|
ejabberd_hooks:add(c2s_update_presence, Host, ?MODULE,
|
|
|
|
update_presence, 100),
|
|
|
|
ejabberd_hooks:add(vcard_set, Host, ?MODULE, vcard_set,
|
|
|
|
100),
|
2010-04-27 12:33:56 +02:00
|
|
|
ok.
|
|
|
|
|
|
|
|
stop(Host) ->
|
|
|
|
ejabberd_hooks:delete(c2s_update_presence, Host,
|
|
|
|
?MODULE, update_presence, 100),
|
2013-03-14 10:33:02 +01:00
|
|
|
ejabberd_hooks:delete(vcard_set, Host, ?MODULE,
|
|
|
|
vcard_set, 100),
|
2010-04-27 12:33:56 +02:00
|
|
|
ok.
|
|
|
|
|
|
|
|
%%====================================================================
|
|
|
|
%% Hooks
|
|
|
|
%%====================================================================
|
|
|
|
|
2013-03-14 10:33:02 +01:00
|
|
|
update_presence(#xmlel{name = <<"presence">>, attrs = Attrs} = Packet,
|
|
|
|
User, Host) ->
|
|
|
|
case xml:get_attr_s(<<"type">>, Attrs) of
|
|
|
|
<<>> -> presence_with_xupdate(Packet, User, Host);
|
|
|
|
_ -> Packet
|
2010-04-27 12:33:56 +02:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
update_presence(Packet, _User, _Host) -> Packet.
|
2010-04-27 12:33:56 +02:00
|
|
|
|
|
|
|
vcard_set(LUser, LServer, VCARD) ->
|
|
|
|
US = {LUser, LServer},
|
2013-03-14 10:33:02 +01:00
|
|
|
case xml:get_path_s(VCARD,
|
|
|
|
[{elem, <<"PHOTO">>}, {elem, <<"BINVAL">>}, cdata])
|
|
|
|
of
|
|
|
|
<<>> -> remove_xupdate(LUser, LServer);
|
|
|
|
BinVal ->
|
|
|
|
add_xupdate(LUser, LServer,
|
2013-06-20 10:40:44 +02:00
|
|
|
p1_sha:sha(jlib:decode_base64(BinVal)))
|
2010-04-27 12:33:56 +02:00
|
|
|
end,
|
|
|
|
ejabberd_sm:force_update_presence(US).
|
|
|
|
|
|
|
|
%%====================================================================
|
2012-04-27 11:52:05 +02:00
|
|
|
%% Storage
|
2010-04-27 12:33:56 +02:00
|
|
|
%%====================================================================
|
|
|
|
|
|
|
|
add_xupdate(LUser, LServer, Hash) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
add_xupdate(LUser, LServer, Hash,
|
|
|
|
gen_mod:db_type(LServer, ?MODULE)).
|
2012-04-27 11:52:05 +02:00
|
|
|
|
|
|
|
add_xupdate(LUser, LServer, Hash, mnesia) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
F = fun () ->
|
|
|
|
mnesia:write(#vcard_xupdate{us = {LUser, LServer},
|
|
|
|
hash = Hash})
|
|
|
|
end,
|
2012-04-27 11:52:05 +02:00
|
|
|
mnesia:transaction(F);
|
|
|
|
add_xupdate(LUser, LServer, Hash, odbc) ->
|
|
|
|
Username = ejabberd_odbc:escape(LUser),
|
|
|
|
SHash = ejabberd_odbc:escape(Hash),
|
2013-03-14 10:33:02 +01:00
|
|
|
F = fun () ->
|
|
|
|
odbc_queries:update_t(<<"vcard_xupdate">>,
|
|
|
|
[<<"username">>, <<"hash">>],
|
|
|
|
[Username, SHash],
|
|
|
|
[<<"username='">>, Username, <<"'">>])
|
|
|
|
end,
|
2012-04-27 11:52:05 +02:00
|
|
|
ejabberd_odbc:sql_transaction(LServer, F).
|
2010-04-27 12:33:56 +02:00
|
|
|
|
|
|
|
get_xupdate(LUser, LServer) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
get_xupdate(LUser, LServer,
|
|
|
|
gen_mod:db_type(LServer, ?MODULE)).
|
2012-04-27 11:52:05 +02:00
|
|
|
|
|
|
|
get_xupdate(LUser, LServer, mnesia) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
case mnesia:dirty_read(vcard_xupdate, {LUser, LServer})
|
|
|
|
of
|
|
|
|
[#vcard_xupdate{hash = Hash}] -> Hash;
|
|
|
|
_ -> undefined
|
2012-04-27 11:52:05 +02:00
|
|
|
end;
|
|
|
|
get_xupdate(LUser, LServer, odbc) ->
|
|
|
|
Username = ejabberd_odbc:escape(LUser),
|
2013-03-14 10:33:02 +01:00
|
|
|
case ejabberd_odbc:sql_query(LServer,
|
|
|
|
[<<"select hash from vcard_xupdate where "
|
|
|
|
"username='">>,
|
|
|
|
Username, <<"';">>])
|
|
|
|
of
|
|
|
|
{selected, [<<"hash">>], [[Hash]]} -> Hash;
|
|
|
|
_ -> undefined
|
2010-04-27 12:33:56 +02:00
|
|
|
end.
|
|
|
|
|
|
|
|
remove_xupdate(LUser, LServer) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
remove_xupdate(LUser, LServer,
|
|
|
|
gen_mod:db_type(LServer, ?MODULE)).
|
2012-04-27 11:52:05 +02:00
|
|
|
|
|
|
|
remove_xupdate(LUser, LServer, mnesia) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
F = fun () ->
|
|
|
|
mnesia:delete({vcard_xupdate, {LUser, LServer}})
|
|
|
|
end,
|
2012-04-27 11:52:05 +02:00
|
|
|
mnesia:transaction(F);
|
|
|
|
remove_xupdate(LUser, LServer, odbc) ->
|
|
|
|
Username = ejabberd_odbc:escape(LUser),
|
2013-03-14 10:33:02 +01:00
|
|
|
F = fun () ->
|
|
|
|
ejabberd_odbc:sql_query_t([<<"delete from vcard_xupdate where username='">>,
|
|
|
|
Username, <<"';">>])
|
|
|
|
end,
|
2012-04-27 11:52:05 +02:00
|
|
|
ejabberd_odbc:sql_transaction(LServer, F).
|
2010-04-27 12:33:56 +02:00
|
|
|
|
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
%%% Presence stanza rebuilding
|
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
|
2013-03-14 10:33:02 +01:00
|
|
|
presence_with_xupdate(#xmlel{name = <<"presence">>,
|
|
|
|
attrs = Attrs, children = Els},
|
|
|
|
User, Host) ->
|
2010-04-27 12:33:56 +02:00
|
|
|
XPhotoEl = build_xphotoel(User, Host),
|
|
|
|
Els2 = presence_with_xupdate2(Els, [], XPhotoEl),
|
2013-03-14 10:33:02 +01:00
|
|
|
#xmlel{name = <<"presence">>, attrs = Attrs,
|
|
|
|
children = Els2}.
|
2010-04-27 12:33:56 +02:00
|
|
|
|
|
|
|
presence_with_xupdate2([], Els2, XPhotoEl) ->
|
|
|
|
lists:reverse([XPhotoEl | Els2]);
|
|
|
|
%% This clause assumes that the x element contains only the XMLNS attribute:
|
2013-03-14 10:33:02 +01:00
|
|
|
presence_with_xupdate2([#xmlel{name = <<"x">>,
|
|
|
|
attrs = [{<<"xmlns">>, ?NS_VCARD_UPDATE}]}
|
|
|
|
| Els],
|
|
|
|
Els2, XPhotoEl) ->
|
2010-04-27 12:33:56 +02:00
|
|
|
presence_with_xupdate2(Els, Els2, XPhotoEl);
|
|
|
|
presence_with_xupdate2([El | Els], Els2, XPhotoEl) ->
|
|
|
|
presence_with_xupdate2(Els, [El | Els2], XPhotoEl).
|
|
|
|
|
|
|
|
build_xphotoel(User, Host) ->
|
|
|
|
Hash = get_xupdate(User, Host),
|
|
|
|
PhotoSubEls = case Hash of
|
2013-03-14 10:33:02 +01:00
|
|
|
Hash when is_binary(Hash) -> [{xmlcdata, Hash}];
|
|
|
|
_ -> []
|
2010-04-27 12:33:56 +02:00
|
|
|
end,
|
2013-03-14 10:33:02 +01:00
|
|
|
PhotoEl = [#xmlel{name = <<"photo">>, attrs = [],
|
|
|
|
children = PhotoSubEls}],
|
|
|
|
#xmlel{name = <<"x">>,
|
|
|
|
attrs = [{<<"xmlns">>, ?NS_VCARD_UPDATE}],
|
|
|
|
children = PhotoEl}.
|
|
|
|
|
|
|
|
update_table() ->
|
|
|
|
Fields = record_info(fields, vcard_xupdate),
|
|
|
|
case mnesia:table_info(vcard_xupdate, attributes) of
|
|
|
|
Fields ->
|
|
|
|
ejabberd_config:convert_table_to_binary(
|
|
|
|
vcard_xupdate, Fields, set,
|
|
|
|
fun(#vcard_xupdate{us = {U, _}}) -> U end,
|
|
|
|
fun(#vcard_xupdate{us = {U, S}, hash = Hash} = R) ->
|
|
|
|
R#vcard_xupdate{us = {iolist_to_binary(U),
|
|
|
|
iolist_to_binary(S)},
|
|
|
|
hash = iolist_to_binary(Hash)}
|
|
|
|
end);
|
|
|
|
_ ->
|
|
|
|
?INFO_MSG("Recreating vcard_xupdate table", []),
|
|
|
|
mnesia:transform_table(vcard_xupdate, ignore, Fields)
|
|
|
|
end.
|
|
|
|
|
|
|
|
export(_Server) ->
|
|
|
|
[{vcard_xupdate,
|
|
|
|
fun(Host, #vcard_xupdate{us = {LUser, LServer}, hash = Hash})
|
|
|
|
when LServer == Host ->
|
|
|
|
Username = ejabberd_odbc:escape(LUser),
|
|
|
|
SHash = ejabberd_odbc:escape(Hash),
|
|
|
|
[[<<"delete from vcard_xupdate where username='">>,
|
|
|
|
Username, <<"';">>],
|
|
|
|
[<<"insert into vcard_xupdate(username, "
|
|
|
|
"hash) values ('">>,
|
|
|
|
Username, <<"', '">>, SHash, <<"');">>]];
|
|
|
|
(_Host, _R) ->
|
|
|
|
[]
|
|
|
|
end}].
|
2013-07-21 12:24:36 +02:00
|
|
|
|
|
|
|
import(LServer) ->
|
|
|
|
[{<<"select username, hash from vcard_xupdate;">>,
|
|
|
|
fun([LUser, Hash]) ->
|
|
|
|
#vcard_xupdate{us = {LUser, LServer}, hash = Hash}
|
|
|
|
end}].
|
|
|
|
|
|
|
|
import(_LServer, mnesia, #vcard_xupdate{} = R) ->
|
|
|
|
mnesia:dirty_write(R);
|
|
|
|
import(_, _, _) ->
|
|
|
|
pass.
|