mirror of
https://github.com/processone/ejabberd.git
synced 2024-11-22 16:20:52 +01:00
several pubsub fixes and improvements. also fixes EJAB-913 and EJAB-871
SVN Revision: 2059
This commit is contained in:
parent
fec3742aaa
commit
3d766c1cb5
35
ChangeLog
35
ChangeLog
@ -1,3 +1,38 @@
|
|||||||
|
2009-04-30 Christophe Romain <christophe.romain@process-one.net>
|
||||||
|
|
||||||
|
* src/mod_caps.erl: Set debug message to DEBUG (from debian patch,
|
||||||
|
thanks to Sergei Golovan)
|
||||||
|
|
||||||
|
* src/mod_pubsub/mod_pubsub.erl: Remove subscriptions when anonymous
|
||||||
|
user removed (EJAB-913) (thanks to Andy Skelton)
|
||||||
|
|
||||||
|
* src/mod_pubsub/mod_pubsub.erl: Fix disco#items bug on root node,
|
||||||
|
and fix other minor typo from previous patch.
|
||||||
|
|
||||||
|
* src/mod_pubsub/mod_pubsub.erl: Avoid calling get_user_resources
|
||||||
|
on non local domain when pep_sendlast_offline is enabled
|
||||||
|
|
||||||
|
* src/mod_pubsub/mod_pubsub.erl: Reduce send_last_item load and number
|
||||||
|
of calls to get_caps
|
||||||
|
|
||||||
|
* src/mod_pubsub/mod_pubsub.erl: Fix get_entity_* not returning node
|
||||||
|
* src/mod_pubsub/node_default.erl: Likewise
|
||||||
|
* src/mod_pubsub/nodetree_default.erl: Likewise
|
||||||
|
|
||||||
|
* src/mod_pubsub/mod_pubsub.erl: Retract policy should obey
|
||||||
|
pubsub#publish_model (EJAB-871) (thanks to Matthew Baron)
|
||||||
|
* src/mod_pubsub/node_default.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_mb.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_dispatch.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_buddy.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_private.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_public.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_default.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_pep.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_club.erl: Likewise
|
||||||
|
* src/mod_pubsub/node_flat.erl: Likewise
|
||||||
|
* src/mod_pubsub/node.template: Likewise
|
||||||
|
|
||||||
2009-05-06 Badlop <badlop@process-one.net>
|
2009-05-06 Badlop <badlop@process-one.net>
|
||||||
|
|
||||||
* src/ejabberd_c2s.erl: Replace TYPE/1 with is_TYPE/1 (EJAB-922)
|
* src/ejabberd_c2s.erl: Replace TYPE/1 with is_TYPE/1 (EJAB-922)
|
||||||
|
@ -337,7 +337,7 @@ handle_cast({disco_response, From, _To,
|
|||||||
mnesia:dirty_write(#caps_features{node_pair = BinaryNode, features = features_to_binary(Features)}),
|
mnesia:dirty_write(#caps_features{node_pair = BinaryNode, features = features_to_binary(Features)}),
|
||||||
gen_server:cast(self(), visit_feature_queries);
|
gen_server:cast(self(), visit_feature_queries);
|
||||||
error ->
|
error ->
|
||||||
?ERROR_MSG("ID '~s' matches no query", [ID])
|
?DEBUG("ID '~s' matches no query", [ID])
|
||||||
end;
|
end;
|
||||||
{error, _} ->
|
{error, _} ->
|
||||||
%% XXX: if we get error, we cache empty feature not to probe the client continuously
|
%% XXX: if we get error, we cache empty feature not to probe the client continuously
|
||||||
@ -346,7 +346,7 @@ handle_cast({disco_response, From, _To,
|
|||||||
mnesia:dirty_write(#caps_features{node_pair = BinaryNode}),
|
mnesia:dirty_write(#caps_features{node_pair = BinaryNode}),
|
||||||
gen_server:cast(self(), visit_feature_queries);
|
gen_server:cast(self(), visit_feature_queries);
|
||||||
error ->
|
error ->
|
||||||
?ERROR_MSG("ID '~s' matches no query", [ID])
|
?DEBUG("ID '~s' matches no query", [ID])
|
||||||
end;
|
end;
|
||||||
%gen_server:cast(self(), visit_feature_queries),
|
%gen_server:cast(self(), visit_feature_queries),
|
||||||
%?DEBUG("Error IQ reponse from ~s:~n~p", [jlib:jid_to_string(From), SubEls]);
|
%?DEBUG("Error IQ reponse from ~s:~n~p", [jlib:jid_to_string(From), SubEls]);
|
||||||
|
@ -49,7 +49,7 @@ behaviour_info(callbacks) ->
|
|||||||
{subscribe_node, 7},
|
{subscribe_node, 7},
|
||||||
{unsubscribe_node, 4},
|
{unsubscribe_node, 4},
|
||||||
{publish_item, 6},
|
{publish_item, 6},
|
||||||
{delete_item, 3},
|
{delete_item, 4},
|
||||||
{remove_extra_items, 3},
|
{remove_extra_items, 3},
|
||||||
{get_node_affiliations, 1},
|
{get_node_affiliations, 1},
|
||||||
{get_entity_affiliations, 2},
|
{get_entity_affiliations, 2},
|
||||||
|
@ -179,6 +179,7 @@ init([ServerHost, Opts]) ->
|
|||||||
ejabberd_hooks:add(presence_probe_hook, ServerHost, ?MODULE, presence_probe, 50),
|
ejabberd_hooks:add(presence_probe_hook, ServerHost, ?MODULE, presence_probe, 50),
|
||||||
ejabberd_hooks:add(roster_out_subscription, ServerHost, ?MODULE, out_subscription, 50),
|
ejabberd_hooks:add(roster_out_subscription, ServerHost, ?MODULE, out_subscription, 50),
|
||||||
ejabberd_hooks:add(remove_user, ServerHost, ?MODULE, remove_user, 50),
|
ejabberd_hooks:add(remove_user, ServerHost, ?MODULE, remove_user, 50),
|
||||||
|
ejabberd_hooks:add(anonymous_purge_hook, ServerHost, ?MODULE, remove_user, 50),
|
||||||
gen_iq_handler:add_iq_handler(ejabberd_sm, ServerHost, ?NS_PUBSUB, ?MODULE, iq_sm, IQDisc),
|
gen_iq_handler:add_iq_handler(ejabberd_sm, ServerHost, ?NS_PUBSUB, ?MODULE, iq_sm, IQDisc),
|
||||||
gen_iq_handler:add_iq_handler(ejabberd_sm, ServerHost, ?NS_PUBSUB_OWNER, ?MODULE, iq_sm, IQDisc),
|
gen_iq_handler:add_iq_handler(ejabberd_sm, ServerHost, ?NS_PUBSUB_OWNER, ?MODULE, iq_sm, IQDisc),
|
||||||
case lists:member(?PEPNODE, Plugins) of
|
case lists:member(?PEPNODE, Plugins) of
|
||||||
@ -409,69 +410,89 @@ send_loop(State) ->
|
|||||||
%% and is not able to "store" events of remote users (via s2s)
|
%% and is not able to "store" events of remote users (via s2s)
|
||||||
%% this makes that hack only work for local domain by now
|
%% this makes that hack only work for local domain by now
|
||||||
if State#state.pep_sendlast_offline ->
|
if State#state.pep_sendlast_offline ->
|
||||||
case catch ejabberd_c2s:get_subscribed(Pid) of
|
{User, Server, Resource} = jlib:jid_tolower(JID),
|
||||||
Contacts when is_list(Contacts) ->
|
case mod_caps:get_caps({User, Server, Resource}) of
|
||||||
{User, Server, Resource} = jlib:jid_tolower(JID),
|
nothing ->
|
||||||
lists:foreach(
|
%% we don't have caps, no need to handle PEP items
|
||||||
fun({U, S, R}) -> %% local contacts
|
ok;
|
||||||
case ejabberd_sm:get_user_resources(U, S) of
|
|
||||||
[] -> %% offline
|
|
||||||
case S of
|
|
||||||
ServerHost -> %% local contact, so we may have pep items
|
|
||||||
PeerJID = jlib:make_jid(U, S, R),
|
|
||||||
self() ! {presence, User, Server, [Resource], PeerJID};
|
|
||||||
_ -> %% remote contact, no items available
|
|
||||||
ok
|
|
||||||
end;
|
|
||||||
_ -> %% online
|
|
||||||
% this is already handled by presence probe
|
|
||||||
ok
|
|
||||||
end;
|
|
||||||
(_) -> %% remote contacts
|
|
||||||
% we can not do anything in any cases
|
|
||||||
ok
|
|
||||||
end, Contacts);
|
|
||||||
_ ->
|
_ ->
|
||||||
ok
|
case catch ejabberd_c2s:get_subscribed(Pid) of
|
||||||
|
Contacts when is_list(Contacts) ->
|
||||||
|
lists:foreach(
|
||||||
|
fun({U, S, R}) ->
|
||||||
|
case S of
|
||||||
|
ServerHost -> %% local contacts
|
||||||
|
case ejabberd_sm:get_user_resources(U, S) of
|
||||||
|
[] -> %% offline
|
||||||
|
PeerJID = jlib:make_jid(U, S, R),
|
||||||
|
self() ! {presence, User, Server, [Resource], PeerJID};
|
||||||
|
_ -> %% online
|
||||||
|
% this is already handled by presence probe
|
||||||
|
ok
|
||||||
|
end;
|
||||||
|
_ -> %% remote contacts
|
||||||
|
% we can not do anything in any cases
|
||||||
|
ok
|
||||||
|
end
|
||||||
|
end, Contacts);
|
||||||
|
_ ->
|
||||||
|
ok
|
||||||
|
end
|
||||||
end;
|
end;
|
||||||
true ->
|
true ->
|
||||||
ok
|
ok
|
||||||
end,
|
end,
|
||||||
send_loop(State);
|
send_loop(State);
|
||||||
{presence, User, Server, Resources, JID} ->
|
{presence, User, Server, Resources, JID} ->
|
||||||
Owner = jlib:jid_remove_resource(jlib:jid_tolower(JID)),
|
%% get resources caps and check if processing is needed
|
||||||
Host = State#state.host,
|
{HasCaps, ResourcesCaps} = lists:foldl(fun(Resource, {R, L}) ->
|
||||||
ServerHost = State#state.server_host,
|
case mod_caps:get_caps({User, Server, Resource}) of
|
||||||
lists:foreach(fun(#pubsub_node{nodeid = {_, Node}, type = Type, id = NodeId, options = Options}) ->
|
nothing -> {R, L};
|
||||||
case get_option(Options, send_last_published_item) of
|
Caps -> {true, [{Resource, Caps} | L]}
|
||||||
on_sub_and_presence ->
|
end
|
||||||
lists:foreach(fun(Resource) ->
|
end, {false, []}, Resources),
|
||||||
LJID = {User, Server, Resource},
|
case HasCaps of
|
||||||
case is_caps_notify(ServerHost, Node, LJID) of
|
true ->
|
||||||
true ->
|
Host = State#state.host,
|
||||||
Subscribed = case get_option(Options, access_model) of
|
ServerHost = State#state.server_host,
|
||||||
open -> true;
|
Owner = jlib:jid_remove_resource(jlib:jid_tolower(JID)),
|
||||||
presence -> true;
|
lists:foreach(fun(#pubsub_node{nodeid = {_, Node}, type = Type, id = NodeId, options = Options}) ->
|
||||||
whitelist -> false; % subscribers are added manually
|
case get_option(Options, send_last_published_item) of
|
||||||
authorize -> false; % likewise
|
on_sub_and_presence ->
|
||||||
roster ->
|
lists:foreach(fun({Resource, Caps}) ->
|
||||||
Grps = get_option(Options, roster_groups_allowed, []),
|
CapsNotify = case catch mod_caps:get_features(ServerHost, Caps) of
|
||||||
{OU, OS, _} = Owner,
|
Features when is_list(Features) -> lists:member(Node ++ "+notify", Features);
|
||||||
element(2, get_roster_info(OU, OS, LJID, Grps))
|
_ -> false
|
||||||
end,
|
end,
|
||||||
if Subscribed ->
|
case CapsNotify of
|
||||||
send_items(Owner, Node, NodeId, Type, LJID, last);
|
true ->
|
||||||
true ->
|
LJID = {User, Server, Resource},
|
||||||
ok
|
Subscribed = case get_option(Options, access_model) of
|
||||||
end;
|
open -> true;
|
||||||
false ->
|
presence -> true;
|
||||||
ok
|
whitelist -> false; % subscribers are added manually
|
||||||
end
|
authorize -> false; % likewise
|
||||||
end, Resources);
|
roster ->
|
||||||
_ ->
|
Grps = get_option(Options, roster_groups_allowed, []),
|
||||||
ok
|
{OU, OS, _} = Owner,
|
||||||
end
|
element(2, get_roster_info(OU, OS, LJID, Grps))
|
||||||
end, tree_action(Host, get_nodes, [Owner, JID])),
|
end,
|
||||||
|
if Subscribed ->
|
||||||
|
send_items(Owner, Node, NodeId, Type, LJID, last);
|
||||||
|
true ->
|
||||||
|
ok
|
||||||
|
end;
|
||||||
|
false ->
|
||||||
|
ok
|
||||||
|
end
|
||||||
|
end, ResourcesCaps);
|
||||||
|
_ ->
|
||||||
|
ok
|
||||||
|
end
|
||||||
|
end, tree_action(Host, get_nodes, [Owner, JID]));
|
||||||
|
false ->
|
||||||
|
ok
|
||||||
|
end,
|
||||||
send_loop(State);
|
send_loop(State);
|
||||||
stop ->
|
stop ->
|
||||||
ok
|
ok
|
||||||
@ -535,7 +556,7 @@ disco_sm_features(Acc, From, To, Node, _Lang) ->
|
|||||||
|
|
||||||
disco_sm_items(Acc, From, To, [], _Lang) ->
|
disco_sm_items(Acc, From, To, [], _Lang) ->
|
||||||
Host = To#jid.lserver,
|
Host = To#jid.lserver,
|
||||||
case tree_action(Host, get_nodes, [Host, From]) of
|
case tree_action(Host, get_subnodes, [Host, [], From]) of
|
||||||
[] ->
|
[] ->
|
||||||
Acc;
|
Acc;
|
||||||
Nodes ->
|
Nodes ->
|
||||||
@ -677,7 +698,7 @@ handle_cast({remove_user, LUser, LServer}, State) ->
|
|||||||
%% remove user's PEP nodes
|
%% remove user's PEP nodes
|
||||||
lists:foreach(fun(#pubsub_node{nodeid={NodeKey, NodeName}}) ->
|
lists:foreach(fun(#pubsub_node{nodeid={NodeKey, NodeName}}) ->
|
||||||
delete_node(NodeKey, NodeName, Owner)
|
delete_node(NodeKey, NodeName, Owner)
|
||||||
end, tree_action(Host, get_nodes, [jlib:jid_tolower(Owner)])),
|
end, tree_action(Host, get_nodes, [jlib:jid_tolower(Owner), Owner])),
|
||||||
%% remove user's nodes
|
%% remove user's nodes
|
||||||
delete_node(Host, ["home", LServer, LUser], Owner),
|
delete_node(Host, ["home", LServer, LUser], Owner),
|
||||||
{noreply, State};
|
{noreply, State};
|
||||||
@ -717,7 +738,6 @@ terminate(_Reason, #state{host = Host,
|
|||||||
nodetree = TreePlugin,
|
nodetree = TreePlugin,
|
||||||
plugins = Plugins,
|
plugins = Plugins,
|
||||||
send_loop = SendLoop}) ->
|
send_loop = SendLoop}) ->
|
||||||
terminate_plugins(Host, ServerHost, Plugins, TreePlugin),
|
|
||||||
ejabberd_router:unregister_route(Host),
|
ejabberd_router:unregister_route(Host),
|
||||||
case lists:member(?PEPNODE, Plugins) of
|
case lists:member(?PEPNODE, Plugins) of
|
||||||
true ->
|
true ->
|
||||||
@ -735,11 +755,12 @@ terminate(_Reason, #state{host = Host,
|
|||||||
ejabberd_hooks:delete(presence_probe_hook, ServerHost, ?MODULE, presence_probe, 50),
|
ejabberd_hooks:delete(presence_probe_hook, ServerHost, ?MODULE, presence_probe, 50),
|
||||||
ejabberd_hooks:delete(roster_out_subscription, ServerHost, ?MODULE, out_subscription, 50),
|
ejabberd_hooks:delete(roster_out_subscription, ServerHost, ?MODULE, out_subscription, 50),
|
||||||
ejabberd_hooks:delete(remove_user, ServerHost, ?MODULE, remove_user, 50),
|
ejabberd_hooks:delete(remove_user, ServerHost, ?MODULE, remove_user, 50),
|
||||||
|
ejabberd_hooks:delete(anonymous_purge_hook, ServerHost, ?MODULE, remove_user, 50),
|
||||||
gen_iq_handler:remove_iq_handler(ejabberd_sm, ServerHost, ?NS_PUBSUB),
|
gen_iq_handler:remove_iq_handler(ejabberd_sm, ServerHost, ?NS_PUBSUB),
|
||||||
gen_iq_handler:remove_iq_handler(ejabberd_sm, ServerHost, ?NS_PUBSUB_OWNER),
|
gen_iq_handler:remove_iq_handler(ejabberd_sm, ServerHost, ?NS_PUBSUB_OWNER),
|
||||||
mod_disco:unregister_feature(ServerHost, ?NS_PUBSUB),
|
mod_disco:unregister_feature(ServerHost, ?NS_PUBSUB),
|
||||||
SendLoop ! stop,
|
SendLoop ! stop,
|
||||||
ok.
|
terminate_plugins(Host, ServerHost, Plugins, TreePlugin).
|
||||||
|
|
||||||
%%--------------------------------------------------------------------
|
%%--------------------------------------------------------------------
|
||||||
%% Func: code_change(OldVsn, State, Extra) -> {ok, NewState}
|
%% Func: code_change(OldVsn, State, Extra) -> {ok, NewState}
|
||||||
@ -1748,23 +1769,24 @@ delete_item(_, "", _, _, _) ->
|
|||||||
%% Request does not specify a node
|
%% Request does not specify a node
|
||||||
{error, extended_error(?ERR_BAD_REQUEST, "node-required")};
|
{error, extended_error(?ERR_BAD_REQUEST, "node-required")};
|
||||||
delete_item(Host, Node, Publisher, ItemId, ForceNotify) ->
|
delete_item(Host, Node, Publisher, ItemId, ForceNotify) ->
|
||||||
Action = fun(#pubsub_node{type = Type, id = NodeId}) ->
|
Action = fun(#pubsub_node{options = Options, type = Type, id = NodeId}) ->
|
||||||
Features = features(Type),
|
Features = features(Type),
|
||||||
PersistentFeature = lists:member("persistent-items", Features),
|
PersistentFeature = lists:member("persistent-items", Features),
|
||||||
DeleteFeature = lists:member("delete-items", Features),
|
DeleteFeature = lists:member("delete-items", Features),
|
||||||
if
|
PublishModel = get_option(Options, publish_model),
|
||||||
%%-> iq_pubsub just does that matchs
|
if
|
||||||
%% %% Request does not specify an item
|
%%-> iq_pubsub just does that matchs
|
||||||
%% {error, extended_error(?ERR_BAD_REQUEST, "item-required")};
|
%% %% Request does not specify an item
|
||||||
not PersistentFeature ->
|
%% {error, extended_error(?ERR_BAD_REQUEST, "item-required")};
|
||||||
%% Node does not support persistent items
|
not PersistentFeature ->
|
||||||
{error, extended_error(?ERR_FEATURE_NOT_IMPLEMENTED, unsupported, "persistent-items")};
|
%% Node does not support persistent items
|
||||||
not DeleteFeature ->
|
{error, extended_error(?ERR_FEATURE_NOT_IMPLEMENTED, unsupported, "persistent-items")};
|
||||||
%% Service does not support item deletion
|
not DeleteFeature ->
|
||||||
{error, extended_error(?ERR_FEATURE_NOT_IMPLEMENTED, unsupported, "delete-items")};
|
%% Service does not support item deletion
|
||||||
true ->
|
{error, extended_error(?ERR_FEATURE_NOT_IMPLEMENTED, unsupported, "delete-items")};
|
||||||
node_call(Type, delete_item, [NodeId, Publisher, ItemId])
|
true ->
|
||||||
end
|
node_call(Type, delete_item, [NodeId, Publisher, PublishModel, ItemId])
|
||||||
|
end
|
||||||
end,
|
end,
|
||||||
Reply = [],
|
Reply = [],
|
||||||
case transaction(Host, Node, Action, sync_dirty) of
|
case transaction(Host, Node, Action, sync_dirty) of
|
||||||
@ -1905,8 +1927,9 @@ get_items(Host, Node, From, SubId, SMaxItems, ItemIDs) ->
|
|||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
%% @spec (Host, NodeId, Type LJID, Number) -> any()
|
%% @spec (Host, Node, NodeId, Type LJID, Number) -> any()
|
||||||
%% Host = pubsubHost()
|
%% Host = pubsubHost()
|
||||||
|
%% Node = pubsubNode()
|
||||||
%% NodeId = pubsubNodeId()
|
%% NodeId = pubsubNodeId()
|
||||||
%% Type = pubsubNodeType()
|
%% Type = pubsubNodeType()
|
||||||
%% LJID = {U, S, []}
|
%% LJID = {U, S, []}
|
||||||
|
@ -48,7 +48,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -130,8 +130,8 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
||||||
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
||||||
|
|
||||||
delete_item(NodeId, JID, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
node_default:delete_item(NodeId, JID, ItemId).
|
node_default:delete_item(NodeId, Publisher, PublishModel, ItemId).
|
||||||
|
|
||||||
purge_node(NodeId, Owner) ->
|
purge_node(NodeId, Owner) ->
|
||||||
node_default:purge_node(NodeId, Owner).
|
node_default:purge_node(NodeId, Owner).
|
||||||
|
@ -49,7 +49,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -133,8 +133,8 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
||||||
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
||||||
|
|
||||||
delete_item(NodeId, JID, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
node_default:delete_item(NodeId, JID, ItemId).
|
node_default:delete_item(NodeId, Publisher, PublishModel, ItemId).
|
||||||
|
|
||||||
purge_node(NodeId, Owner) ->
|
purge_node(NodeId, Owner) ->
|
||||||
node_default:purge_node(NodeId, Owner).
|
node_default:purge_node(NodeId, Owner).
|
||||||
|
@ -49,7 +49,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -132,8 +132,8 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
||||||
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
||||||
|
|
||||||
delete_item(NodeId, JID, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
node_default:delete_item(NodeId, JID, ItemId).
|
node_default:delete_item(NodeId, Publisher, PublishModel, ItemId).
|
||||||
|
|
||||||
purge_node(NodeId, Owner) ->
|
purge_node(NodeId, Owner) ->
|
||||||
node_default:purge_node(NodeId, Owner).
|
node_default:purge_node(NodeId, Owner).
|
||||||
|
@ -56,7 +56,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -482,23 +482,25 @@ remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
|||||||
%% Return the new items list:
|
%% Return the new items list:
|
||||||
{result, {NewItems, OldItems}}.
|
{result, {NewItems, OldItems}}.
|
||||||
|
|
||||||
%% @spec (NodeId, JID, ItemId) ->
|
%% @spec (NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
%% {error, Reason::stanzaError()} |
|
%% {error, Reason::stanzaError()} |
|
||||||
%% {result, []}
|
%% {result, []}
|
||||||
%% NodeId = mod_pubsub:pubsubNodeId()
|
%% NodeId = mod_pubsub:pubsubNodeId()
|
||||||
%% JID = mod_pubsub:jid()
|
%% Publisher = mod_pubsub:jid()
|
||||||
|
%% PublishModel = atom()
|
||||||
%% ItemId = string()
|
%% ItemId = string()
|
||||||
%% @doc <p>Triggers item deletion.</p>
|
%% @doc <p>Triggers item deletion.</p>
|
||||||
%% <p>Default plugin: The user performing the deletion must be the node owner
|
%% <p>Default plugin: The user performing the deletion must be the node owner
|
||||||
%% or a publisher.</p>
|
%% or a publisher, or PublishModel being open.</p>
|
||||||
delete_item(NodeId, Publisher, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
SubKey = jlib:jid_tolower(Publisher),
|
SubKey = jlib:jid_tolower(Publisher),
|
||||||
GenKey = jlib:jid_remove_resource(SubKey),
|
GenKey = jlib:jid_remove_resource(SubKey),
|
||||||
GenState = get_state(NodeId, GenKey),
|
GenState = get_state(NodeId, GenKey),
|
||||||
#pubsub_state{affiliation = Affiliation, items = Items} = GenState,
|
#pubsub_state{affiliation = Affiliation, items = Items} = GenState,
|
||||||
Allowed = (Affiliation == publisher) orelse (Affiliation == owner)
|
Allowed = (Affiliation == publisher) orelse (Affiliation == owner)
|
||||||
|
orelse (PublishModel == open)
|
||||||
orelse case get_item(NodeId, ItemId) of
|
orelse case get_item(NodeId, ItemId) of
|
||||||
{result, #pubsub_item{creation = {GenKey, _}}} -> true;
|
{result, #pubsub_item{creation = {_, GenKey}}} -> true;
|
||||||
_ -> false
|
_ -> false
|
||||||
end,
|
end,
|
||||||
if
|
if
|
||||||
@ -554,7 +556,7 @@ get_entity_affiliations(_Host, Owner) ->
|
|||||||
%% TODO check if Host needed
|
%% TODO check if Host needed
|
||||||
#pubsub_state{stateid = {GenKey, '_'}, _ = '_'}),
|
#pubsub_state{stateid = {GenKey, '_'}, _ = '_'}),
|
||||||
Tr = fun(#pubsub_state{stateid = {_, N}, affiliation = A}) ->
|
Tr = fun(#pubsub_state{stateid = {_, N}, affiliation = A}) ->
|
||||||
{N, A}
|
{get_nodename(N), A}
|
||||||
end,
|
end,
|
||||||
{result, lists:map(Tr, States)}.
|
{result, lists:map(Tr, States)}.
|
||||||
|
|
||||||
@ -604,7 +606,7 @@ get_entity_subscriptions(_Host, Owner) ->
|
|||||||
#pubsub_state{stateid = {SubKey, '_'}, _ = '_'})
|
#pubsub_state{stateid = {SubKey, '_'}, _ = '_'})
|
||||||
end,
|
end,
|
||||||
Tr = fun(#pubsub_state{stateid = {J, N}, subscription = S}) ->
|
Tr = fun(#pubsub_state{stateid = {J, N}, subscription = S}) ->
|
||||||
{N, S, J}
|
{get_nodename(N), S, J}
|
||||||
end,
|
end,
|
||||||
{result, lists:map(Tr, States)}.
|
{result, lists:map(Tr, States)}.
|
||||||
|
|
||||||
@ -810,3 +812,11 @@ can_fetch_item(outcast, _) -> false;
|
|||||||
can_fetch_item(none, subscribed) -> true;
|
can_fetch_item(none, subscribed) -> true;
|
||||||
can_fetch_item(none, none) -> false;
|
can_fetch_item(none, none) -> false;
|
||||||
can_fetch_item(_Affiliation, _Subscription) -> false.
|
can_fetch_item(_Affiliation, _Subscription) -> false.
|
||||||
|
|
||||||
|
%% @spec (NodeId) -> Node
|
||||||
|
%% @doc retreive pubsubNode() representation giving a NodeId
|
||||||
|
get_nodename(NodeId) ->
|
||||||
|
case mnesia:index_read(pubsub_node, NodeId, #pubsub_node.id) of
|
||||||
|
[#pubsub_node{nodeid = {_, Node}}] -> Node;
|
||||||
|
_ -> []
|
||||||
|
end.
|
||||||
|
@ -47,7 +47,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -134,7 +134,7 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(_NodeId, _MaxItems, ItemIds) ->
|
remove_extra_items(_NodeId, _MaxItems, ItemIds) ->
|
||||||
{result, {ItemIds, []}}.
|
{result, {ItemIds, []}}.
|
||||||
|
|
||||||
delete_item(_NodeId, _JID, _ItemId) ->
|
delete_item(_NodeId, _Publisher, _PublishModel, _ItemId) ->
|
||||||
{error, ?ERR_ITEM_NOT_FOUND}.
|
{error, ?ERR_ITEM_NOT_FOUND}.
|
||||||
|
|
||||||
purge_node(_NodeId, _Owner) ->
|
purge_node(_NodeId, _Owner) ->
|
||||||
|
@ -40,7 +40,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -119,8 +119,8 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
||||||
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
||||||
|
|
||||||
delete_item(NodeId, JID, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
node_default:delete_item(NodeId, JID, ItemId).
|
node_default:delete_item(NodeId, Publisher, PublishModel, ItemId).
|
||||||
|
|
||||||
purge_node(NodeId, Owner) ->
|
purge_node(NodeId, Owner) ->
|
||||||
node_default:purge_node(NodeId, Owner).
|
node_default:purge_node(NodeId, Owner).
|
||||||
|
@ -52,7 +52,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -140,8 +140,8 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
||||||
node_pep:remove_extra_items(NodeId, MaxItems, ItemIds).
|
node_pep:remove_extra_items(NodeId, MaxItems, ItemIds).
|
||||||
|
|
||||||
delete_item(NodeId, JID, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
node_pep:delete_item(NodeId, JID, ItemId).
|
node_pep:delete_item(NodeId, Publisher, PublishModel, ItemId).
|
||||||
|
|
||||||
purge_node(NodeId, Owner) ->
|
purge_node(NodeId, Owner) ->
|
||||||
node_pep:purge_node(NodeId, Owner).
|
node_pep:purge_node(NodeId, Owner).
|
||||||
|
@ -45,7 +45,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -161,8 +161,8 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
||||||
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
||||||
|
|
||||||
delete_item(NodeId, JID, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
node_default:delete_item(NodeId, JID, ItemId).
|
node_default:delete_item(NodeId, Publisher, PublishModel, ItemId).
|
||||||
|
|
||||||
purge_node(NodeId, Owner) ->
|
purge_node(NodeId, Owner) ->
|
||||||
node_default:purge_node(NodeId, Owner).
|
node_default:purge_node(NodeId, Owner).
|
||||||
|
@ -49,7 +49,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -135,8 +135,8 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
||||||
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
||||||
|
|
||||||
delete_item(NodeId, JID, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
node_default:delete_item(NodeId, JID, ItemId).
|
node_default:delete_item(NodeId, Publisher, PublishModel, ItemId).
|
||||||
|
|
||||||
purge_node(NodeId, Owner) ->
|
purge_node(NodeId, Owner) ->
|
||||||
node_default:purge_node(NodeId, Owner).
|
node_default:purge_node(NodeId, Owner).
|
||||||
|
@ -49,7 +49,7 @@
|
|||||||
subscribe_node/7,
|
subscribe_node/7,
|
||||||
unsubscribe_node/4,
|
unsubscribe_node/4,
|
||||||
publish_item/6,
|
publish_item/6,
|
||||||
delete_item/3,
|
delete_item/4,
|
||||||
remove_extra_items/3,
|
remove_extra_items/3,
|
||||||
get_entity_affiliations/2,
|
get_entity_affiliations/2,
|
||||||
get_node_affiliations/1,
|
get_node_affiliations/1,
|
||||||
@ -132,8 +132,8 @@ publish_item(NodeId, Publisher, Model, MaxItems, ItemId, Payload) ->
|
|||||||
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
remove_extra_items(NodeId, MaxItems, ItemIds) ->
|
||||||
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
node_default:remove_extra_items(NodeId, MaxItems, ItemIds).
|
||||||
|
|
||||||
delete_item(NodeId, JID, ItemId) ->
|
delete_item(NodeId, Publisher, PublishModel, ItemId) ->
|
||||||
node_default:delete_item(NodeId, JID, ItemId).
|
node_default:delete_item(NodeId, Publisher, PublishModel, ItemId).
|
||||||
|
|
||||||
purge_node(NodeId, Owner) ->
|
purge_node(NodeId, Owner) ->
|
||||||
node_default:purge_node(NodeId, Owner).
|
node_default:purge_node(NodeId, Owner).
|
||||||
|
@ -73,6 +73,7 @@ init(_Host, _ServerHost, _Opts) ->
|
|||||||
mnesia:create_table(pubsub_node,
|
mnesia:create_table(pubsub_node,
|
||||||
[{disc_copies, [node()]},
|
[{disc_copies, [node()]},
|
||||||
{attributes, record_info(fields, pubsub_node)}]),
|
{attributes, record_info(fields, pubsub_node)}]),
|
||||||
|
mnesia:add_table_index(pubsub_node, id),
|
||||||
NodesFields = record_info(fields, pubsub_node),
|
NodesFields = record_info(fields, pubsub_node),
|
||||||
case mnesia:table_info(pubsub_node, attributes) of
|
case mnesia:table_info(pubsub_node, attributes) of
|
||||||
NodesFields -> ok;
|
NodesFields -> ok;
|
||||||
|
Loading…
Reference in New Issue
Block a user