1 |
|
%%%---------------------------------------------------------------------- |
2 |
|
%%% File : mod_caps.erl |
3 |
|
%%% Author : Magnus Henoch <henoch@dtek.chalmers.se> |
4 |
|
%%% Purpose : Request and cache Entity Capabilities (XEP-0115) |
5 |
|
%%% Created : 7 Oct 2006 by Magnus Henoch <henoch@dtek.chalmers.se> |
6 |
|
%%% |
7 |
|
%%% |
8 |
|
%%% ejabberd, Copyright (C) 2002-2015 ProcessOne |
9 |
|
%%% |
10 |
|
%%% This program is free software; you can redistribute it and/or |
11 |
|
%%% modify it under the terms of the GNU General Public License as |
12 |
|
%%% published by the Free Software Foundation; either version 2 of the |
13 |
|
%%% License, or (at your option) any later version. |
14 |
|
%%% |
15 |
|
%%% This program is distributed in the hope that it will be useful, |
16 |
|
%%% but WITHOUT ANY WARRANTY; without even the implied warranty of |
17 |
|
%%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
18 |
|
%%% General Public License for more details. |
19 |
|
%%% |
20 |
|
%%% You should have received a copy of the GNU General Public License along |
21 |
|
%%% with this program; if not, write to the Free Software Foundation, Inc., |
22 |
|
%%% 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. |
23 |
|
%%% |
24 |
|
%%% 2009, improvements from ProcessOne to support correct PEP handling |
25 |
|
%%% through s2s, use less memory, and speedup global caps handling |
26 |
|
%%%---------------------------------------------------------------------- |
27 |
|
|
28 |
|
-module(mod_caps). |
29 |
|
|
30 |
|
-author('henoch@dtek.chalmers.se'). |
31 |
|
|
32 |
|
-xep([{xep, 115}, {version, "1.6.0"}]). |
33 |
|
|
34 |
|
-behaviour(gen_server). |
35 |
|
-behaviour(gen_mod). |
36 |
|
-behaviour(mongoose_module_metrics). |
37 |
|
|
38 |
|
-export([read_caps/1, caps_stream_features/3, |
39 |
|
disco_local_features/3, disco_local_identity/3, disco_info/3]). |
40 |
|
|
41 |
|
%% gen_mod callbacks |
42 |
|
-export([start/2, start_link/2, stop/1, config_spec/0, supported_features/0]). |
43 |
|
|
44 |
|
%% gen_server callbacks |
45 |
|
-export([init/1, handle_info/2, handle_call/3, |
46 |
|
handle_cast/2, terminate/2, code_change/3]). |
47 |
|
|
48 |
|
-export([user_send_presence/3, |
49 |
|
user_receive_presence/3, |
50 |
|
get_pep_recipients/3, |
51 |
|
filter_pep_recipient/3]). |
52 |
|
|
53 |
|
%% for test cases |
54 |
|
-export([delete_caps/1, make_disco_hash/2]). |
55 |
|
-ignore_xref([delete_caps/1, make_disco_hash/2, read_caps/1, start_link/2]). |
56 |
|
|
57 |
|
-include("mongoose.hrl"). |
58 |
|
-include("mongoose_config_spec.hrl"). |
59 |
|
|
60 |
|
-include("jlib.hrl"). |
61 |
|
|
62 |
|
-define(PROCNAME, ejabberd_mod_caps). |
63 |
|
|
64 |
|
-define(BAD_HASH_LIFETIME, 600). |
65 |
|
|
66 |
|
-record(caps, |
67 |
|
{ |
68 |
|
node = <<>> :: binary(), |
69 |
|
version = <<>> :: binary(), |
70 |
|
hash = <<>> :: binary(), |
71 |
|
exts = [] :: [binary()] |
72 |
|
}). |
73 |
|
|
74 |
|
-type caps() :: #caps{}. |
75 |
|
-type caps_resources() :: gb_trees:tree(jid:simple_jid(), caps()). |
76 |
|
|
77 |
|
-export_type([caps/0]). |
78 |
|
|
79 |
|
-type features() :: [binary()]. |
80 |
|
-type maybe_pending_features() :: features() | pos_integer(). |
81 |
|
-type node_pair() :: {binary(), binary()}. |
82 |
|
|
83 |
|
-record(caps_features, |
84 |
|
{ |
85 |
|
node_pair = {<<>>, <<>>} :: node_pair(), |
86 |
|
features = [] :: maybe_pending_features() |
87 |
|
}). |
88 |
|
|
89 |
|
-record(state, {host_type :: mongooseim:host_type()}). |
90 |
|
|
91 |
|
-type state() :: #state{}. |
92 |
|
|
93 |
|
-spec start_link(mongooseim:host_type(), gen_mod:module_opts()) -> any(). |
94 |
|
start_link(HostType, Opts) -> |
95 |
2 |
Proc = gen_mod:get_module_proc(HostType, ?PROCNAME), |
96 |
2 |
gen_server:start_link({local, Proc}, ?MODULE, |
97 |
|
[HostType, Opts], []). |
98 |
|
|
99 |
|
-spec start(mongooseim:host_type(), gen_mod:module_opts()) -> any(). |
100 |
|
start(HostType, Opts) -> |
101 |
2 |
Proc = gen_mod:get_module_proc(HostType, ?PROCNAME), |
102 |
2 |
ChildSpec = {Proc, {?MODULE, start_link, [HostType, Opts]}, |
103 |
|
transient, 1000, worker, [?MODULE]}, |
104 |
2 |
ejabberd_sup:start_child(ChildSpec). |
105 |
|
|
106 |
|
-spec stop(mongooseim:host_type()) -> any(). |
107 |
|
stop(HostType) -> |
108 |
2 |
Proc = gen_mod:get_module_proc(HostType, ?PROCNAME), |
109 |
2 |
gen_server:call(Proc, stop), |
110 |
2 |
ejabberd_sup:stop_child(Proc). |
111 |
|
|
112 |
|
-spec config_spec() -> mongoose_config_spec:config_section(). |
113 |
|
config_spec() -> |
114 |
84 |
#section{ |
115 |
|
items = #{<<"cache_size">> => #option{type = integer, |
116 |
|
validate = positive}, |
117 |
|
<<"cache_life_time">> => #option{type = integer, |
118 |
|
validate = positive} |
119 |
|
}, |
120 |
|
defaults = #{<<"cache_size">> => 1000, |
121 |
|
<<"cache_life_time">> => timer:hours(24) div 1000} |
122 |
|
}. |
123 |
|
|
124 |
2 |
supported_features() -> [dynamic_domains]. |
125 |
|
|
126 |
|
-spec get_features_list(mongooseim:host_type(), nothing | caps()) -> features(). |
127 |
|
get_features_list(HostType, Caps) -> |
128 |
:-( |
case get_features(HostType, Caps) of |
129 |
:-( |
unknown -> []; |
130 |
:-( |
Features -> Features |
131 |
|
end. |
132 |
|
|
133 |
|
-spec get_features(mongooseim:host_type(), nothing | caps()) -> unknown | features(). |
134 |
:-( |
get_features(_HostType, nothing) -> []; |
135 |
|
get_features(HostType, #caps{node = Node, version = Version, exts = Exts}) -> |
136 |
:-( |
SubNodes = [Version | Exts], |
137 |
:-( |
lists:foldl(fun (SubNode, Acc) -> |
138 |
:-( |
NodePair = {Node, SubNode}, |
139 |
:-( |
case cache_tab:lookup(caps_features, NodePair, |
140 |
|
caps_read_fun(HostType, NodePair)) |
141 |
|
of |
142 |
|
{ok, Features} when is_list(Features) -> |
143 |
:-( |
Features ++ Acc; |
144 |
|
_ when Acc == [] -> |
145 |
:-( |
unknown; |
146 |
|
_ -> |
147 |
:-( |
Acc |
148 |
|
end |
149 |
|
end, |
150 |
|
[], SubNodes). |
151 |
|
|
152 |
|
-spec read_caps([exml:element()]) -> nothing | caps(). |
153 |
70 |
read_caps(Els) -> read_caps(Els, nothing). |
154 |
|
|
155 |
|
read_caps([#xmlel{name = <<"c">>, attrs = Attrs} | Tail], Result) -> |
156 |
:-( |
case xml:get_attr_s(<<"xmlns">>, Attrs) of |
157 |
|
?NS_CAPS -> |
158 |
:-( |
Node = xml:get_attr_s(<<"node">>, Attrs), |
159 |
:-( |
Version = xml:get_attr_s(<<"ver">>, Attrs), |
160 |
:-( |
Hash = xml:get_attr_s(<<"hash">>, Attrs), |
161 |
:-( |
Exts = mongoose_bin:tokens(xml:get_attr_s(<<"ext">>, Attrs), <<" ">>), |
162 |
:-( |
read_caps(Tail, #caps{node = Node, hash = Hash, version = Version, exts = Exts}); |
163 |
:-( |
_ -> read_caps(Tail, Result) |
164 |
|
end; |
165 |
|
read_caps([#xmlel{name = <<"x">>, attrs = Attrs} | Tail], Result) -> |
166 |
:-( |
case xml:get_attr_s(<<"xmlns">>, Attrs) of |
167 |
:-( |
?NS_MUC_USER -> nothing; |
168 |
:-( |
_ -> read_caps(Tail, Result) |
169 |
|
end; |
170 |
|
read_caps([_ | Tail], Result) -> |
171 |
6 |
read_caps(Tail, Result); |
172 |
70 |
read_caps([], Result) -> Result. |
173 |
|
|
174 |
|
-spec user_send_presence(Acc, Params, Extra) -> {ok, Acc} when |
175 |
|
Acc :: mongoose_acc:t(), |
176 |
|
Params :: map(), |
177 |
|
Extra :: map(). |
178 |
|
user_send_presence(Acc, _, _) -> |
179 |
50 |
{From, To, Packet} = mongoose_acc:packet(Acc), |
180 |
50 |
{ok, user_send_presence(Acc, From, To, Packet)}. |
181 |
|
|
182 |
|
-spec user_send_presence(mongoose_acc:t(), jid:jid(), jid:jid(), exml:element()) -> mongoose_acc:t(). |
183 |
|
user_send_presence(Acc, |
184 |
|
#jid{luser = User, lserver = LServer} = From, |
185 |
|
#jid{luser = User, lserver = LServer, lresource = <<>>}, |
186 |
|
#xmlel{attrs = Attrs, children = Elements}) -> |
187 |
26 |
Type = xml:get_attr_s(<<"type">>, Attrs), |
188 |
26 |
handle_presence(Acc, LServer, From, Type, Elements); |
189 |
|
user_send_presence(Acc, _, _, _) -> |
190 |
24 |
Acc. |
191 |
|
|
192 |
|
-spec caps_stream_features(Acc, Params, Extra) -> {ok, Acc} when |
193 |
|
Acc :: [exml:element()], |
194 |
|
Params :: #{lserver := jid:lserver()}, |
195 |
|
Extra :: #{host_type := mongooseim:host_type()}. |
196 |
|
caps_stream_features(Acc, #{lserver := LServer}, #{host_type := HostType}) -> |
197 |
58 |
NewAcc = case make_my_disco_hash(HostType, LServer) of |
198 |
|
<<>> -> |
199 |
:-( |
Acc; |
200 |
|
Hash -> |
201 |
58 |
[#xmlel{name = <<"c">>, |
202 |
|
attrs = [{<<"xmlns">>, ?NS_CAPS}, {<<"hash">>, <<"sha-1">>}, |
203 |
|
{<<"node">>, ?MONGOOSE_URI}, {<<"ver">>, Hash}], |
204 |
|
children = []} |
205 |
|
| Acc] |
206 |
|
end, |
207 |
58 |
{ok, NewAcc}. |
208 |
|
|
209 |
|
-spec disco_local_features(mongoose_disco:feature_acc(), |
210 |
|
map(), |
211 |
|
map()) -> {ok, mongoose_disco:feature_acc()}. |
212 |
|
disco_local_features(Acc = #{node := Node}, _, _) -> |
213 |
63 |
NewAcc = case is_valid_node(Node) of |
214 |
2 |
true -> Acc#{node := <<>>}; |
215 |
61 |
false -> Acc |
216 |
|
end, |
217 |
63 |
{ok, NewAcc}. |
218 |
|
|
219 |
|
-spec disco_local_identity(Acc, Params, Extra) -> {ok, Acc} when |
220 |
|
Acc :: mongoose_disco:identity_acc(), |
221 |
|
Params :: map(), |
222 |
|
Extra :: gen_hook:extra(). |
223 |
|
disco_local_identity(Acc = #{node := Node}, _, _) -> |
224 |
63 |
NewAcc = case is_valid_node(Node) of |
225 |
2 |
true -> Acc#{node := <<>>}; |
226 |
61 |
false -> Acc |
227 |
|
end, |
228 |
63 |
{ok, NewAcc}. |
229 |
|
|
230 |
|
-spec disco_info(Acc, Params, Extra) -> {ok, Acc} when |
231 |
|
Acc :: mongoose_disco:identity_acc(), |
232 |
|
Params :: map(), |
233 |
|
Extra :: gen_hook:extra(). |
234 |
|
disco_info(Acc = #{node := Node}, _, _) -> |
235 |
63 |
NewAcc = case is_valid_node(Node) of |
236 |
2 |
true -> Acc#{node := <<>>}; |
237 |
61 |
false -> Acc |
238 |
|
end, |
239 |
63 |
{ok, NewAcc}. |
240 |
|
|
241 |
|
-spec handle_presence(mongoose_acc:t(), jid:lserver(), jid:jid(), binary(), [exml:element()]) -> |
242 |
|
mongoose_acc:t(). |
243 |
|
handle_presence(Acc, LServer, From, Type, Elements) when Type =:= <<>>; |
244 |
|
Type =:= <<"available">> -> |
245 |
26 |
case read_caps(Elements) of |
246 |
|
nothing -> |
247 |
26 |
Acc; |
248 |
|
#caps{version = Version, exts = Exts} = Caps -> |
249 |
:-( |
feature_request(Acc, LServer, From, Caps, [Version | Exts]) |
250 |
|
end; |
251 |
|
handle_presence(Acc, _LServer, _From, _Type, _Elements) -> |
252 |
:-( |
Acc. |
253 |
|
|
254 |
|
-spec user_receive_presence(mongoose_acc:t(), mongoose_c2s_hooks:params(), gen_hook:extra()) -> |
255 |
|
mongoose_c2s_hooks:result(). |
256 |
|
user_receive_presence(Acc0, #{c2s_data := C2SData}, _Extra) -> |
257 |
94 |
{From, To, #xmlel{attrs = Attrs, children = Els} = Packet} = mongoose_acc:packet(Acc0), |
258 |
94 |
?LOG_DEBUG(#{what => user_receive_presence, |
259 |
|
to => jid:to_binary(To), from => jid:to_binary(From), |
260 |
94 |
exml_packet => Packet, c2s_state => C2SData}), |
261 |
94 |
Type = xml:get_attr_s(<<"type">>, Attrs), |
262 |
94 |
#jid{lserver = LServer} = mongoose_c2s:get_jid(C2SData), |
263 |
94 |
Acc = case mongoose_domain_api:get_host_type(From#jid.lserver) of |
264 |
|
{error, not_found} -> |
265 |
:-( |
handle_presence(Acc0, LServer, From, Type, Els); |
266 |
|
{ok, _} -> |
267 |
94 |
Acc0 %% it was already handled in 'user_send_presence' |
268 |
|
end, |
269 |
94 |
case mongoose_c2s:get_mod_state(C2SData, mod_presence) of |
270 |
|
{ok, Presences} -> |
271 |
94 |
Subscription = get_subscription(From, Presences), |
272 |
94 |
Insert = (Type == <<>> orelse Type == <<"available">>) |
273 |
94 |
and (Subscription == both orelse Subscription == to), |
274 |
94 |
Delete = Type == <<"unavailable">> orelse Type == <<"error">>, |
275 |
94 |
case Insert orelse Delete of |
276 |
|
true -> |
277 |
44 |
LFrom = jid:to_lower(From), |
278 |
44 |
Rs = case mongoose_c2s:get_mod_state(C2SData, ?MODULE) of |
279 |
18 |
{ok, Rs1} -> Rs1; |
280 |
26 |
{error, not_found} -> gb_trees:empty() |
281 |
|
end, |
282 |
44 |
Caps = read_caps(Els), |
283 |
44 |
NewRs = case Caps of |
284 |
|
nothing when Insert == true -> |
285 |
38 |
Rs; |
286 |
|
_ when Insert == true -> |
287 |
:-( |
?LOG_DEBUG(#{what => caps_set_caps, |
288 |
|
caps => Caps, |
289 |
|
to => jid:to_binary(To), |
290 |
|
from => jid:to_binary(From), |
291 |
|
exml_packet => Packet, |
292 |
:-( |
c2s_state => C2SData}), |
293 |
:-( |
upsert_caps(LFrom, Caps, Rs); |
294 |
|
_ -> |
295 |
6 |
gb_trees:delete_any(LFrom, Rs) |
296 |
|
end, |
297 |
44 |
{ok, mongoose_c2s_acc:to_acc(Acc, state_mod, {?MODULE, NewRs})}; |
298 |
|
false -> |
299 |
50 |
{ok, Acc} |
300 |
|
end; |
301 |
|
{error, not_found} -> |
302 |
:-( |
{ok, Acc} |
303 |
|
end. |
304 |
|
|
305 |
|
get_subscription(From, Presences) -> |
306 |
94 |
BareFrom = jid:to_bare(From), |
307 |
94 |
F = mod_presence:is_subscribed_to_my_presence(From, BareFrom, Presences), |
308 |
94 |
T = mod_presence:am_i_subscribed_to_presence(From, BareFrom, Presences), |
309 |
94 |
case {F, T} of |
310 |
70 |
{true, true} -> both; |
311 |
:-( |
{true, false} -> from; |
312 |
12 |
{false, true} -> to; |
313 |
12 |
{false, false} -> none |
314 |
|
end. |
315 |
|
|
316 |
|
-spec upsert_caps(jid:simple_jid(), caps(), caps_resources()) -> caps_resources(). |
317 |
|
upsert_caps(LFrom, Caps, Rs) -> |
318 |
:-( |
case gb_trees:lookup(LFrom, Rs) of |
319 |
:-( |
{value, Caps} -> Rs; |
320 |
|
none -> |
321 |
:-( |
gb_trees:insert(LFrom, Caps, Rs); |
322 |
|
_ -> |
323 |
:-( |
gb_trees:update(LFrom, Caps, Rs) |
324 |
|
end. |
325 |
|
|
326 |
|
-spec get_pep_recipients(Acc, Params, Extra) -> {ok, Acc} when |
327 |
|
Acc :: [jid:simple_jid()], |
328 |
|
Params :: #{c2s_data := mongoose_c2s:data(), feature := binary()}, |
329 |
|
Extra :: map(). |
330 |
|
get_pep_recipients(InAcc, #{c2s_data := C2SData, feature := Feature}, _) -> |
331 |
:-( |
HostType = mongoose_c2s:get_host_type(C2SData), |
332 |
:-( |
NewAcc = case mongoose_c2s:get_mod_state(C2SData, ?MODULE) of |
333 |
|
{ok, Rs} -> |
334 |
:-( |
filter_recipients_by_caps(HostType, InAcc, Feature, Rs); |
335 |
:-( |
_ -> InAcc |
336 |
|
end, |
337 |
:-( |
{ok, NewAcc}; |
338 |
:-( |
get_pep_recipients(Acc, _, _) -> {ok, Acc}. |
339 |
|
|
340 |
|
-spec filter_recipients_by_caps(mongooseim:host_type(), Acc, binary(), caps_resources()) -> Acc |
341 |
|
when Acc :: [jid:simple_jid()]. |
342 |
|
filter_recipients_by_caps(HostType, InAcc, Feature, Rs) -> |
343 |
:-( |
gb_trees_fold(fun(USR, Caps, Acc) -> |
344 |
:-( |
case lists:member(Feature, get_features_list(HostType, Caps)) of |
345 |
:-( |
true -> [USR | Acc]; |
346 |
:-( |
false -> Acc |
347 |
|
end |
348 |
|
end, |
349 |
|
InAcc, Rs). |
350 |
|
|
351 |
|
-spec filter_pep_recipient(Acc, Params, Extra) -> {ok | stop, Acc} when |
352 |
|
Acc :: boolean(), |
353 |
|
Params :: #{c2s_data := mongoose_c2s:data(), feature := binary(), to := jid:jid()}, |
354 |
|
Extra :: gen_hook:extra(). |
355 |
|
filter_pep_recipient(InAcc, #{c2s_data := C2SData, feature := Feature, to := To}, _) -> |
356 |
:-( |
case mongoose_c2s:get_mod_state(C2SData, ?MODULE) of |
357 |
|
{ok, Rs} -> |
358 |
:-( |
?LOG_DEBUG(#{what => caps_lookup, text => <<"Look for CAPS for To jid">>, |
359 |
:-( |
acc => InAcc, c2s_state => C2SData, caps_resources => Rs}), |
360 |
:-( |
LTo = jid:to_lower(To), |
361 |
:-( |
case gb_trees:lookup(LTo, Rs) of |
362 |
|
{value, Caps} -> |
363 |
:-( |
HostType = mongoose_c2s:get_host_type(C2SData), |
364 |
:-( |
Drop = not lists:member(Feature, get_features_list(HostType, Caps)), |
365 |
:-( |
{stop, Drop}; |
366 |
|
none -> |
367 |
:-( |
{stop, true} |
368 |
|
end; |
369 |
:-( |
_ -> {ok, InAcc} |
370 |
|
end. |
371 |
|
|
372 |
|
init_db(mnesia) -> |
373 |
2 |
case catch mnesia:table_info(caps_features, storage_type) of |
374 |
|
{'EXIT', _} -> |
375 |
1 |
ok; |
376 |
|
disc_only_copies -> |
377 |
1 |
ok; |
378 |
|
_ -> |
379 |
:-( |
mnesia:delete_table(caps_features) |
380 |
|
end, |
381 |
2 |
mongoose_mnesia:create_table(caps_features, |
382 |
|
[{disc_only_copies, [node()]}, |
383 |
|
{local_content, true}, |
384 |
|
{attributes, record_info(fields, caps_features)}]). |
385 |
|
|
386 |
|
-spec init(list()) -> {ok, state()}. |
387 |
|
init([HostType, #{cache_size := MaxSize, cache_life_time := LifeTime}]) -> |
388 |
2 |
init_db(db_type(HostType)), |
389 |
2 |
cache_tab:new(caps_features, [{max_size, MaxSize}, {life_time, LifeTime}]), |
390 |
2 |
gen_hook:add_handlers(hooks(HostType)), |
391 |
2 |
{ok, #state{host_type = HostType}}. |
392 |
|
|
393 |
|
-spec handle_call(term(), any(), state()) -> |
394 |
|
{stop, normal, ok, state()} | {reply, {error, any()}, state()}. |
395 |
|
handle_call(stop, _From, State) -> |
396 |
2 |
{stop, normal, ok, State}; |
397 |
|
handle_call(_Req, _From, State) -> |
398 |
:-( |
{reply, {error, badarg}, State}. |
399 |
|
|
400 |
|
-spec handle_cast(any(), state()) -> {noreply, state()}. |
401 |
:-( |
handle_cast(_Msg, State) -> {noreply, State}. |
402 |
|
|
403 |
|
-spec handle_info(any(), state()) -> {noreply, state()}. |
404 |
:-( |
handle_info(_Info, State) -> {noreply, State}. |
405 |
|
|
406 |
|
-spec terminate(any(), state()) -> ok. |
407 |
|
terminate(_Reason, #state{host_type = HostType}) -> |
408 |
2 |
gen_hook:delete_handlers(hooks(HostType)). |
409 |
|
|
410 |
|
hooks(HostType) -> |
411 |
4 |
[{disco_local_features, HostType, fun ?MODULE:disco_local_features/3, #{}, 1}, |
412 |
|
{get_pep_recipients, HostType, fun ?MODULE:get_pep_recipients/3, #{}, 75}, |
413 |
|
{filter_pep_recipient, HostType, fun ?MODULE:filter_pep_recipient/3, #{}, 75}, |
414 |
|
{user_send_presence, HostType, fun ?MODULE:user_send_presence/3, #{}, 75}, |
415 |
|
{user_receive_presence, HostType, fun ?MODULE:user_receive_presence/3, #{}, 1}, |
416 |
|
{c2s_stream_features, HostType, fun ?MODULE:caps_stream_features/3, #{}, 75}, |
417 |
|
{s2s_stream_features, HostType, fun ?MODULE:caps_stream_features/3, #{}, 75}, |
418 |
|
{disco_local_identity, HostType, fun ?MODULE:disco_local_identity/3, #{}, 1}, |
419 |
|
{disco_info, HostType, fun ?MODULE:disco_info/3, #{}, 1}, |
420 |
|
{disco_local_features, HostType, fun ?MODULE:disco_local_features/3, #{}, 1} |
421 |
|
]. |
422 |
|
|
423 |
|
-spec code_change(any(), state(), any()) -> {ok, state()}. |
424 |
:-( |
code_change(_OldVsn, State, _Extra) -> {ok, State}. |
425 |
|
|
426 |
|
-spec feature_request(mongoose_acc:t(), jid:lserver(), jid:jid(), caps(), [binary()]) -> |
427 |
|
mongoose_acc:t(). |
428 |
|
feature_request(Acc, LServer, From, Caps, [SubNode | Tail] = SubNodes) -> |
429 |
:-( |
Node = Caps#caps.node, |
430 |
:-( |
NodePair = {Node, SubNode}, |
431 |
:-( |
HostType = mongoose_acc:host_type(Acc), |
432 |
:-( |
case cache_tab:lookup(caps_features, NodePair, caps_read_fun(HostType, NodePair)) of |
433 |
|
{ok, Fs} when is_list(Fs) -> |
434 |
:-( |
feature_request(Acc, LServer, From, Caps, Tail); |
435 |
|
Other -> |
436 |
:-( |
NeedRequest = case Other of |
437 |
:-( |
{ok, TS} -> os:system_time(second) >= TS + (?BAD_HASH_LIFETIME); |
438 |
:-( |
_ -> true |
439 |
|
end, |
440 |
:-( |
F = fun (_From, _To, _Acc1, timeout) -> |
441 |
|
%% IQ request timed out, skip this node |
442 |
:-( |
feature_request(Acc, LServer, From, Caps, Tail); |
443 |
|
(_From, _To, Acc1, IQReply) -> |
444 |
:-( |
feature_response(Acc1, IQReply, LServer, From, Caps, SubNodes) |
445 |
|
end, |
446 |
:-( |
case NeedRequest of |
447 |
|
true -> |
448 |
:-( |
IQ = #iq{type = get, xmlns = ?NS_DISCO_INFO, |
449 |
|
sub_el = |
450 |
|
[#xmlel{name = <<"query">>, |
451 |
|
attrs = |
452 |
|
[{<<"xmlns">>, ?NS_DISCO_INFO}, |
453 |
|
{<<"node">>, |
454 |
|
<<Node/binary, "#", |
455 |
|
SubNode/binary>>}], |
456 |
|
children = []}]}, |
457 |
:-( |
cache_tab:insert(caps_features, NodePair, os:system_time(second), |
458 |
|
caps_write_fun(HostType, NodePair, os:system_time(second))), |
459 |
:-( |
ejabberd_local:route_iq(jid:make_noprep(<<>>, LServer, <<>>), From, Acc, IQ, F), |
460 |
:-( |
Acc; |
461 |
:-( |
false -> feature_request(Acc, LServer, From, Caps, Tail) |
462 |
|
end |
463 |
|
end; |
464 |
|
feature_request(Acc, _LServer, From, Caps, []) -> |
465 |
|
%% feature_request is never executed with empty SubNodes list |
466 |
|
%% so if we end up here, it means the caps are known |
467 |
:-( |
HostType = mongoose_acc:host_type(Acc), |
468 |
:-( |
mongoose_hooks:caps_recognised(Acc, From, self(), get_features_list(HostType, Caps)). |
469 |
|
|
470 |
|
-spec feature_response(mongoose_acc:t(), jlib:iq(), jid:lserver(), jid:jid(), caps(), [binary()]) -> |
471 |
|
mongoose_acc:t(). |
472 |
|
feature_response(Acc, #iq{type = result, sub_el = [#xmlel{children = Els}]}, |
473 |
|
LServer, From, Caps, [SubNode | SubNodes]) -> |
474 |
:-( |
HostType = mongoose_acc:host_type(Acc), |
475 |
:-( |
NodePair = {Caps#caps.node, SubNode}, |
476 |
:-( |
case check_hash(Caps, Els) of |
477 |
|
true -> |
478 |
:-( |
Features = lists:flatmap(fun (#xmlel{name = <<"feature">>, |
479 |
|
attrs = FAttrs}) -> |
480 |
:-( |
[xml:get_attr_s(<<"var">>, FAttrs)]; |
481 |
:-( |
(_) -> [] |
482 |
|
end, |
483 |
|
Els), |
484 |
:-( |
cache_tab:insert(caps_features, NodePair, |
485 |
|
Features, |
486 |
|
caps_write_fun(HostType, NodePair, Features)); |
487 |
:-( |
false -> ok |
488 |
|
end, |
489 |
:-( |
feature_request(Acc, LServer, From, Caps, SubNodes); |
490 |
|
feature_response(Acc, _IQResult, LServer, From, Caps, [_SubNode | SubNodes]) -> |
491 |
:-( |
feature_request(Acc, LServer, From, Caps, SubNodes). |
492 |
|
|
493 |
|
-spec caps_read_fun(mongooseim:host_type(), node_pair()) -> |
494 |
|
fun(() -> {ok, maybe_pending_features()} | error). |
495 |
|
caps_read_fun(HostType, Node) -> |
496 |
:-( |
DBType = db_type(HostType), |
497 |
:-( |
caps_read_fun(HostType, Node, DBType). |
498 |
|
|
499 |
|
caps_read_fun(_HostType, Node, mnesia) -> |
500 |
:-( |
fun () -> |
501 |
:-( |
case mnesia:dirty_read({caps_features, Node}) of |
502 |
:-( |
[#caps_features{features = Features}] -> {ok, Features}; |
503 |
:-( |
_ -> error |
504 |
|
end |
505 |
|
end. |
506 |
|
|
507 |
|
-spec caps_write_fun(mongooseim:host_type(), node_pair(), maybe_pending_features()) -> |
508 |
|
fun(() -> ok). |
509 |
|
caps_write_fun(HostType, Node, Features) -> |
510 |
:-( |
DBType = db_type(HostType), |
511 |
:-( |
caps_write_fun(HostType, Node, Features, DBType). |
512 |
|
|
513 |
|
caps_write_fun(_HostType, Node, Features, mnesia) -> |
514 |
:-( |
fun () -> |
515 |
:-( |
mnesia:dirty_write(#caps_features{node_pair = Node, |
516 |
|
features = Features}) |
517 |
|
end. |
518 |
|
|
519 |
|
-spec delete_caps(node_pair()) -> ok. |
520 |
|
delete_caps(Node) -> |
521 |
:-( |
cache_tab:delete(caps_features, Node, caps_delete_fun(Node)). |
522 |
|
|
523 |
|
-spec caps_delete_fun(node_pair()) -> fun(() -> ok). |
524 |
|
caps_delete_fun(Node) -> |
525 |
:-( |
fun () -> |
526 |
:-( |
mnesia:dirty_delete(caps_features, Node) |
527 |
|
end. |
528 |
|
|
529 |
|
-spec make_my_disco_hash(mongooseim:host_type(), jid:lserver()) -> binary(). |
530 |
|
make_my_disco_hash(HostType, LServer) -> |
531 |
58 |
JID = jid:make(<<>>, LServer, <<>>), |
532 |
58 |
case mongoose_disco:get_local_features(HostType, JID, JID, <<>>, <<>>) of |
533 |
|
empty -> |
534 |
:-( |
<<>>; |
535 |
|
{result, FeaturesXML} -> |
536 |
58 |
IdentityXML = mongoose_disco:get_local_identity(HostType, JID, JID, <<>>, <<>>), |
537 |
58 |
InfoXML = mongoose_disco:get_info(HostType, mod_disco, <<>>, <<>>), |
538 |
58 |
make_disco_hash(IdentityXML ++ InfoXML ++ FeaturesXML, sha1) |
539 |
|
end. |
540 |
|
|
541 |
|
-spec make_disco_hash([exml:element()], HashAlgorithm :: atom()) -> binary(). |
542 |
|
make_disco_hash(DiscoEls, Algo) -> |
543 |
58 |
Concat = list_to_binary([concat_identities(DiscoEls), |
544 |
|
concat_features(DiscoEls), concat_info(DiscoEls)]), |
545 |
58 |
jlib:encode_base64(case Algo of |
546 |
:-( |
md5 -> erlang:md5(Concat); |
547 |
58 |
sha1 -> crypto:hash(sha, Concat); |
548 |
:-( |
sha224 -> crypto:hash(sha224, Concat); |
549 |
:-( |
sha256 -> crypto:hash(sha256, Concat); |
550 |
:-( |
sha384 -> crypto:hash(sha384, Concat); |
551 |
:-( |
sha512 -> crypto:hash(sha512, Concat) |
552 |
|
end). |
553 |
|
|
554 |
|
check_hash(Caps, Els) -> |
555 |
:-( |
case Caps#caps.hash of |
556 |
|
<<"md5">> -> |
557 |
:-( |
Caps#caps.version == make_disco_hash(Els, md5); |
558 |
|
<<"sha-1">> -> |
559 |
:-( |
Caps#caps.version == make_disco_hash(Els, sha1); |
560 |
|
<<"sha-224">> -> |
561 |
:-( |
Caps#caps.version == make_disco_hash(Els, sha224); |
562 |
|
<<"sha-256">> -> |
563 |
:-( |
Caps#caps.version == make_disco_hash(Els, sha256); |
564 |
|
<<"sha-384">> -> |
565 |
:-( |
Caps#caps.version == make_disco_hash(Els, sha384); |
566 |
|
<<"sha-512">> -> |
567 |
:-( |
Caps#caps.version == make_disco_hash(Els, sha512); |
568 |
:-( |
_ -> true |
569 |
|
end. |
570 |
|
|
571 |
|
concat_features(Els) -> |
572 |
58 |
lists:usort(lists:flatmap(fun (#xmlel{name = |
573 |
|
<<"feature">>, |
574 |
|
attrs = Attrs}) -> |
575 |
754 |
[[xml:get_attr_s(<<"var">>, Attrs), $<]]; |
576 |
116 |
(_) -> [] |
577 |
|
end, |
578 |
|
Els)). |
579 |
|
|
580 |
|
concat_identities(Els) -> |
581 |
58 |
lists:sort(lists:flatmap(fun (#xmlel{name = |
582 |
|
<<"identity">>, |
583 |
|
attrs = Attrs}) -> |
584 |
58 |
[[xml:get_attr_s(<<"category">>, Attrs), |
585 |
|
$/, xml:get_attr_s(<<"type">>, Attrs), |
586 |
|
$/, |
587 |
|
xml:get_attr_s(<<"xml:lang">>, Attrs), |
588 |
|
$/, xml:get_attr_s(<<"name">>, Attrs), |
589 |
|
$<]]; |
590 |
812 |
(_) -> [] |
591 |
|
end, |
592 |
|
Els)). |
593 |
|
|
594 |
|
concat_info(Els) -> |
595 |
58 |
lists:sort(lists:flatmap(fun(El) -> |
596 |
870 |
concat_xdata_fields(mongoose_data_forms:parse_form(El)) |
597 |
|
end, |
598 |
|
Els)). |
599 |
|
|
600 |
|
concat_xdata_fields(#{type := <<"result">>, kvs := KVs, ns := NS}) -> |
601 |
58 |
Res = maps:fold(fun(Var, Values, VarFields) -> |
602 |
62 |
NewField = [[V, $<] || V <- [Var | lists:sort(Values)]], |
603 |
62 |
[NewField | VarFields] |
604 |
|
end, [], KVs), |
605 |
58 |
[[NS, $<, lists:sort(Res)]]; |
606 |
|
concat_xdata_fields(_) -> |
607 |
812 |
[]. |
608 |
|
|
609 |
|
gb_trees_fold(F, Acc, Tree) -> |
610 |
:-( |
Iter = gb_trees:iterator(Tree), |
611 |
:-( |
gb_trees_fold_iter(F, Acc, Iter). |
612 |
|
|
613 |
|
gb_trees_fold_iter(F, Acc, Iter) -> |
614 |
:-( |
case gb_trees:next(Iter) of |
615 |
|
{Key, Val, NewIter} -> |
616 |
:-( |
NewAcc = F(Key, Val, Acc), |
617 |
:-( |
gb_trees_fold_iter(F, NewAcc, NewIter); |
618 |
:-( |
_ -> Acc |
619 |
|
end. |
620 |
|
|
621 |
|
is_valid_node(Node) -> |
622 |
189 |
case mongoose_bin:tokens(Node, <<"#">>) of |
623 |
|
[?MONGOOSE_URI|_] -> |
624 |
6 |
true; |
625 |
|
_ -> |
626 |
183 |
false |
627 |
|
end. |
628 |
|
|
629 |
|
db_type(_HostType) -> |
630 |
2 |
mnesia. |