Welcome to mirror list, hosted at ThFree Co, Russian Federation.

github.com/processone/ejabberd.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'src/node_pep_odbc.erl')
-rw-r--r--src/node_pep_odbc.erl449
1 files changed, 449 insertions, 0 deletions
diff --git a/src/node_pep_odbc.erl b/src/node_pep_odbc.erl
new file mode 100644
index 000000000..d997d9ce1
--- /dev/null
+++ b/src/node_pep_odbc.erl
@@ -0,0 +1,449 @@
+%%% ====================================================================
+%%% ``The contents of this file are subject to the Erlang Public License,
+%%% Version 1.1, (the "License"); you may not use this file except in
+%%% compliance with the License. You should have received a copy of the
+%%% Erlang Public License along with this software. If not, it can be
+%%% retrieved via the world wide web at http://www.erlang.org/.
+%%%
+%%%
+%%% Software distributed under the License is distributed on an "AS IS"
+%%% basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
+%%% the License for the specific language governing rights and limitations
+%%% under the License.
+%%%
+%%%
+%%% The Initial Developer of the Original Code is ProcessOne.
+%%% Portions created by ProcessOne are Copyright 2006-2013, ProcessOne
+%%% All Rights Reserved.''
+%%% This software is copyright 2006-2013, ProcessOne.
+%%%
+%%%
+%%% @copyright 2006-2013 ProcessOne
+%%% @author Christophe Romain <christophe.romain@process-one.net>
+%%% [http://www.process-one.net/]
+%%% @version {@vsn}, {@date} {@time}
+%%% @end
+%%% ====================================================================
+
+%%% @doc The module <strong>{@module}</strong> is the pep PubSub plugin.
+%%% <p>PubSub plugin nodes are using the {@link gen_pubsub_node} behaviour.</p>
+
+-module(node_pep_odbc).
+
+-author('christophe.romain@process-one.net').
+
+-include("ejabberd.hrl").
+-include("logger.hrl").
+
+-include("pubsub.hrl").
+
+-include("jlib.hrl").
+
+-define(PUBSUB, mod_pubsub_odbc).
+
+-behaviour(gen_pubsub_node).
+
+%% API definition
+-export([init/3, terminate/2, options/0, features/0,
+ create_node_permission/6, create_node/2, delete_node/1,
+ purge_node/2, subscribe_node/8, unsubscribe_node/4,
+ publish_item/6, delete_item/4, remove_extra_items/3,
+ get_entity_affiliations/2, get_node_affiliations/1,
+ get_affiliation/2, set_affiliation/3,
+ get_entity_subscriptions/2,
+ get_entity_subscriptions_for_send_last/2,
+ get_node_subscriptions/1, get_subscriptions/2,
+ set_subscriptions/4, get_pending_nodes/2, get_states/1,
+ get_state/2, set_state/1, get_items/7, get_items/6,
+ get_items/3, get_items/2, get_item/7, get_item/2,
+ set_item/1, get_item_name/3, get_last_items/3,
+ node_to_path/1, path_to_node/1]).
+
+init(Host, ServerHost, Opts) ->
+ node_hometree_odbc:init(Host, ServerHost, Opts),
+ complain_if_modcaps_disabled(ServerHost),
+ ok.
+
+terminate(Host, ServerHost) ->
+ node_hometree_odbc:terminate(Host, ServerHost), ok.
+
+-spec(options/0 :: () -> NodeOptions::mod_pubsub:nodeOptions()).
+options() ->
+ [{odbc, true},
+ {node_type, pep},
+ {deliver_payloads, true},
+ {notify_config, false},
+ {notify_delete, false},
+ {notify_retract, false},
+ {purge_offline, false},
+ {persist_items, false},
+ {max_items, ?MAXITEMS},
+ {subscribe, true},
+ {access_model, presence},
+ {roster_groups_allowed, []},
+ {publish_model, publishers},
+ {notification_type, headline},
+ {max_payload_size, ?MAX_PAYLOAD_SIZE},
+ {send_last_published_item, on_sub_and_presence},
+ {deliver_notifications, true},
+ {presence_based_delivery, true}].
+
+-spec(features/0 :: () -> Features::[binary(),...]).
+features() ->
+ [<<"create-nodes">>, %*
+ <<"auto-create">>, %*
+ <<"auto-subscribe">>, %*
+ <<"delete-nodes">>, %*
+ <<"delete-items">>, %*
+ <<"filtered-notifications">>, %*
+ <<"modify-affiliations">>, <<"outcast-affiliation">>,
+ <<"persistent-items">>,
+ <<"publish">>, %*
+ <<"purge-nodes">>, <<"retract-items">>,
+ <<"retrieve-affiliations">>,
+ <<"retrieve-items">>, %*
+ <<"retrieve-subscriptions">>, <<"subscribe">>].
+
+-spec(create_node_permission/6 ::
+(
+ Host :: mod_pubsub:hostPEP(),
+ ServerHost :: binary(),
+ NodeId :: mod_pubsub:nodeId(),
+ _ParentNodeId :: mod_pubsub:nodeId(),
+ Owner :: jid(),
+ Access :: atom())
+ -> {result, boolean()}
+).
+
+create_node_permission(Host, ServerHost, _NodeId, _ParentNode, Owner, Access) ->
+ LOwner = jlib:jid_tolower(Owner),
+ {User, Server, _Resource} = LOwner,
+ Allowed = case LOwner of
+ {<<"">>, Host, <<"">>} ->
+ true; % pubsub service always allowed
+ _ ->
+ case acl:match_rule(ServerHost, Access, LOwner) of
+ allow ->
+ case Host of
+ {User, Server, _} -> true;
+ _ -> false
+ end;
+ E -> ?DEBUG("Create not allowed : ~p~n", [E]), false
+ end
+ end,
+ {result, Allowed}.
+
+-spec(create_node/2 ::
+(
+ NodeIdx :: mod_pubsub:nodeIdx(),
+ Owner :: jid())
+ -> {result, []}
+).
+create_node(NodeIdx, Owner) ->
+ case node_hometree_odbc:create_node(NodeIdx, Owner) of
+ {result, _} -> {result, []};
+ Error -> Error
+ end.
+
+-spec(delete_node/1 ::
+(
+ Removed :: [mod_pubsub:pubsubNode(),...])
+ -> {result,
+ {[],
+ [{mod_pubsub:pubsubNode(),
+ [{ljid(), [{mod_pubsub:subscription(), mod_pubsub:subId()}]}]}]
+ }
+ }
+).
+delete_node(Removed) ->
+ case node_hometree_odbc:delete_node(Removed) of
+ {result, {_, _, Removed}} -> {result, {[], Removed}};
+ Error -> Error
+ end.
+
+-spec(subscribe_node/8 ::
+(
+ NodeIdx :: mod_pubsub:nodeIdx(),
+ Sender :: jid(),
+ Subscriber :: ljid(),
+ AccessModel :: mod_pubsub:accessModel(),
+ SendLast :: 'never' | 'on_sub' | 'on_sub_and_presence',
+ PresenceSubscription :: boolean(),
+ RosterGroup :: boolean(),
+ Options :: mod_pubsub:subOptions())
+ -> {result, {default, subscribed, mod_pubsub:subId()}}
+ | {result, {default, subscribed, mod_pubsub:subId(), send_last}}
+ | {result, {default, pending, mod_pubsub:subId()}}
+ %%%
+ | {error, _}
+ | {error, _, binary()}
+).
+subscribe_node(NodeId, Sender, Subscriber, AccessModel,
+ SendLast, PresenceSubscription, RosterGroup, Options) ->
+ node_hometree_odbc:subscribe_node(NodeId, Sender,
+ Subscriber, AccessModel, SendLast,
+ PresenceSubscription, RosterGroup,
+ Options).
+
+
+-spec(unsubscribe_node/4 ::
+(
+ NodeIdx :: mod_pubsub:nodeIdx(),
+ Sender :: jid(),
+ Subscriber :: jid(),
+ SubId :: subId())
+ -> {result, []}
+ %
+ | {error, _}
+ | {error, _, binary()}
+).
+unsubscribe_node(NodeIdx, Sender, Subscriber, SubID) ->
+ case node_hometree_odbc:unsubscribe_node(NodeIdx, Sender, Subscriber, SubID) of
+ {error, Error} -> {error, Error};
+ {result, _} -> {result, []}
+ end.
+
+-spec(publish_item/6 ::
+(
+ NodeIdx :: mod_pubsub:nodeIdx(),
+ Publisher :: jid(),
+ PublishModel :: mod_pubsub:publishModel(),
+ Max_Items :: non_neg_integer(),
+ ItemId :: <<>> | mod_pubsub:itemId(),
+ Payload :: mod_pubsub:payload())
+ -> {result, {default, broadcast, [mod_pubsub:itemId()]}}
+ %%%
+ | {error, _}
+).
+publish_item(NodeIdx, Publisher, Model, MaxItems, ItemId, Payload) ->
+ node_hometree_odbc:publish_item(NodeIdx, Publisher,
+ Model, MaxItems, ItemId, Payload).
+
+remove_extra_items(NodeId, MaxItems, ItemIds) ->
+ node_hometree_odbc:remove_extra_items(NodeId, MaxItems,
+ ItemIds).
+
+-spec(delete_item/4 ::
+(
+ NodeIdx :: mod_pubsub:nodeIdx(),
+ Publisher :: jid(),
+ PublishModel :: mod_pubsub:publishModel(),
+ ItemId :: <<>> | mod_pubsub:itemId())
+ -> {result, {default, broadcast}}
+ %%%
+ | {error, _}
+).
+delete_item(NodeId, Publisher, PublishModel, ItemId) ->
+ node_hometree_odbc:delete_item(NodeId, Publisher,
+ PublishModel, ItemId).
+
+-spec(purge_node/2 ::
+(
+ NodeIdx :: mod_pubsub:nodeIdx(),
+ Owner :: jid())
+ -> {result, {default, broadcast}}
+ | {error, _}
+).
+purge_node(NodeIdx, Owner) ->
+ node_hometree_odbc:purge_node(NodeIdx, Owner).
+
+-spec(get_entity_affiliations/2 ::
+(
+ Host :: mod_pubsub:hostPubsub(),
+ Owner :: jid())
+ -> {result, [{mod_pubsub:pubsubNode(), mod_pubsub:affiliation()}]}
+).
+get_entity_affiliations(_Host, Owner) ->
+ OwnerKey = jlib:jid_tolower(jlib:jid_remove_resource(Owner)),
+ node_hometree_odbc:get_entity_affiliations(OwnerKey, Owner).
+
+-spec(get_node_affiliations/1 ::
+(
+ NodeIdx::mod_pubsub:nodeIdx())
+ -> {result, [{ljid(), mod_pubsub:affiliation()}]}
+).
+get_node_affiliations(NodeIdx) ->
+ node_hometree_odbc:get_node_affiliations(NodeIdx).
+
+-spec(get_affiliation/2 ::
+(
+ NodeIdx :: mod_pubsub:nodeIdx(),
+ Owner :: ljid())
+ -> {result, mod_pubsub:affiliation()}
+).
+get_affiliation(NodeIdx, Owner) ->
+ node_hometree_odbc:get_affiliation(NodeIdx, Owner).
+
+set_affiliation(NodeId, Owner, Affiliation) ->
+ node_hometree_odbc:set_affiliation(NodeId, Owner, Affiliation).
+
+get_entity_subscriptions(_Host, Owner) ->
+ SubKey = jlib:jid_tolower(Owner),
+ GenKey = jlib:jid_remove_resource(SubKey),
+ Host = (?PUBSUB):escape(element(2, SubKey)),
+ SJ = node_hometree_odbc:encode_jid(SubKey),
+ GJ = node_hometree_odbc:encode_jid(GenKey),
+ Query = case SubKey of
+ GenKey ->
+ [<<"select host, node, type, i.nodeid, jid, "
+ "subscriptions from pubsub_state i, pubsub_nod"
+ "e n where i.nodeid = n.nodeid and jid "
+ "like '">>,
+ GJ, <<"%' and host like '%@">>, Host, <<"';">>];
+ _ ->
+ [<<"select host, node, type, i.nodeid, jid, "
+ "subscriptions from pubsub_state i, pubsub_nod"
+ "e n where i.nodeid = n.nodeid and jid "
+ "in ('">>,
+ SJ, <<"', '">>, GJ, <<"') and host like '%@">>, Host,
+ <<"';">>]
+ end,
+ Reply = case catch ejabberd_odbc:sql_query_t(Query) of
+ {selected,
+ [<<"host">>, <<"node">>, <<"type">>, <<"nodeid">>,
+ <<"jid">>, <<"subscriptions">>],
+ RItems} ->
+ lists:map(fun ({H, N, T, I, J, S}) ->
+ O = node_hometree_odbc:decode_jid(H),
+ Node = nodetree_tree_odbc:raw_to_node(O,
+ {N,
+ <<"">>,
+ T,
+ I}),
+ {Node,
+ node_hometree_odbc:decode_subscriptions(S),
+ node_hometree_odbc:decode_jid(J)}
+ end,
+ RItems);
+ _ -> []
+ end,
+ {result, Reply}.
+
+get_entity_subscriptions_for_send_last(_Host, Owner) ->
+ SubKey = jlib:jid_tolower(Owner),
+ GenKey = jlib:jid_remove_resource(SubKey),
+ Host = (?PUBSUB):escape(element(2, SubKey)),
+ SJ = node_hometree_odbc:encode_jid(SubKey),
+ GJ = node_hometree_odbc:encode_jid(GenKey),
+ Query = case SubKey of
+ GenKey ->
+ [<<"select host, node, type, i.nodeid, jid, "
+ "subscriptions from pubsub_state i, pubsub_nod"
+ "e n, pubsub_node_option o where i.nodeid "
+ "= n.nodeid and n.nodeid = o.nodeid and "
+ "name='send_last_published_item' and "
+ "val='on_sub_and_presence' and jid like "
+ "'">>,
+ GJ, <<"%' and host like '%@">>, Host, <<"';">>];
+ _ ->
+ [<<"select host, node, type, i.nodeid, jid, "
+ "subscriptions from pubsub_state i, pubsub_nod"
+ "e n, pubsub_node_option o where i.nodeid "
+ "= n.nodeid and n.nodeid = o.nodeid and "
+ "name='send_last_published_item' and "
+ "val='on_sub_and_presence' and jid in "
+ "('">>,
+ SJ, <<"', '">>, GJ, <<"') and host like '%@">>, Host,
+ <<"';">>]
+ end,
+ Reply = case catch ejabberd_odbc:sql_query_t(Query) of
+ {selected,
+ [<<"host">>, <<"node">>, <<"type">>, <<"nodeid">>,
+ <<"jid">>, <<"subscriptions">>],
+ RItems} ->
+ lists:map(fun ({H, N, T, I, J, S}) ->
+ O = node_hometree_odbc:decode_jid(H),
+ Node = nodetree_tree_odbc:raw_to_node(O,
+ {N,
+ <<"">>,
+ T,
+ I}),
+ {Node,
+ node_hometree_odbc:decode_subscriptions(S),
+ node_hometree_odbc:decode_jid(J)}
+ end,
+ RItems);
+ _ -> []
+ end,
+ {result, Reply}.
+
+get_node_subscriptions(NodeId) ->
+ node_hometree_odbc:get_node_subscriptions(NodeId).
+
+get_subscriptions(NodeId, Owner) ->
+ node_hometree_odbc:get_subscriptions(NodeId, Owner).
+
+set_subscriptions(NodeId, Owner, Subscription, SubId) ->
+ node_hometree_odbc:set_subscriptions(NodeId, Owner,
+ Subscription, SubId).
+
+get_pending_nodes(Host, Owner) ->
+ node_hometree_odbc:get_pending_nodes(Host, Owner).
+
+get_states(NodeId) ->
+ node_hometree_odbc:get_states(NodeId).
+
+get_state(NodeId, JID) ->
+ node_hometree_odbc:get_state(NodeId, JID).
+
+set_state(State) -> node_hometree_odbc:set_state(State).
+
+get_items(NodeId, From) ->
+ node_hometree_odbc:get_items(NodeId, From).
+
+get_items(NodeId, From, RSM) ->
+ node_hometree_odbc:get_items(NodeId, From, RSM).
+
+get_items(NodeId, JID, AccessModel,
+ PresenceSubscription, RosterGroup, SubId) ->
+ get_items(NodeId, JID, AccessModel,
+ PresenceSubscription, RosterGroup, SubId, none).
+
+get_items(NodeId, JID, AccessModel,
+ PresenceSubscription, RosterGroup, SubId, RSM) ->
+ node_hometree_odbc:get_items(NodeId, JID, AccessModel,
+ PresenceSubscription, RosterGroup, SubId, RSM).
+
+get_last_items(NodeId, JID, Count) ->
+ node_hometree_odbc:get_last_items(NodeId, JID, Count).
+
+get_item(NodeId, ItemId) ->
+ node_hometree_odbc:get_item(NodeId, ItemId).
+
+get_item(NodeId, ItemId, JID, AccessModel,
+ PresenceSubscription, RosterGroup, SubId) ->
+ node_hometree_odbc:get_item(NodeId, ItemId, JID,
+ AccessModel, PresenceSubscription, RosterGroup,
+ SubId).
+
+set_item(Item) -> node_hometree_odbc:set_item(Item).
+
+get_item_name(Host, Node, Id) ->
+ node_hometree_odbc:get_item_name(Host, Node, Id).
+
+node_to_path(Node) -> node_flat_odbc:node_to_path(Node).
+
+path_to_node(Path) -> node_flat_odbc:path_to_node(Path).
+
+%%%
+%%% Internal
+%%%
+
+%% @doc Check mod_caps is enabled, otherwise show warning.
+%% The PEP plugin for mod_pubsub requires mod_caps to be enabled in the host.
+%% Check that the mod_caps module is enabled in that Jabber Host
+%% If not, show a warning message in the ejabberd log file.
+complain_if_modcaps_disabled(ServerHost) ->
+ Modules = ejabberd_config:get_local_option({modules,
+ ServerHost},
+ fun(Ms) when is_list(Ms) -> Ms end),
+ ModCaps = [mod_caps_enabled
+ || {mod_caps, _Opts} <- Modules],
+ case ModCaps of
+ [] ->
+ ?WARNING_MSG("The PEP plugin is enabled in mod_pubsub "
+ "of host ~p. This plugin requires mod_caps "
+ "to be enabled, but it isn't.",
+ [ServerHost]);
+ _ -> ok
+ end.