Commit 6671a374 authored by Armando Jagucki's avatar Armando Jagucki Committed by ajagucki

PEP: Added initial support for Contact Notification Filtering.

git-svn-id: http://svn.igniterealtime.org/svn/repos/openfire/branches/pep@8851 b35dd754-fafc-0310-a699-88a17e54d16e
parent a660434a
......@@ -23,6 +23,9 @@ import org.jivesoftware.openfire.disco.ServerIdentitiesProvider;
import org.jivesoftware.openfire.disco.UserIdentitiesProvider;
import org.jivesoftware.openfire.disco.UserItemsProvider;
import org.jivesoftware.openfire.handler.IQHandler;
import org.jivesoftware.openfire.interceptor.InterceptorManager;
import org.jivesoftware.openfire.interceptor.PacketInterceptor;
import org.jivesoftware.openfire.interceptor.PacketRejectedException;
import org.jivesoftware.openfire.pubsub.CollectionNode;
import org.jivesoftware.openfire.pubsub.LeafNode;
import org.jivesoftware.openfire.pubsub.Node;
......@@ -32,6 +35,7 @@ import org.jivesoftware.openfire.pubsub.models.AccessModel;
import org.jivesoftware.openfire.roster.Roster;
import org.jivesoftware.openfire.roster.RosterItem;
import org.jivesoftware.openfire.session.ClientSession;
import org.jivesoftware.openfire.session.Session;
import org.jivesoftware.openfire.user.PresenceEventDispatcher;
import org.jivesoftware.openfire.user.PresenceEventListener;
import org.jivesoftware.openfire.user.UserManager;
......@@ -41,6 +45,7 @@ import org.xmpp.forms.DataForm;
import org.xmpp.forms.FormField;
import org.xmpp.packet.IQ;
import org.xmpp.packet.JID;
import org.xmpp.packet.Packet;
import org.xmpp.packet.PacketError;
import org.xmpp.packet.Presence;
......@@ -49,6 +54,7 @@ import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
......@@ -82,11 +88,20 @@ import java.util.concurrent.ConcurrentHashMap;
*
*/
public class IQPEPHandler extends IQHandler implements ServerIdentitiesProvider, ServerFeaturesProvider, UserIdentitiesProvider, UserItemsProvider,
PresenceEventListener {
PresenceEventListener, PacketInterceptor {
// Map of PEP services. Table, Key: bare JID (String); Value: PEPService
private Map<String, PEPService> pepServices;
/**
* Nodes to send filtered notifications for, table: key JID (String); value Set of nodes
*
* filteredNodesMap are used for Contact Notification Filtering as described in XEP-0163. The JID
* of a user is associated with a set of PEP node IDs they are interested in receiving notifications
* for.
*/
private Map<String, HashSet<String>> filteredNodesMap = new ConcurrentHashMap<String, HashSet<String>>();
private IQHandlerInfo info;
private PubSubEngine pubSubEngine = null;
......@@ -151,6 +166,11 @@ public class IQPEPHandler extends IQHandler implements ServerIdentitiesProvider,
Log.error(e);
}
}
// Add this PEP handler as a packet interceptor so we may deal with
// client packets that send disco#info's explaining capabilities
// including PEP contact notification filters.
InterceptorManager.getInstance().addInterceptor(this);
}
public void start() {
......@@ -421,4 +441,114 @@ public class IQPEPHandler extends IQHandler implements ServerIdentitiesProvider,
}
public void interceptPacket(Packet packet, Session session, boolean incoming, boolean processed) throws PacketRejectedException {
if (processed && packet instanceof IQ && ((IQ) packet).getType() == IQ.Type.result) {
// Examine the packet and return if it does not look like a disco#info result containing
// Entity Capabilities for a client. The sooner we return the better, as this method will be called
// quite a lot.
Element element = packet.getElement();
if (element == null) {
return;
}
Element query = element.element("query");
if (query == null) {
return;
}
else {
String queryNamespace = query.getNamespaceURI();
if (queryNamespace == null || !queryNamespace.equals("http://jabber.org/protocol/disco#info")) {
return;
}
}
/*Element identity = query.element("identity");
if (identity == null) {
return;
}
else {
String identityCategory = identity.attributeValue("category");
if (identityCategory == null || !identityCategory.equals("client")) {
return;
}
}*/
if (Log.isDebugEnabled()) {
Log.debug("PEP: Intercepted a caps result packet: " + packet.toString());
}
Iterator featuresIterator = query.elementIterator("feature");
if (featuresIterator == null) {
return;
}
// FIXME: Instead of the bare JID, use the full JID. Only change this
// when PEPService's sendNotification() is able to send to the full
// JID. See its FIXME as well.
String jidFrom = packet.getFrom().toBareJID();
// For each feature variable, or in this case node ID, ending in "+notify" -- add
// the node ID to the set of filtered nodes that jidFrom is interested in being
// notified about.
//
// If a node ID does not end in "+notify", remove it from the set of filtered nodes
// that jidFrom is interested in being notified about.
while (featuresIterator.hasNext()) {
Element featureElement = (Element) featuresIterator.next();
String featureVar = featureElement.attributeValue("var");
if (featureVar == null) {
continue;
}
if (featureVar.endsWith("+notify")) {
String nodeID = featureVar.replace("+notify", "");
HashSet<String> filteredNodeSet = filteredNodesMap.get(jidFrom);
if (filteredNodeSet == null) {
filteredNodeSet = new HashSet<String>();
filteredNodeSet.add(nodeID);
filteredNodesMap.put(jidFrom, filteredNodeSet);
if (Log.isDebugEnabled()) {
Log.debug("PEP: Created filteredNodeSet for " + jidFrom);
Log.debug("PEP: Added " + nodeID + " to " + jidFrom + "'s set of filtered nodes.");
}
}
else {
if (filteredNodeSet.add(nodeID)) {
if (Log.isDebugEnabled()) {
Log.debug("PEP: Added " + nodeID + " to " + jidFrom + "'s set of filtered nodes: ");
Iterator tempIter = filteredNodeSet.iterator();
while (tempIter.hasNext()) {
Log.debug("PEP: " + tempIter.next());
}
}
}
}
}
else {
HashSet<String> filteredNodeSet = filteredNodesMap.get(jidFrom);
if (filteredNodeSet != null && filteredNodeSet.remove(featureVar)) {
if (Log.isDebugEnabled()) {
Log.debug("PEP: Removed " + featureVar + " from " + jidFrom + "'s set of filtered nodes: ");
Iterator tempIter = filteredNodeSet.iterator();
while (tempIter.hasNext()) {
Log.debug("PEP: " + tempIter.next());
}
}
}
}
}
}
}
/**
* Returns the filteredNodesMap.
*
* @return the filteredNodesMap
*/
public Map<String, HashSet<String>> getFilteredNodesMap() {
return filteredNodesMap;
}
}
......@@ -36,7 +36,9 @@ import org.xmpp.packet.JID;
import org.xmpp.packet.Message;
import org.xmpp.packet.Packet;
import org.xmpp.packet.PacketExtension;
import java.util.Collection;
import java.util.HashSet;
import java.util.Map;
import java.util.Queue;
import java.util.Timer;
......@@ -183,7 +185,7 @@ public class PEPService implements PubSubService {
if (collectionDefaultConfiguration == null) {
// Create and save default configuration for collection nodes;
collectionDefaultConfiguration = new DefaultNodeConfiguration(false);
collectionDefaultConfiguration.setAccessModel(AccessModel.open);
collectionDefaultConfiguration.setAccessModel(AccessModel.presence);
collectionDefaultConfiguration.setPublisherModel(PublisherModel.publishers);
collectionDefaultConfiguration.setDeliverPayloads(false);
collectionDefaultConfiguration.setLanguage("English");
......@@ -310,6 +312,9 @@ public class PEPService implements PubSubService {
public void sendNotification(Node node, Message message, JID recipientJID) {
message.setFrom(getAddress());
// FIXME: Send to the full JID if this PEPService can retrieve presence
// for the recipient.
message.setTo(recipientJID);
message.setID(node.getNodeID() + "__" + recipientJID.toBareJID() + "__" + StringUtils.randomString(5));
......@@ -320,16 +325,31 @@ public class PEPService implements PubSubService {
// Get the full JID of the item publisher from the node that was published to.
// This full JID will be used as the "replyto" address in the addressing extension.
JID publisher = null;
Element itemsElement = message.getElement().element("event").element("items");
String publishedItemID = itemsElement.element("item").attributeValue("id");
String nodeIDPublishedTo = itemsElement.attributeValue("node");
// Check if the recipientJID is interested in notifications for this node.
// If the recipient has not yet requested any notification filtering, continue and send
// the notification.
Map<String, HashSet<String>> filteredNodesMap = XMPPServer.getInstance().getIQPEPHandler().getFilteredNodesMap();
HashSet<String> filteredNodesSet = filteredNodesMap.get(recipientJID.toBareJID());
if (filteredNodesSet != null && !filteredNodesSet.contains(nodeIDPublishedTo)) {
return;
}
if (node.isCollectionNode()) {
String nodeIDPublishedTo = itemsElement.attributeValue("node");
for (Node leafNode : node.getNodes()) {
if (leafNode.getNodeID().equals(nodeIDPublishedTo)) {
publisher = leafNode.getPublishedItem(publishedItemID).getPublisher();
// Ensure the recipientJID has access to receive notifications for items published to the leaf node.
AccessModel accessModel = leafNode.getAccessModel();
if (!accessModel.canAccessItems(leafNode, recipientJID, publisher)) {
return;
}
break;
}
}
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment