Commit 40aa7bd0 authored by Gaston Dombiak's avatar Gaston Dombiak Committed by gato

Refactoring to add PEP support. JM-1122

git-svn-id: http://svn.igniterealtime.org/svn/repos/openfire/trunk@9158 b35dd754-fafc-0310-a699-88a17e54d16e
parent 02095e50
...@@ -22,10 +22,7 @@ import org.jivesoftware.openfire.component.InternalComponentManager; ...@@ -22,10 +22,7 @@ import org.jivesoftware.openfire.component.InternalComponentManager;
import org.jivesoftware.openfire.container.AdminConsolePlugin; import org.jivesoftware.openfire.container.AdminConsolePlugin;
import org.jivesoftware.openfire.container.Module; import org.jivesoftware.openfire.container.Module;
import org.jivesoftware.openfire.container.PluginManager; import org.jivesoftware.openfire.container.PluginManager;
import org.jivesoftware.openfire.disco.IQDiscoInfoHandler; import org.jivesoftware.openfire.disco.*;
import org.jivesoftware.openfire.disco.IQDiscoItemsHandler;
import org.jivesoftware.openfire.disco.ServerFeaturesProvider;
import org.jivesoftware.openfire.disco.ServerItemsProvider;
import org.jivesoftware.openfire.filetransfer.DefaultFileTransferManager; import org.jivesoftware.openfire.filetransfer.DefaultFileTransferManager;
import org.jivesoftware.openfire.filetransfer.FileTransferManager; import org.jivesoftware.openfire.filetransfer.FileTransferManager;
import org.jivesoftware.openfire.filetransfer.proxy.FileTransferProxy; import org.jivesoftware.openfire.filetransfer.proxy.FileTransferProxy;
...@@ -36,6 +33,8 @@ import org.jivesoftware.openfire.muc.spi.MultiUserChatServerImpl; ...@@ -36,6 +33,8 @@ import org.jivesoftware.openfire.muc.spi.MultiUserChatServerImpl;
import org.jivesoftware.openfire.net.MulticastDNSService; import org.jivesoftware.openfire.net.MulticastDNSService;
import org.jivesoftware.openfire.net.SSLConfig; import org.jivesoftware.openfire.net.SSLConfig;
import org.jivesoftware.openfire.net.ServerTrafficCounter; import org.jivesoftware.openfire.net.ServerTrafficCounter;
import org.jivesoftware.openfire.pep.IQPEPHandler;
import org.jivesoftware.openfire.pep.IQPEPOwnerHandler;
import org.jivesoftware.openfire.pubsub.PubSubModule; import org.jivesoftware.openfire.pubsub.PubSubModule;
import org.jivesoftware.openfire.roster.RosterManager; import org.jivesoftware.openfire.roster.RosterManager;
import org.jivesoftware.openfire.session.RemoteSessionLocator; import org.jivesoftware.openfire.session.RemoteSessionLocator;
...@@ -98,6 +97,7 @@ public class XMPPServer { ...@@ -98,6 +97,7 @@ public class XMPPServer {
private Version version; private Version version;
private Date startDate; private Date startDate;
private boolean initialized = false; private boolean initialized = false;
private boolean started = false;
private NodeID nodeID; private NodeID nodeID;
private static final NodeID DEFAULT_NODE_ID = NodeID.getInstance(new byte[0]); private static final NodeID DEFAULT_NODE_ID = NodeID.getInstance(new byte[0]);
...@@ -451,6 +451,8 @@ public class XMPPServer { ...@@ -451,6 +451,8 @@ public class XMPPServer {
Log.info(startupBanner); Log.info(startupBanner);
System.out.println(startupBanner); System.out.println(startupBanner);
started = true;
// Notify server listeners that the server has been started // Notify server listeners that the server has been started
for (XMPPServerListener listener : listeners) { for (XMPPServerListener listener : listeners) {
listener.serverStarted(); listener.serverStarted();
...@@ -497,9 +499,9 @@ public class XMPPServer { ...@@ -497,9 +499,9 @@ public class XMPPServer {
loadModule(IQLastActivityHandler.class.getName()); loadModule(IQLastActivityHandler.class.getName());
loadModule(PresenceSubscribeHandler.class.getName()); loadModule(PresenceSubscribeHandler.class.getName());
loadModule(PresenceUpdateHandler.class.getName()); loadModule(PresenceUpdateHandler.class.getName());
loadModule(IQDiscoInfoHandler.class.getName());
loadModule(IQDiscoItemsHandler.class.getName());
loadModule(IQOfflineMessagesHandler.class.getName()); loadModule(IQOfflineMessagesHandler.class.getName());
loadModule(IQPEPHandler.class.getName());
loadModule(IQPEPOwnerHandler.class.getName());
loadModule(MultiUserChatServerImpl.class.getName()); loadModule(MultiUserChatServerImpl.class.getName());
loadModule(MulticastDNSService.class.getName()); loadModule(MulticastDNSService.class.getName());
loadModule(IQSharedGroupHandler.class.getName()); loadModule(IQSharedGroupHandler.class.getName());
...@@ -510,6 +512,8 @@ public class XMPPServer { ...@@ -510,6 +512,8 @@ public class XMPPServer {
loadModule(MediaProxyService.class.getName()); loadModule(MediaProxyService.class.getName());
loadModule(STUNService.class.getName()); loadModule(STUNService.class.getName());
loadModule(PubSubModule.class.getName()); loadModule(PubSubModule.class.getName());
loadModule(IQDiscoInfoHandler.class.getName());
loadModule(IQDiscoItemsHandler.class.getName());
loadModule(UpdateManager.class.getName()); loadModule(UpdateManager.class.getName());
loadModule(InternalComponentManager.class.getName()); loadModule(InternalComponentManager.class.getName());
// Load this module always last since we don't want to start listening for clients // Load this module always last since we don't want to start listening for clients
...@@ -1028,6 +1032,17 @@ public class XMPPServer { ...@@ -1028,6 +1032,17 @@ public class XMPPServer {
return (IQAuthHandler) modules.get(IQAuthHandler.class); return (IQAuthHandler) modules.get(IQAuthHandler.class);
} }
/**
* Returns the <code>IQPEPHandler</code> registered with this server. The
* <code>IQPEPHandler</code> was registered with the server as a module while starting up
* the server.
*
* @return the <code>IQPEPHandler</code> registered with this server.
*/
public IQPEPHandler getIQPEPHandler() {
return (IQPEPHandler) modules.get(IQPEPHandler.class);
}
/** /**
* Returns the <code>PluginManager</code> instance registered with this server. * Returns the <code>PluginManager</code> instance registered with this server.
* *
...@@ -1199,6 +1214,21 @@ public class XMPPServer { ...@@ -1199,6 +1214,21 @@ public class XMPPServer {
return answer; return answer;
} }
/**
* Returns a list with all the modules that provide "discoverable" identities.
*
* @return a list with all the modules that provide "discoverable" identities.
*/
public List<ServerIdentitiesProvider> getServerIdentitiesProviders() {
List<ServerIdentitiesProvider> answer = new ArrayList<ServerIdentitiesProvider>();
for (Module module : modules.values()) {
if (module instanceof ServerIdentitiesProvider) {
answer.add((ServerIdentitiesProvider) module);
}
}
return answer;
}
/** /**
* Returns a list with all the modules that provide "discoverable" items associated with * Returns a list with all the modules that provide "discoverable" items associated with
* the server. * the server.
...@@ -1216,6 +1246,38 @@ public class XMPPServer { ...@@ -1216,6 +1246,38 @@ public class XMPPServer {
return answer; return answer;
} }
/**
* Returns a list with all the modules that provide "discoverable" user identities.
*
* @return a list with all the modules that provide "discoverable" user identities.
*/
public List<UserIdentitiesProvider> getUserIdentitiesProviders() {
List<UserIdentitiesProvider> answer = new ArrayList<UserIdentitiesProvider>();
for (Module module : modules.values()) {
if (module instanceof UserIdentitiesProvider) {
answer.add((UserIdentitiesProvider) module);
}
}
return answer;
}
/**
* Returns a list with all the modules that provide "discoverable" items associated with
* users.
*
* @return a list with all the modules that provide "discoverable" items associated with
* users.
*/
public List<UserItemsProvider> getUserItemsProviders() {
List<UserItemsProvider> answer = new ArrayList<UserItemsProvider>();
for (Module module : modules.values()) {
if (module instanceof UserItemsProvider) {
answer.add((UserItemsProvider) module);
}
}
return answer;
}
/** /**
* Returns the <code>IQDiscoInfoHandler</code> registered with this server. The * Returns the <code>IQDiscoInfoHandler</code> registered with this server. The
* <code>IQDiscoInfoHandler</code> was registered with the server as a module while starting up * <code>IQDiscoInfoHandler</code> was registered with the server as a module while starting up
...@@ -1355,4 +1417,13 @@ public class XMPPServer { ...@@ -1355,4 +1417,13 @@ public class XMPPServer {
public void setRemoteSessionLocator(RemoteSessionLocator remoteSessionLocator) { public void setRemoteSessionLocator(RemoteSessionLocator remoteSessionLocator) {
this.remoteSessionLocator = remoteSessionLocator; this.remoteSessionLocator = remoteSessionLocator;
} }
/**
* Returns whether or not the server has been started.
*
* @return whether or not the server has been started.
*/
public boolean isStarted() {
return started;
}
} }
...@@ -11,12 +11,12 @@ ...@@ -11,12 +11,12 @@
package org.jivesoftware.openfire.pubsub; package org.jivesoftware.openfire.pubsub;
import org.dom4j.Element;
import org.jivesoftware.util.LocaleUtils;
import org.xmpp.forms.DataForm;
import org.xmpp.forms.FormField;
import org.xmpp.packet.JID; import org.xmpp.packet.JID;
import org.xmpp.packet.Message; import org.xmpp.packet.Message;
import org.xmpp.forms.FormField;
import org.xmpp.forms.DataForm;
import org.jivesoftware.util.LocaleUtils;
import org.dom4j.Element;
import java.util.*; import java.util.*;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
...@@ -51,7 +51,7 @@ public class CollectionNode extends Node { ...@@ -51,7 +51,7 @@ public class CollectionNode extends Node {
*/ */
private int maxLeafNodes = -1; private int maxLeafNodes = -1;
CollectionNode(PubSubService service, CollectionNode parentNode, String nodeID, JID creator) { public CollectionNode(PubSubService service, CollectionNode parentNode, String nodeID, JID creator) {
super(service, parentNode, nodeID, creator); super(service, parentNode, nodeID, creator);
// Configure node with default values (get them from the pubsub service) // Configure node with default values (get them from the pubsub service)
DefaultNodeConfiguration defaultConfiguration = service.getDefaultNodeConfiguration(false); DefaultNodeConfiguration defaultConfiguration = service.getDefaultNodeConfiguration(false);
......
...@@ -61,7 +61,7 @@ public class LeafNode extends Node { ...@@ -61,7 +61,7 @@ public class LeafNode extends Node {
// TODO Add checking of max payload size. Return <not-acceptable> plus a application specific error condition of <payload-too-big/>. // TODO Add checking of max payload size. Return <not-acceptable> plus a application specific error condition of <payload-too-big/>.
LeafNode(PubSubService service, CollectionNode parentNode, String nodeID, JID creator) { public LeafNode(PubSubService service, CollectionNode parentNode, String nodeID, JID creator) {
super(service, parentNode, nodeID, creator); super(service, parentNode, nodeID, creator);
// Configure node with default values (get them from the pubsub service) // Configure node with default values (get them from the pubsub service)
DefaultNodeConfiguration defaultConfiguration = service.getDefaultNodeConfiguration(true); DefaultNodeConfiguration defaultConfiguration = service.getDefaultNodeConfiguration(true);
......
...@@ -12,10 +12,10 @@ ...@@ -12,10 +12,10 @@
package org.jivesoftware.openfire.pubsub; package org.jivesoftware.openfire.pubsub;
import org.dom4j.Element; import org.dom4j.Element;
import org.jivesoftware.util.LocaleUtils;
import org.jivesoftware.util.StringUtils;
import org.jivesoftware.openfire.pubsub.models.AccessModel; import org.jivesoftware.openfire.pubsub.models.AccessModel;
import org.jivesoftware.openfire.pubsub.models.PublisherModel; import org.jivesoftware.openfire.pubsub.models.PublisherModel;
import org.jivesoftware.util.LocaleUtils;
import org.jivesoftware.util.StringUtils;
import org.xmpp.forms.DataForm; import org.xmpp.forms.DataForm;
import org.xmpp.forms.FormField; import org.xmpp.forms.FormField;
import org.xmpp.packet.IQ; import org.xmpp.packet.IQ;
...@@ -1735,7 +1735,7 @@ public abstract class Node { ...@@ -1735,7 +1735,7 @@ public abstract class Node {
* @throws IllegalStateException If this message was used when the node supports multiple * @throws IllegalStateException If this message was used when the node supports multiple
* subscriptions. * subscriptions.
*/ */
NodeSubscription getSubscription(JID subscriberJID) { public NodeSubscription getSubscription(JID subscriberJID) {
// Check that node does not support multiple subscriptions // Check that node does not support multiple subscriptions
if (isMultipleSubscriptionsEnabled()) { if (isMultipleSubscriptionsEnabled()) {
throw new IllegalStateException("Multiple subscriptions is enabled so subscriptions " + throw new IllegalStateException("Multiple subscriptions is enabled so subscriptions " +
......
...@@ -88,6 +88,14 @@ public class NodeAffiliate { ...@@ -88,6 +88,14 @@ public class NodeAffiliate {
Element items = event.addElement("items"); Element items = event.addElement("items");
items.addAttribute("node", getNode().getNodeID()); items.addAttribute("node", getNode().getNodeID());
for (PublishedItem publishedItem : itemsBySubs.get(nodeSubscriptions)) { for (PublishedItem publishedItem : itemsBySubs.get(nodeSubscriptions)) {
// FIXME: This was added for compatibility with PEP supporting clients.
// Alternate solution needed when XEP-0163 version > 1.0 is released.
//
// If the node ID looks like a JID, replace it with the published item's node ID.
if (getNode().getNodeID().indexOf("@") >= 0) {
items.addAttribute("node", publishedItem.getNode().getNodeID());
}
// Add item information to the event notification // Add item information to the event notification
Element item = items.addElement("item"); Element item = items.addElement("item");
if (leafNode.isItemRequired()) { if (leafNode.isItemRequired()) {
......
...@@ -129,7 +129,7 @@ public class NodeSubscription { ...@@ -129,7 +129,7 @@ public class NodeSubscription {
dateFormat = new SimpleDateFormat("yyyy-MM-DD'T'HH:mm:ss.SSS'Z'"); dateFormat = new SimpleDateFormat("yyyy-MM-DD'T'HH:mm:ss.SSS'Z'");
dateFormat.setTimeZone(TimeZone.getTimeZone("UTC")); dateFormat.setTimeZone(TimeZone.getTimeZone("UTC"));
fastDateFormat = FastDateFormat fastDateFormat = FastDateFormat
.getInstance("yyyy-MM-DD'T'HH:mm:ss.SSS'Z'", TimeZone.getTimeZone("UTC")); .getInstance("yyyy-MM-dd'T'HH:mm:ss.SSS'Z'", TimeZone.getTimeZone("UTC"));
} }
/** /**
...@@ -613,7 +613,7 @@ public class NodeSubscription { ...@@ -613,7 +613,7 @@ public class NodeSubscription {
* @return true if an event notification can be sent to the subscriber for the specified * @return true if an event notification can be sent to the subscriber for the specified
* published item. * published item.
*/ */
boolean canSendPublicationEvent(LeafNode leafNode, PublishedItem publishedItem) { public boolean canSendPublicationEvent(LeafNode leafNode, PublishedItem publishedItem) {
if (!canSendEvents()) { if (!canSendEvents()) {
return false; return false;
} }
......
...@@ -11,9 +11,9 @@ ...@@ -11,9 +11,9 @@
package org.jivesoftware.openfire.pubsub; package org.jivesoftware.openfire.pubsub;
import org.dom4j.Element; import org.dom4j.Element;
import org.jivesoftware.util.LocaleUtils;
import org.jivesoftware.openfire.commands.AdHocCommand; import org.jivesoftware.openfire.commands.AdHocCommand;
import org.jivesoftware.openfire.commands.SessionData; import org.jivesoftware.openfire.commands.SessionData;
import org.jivesoftware.util.LocaleUtils;
import org.xmpp.forms.DataForm; import org.xmpp.forms.DataForm;
import org.xmpp.forms.FormField; import org.xmpp.forms.FormField;
import org.xmpp.packet.JID; import org.xmpp.packet.JID;
...@@ -26,11 +26,11 @@ import java.util.List; ...@@ -26,11 +26,11 @@ import java.util.List;
* *
* @author Matt Tucker * @author Matt Tucker
*/ */
class PendingSubscriptionsCommand extends AdHocCommand { public class PendingSubscriptionsCommand extends AdHocCommand {
private PubSubService service; private PubSubService service;
PendingSubscriptionsCommand(PubSubService service) { public PendingSubscriptionsCommand(PubSubService service) {
this.service = service; this.service = service;
} }
......
...@@ -14,96 +14,45 @@ package org.jivesoftware.openfire.pubsub; ...@@ -14,96 +14,45 @@ package org.jivesoftware.openfire.pubsub;
import org.dom4j.DocumentHelper; import org.dom4j.DocumentHelper;
import org.dom4j.Element; import org.dom4j.Element;
import org.dom4j.QName; import org.dom4j.QName;
import org.jivesoftware.util.LocaleUtils;
import org.jivesoftware.util.Log;
import org.jivesoftware.util.StringUtils;
import org.jivesoftware.openfire.PacketRouter; import org.jivesoftware.openfire.PacketRouter;
import org.jivesoftware.openfire.XMPPServer; import org.jivesoftware.openfire.XMPPServer;
import org.jivesoftware.openfire.XMPPServerListener; import org.jivesoftware.openfire.XMPPServerListener;
import org.jivesoftware.openfire.commands.AdHocCommandManager;
import org.jivesoftware.openfire.pubsub.models.AccessModel; import org.jivesoftware.openfire.pubsub.models.AccessModel;
import org.jivesoftware.openfire.user.UserManager; import org.jivesoftware.openfire.user.UserManager;
import org.jivesoftware.util.Log;
import org.jivesoftware.util.StringUtils;
import org.xmpp.forms.DataForm; import org.xmpp.forms.DataForm;
import org.xmpp.forms.FormField; import org.xmpp.forms.FormField;
import org.xmpp.packet.*; import org.xmpp.packet.*;
import java.util.*; import java.util.*;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.LinkedBlockingQueue;
/** /**
* A PubSubEngine is responsible for handling packets sent to the pub-sub service. * A PubSubEngine is responsible for handling packets sent to a pub-sub service.
* *
* @author Matt Tucker * @author Matt Tucker
*/ */
public class PubSubEngine { public class PubSubEngine {
private PubSubService service;
/**
* Manager that keeps the list of ad-hoc commands and processing command requests.
*/
private AdHocCommandManager manager;
/**
* Keep a registry of the presence's show value of users that subscribed to a node of
* the pubsub service and for which the node only delivers notifications for online users
* or node subscriptions deliver events based on the user presence show value. Offline
* users will not have an entry in the map. Note: Key-> bare JID and Value-> Map whose key
* is full JID of connected resource and value is show value of the last received presence.
*/
private Map<String, Map<String, String>> barePresences =
new ConcurrentHashMap<String, Map<String, String>>();
/**
* The time to elapse between each execution of the maintenance process. Default
* is 2 minutes.
*/
private int items_task_timeout = 2 * 60 * 1000;
/**
* The number of items to save on each run of the maintenance process.
*/
private int items_batch_size = 50;
/**
* Queue that holds the items that need to be added to the database.
*/
private Queue<PublishedItem> itemsToAdd = new LinkedBlockingQueue<PublishedItem>();
/**
* Queue that holds the items that need to be deleted from the database.
*/
private Queue<PublishedItem> itemsToDelete = new LinkedBlockingQueue<PublishedItem>();
/**
* Task that saves or deletes published items from the database.
*/
private PublishedItemTask publishedItemTask;
/**
* Timer to save published items to the database or remove deleted or old items.
*/
private Timer timer = new Timer("PubSub maintenance");
/** /**
* The packet router for the server. * The packet router for the server.
*/ */
private PacketRouter router = null; private PacketRouter router = null;
public PubSubEngine(PubSubService pubSubService, PacketRouter router) { public PubSubEngine(PacketRouter router) {
this.service = pubSubService;
this.router = router; this.router = router;
// Initialize the ad-hoc commands manager to use for this pubsub service
manager = new AdHocCommandManager();
manager.addCommand(new PendingSubscriptionsCommand(service));
// Save or delete published items from the database every 2 minutes starting in
// 2 minutes (default values)
publishedItemTask = new PublishedItemTask();
timer.schedule(publishedItemTask, items_task_timeout, items_task_timeout);
} }
/** /**
* Handles IQ packets sent to the pubsub service. Requests of disco#info and disco#items * Handles IQ packets sent to the pubsub service. Requests of disco#info and disco#items
* are not being handled by the engine. Instead the service itself should handle disco packets. * are not being handled by the engine. Instead the service itself should handle disco packets.
* *
* @param service the PubSub service this action is to be performed for.
* @param iq the IQ packet sent to the pubsub service. * @param iq the IQ packet sent to the pubsub service.
* @return true if the IQ packet was handled by the engine. * @return true if the IQ packet was handled by the engine.
*/ */
public boolean process(IQ iq) { public boolean process(PubSubService service, IQ iq) {
// Ignore IQs of type ERROR or RESULT // Ignore IQs of type ERROR or RESULT
if (IQ.Type.error == iq.getType() || IQ.Type.result == iq.getType()) { if (IQ.Type.error == iq.getType() || IQ.Type.result == iq.getType()) {
return true; return true;
...@@ -118,61 +67,61 @@ public class PubSubEngine { ...@@ -118,61 +67,61 @@ public class PubSubEngine {
Element action = childElement.element("publish"); Element action = childElement.element("publish");
if (action != null) { if (action != null) {
// Entity publishes an item // Entity publishes an item
publishItemsToNode(iq, action); publishItemsToNode(service, iq, action);
return true; return true;
} }
action = childElement.element("subscribe"); action = childElement.element("subscribe");
if (action != null) { if (action != null) {
// Entity subscribes to a node // Entity subscribes to a node
subscribeNode(iq, childElement, action); subscribeNode(service, iq, childElement, action);
return true; return true;
} }
action = childElement.element("options"); action = childElement.element("options");
if (action != null) { if (action != null) {
if (IQ.Type.get == iq.getType()) { if (IQ.Type.get == iq.getType()) {
// Subscriber requests subscription options form // Subscriber requests subscription options form
getSubscriptionConfiguration(iq, childElement, action); getSubscriptionConfiguration(service, iq, childElement, action);
} }
else { else {
// Subscriber submits completed options form // Subscriber submits completed options form
configureSubscription(iq, action); configureSubscription(service, iq, action);
} }
return true; return true;
} }
action = childElement.element("create"); action = childElement.element("create");
if (action != null) { if (action != null) {
// Entity is requesting to create a new node // Entity is requesting to create a new node
createNode(iq, childElement, action); createNode(service, iq, childElement, action);
return true; return true;
} }
action = childElement.element("unsubscribe"); action = childElement.element("unsubscribe");
if (action != null) { if (action != null) {
// Entity unsubscribes from a node // Entity unsubscribes from a node
unsubscribeNode(iq, action); unsubscribeNode(service, iq, action);
return true; return true;
} }
action = childElement.element("subscriptions"); action = childElement.element("subscriptions");
if (action != null) { if (action != null) {
// Entity requests all current subscriptions // Entity requests all current subscriptions
getSubscriptions(iq, childElement); getSubscriptions(service, iq, childElement);
return true; return true;
} }
action = childElement.element("affiliations"); action = childElement.element("affiliations");
if (action != null) { if (action != null) {
// Entity requests all current affiliations // Entity requests all current affiliations
getAffiliations(iq, childElement); getAffiliations(service, iq, childElement);
return true; return true;
} }
action = childElement.element("items"); action = childElement.element("items");
if (action != null) { if (action != null) {
// Subscriber requests all active items // Subscriber requests all active items
getPublishedItems(iq, action); getPublishedItems(service, iq, action);
return true; return true;
} }
action = childElement.element("retract"); action = childElement.element("retract");
if (action != null) { if (action != null) {
// Entity deletes an item // Entity deletes an item
deleteItems(iq, action); deleteItems(service, iq, action);
return true; return true;
} }
// Unknown action requested // Unknown action requested
...@@ -200,11 +149,11 @@ public class PubSubEngine { ...@@ -200,11 +149,11 @@ public class PubSubEngine {
} }
if (IQ.Type.get == iq.getType()) { if (IQ.Type.get == iq.getType()) {
// Owner requests configuration form of a node // Owner requests configuration form of a node
getNodeConfiguration(iq, childElement, nodeID); getNodeConfiguration(service, iq, childElement, nodeID);
} }
else { else {
// Owner submits or cancels node configuration form // Owner submits or cancels node configuration form
configureNode(iq, action, nodeID); configureNode(service, iq, action, nodeID);
} }
return true; return true;
} }
...@@ -212,23 +161,23 @@ public class PubSubEngine { ...@@ -212,23 +161,23 @@ public class PubSubEngine {
if (action != null) { if (action != null) {
// Owner requests default configuration options for // Owner requests default configuration options for
// leaf or collection nodes // leaf or collection nodes
getDefaultNodeConfiguration(iq, childElement, action); getDefaultNodeConfiguration(service, iq, childElement, action);
return true; return true;
} }
action = childElement.element("delete"); action = childElement.element("delete");
if (action != null) { if (action != null) {
// Owner deletes a node // Owner deletes a node
deleteNode(iq, action); deleteNode(service, iq, action);
return true; return true;
} }
action = childElement.element("subscriptions"); action = childElement.element("subscriptions");
if (action != null) { if (action != null) {
if (IQ.Type.get == iq.getType()) { if (IQ.Type.get == iq.getType()) {
// Owner requests all affiliated entities // Owner requests all affiliated entities
getNodeSubscriptions(iq, action); getNodeSubscriptions(service, iq, action);
} }
else { else {
modifyNodeSubscriptions(iq, action); modifyNodeSubscriptions(service, iq, action);
} }
return true; return true;
} }
...@@ -236,17 +185,17 @@ public class PubSubEngine { ...@@ -236,17 +185,17 @@ public class PubSubEngine {
if (action != null) { if (action != null) {
if (IQ.Type.get == iq.getType()) { if (IQ.Type.get == iq.getType()) {
// Owner requests all affiliated entities // Owner requests all affiliated entities
getNodeAffiliations(iq, action); getNodeAffiliations(service, iq, action);
} }
else { else {
modifyNodeAffiliations(iq, action); modifyNodeAffiliations(service, iq, action);
} }
return true; return true;
} }
action = childElement.element("purge"); action = childElement.element("purge");
if (action != null) { if (action != null) {
// Owner purges items from a node // Owner purges items from a node
purgeNode(iq, action); purgeNode(service, iq, action);
return true; return true;
} }
// Unknown action requested so return error to sender // Unknown action requested so return error to sender
...@@ -255,7 +204,7 @@ public class PubSubEngine { ...@@ -255,7 +204,7 @@ public class PubSubEngine {
} }
else if ("http://jabber.org/protocol/commands".equals(namespace)) { else if ("http://jabber.org/protocol/commands".equals(namespace)) {
// Process ad-hoc command // Process ad-hoc command
IQ reply = manager.process(iq); IQ reply = service.getManager().process(iq);
router.route(reply); router.route(reply);
return true; return true;
} }
...@@ -266,18 +215,19 @@ public class PubSubEngine { ...@@ -266,18 +215,19 @@ public class PubSubEngine {
* Handles Presence packets sent to the pubsub service. Only process available and not * Handles Presence packets sent to the pubsub service. Only process available and not
* available presences. * available presences.
* *
* @param service the PubSub service this action is to be performed for.
* @param presence the Presence packet sent to the pubsub service. * @param presence the Presence packet sent to the pubsub service.
*/ */
public void process(Presence presence) { public void process(PubSubService service, Presence presence) {
if (presence.isAvailable()) { if (presence.isAvailable()) {
JID subscriber = presence.getFrom(); JID subscriber = presence.getFrom();
Map<String, String> fullPresences = barePresences.get(subscriber.toBareJID()); Map<String, String> fullPresences = service.getBarePresences().get(subscriber.toBareJID());
if (fullPresences == null) { if (fullPresences == null) {
synchronized (subscriber.toBareJID().intern()) { synchronized (subscriber.toBareJID().intern()) {
fullPresences = barePresences.get(subscriber.toBareJID()); fullPresences = service.getBarePresences().get(subscriber.toBareJID());
if (fullPresences == null) { if (fullPresences == null) {
fullPresences = new ConcurrentHashMap<String, String>(); fullPresences = new ConcurrentHashMap<String, String>();
barePresences.put(subscriber.toBareJID(), fullPresences); service.getBarePresences().put(subscriber.toBareJID(), fullPresences);
} }
} }
} }
...@@ -286,11 +236,11 @@ public class PubSubEngine { ...@@ -286,11 +236,11 @@ public class PubSubEngine {
} }
else if (presence.getType() == Presence.Type.unavailable) { else if (presence.getType() == Presence.Type.unavailable) {
JID subscriber = presence.getFrom(); JID subscriber = presence.getFrom();
Map<String, String> fullPresences = barePresences.get(subscriber.toBareJID()); Map<String, String> fullPresences = service.getBarePresences().get(subscriber.toBareJID());
if (fullPresences != null) { if (fullPresences != null) {
fullPresences.remove(subscriber.toString()); fullPresences.remove(subscriber.toString());
if (fullPresences.isEmpty()) { if (fullPresences.isEmpty()) {
barePresences.remove(subscriber.toBareJID()); service.getBarePresences().remove(subscriber.toBareJID());
} }
} }
} }
...@@ -303,16 +253,17 @@ public class PubSubEngine { ...@@ -303,16 +253,17 @@ public class PubSubEngine {
* Answers to authorization requests sent to node owners to approve pending subscriptions * Answers to authorization requests sent to node owners to approve pending subscriptions
* will also be processed by this method. * will also be processed by this method.
* *
* @param service the PubSub service this action is to be performed for.
* @param message the Message packet sent to the pubsub service. * @param message the Message packet sent to the pubsub service.
*/ */
public void process(Message message) { public void process(PubSubService service, Message message) {
if (message.getType() == Message.Type.error) { if (message.getType() == Message.Type.error) {
// Process Messages of type error to identify possible subscribers that no longer exist // Process Messages of type error to identify possible subscribers that no longer exist
if (message.getError().getType() == PacketError.Type.cancel) { if (message.getError().getType() == PacketError.Type.cancel) {
// TODO Assuming that owner is the bare JID (as defined in the JEP). This can be replaced with an explicit owner specified in the packet // TODO Assuming that owner is the bare JID (as defined in the JEP). This can be replaced with an explicit owner specified in the packet
JID owner = new JID(message.getFrom().toBareJID()); JID owner = new JID(message.getFrom().toBareJID());
// Terminate the subscription of the entity to all nodes hosted at the service // Terminate the subscription of the entity to all nodes hosted at the service
cancelAllSubscriptions(owner); cancelAllSubscriptions(service, owner);
} }
else if (message.getError().getType() == PacketError.Type.auth) { else if (message.getError().getType() == PacketError.Type.auth) {
// TODO Queue the message to be sent again later (will retry a few times and // TODO Queue the message to be sent again later (will retry a few times and
...@@ -327,13 +278,13 @@ public class PubSubEngine { ...@@ -327,13 +278,13 @@ public class PubSubEngine {
// Check that completed data form belongs to an authorization request // Check that completed data form belongs to an authorization request
if ("http://jabber.org/protocol/pubsub#subscribe_authorization".equals(formType)) { if ("http://jabber.org/protocol/pubsub#subscribe_authorization".equals(formType)) {
// Process the answer to the authorization request // Process the answer to the authorization request
processAuthorizationAnswer(authForm, message); processAuthorizationAnswer(service, authForm, message);
} }
} }
} }
} }
private void publishItemsToNode(IQ iq, Element publishElement) { private void publishItemsToNode(PubSubService service, IQ iq, Element publishElement) {
String nodeID = publishElement.attributeValue("node"); String nodeID = publishElement.attributeValue("node");
Node node; Node node;
if (nodeID == null) { if (nodeID == null) {
...@@ -420,7 +371,7 @@ public class PubSubEngine { ...@@ -420,7 +371,7 @@ public class PubSubEngine {
leafNode.publishItems(from, items); leafNode.publishItems(from, items);
} }
private void deleteItems(IQ iq, Element retractElement) { private void deleteItems(PubSubService service, IQ iq, Element retractElement) {
String nodeID = retractElement.attributeValue("node"); String nodeID = retractElement.attributeValue("node");
Node node; Node node;
if (nodeID == null) { if (nodeID == null) {
...@@ -503,7 +454,7 @@ public class PubSubEngine { ...@@ -503,7 +454,7 @@ public class PubSubEngine {
leafNode.deleteItems(items); leafNode.deleteItems(items);
} }
private void subscribeNode(IQ iq, Element childElement, Element subscribeElement) { private void subscribeNode(PubSubService service, IQ iq, Element childElement, Element subscribeElement) {
String nodeID = subscribeElement.attributeValue("node"); String nodeID = subscribeElement.attributeValue("node");
Node node; Node node;
if (nodeID == null) { if (nodeID == null) {
...@@ -631,7 +582,7 @@ public class PubSubEngine { ...@@ -631,7 +582,7 @@ public class PubSubEngine {
optionsForm); optionsForm);
} }
private void unsubscribeNode(IQ iq, Element unsubscribeElement) { private void unsubscribeNode(PubSubService service, IQ iq, Element unsubscribeElement) {
String nodeID = unsubscribeElement.attributeValue("node"); String nodeID = unsubscribeElement.attributeValue("node");
String subID = unsubscribeElement.attributeValue("subid"); String subID = unsubscribeElement.attributeValue("subid");
Node node; Node node;
...@@ -719,7 +670,8 @@ public class PubSubEngine { ...@@ -719,7 +670,8 @@ public class PubSubEngine {
router.route(IQ.createResultIQ(iq)); router.route(IQ.createResultIQ(iq));
} }
private void getSubscriptionConfiguration(IQ iq, Element childElement, Element optionsElement) { private void getSubscriptionConfiguration(PubSubService service, IQ iq,
Element childElement, Element optionsElement) {
String nodeID = optionsElement.attributeValue("node"); String nodeID = optionsElement.attributeValue("node");
String subID = optionsElement.attributeValue("subid"); String subID = optionsElement.attributeValue("subid");
Node node; Node node;
...@@ -800,7 +752,7 @@ public class PubSubEngine { ...@@ -800,7 +752,7 @@ public class PubSubEngine {
router.route(reply); router.route(reply);
} }
private void configureSubscription(IQ iq, Element optionsElement) { private void configureSubscription(PubSubService service, IQ iq, Element optionsElement) {
String nodeID = optionsElement.attributeValue("node"); String nodeID = optionsElement.attributeValue("node");
String subID = optionsElement.attributeValue("subid"); String subID = optionsElement.attributeValue("subid");
Node node; Node node;
...@@ -885,7 +837,7 @@ public class PubSubEngine { ...@@ -885,7 +837,7 @@ public class PubSubEngine {
} }
} }
private void getSubscriptions(IQ iq, Element childElement) { private void getSubscriptions(PubSubService service, IQ iq, Element childElement) {
// TODO Assuming that owner is the bare JID (as defined in the JEP). This can be replaced with an explicit owner specified in the packet // TODO Assuming that owner is the bare JID (as defined in the JEP). This can be replaced with an explicit owner specified in the packet
JID owner = new JID(iq.getFrom().toBareJID()); JID owner = new JID(iq.getFrom().toBareJID());
// Collect subscriptions of owner for all nodes at the service // Collect subscriptions of owner for all nodes at the service
...@@ -924,7 +876,7 @@ public class PubSubEngine { ...@@ -924,7 +876,7 @@ public class PubSubEngine {
router.route(reply); router.route(reply);
} }
private void getAffiliations(IQ iq, Element childElement) { private void getAffiliations(PubSubService service, IQ iq, Element childElement) {
// TODO Assuming that owner is the bare JID (as defined in the JEP). This can be replaced with an explicit owner specified in the packet // TODO Assuming that owner is the bare JID (as defined in the JEP). This can be replaced with an explicit owner specified in the packet
JID owner = new JID(iq.getFrom().toBareJID()); JID owner = new JID(iq.getFrom().toBareJID());
// Collect affiliations of owner for all nodes at the service // Collect affiliations of owner for all nodes at the service
...@@ -960,7 +912,7 @@ public class PubSubEngine { ...@@ -960,7 +912,7 @@ public class PubSubEngine {
router.route(reply); router.route(reply);
} }
private void getPublishedItems(IQ iq, Element itemsElement) { private void getPublishedItems(PubSubService service, IQ iq, Element itemsElement) {
String nodeID = itemsElement.attributeValue("node"); String nodeID = itemsElement.attributeValue("node");
String subID = itemsElement.attributeValue("subid"); String subID = itemsElement.attributeValue("subid");
Node node; Node node;
...@@ -1095,7 +1047,7 @@ public class PubSubEngine { ...@@ -1095,7 +1047,7 @@ public class PubSubEngine {
leafNode.sendPublishedItems(iq, items, forceToIncludePayload); leafNode.sendPublishedItems(iq, items, forceToIncludePayload);
} }
private void createNode(IQ iq, Element childElement, Element createElement) { private void createNode(PubSubService service, IQ iq, Element childElement, Element createElement) {
// Get sender of the IQ packet // Get sender of the IQ packet
JID from = iq.getFrom(); JID from = iq.getFrom();
// Verify that sender has permissions to create nodes // Verify that sender has permissions to create nodes
...@@ -1150,11 +1102,6 @@ public class PubSubEngine { ...@@ -1150,11 +1102,6 @@ public class PubSubEngine {
return; return;
} }
parentNode = (CollectionNode) tempNode; parentNode = (CollectionNode) tempNode;
// If requested new nodeID does not contain parent nodeID then add
// the parent nodeID to the beginging of the new nodeID
if (!newNodeID.startsWith(parentNodeID)) {
newNodeID = parentNodeID + "/" + newNodeID;
}
} }
} }
field = completedForm.getField("pubsub#node_type"); field = completedForm.getField("pubsub#node_type");
...@@ -1170,10 +1117,6 @@ public class PubSubEngine { ...@@ -1170,10 +1117,6 @@ public class PubSubEngine {
// If no parent was defined then use the root collection node // If no parent was defined then use the root collection node
if (parentNode == null && service.isCollectionNodesSupported()) { if (parentNode == null && service.isCollectionNodesSupported()) {
parentNode = service.getRootCollectionNode(); parentNode = service.getRootCollectionNode();
// Calculate new nodeID for the new node
if (!newNodeID.startsWith(parentNode.getNodeID() + "/")) {
newNodeID = parentNode.getNodeID() + "/" + newNodeID;
}
} }
// Check that the requested nodeID does not exist // Check that the requested nodeID does not exist
Node existingNode = service.getNode(newNodeID); Node existingNode = service.getNode(newNodeID);
...@@ -1260,7 +1203,7 @@ public class PubSubEngine { ...@@ -1260,7 +1203,7 @@ public class PubSubEngine {
} }
} }
private void getNodeConfiguration(IQ iq, Element childElement, String nodeID) { private void getNodeConfiguration(PubSubService service, IQ iq, Element childElement, String nodeID) {
Node node = service.getNode(nodeID); Node node = service.getNode(nodeID);
if (node == null) { if (node == null) {
// Node does not exist. Return item-not-found error // Node does not exist. Return item-not-found error
...@@ -1281,7 +1224,8 @@ public class PubSubEngine { ...@@ -1281,7 +1224,8 @@ public class PubSubEngine {
router.route(reply); router.route(reply);
} }
private void getDefaultNodeConfiguration(IQ iq, Element childElement, Element defaultElement) { private void getDefaultNodeConfiguration(PubSubService service, IQ iq,
Element childElement, Element defaultElement) {
String type = defaultElement.attributeValue("type"); String type = defaultElement.attributeValue("type");
type = type == null ? "leaf" : type; type = type == null ? "leaf" : type;
...@@ -1304,7 +1248,7 @@ public class PubSubEngine { ...@@ -1304,7 +1248,7 @@ public class PubSubEngine {
router.route(reply); router.route(reply);
} }
private void configureNode(IQ iq, Element configureElement, String nodeID) { private void configureNode(PubSubService service, IQ iq, Element configureElement, String nodeID) {
Node node = service.getNode(nodeID); Node node = service.getNode(nodeID);
if (node == null) { if (node == null) {
// Node does not exist. Return item-not-found error // Node does not exist. Return item-not-found error
...@@ -1338,7 +1282,7 @@ public class PubSubEngine { ...@@ -1338,7 +1282,7 @@ public class PubSubEngine {
} }
} }
private void deleteNode(IQ iq, Element deleteElement) { private void deleteNode(PubSubService service, IQ iq, Element deleteElement) {
String nodeID = deleteElement.attributeValue("node"); String nodeID = deleteElement.attributeValue("node");
if (nodeID == null) { if (nodeID == null) {
// NodeID was not provided. Return bad-request error // NodeID was not provided. Return bad-request error
...@@ -1373,7 +1317,7 @@ public class PubSubEngine { ...@@ -1373,7 +1317,7 @@ public class PubSubEngine {
} }
} }
private void purgeNode(IQ iq, Element purgeElement) { private void purgeNode(PubSubService service, IQ iq, Element purgeElement) {
String nodeID = purgeElement.attributeValue("node"); String nodeID = purgeElement.attributeValue("node");
if (nodeID == null) { if (nodeID == null) {
// NodeID was not provided. Return bad-request error // NodeID was not provided. Return bad-request error
...@@ -1414,7 +1358,7 @@ public class PubSubEngine { ...@@ -1414,7 +1358,7 @@ public class PubSubEngine {
router.route(IQ.createResultIQ(iq)); router.route(IQ.createResultIQ(iq));
} }
private void getNodeSubscriptions(IQ iq, Element affiliationsElement) { private void getNodeSubscriptions(PubSubService service, IQ iq, Element affiliationsElement) {
String nodeID = affiliationsElement.attributeValue("node"); String nodeID = affiliationsElement.attributeValue("node");
if (nodeID == null) { if (nodeID == null) {
// NodeID was not provided. Return bad-request error. // NodeID was not provided. Return bad-request error.
...@@ -1437,7 +1381,7 @@ public class PubSubEngine { ...@@ -1437,7 +1381,7 @@ public class PubSubEngine {
node.sendSubscriptions(iq); node.sendSubscriptions(iq);
} }
private void modifyNodeSubscriptions(IQ iq, Element entitiesElement) { private void modifyNodeSubscriptions(PubSubService service, IQ iq, Element entitiesElement) {
String nodeID = entitiesElement.attributeValue("node"); String nodeID = entitiesElement.attributeValue("node");
if (nodeID == null) { if (nodeID == null) {
// NodeID was not provided. Return bad-request error. // NodeID was not provided. Return bad-request error.
...@@ -1497,7 +1441,7 @@ public class PubSubEngine { ...@@ -1497,7 +1441,7 @@ public class PubSubEngine {
router.route(reply); router.route(reply);
} }
private void getNodeAffiliations(IQ iq, Element affiliationsElement) { private void getNodeAffiliations(PubSubService service, IQ iq, Element affiliationsElement) {
String nodeID = affiliationsElement.attributeValue("node"); String nodeID = affiliationsElement.attributeValue("node");
if (nodeID == null) { if (nodeID == null) {
// NodeID was not provided. Return bad-request error. // NodeID was not provided. Return bad-request error.
...@@ -1520,7 +1464,7 @@ public class PubSubEngine { ...@@ -1520,7 +1464,7 @@ public class PubSubEngine {
node.sendAffiliations(iq); node.sendAffiliations(iq);
} }
private void modifyNodeAffiliations(IQ iq, Element entitiesElement) { private void modifyNodeAffiliations(PubSubService service, IQ iq, Element entitiesElement) {
String nodeID = entitiesElement.attributeValue("node"); String nodeID = entitiesElement.attributeValue("node");
if (nodeID == null) { if (nodeID == null) {
// NodeID was not provided. Return bad-request error. // NodeID was not provided. Return bad-request error.
...@@ -1603,9 +1547,10 @@ public class PubSubEngine { ...@@ -1603,9 +1547,10 @@ public class PubSubEngine {
* The affiliation with the node will be removed if the entity was not a node owner or * The affiliation with the node will be removed if the entity was not a node owner or
* publisher. * publisher.
* *
* @param service the PubSub service this action is to be performed for.
* @param user the entity that no longer exists. * @param user the entity that no longer exists.
*/ */
private void cancelAllSubscriptions(JID user) { private void cancelAllSubscriptions(PubSubService service, JID user) {
for (Node node : service.getNodes()) { for (Node node : service.getNodes()) {
NodeAffiliate affiliate = node.getAffiliate(user); NodeAffiliate affiliate = node.getAffiliate(user);
if (affiliate == null) { if (affiliate == null) {
...@@ -1618,7 +1563,7 @@ public class PubSubEngine { ...@@ -1618,7 +1563,7 @@ public class PubSubEngine {
} }
} }
private void processAuthorizationAnswer(DataForm authForm, Message message) { private void processAuthorizationAnswer(PubSubService service, DataForm authForm, Message message) {
String nodeID = authForm.getField("pubsub#node").getValues().get(0); String nodeID = authForm.getField("pubsub#node").getValues().get(0);
String subID = authForm.getField("pubsub#subid").getValues().get(0); String subID = authForm.getField("pubsub#subid").getValues().get(0);
String allow = authForm.getField("pubsub#allow").getValues().get(0); String allow = authForm.getField("pubsub#allow").getValues().get(0);
...@@ -1719,11 +1664,26 @@ public class PubSubEngine { ...@@ -1719,11 +1664,26 @@ public class PubSubEngine {
return completedForm; return completedForm;
} }
public void start() { public void start(final PubSubService service) {
// Probe presences of users that this service has subscribed to (once the server // Probe presences of users that this service has subscribed to (once the server
// has started) // has started)
if (XMPPServer.getInstance().isStarted()) {
probePresences(service);
}
else {
XMPPServer.getInstance().addServerListener(new XMPPServerListener() { XMPPServer.getInstance().addServerListener(new XMPPServerListener() {
public void serverStarted() { public void serverStarted() {
probePresences(service);
}
public void serverStopping() {
}
});
}
}
private void probePresences(final PubSubService service) {
Set<JID> affiliates = new HashSet<JID>(); Set<JID> affiliates = new HashSet<JID>();
for (Node node : service.getNodes()) { for (Node node : service.getNodes()) {
affiliates.addAll(node.getPresenceBasedSubscribers()); affiliates.addAll(node.getPresenceBasedSubscribers());
...@@ -1737,31 +1697,26 @@ public class PubSubEngine { ...@@ -1737,31 +1697,26 @@ public class PubSubEngine {
} }
} }
public void serverStopping() { public void shutdown(PubSubService service) {
} // Stop the maintenance processes
}); service.getTimer().cancel();
}
public void shutdown() {
// Stop te maintenance processes
timer.cancel();
// Delete from the database items contained in the itemsToDelete queue // Delete from the database items contained in the itemsToDelete queue
PublishedItem entry; PublishedItem entry;
while (!itemsToDelete.isEmpty()) { while (!service.getItemsToDelete().isEmpty()) {
entry = itemsToDelete.poll(); entry = service.getItemsToDelete().poll();
if (entry != null) { if (entry != null) {
PubSubPersistenceManager.removePublishedItem(service, entry); PubSubPersistenceManager.removePublishedItem(service, entry);
} }
} }
// Save to the database items contained in the itemsToAdd queue // Save to the database items contained in the itemsToAdd queue
while (!itemsToAdd.isEmpty()) { while (!service.getItemsToAdd().isEmpty()) {
entry = itemsToAdd.poll(); entry = service.getItemsToAdd().poll();
if (entry != null) { if (entry != null) {
PubSubPersistenceManager.createPublishedItem(service, entry); PubSubPersistenceManager.createPublishedItem(service, entry);
} }
} }
// Stop executing ad-hoc commands // Stop executing ad-hoc commands
manager.stop(); service.getManager().stop();
} }
/******************************************************************************* /*******************************************************************************
...@@ -1775,12 +1730,13 @@ public class PubSubEngine { ...@@ -1775,12 +1730,13 @@ public class PubSubEngine {
* is offline then an empty collectin is returned. Available show status is represented * is offline then an empty collectin is returned. Available show status is represented
* by a <tt>online</tt> value. The rest of the possible show values as defined in RFC 3921. * by a <tt>online</tt> value. The rest of the possible show values as defined in RFC 3921.
* *
* @param service the PubSub service this action is to be performed for.
* @param subscriber the JID of the subscriber. This is not the JID of the affiliate. * @param subscriber the JID of the subscriber. This is not the JID of the affiliate.
* @return an empty collection when offline. Otherwise, a collection with the show value * @return an empty collection when offline. Otherwise, a collection with the show value
* of each connected resource. * of each connected resource.
*/ */
public Collection<String> getShowPresences(JID subscriber) { public static Collection<String> getShowPresences(PubSubService service, JID subscriber) {
Map<String, String> fullPresences = barePresences.get(subscriber.toBareJID()); Map<String, String> fullPresences = service.getBarePresences().get(subscriber.toBareJID());
if (fullPresences == null) { if (fullPresences == null) {
// User is offline so return empty list // User is offline so return empty list
return Collections.emptyList(); return Collections.emptyList();
...@@ -1805,10 +1761,11 @@ public class PubSubEngine { ...@@ -1805,10 +1761,11 @@ public class PubSubEngine {
* Requests the pubsub service to subscribe to the presence of the user. If the service * Requests the pubsub service to subscribe to the presence of the user. If the service
* has already subscribed to the user's presence then do nothing. * has already subscribed to the user's presence then do nothing.
* *
* @param service the PubSub service this action is to be performed for.
* @param node the node that originated the subscription request. * @param node the node that originated the subscription request.
* @param user the JID of the affiliate to subscribe to his presence. * @param user the JID of the affiliate to subscribe to his presence.
*/ */
public void presenceSubscriptionNotRequired(Node node, JID user) { public static void presenceSubscriptionNotRequired(PubSubService service, Node node, JID user) {
// Check that no node is requiring to be subscribed to this user // Check that no node is requiring to be subscribed to this user
for (Node hostedNode : service.getNodes()) { for (Node hostedNode : service.getNodes()) {
if (hostedNode.isPresenceBasedDelivery(user)) { if (hostedNode.isPresenceBasedDelivery(user)) {
...@@ -1828,11 +1785,12 @@ public class PubSubEngine { ...@@ -1828,11 +1785,12 @@ public class PubSubEngine {
* was not subscribed to the user's presence or any node still requires to be subscribed to * was not subscribed to the user's presence or any node still requires to be subscribed to
* the user presence then do nothing. * the user presence then do nothing.
* *
* @param service the PubSub service this action is to be performed for.
* @param node the node that originated the unsubscription request. * @param node the node that originated the unsubscription request.
* @param user the JID of the affiliate to unsubscribe from his presence. * @param user the JID of the affiliate to unsubscribe from his presence.
*/ */
public void presenceSubscriptionRequired(Node node, JID user) { public static void presenceSubscriptionRequired(PubSubService service, Node node, JID user) {
Map<String, String> fullPresences = barePresences.get(user.toString()); Map<String, String> fullPresences = service.getBarePresences().get(user.toString());
if (fullPresences == null || fullPresences.isEmpty()) { if (fullPresences == null || fullPresences.isEmpty()) {
Presence subscription = new Presence(Presence.Type.subscribe); Presence subscription = new Presence(Presence.Type.subscribe);
subscription.setTo(user); subscription.setTo(user);
...@@ -1840,10 +1798,10 @@ public class PubSubEngine { ...@@ -1840,10 +1798,10 @@ public class PubSubEngine {
service.send(subscription); service.send(subscription);
// Sending subscription requests based on received presences may generate // Sending subscription requests based on received presences may generate
// that a sunscription request is sent to an offline user (since offline // that a sunscription request is sent to an offline user (since offline
// presences are not stored in "barePresences"). However, this not optimal // presences are not stored in the service's "barePresences"). However, this
// algorithm shouldn't bother the user since the user's server should reply // not optimal algorithm shouldn't bother the user since the user's server
// when already subscribed to the user's presence instead of asking the user // should reply when already subscribed to the user's presence instead of
// to accept the subscription request // asking the user to accept the subscription request.
} }
} }
...@@ -1857,34 +1815,38 @@ public class PubSubEngine { ...@@ -1857,34 +1815,38 @@ public class PubSubEngine {
* beginning after the specified delay. Subsequent executions take place * beginning after the specified delay. Subsequent executions take place
* at approximately regular intervals separated by the specified period. * at approximately regular intervals separated by the specified period.
* *
* @param service the PubSub service this action is to be performed for.
* @param timeout the new frequency of the maintenance task. * @param timeout the new frequency of the maintenance task.
*/ */
void setPublishedItemTaskTimeout(int timeout) { void setPublishedItemTaskTimeout(PubSubService service, int timeout) {
if (this.items_task_timeout == timeout) { int items_task_timeout = service.getItemsTaskTimeout();
if (items_task_timeout == timeout) {
return; return;
} }
// Cancel the existing task because the timeout has changed // Cancel the existing task because the timeout has changed
PublishedItemTask publishedItemTask = service.getPublishedItemTask();
if (publishedItemTask != null) { if (publishedItemTask != null) {
publishedItemTask.cancel(); publishedItemTask.cancel();
} }
this.items_task_timeout = timeout; service.setItemsTaskTimeout(timeout);
// Create a new task and schedule it with the new timeout // Create a new task and schedule it with the new timeout
publishedItemTask = new PublishedItemTask(); service.setPublishedItemTask(new PublishedItemTask(service));
timer.schedule(publishedItemTask, items_task_timeout, items_task_timeout); service.getTimer().schedule(publishedItemTask, items_task_timeout, items_task_timeout);
} }
/** /**
* Adds the item to the queue of items to remove from the database. The queue is going * Adds the item to the queue of items to remove from the database. The queue is going
* to be processed by another thread. * to be processed by another thread.
* *
* @param service the PubSub service this action is to be performed for.
* @param removedItem the item to remove from the database. * @param removedItem the item to remove from the database.
*/ */
void queueItemToRemove(PublishedItem removedItem) { public static void queueItemToRemove(PubSubService service, PublishedItem removedItem) {
// Remove the removed item from the queue of items to add to the database // Remove the removed item from the queue of items to add to the database
if (!itemsToAdd.remove(removedItem)) { if (!service.getItemsToAdd().remove(removedItem)) {
// The item is already present in the database so add the removed item // The item is already present in the database so add the removed item
// to the queue of items to delete from the database // to the queue of items to delete from the database
itemsToDelete.add(removedItem); service.getItemsToDelete().add(removedItem);
} }
} }
...@@ -1892,10 +1854,11 @@ public class PubSubEngine { ...@@ -1892,10 +1854,11 @@ public class PubSubEngine {
* Adds the item to the queue of items to add to the database. The queue is going * Adds the item to the queue of items to add to the database. The queue is going
* to be processed by another thread. * to be processed by another thread.
* *
* @param service the PubSub service this action is to be performed for.
* @param newItem the item to add to the database. * @param newItem the item to add to the database.
*/ */
void queueItemToAdd(PublishedItem newItem) { public static void queueItemToAdd(PubSubService service, PublishedItem newItem) {
itemsToAdd.add(newItem); service.getItemsToAdd().add(newItem);
} }
/** /**
...@@ -1903,44 +1866,13 @@ public class PubSubEngine { ...@@ -1903,44 +1866,13 @@ public class PubSubEngine {
* usually required when a node was deleted so any pending operation of the node items * usually required when a node was deleted so any pending operation of the node items
* should be cancelled. * should be cancelled.
* *
* @param service the PubSub service this action is to be performed for.
* @param items the list of items to remove the from queues. * @param items the list of items to remove the from queues.
*/ */
void cancelQueuedItems(Collection<PublishedItem> items) { void cancelQueuedItems(PubSubService service, Collection<PublishedItem> items) {
for (PublishedItem item : items) { for (PublishedItem item : items) {
itemsToAdd.remove(item); service.getItemsToAdd().remove(item);
itemsToDelete.remove(item); service.getItemsToDelete().remove(item);
}
}
private class PublishedItemTask extends TimerTask {
public void run() {
try {
PublishedItem entry;
boolean success;
// Delete from the database items contained in the itemsToDelete queue
for (int index = 0; index <= items_batch_size && !itemsToDelete.isEmpty(); index++) {
entry = itemsToDelete.poll();
if (entry != null) {
success = PubSubPersistenceManager.removePublishedItem(service, entry);
if (!success) {
itemsToDelete.add(entry);
}
}
}
// Save to the database items contained in the itemsToAdd queue
for (int index = 0; index <= items_batch_size && !itemsToAdd.isEmpty(); index++) {
entry = itemsToAdd.poll();
if (entry != null) {
success = PubSubPersistenceManager.createPublishedItem(service, entry);
if (!success) {
itemsToAdd.add(entry);
}
}
}
}
catch (Throwable e) {
Log.error(LocaleUtils.getLocalizedString("admin.error"), e);
}
} }
} }
......
...@@ -19,6 +19,7 @@ import org.jivesoftware.openfire.RoutingTable; ...@@ -19,6 +19,7 @@ import org.jivesoftware.openfire.RoutingTable;
import org.jivesoftware.openfire.XMPPServer; import org.jivesoftware.openfire.XMPPServer;
import org.jivesoftware.openfire.cluster.ClusterEventListener; import org.jivesoftware.openfire.cluster.ClusterEventListener;
import org.jivesoftware.openfire.cluster.ClusterManager; import org.jivesoftware.openfire.cluster.ClusterManager;
import org.jivesoftware.openfire.commands.AdHocCommandManager;
import org.jivesoftware.openfire.component.InternalComponentManager; import org.jivesoftware.openfire.component.InternalComponentManager;
import org.jivesoftware.openfire.container.BasicModule; import org.jivesoftware.openfire.container.BasicModule;
import org.jivesoftware.openfire.disco.DiscoInfoProvider; import org.jivesoftware.openfire.disco.DiscoInfoProvider;
...@@ -39,6 +40,7 @@ import org.xmpp.packet.*; ...@@ -39,6 +40,7 @@ import org.xmpp.packet.*;
import java.util.*; import java.util.*;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList; import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.LinkedBlockingQueue;
/** /**
* Module that implements JEP-60: Publish-Subscribe. By default node collections and * Module that implements JEP-60: Publish-Subscribe. By default node collections and
...@@ -63,6 +65,47 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -63,6 +65,47 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
* Nodes managed by this manager, table: key nodeID (String); value Node * Nodes managed by this manager, table: key nodeID (String); value Node
*/ */
private Map<String, Node> nodes = new ConcurrentHashMap<String, Node>(); private Map<String, Node> nodes = new ConcurrentHashMap<String, Node>();
/**
* Keep a registry of the presence's show value of users that subscribed to a node of
* the pubsub service and for which the node only delivers notifications for online users
* or node subscriptions deliver events based on the user presence show value. Offline
* users will not have an entry in the map. Note: Key-> bare JID and Value-> Map whose key
* is full JID of connected resource and value is show value of the last received presence.
*/
private Map<String, Map<String, String>> barePresences =
new ConcurrentHashMap<String, Map<String, String>>();
/**
* Queue that holds the items that need to be added to the database.
*/
private Queue<PublishedItem> itemsToAdd = new LinkedBlockingQueue<PublishedItem>();
/**
* Queue that holds the items that need to be deleted from the database.
*/
private Queue<PublishedItem> itemsToDelete = new LinkedBlockingQueue<PublishedItem>();
/**
* Manager that keeps the list of ad-hoc commands and processing command requests.
*/
private AdHocCommandManager manager;
/**
* The time to elapse between each execution of the maintenance process. Default
* is 2 minutes.
*/
private int items_task_timeout = 2 * 60 * 1000;
/**
* Task that saves or deletes published items from the database.
*/
private PublishedItemTask publishedItemTask;
/**
* Timer to save published items to the database or remove deleted or old items.
*/
private Timer timer = new Timer("PubSub maintenance");
/** /**
* Returns the permission policy for creating nodes. A true value means that not anyone can * Returns the permission policy for creating nodes. A true value means that not anyone can
* create a node, only the JIDs listed in <code>allowedToCreate</code> are allowed to create * create a node, only the JIDs listed in <code>allowedToCreate</code> are allowed to create
...@@ -117,6 +160,15 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -117,6 +160,15 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
public PubSubModule() { public PubSubModule() {
super("Publish Subscribe Service"); super("Publish Subscribe Service");
// Initialize the ad-hoc commands manager to use for this pubsub service
manager = new AdHocCommandManager();
manager.addCommand(new PendingSubscriptionsCommand(this));
// Save or delete published items from the database every 2 minutes starting in
// 2 minutes (default values)
publishedItemTask = new PublishedItemTask(this);
timer.schedule(publishedItemTask, items_task_timeout, items_task_timeout);
} }
public void process(Packet packet) { public void process(Packet packet) {
...@@ -127,15 +179,15 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -127,15 +179,15 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
try { try {
// Check if the packet is a disco request or a packet with namespace iq:register // Check if the packet is a disco request or a packet with namespace iq:register
if (packet instanceof IQ) { if (packet instanceof IQ) {
if (!engine.process((IQ) packet)) { if (!engine.process(this, (IQ) packet)) {
process((IQ) packet); process((IQ) packet);
} }
} }
else if (packet instanceof Presence) { else if (packet instanceof Presence) {
engine.process((Presence) packet); engine.process(this, (Presence) packet);
} }
else { else {
engine.process((Message) packet); engine.process(this, (Message) packet);
} }
} }
catch (Exception e) { catch (Exception e) {
...@@ -216,23 +268,23 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -216,23 +268,23 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
} }
public Collection<String> getShowPresences(JID subscriber) { public Collection<String> getShowPresences(JID subscriber) {
return engine.getShowPresences(subscriber); return PubSubEngine.getShowPresences(this, subscriber);
} }
public void presenceSubscriptionNotRequired(Node node, JID user) { public void presenceSubscriptionNotRequired(Node node, JID user) {
engine.presenceSubscriptionNotRequired(node, user); PubSubEngine.presenceSubscriptionNotRequired(this, node, user);
} }
public void presenceSubscriptionRequired(Node node, JID user) { public void presenceSubscriptionRequired(Node node, JID user) {
engine.presenceSubscriptionRequired(node, user); PubSubEngine.presenceSubscriptionRequired(this, node, user);
} }
public void queueItemToAdd(PublishedItem newItem) { public void queueItemToAdd(PublishedItem newItem) {
engine.queueItemToAdd(newItem); PubSubEngine.queueItemToAdd(this, newItem);
} }
public void queueItemToRemove(PublishedItem removedItem) { public void queueItemToRemove(PublishedItem removedItem) {
engine.queueItemToRemove(removedItem); PubSubEngine.queueItemToRemove(this, removedItem);
} }
public String getServiceName() { public String getServiceName() {
...@@ -334,7 +386,7 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -334,7 +386,7 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
routingTable = server.getRoutingTable(); routingTable = server.getRoutingTable();
router = server.getPacketRouter(); router = server.getPacketRouter();
engine = new PubSubEngine(this, server.getPacketRouter()); engine = new PubSubEngine(server.getPacketRouter());
// Load default configuration for leaf nodes // Load default configuration for leaf nodes
leafDefaultConfiguration = PubSubPersistenceManager.loadDefaultConfiguration(this, true); leafDefaultConfiguration = PubSubPersistenceManager.loadDefaultConfiguration(this, true);
...@@ -406,7 +458,7 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -406,7 +458,7 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
// Add the route to this service // Add the route to this service
routingTable.addComponentRoute(getAddress(), this); routingTable.addComponentRoute(getAddress(), this);
// Start the pubsub engine // Start the pubsub engine
engine.start(); engine.start(this);
ArrayList<String> params = new ArrayList<String>(); ArrayList<String> params = new ArrayList<String>();
params.clear(); params.clear();
params.add(getServiceDomain()); params.add(getServiceDomain());
...@@ -419,7 +471,7 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -419,7 +471,7 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
routingTable.removeComponentRoute(getAddress()); routingTable.removeComponentRoute(getAddress());
// Stop the pubsub engine. This will gives us the chance to // Stop the pubsub engine. This will gives us the chance to
// save queued items to the database. // save queued items to the database.
engine.shutdown(); engine.shutdown(this);
} }
private void enableService(boolean enabled) { private void enableService(boolean enabled) {
...@@ -750,4 +802,40 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -750,4 +802,40 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
} }
return buf.toString(); return buf.toString();
} }
public Map<String, Map<String, String>> getBarePresences() {
return barePresences;
}
public Queue<PublishedItem> getItemsToAdd() {
return itemsToAdd;
}
public Queue<PublishedItem> getItemsToDelete() {
return itemsToDelete;
}
public AdHocCommandManager getManager() {
return manager;
}
public PublishedItemTask getPublishedItemTask() {
return publishedItemTask;
}
public void setPublishedItemTask(PublishedItemTask task) {
publishedItemTask = task;
}
public Timer getTimer() {
return timer;
}
public int getItemsTaskTimeout() {
return items_task_timeout;
}
public void setItemsTaskTimeout(int timeout) {
items_task_timeout = timeout;
}
} }
...@@ -13,10 +13,10 @@ package org.jivesoftware.openfire.pubsub; ...@@ -13,10 +13,10 @@ package org.jivesoftware.openfire.pubsub;
import org.dom4j.io.SAXReader; import org.dom4j.io.SAXReader;
import org.jivesoftware.database.DbConnectionManager; import org.jivesoftware.database.DbConnectionManager;
import org.jivesoftware.util.Log;
import org.jivesoftware.util.StringUtils;
import org.jivesoftware.openfire.pubsub.models.AccessModel; import org.jivesoftware.openfire.pubsub.models.AccessModel;
import org.jivesoftware.openfire.pubsub.models.PublisherModel; import org.jivesoftware.openfire.pubsub.models.PublisherModel;
import org.jivesoftware.util.Log;
import org.jivesoftware.util.StringUtils;
import org.xmpp.packet.JID; import org.xmpp.packet.JID;
import java.io.StringReader; import java.io.StringReader;
...@@ -35,14 +35,22 @@ import java.util.concurrent.LinkedBlockingQueue; ...@@ -35,14 +35,22 @@ import java.util.concurrent.LinkedBlockingQueue;
*/ */
public class PubSubPersistenceManager { public class PubSubPersistenceManager {
private static final String LOAD_NODES = private static final String LOAD_NON_LEAF_NODES =
"SELECT nodeID, leaf, creationDate, modificationDate, parent, deliverPayloads, " +
"maxPayloadSize, persistItems, maxItems, notifyConfigChanges, notifyDelete, " +
"notifyRetract, presenceBased, sendItemSubscribe, publisherModel, " +
"subscriptionEnabled, configSubscription, accessModel, payloadType, " +
"bodyXSLT, dataformXSLT, creator, description, language, name, " +
"replyPolicy, associationPolicy, maxLeafNodes FROM pubsubNode " +
"WHERE serviceID=? AND leaf=0 ORDER BY nodeID";
private static final String LOAD_LEAF_NODES =
"SELECT nodeID, leaf, creationDate, modificationDate, parent, deliverPayloads, " + "SELECT nodeID, leaf, creationDate, modificationDate, parent, deliverPayloads, " +
"maxPayloadSize, persistItems, maxItems, notifyConfigChanges, notifyDelete, " + "maxPayloadSize, persistItems, maxItems, notifyConfigChanges, notifyDelete, " +
"notifyRetract, presenceBased, sendItemSubscribe, publisherModel, " + "notifyRetract, presenceBased, sendItemSubscribe, publisherModel, " +
"subscriptionEnabled, configSubscription, accessModel, payloadType, " + "subscriptionEnabled, configSubscription, accessModel, payloadType, " +
"bodyXSLT, dataformXSLT, creator, description, language, name, " + "bodyXSLT, dataformXSLT, creator, description, language, name, " +
"replyPolicy, associationPolicy, maxLeafNodes FROM pubsubNode " + "replyPolicy, associationPolicy, maxLeafNodes FROM pubsubNode " +
"WHERE serviceID=? ORDER BY nodeID"; "WHERE serviceID=? AND leaf=1 ORDER BY nodeID";
private static final String UPDATE_NODE = private static final String UPDATE_NODE =
"UPDATE pubsubNode SET modificationDate=?, parent=?, deliverPayloads=?, " + "UPDATE pubsubNode SET modificationDate=?, parent=?, deliverPayloads=?, " +
"maxPayloadSize=?, persistItems=?, maxItems=?, " + "maxPayloadSize=?, persistItems=?, maxItems=?, " +
...@@ -446,11 +454,22 @@ public class PubSubPersistenceManager { ...@@ -446,11 +454,22 @@ public class PubSubPersistenceManager {
Map<String, Node> nodes = new HashMap<String, Node>(); Map<String, Node> nodes = new HashMap<String, Node>();
try { try {
con = DbConnectionManager.getConnection(); con = DbConnectionManager.getConnection();
// Get all nodes at once (with 1 query) // Get all non-leaf nodes (to ensure parent nodes are loaded before their children)
pstmt = con.prepareStatement(LOAD_NODES); pstmt = con.prepareStatement(LOAD_NON_LEAF_NODES);
pstmt.setString(1, service.getServiceID()); pstmt.setString(1, service.getServiceID());
ResultSet rs = pstmt.executeQuery(); ResultSet rs = pstmt.executeQuery();
// Rebuild all loaded nodes // Rebuild loaded non-leaf nodes
while(rs.next()) {
loadNode(service, nodes, rs);
}
rs.close();
pstmt.close();
// Get all leaf nodes (remaining unloaded nodes)
pstmt = con.prepareStatement(LOAD_LEAF_NODES);
pstmt.setString(1, service.getServiceID());
rs = pstmt.executeQuery();
// Rebuild loaded leaf nodes
while(rs.next()) { while(rs.next()) {
loadNode(service, nodes, rs); loadNode(service, nodes, rs);
} }
......
...@@ -11,11 +11,15 @@ ...@@ -11,11 +11,15 @@
package org.jivesoftware.openfire.pubsub; package org.jivesoftware.openfire.pubsub;
import org.jivesoftware.openfire.commands.AdHocCommandManager;
import org.xmpp.packet.JID; import org.xmpp.packet.JID;
import org.xmpp.packet.Message; import org.xmpp.packet.Message;
import org.xmpp.packet.Packet; import org.xmpp.packet.Packet;
import java.util.Collection; import java.util.Collection;
import java.util.Map;
import java.util.Queue;
import java.util.Timer;
/** /**
* A PubSubService is responsible for keeping the hosted nodes by the service, the default * A PubSubService is responsible for keeping the hosted nodes by the service, the default
...@@ -50,6 +54,18 @@ public interface PubSubService { ...@@ -50,6 +54,18 @@ public interface PubSubService {
*/ */
String getServiceID(); String getServiceID();
/**
* Returns a registry of the presence's show value of users that subscribed to a node of
* the pubsub service and for which the node only delivers notifications for online users
* or node subscriptions deliver events based on the user presence show value. Offline
* users will not have an entry in the map. Note: Key-> bare JID and Value-> Map whose key
* is full JID of connected resource and value is show value of the last received presence.
*
* @return a registry of the presence's show value of users that subscribed to a node
* of the pubsub service.
*/
Map<String, Map<String, String>> getBarePresences();
/** /**
* Returns true if the pubsub service allows the specified user to create nodes. * Returns true if the pubsub service allows the specified user to create nodes.
* *
...@@ -222,6 +238,41 @@ public interface PubSubService { ...@@ -222,6 +238,41 @@ public interface PubSubService {
*/ */
void queueItemToAdd(PublishedItem newItem); void queueItemToAdd(PublishedItem newItem);
/**
* Gets the queue that holds the items that need to be added to the database.
*
* @return the queue that holds the items that need to be added to the database.
*/
Queue<PublishedItem> getItemsToAdd();
/**
* Gets the queue that holds the items that need to be deleted from the database.
*
* @return the queue that holds the items that need to be deleted from the database.
*/
Queue<PublishedItem> getItemsToDelete();
/**
* Returns the ad-hoc commands manager used for this service.
*
* @return the ad-hoc commands manager used for this service.
*/
AdHocCommandManager getManager();
/**
* Returns the published item task used for this service.
*
* @return the published item task used for this service.
*/
PublishedItemTask getPublishedItemTask();
/**
* Sets the published item task used for this service.
*
* @param task the PublishedItemTask to set for this service.
*/
void setPublishedItemTask(PublishedItemTask task);
/** /**
* Adds the item to the queue of items to remove from the database. The queue is going * Adds the item to the queue of items to remove from the database. The queue is going
* to be processed by another thread. * to be processed by another thread.
...@@ -229,4 +280,26 @@ public interface PubSubService { ...@@ -229,4 +280,26 @@ public interface PubSubService {
* @param removedItem the item to remove from the database. * @param removedItem the item to remove from the database.
*/ */
void queueItemToRemove(PublishedItem removedItem); void queueItemToRemove(PublishedItem removedItem);
/**
* Returns the timer used for the maintenance process of this service.
*
* @return the timer used for the maintenance process of this service.
*/
Timer getTimer();
/**
* Returns the timeout value for the published items maintenance task.
*
* @return the timeout value for the published items maintenance task.
*/
int getItemsTaskTimeout();
/**
* Sets the timeout value for the published items maintenance task.
*
* @param timeout the timeout value for the published items maintenance task.
*/
void setItemsTaskTimeout(int timeout);
} }
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