Commit 8dbcb94b authored by Tom Evans's avatar Tom Evans Committed by tevans

OF-205: Implement replicated cache for published items

git-svn-id: http://svn.igniterealtime.org/svn/repos/openfire/branches/pubsub_clustering@13282 b35dd754-fafc-0310-a699-88a17e54d16e
parents a4bebe6b d4c65b40
...@@ -495,7 +495,7 @@ public class PEPService implements PubSubService, Cacheable { ...@@ -495,7 +495,7 @@ public class PEPService implements PubSubService, Cacheable {
Message notification = new Message(); Message notification = new Message();
Element event = notification.getElement().addElement("event", "http://jabber.org/protocol/pubsub#event"); Element event = notification.getElement().addElement("event", "http://jabber.org/protocol/pubsub#event");
Element items = event.addElement("items"); Element items = event.addElement("items");
items.addAttribute("node", leafLastPublishedItem.getNode().getNodeID()); items.addAttribute("node", leafLastPublishedItem.getNodeID());
Element item = items.addElement("item"); Element item = items.addElement("item");
if (((LeafNode) leafLastPublishedItem.getNode()).isItemRequired()) { if (((LeafNode) leafLastPublishedItem.getNode()).isItemRequired()) {
item.addAttribute("id", leafLastPublishedItem.getID()); item.addAttribute("id", leafLastPublishedItem.getID());
......
...@@ -102,7 +102,7 @@ public class NodeAffiliate { ...@@ -102,7 +102,7 @@ public class NodeAffiliate {
// //
// If the node ID looks like a JID, replace it with the published item's node ID. // If the node ID looks like a JID, replace it with the published item's node ID.
if (getNode().getNodeID().indexOf("@") >= 0) { if (getNode().getNodeID().indexOf("@") >= 0) {
items.addAttribute("node", publishedItem.getNode().getNodeID()); items.addAttribute("node", publishedItem.getNodeID());
} }
// Add item information to the event notification // Add item information to the event notification
......
...@@ -150,7 +150,6 @@ public class NodeSubscription { ...@@ -150,7 +150,6 @@ public class NodeSubscription {
/** /**
* Creates a new subscription of the specified user with the node. * Creates a new subscription of the specified user with the node.
* *
* @param service the pubsub service hosting the node where this subscription lives.
* @param node Node to which this subscription is interested in. * @param node Node to which this subscription is interested in.
* @param owner the JID of the entity that owns this subscription. * @param owner the JID of the entity that owns this subscription.
* @param jid the JID of the user that owns the subscription. * @param jid the JID of the user that owns the subscription.
......
...@@ -20,10 +20,19 @@ ...@@ -20,10 +20,19 @@
package org.jivesoftware.openfire.pubsub; package org.jivesoftware.openfire.pubsub;
import org.xmpp.packet.JID; import java.io.Serializable;
import org.dom4j.Element; import java.io.StringReader;
import java.util.Date; import java.util.Date;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
import org.jivesoftware.openfire.XMPPServer;
import org.jivesoftware.openfire.pep.PEPServiceManager;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.xmpp.packet.JID;
/** /**
* A published item to a node. Once an item was published to a node, node subscribers will be * A published item to a node. Once an item was published to a node, node subscribers will be
...@@ -37,7 +46,26 @@ import java.util.Date; ...@@ -37,7 +46,26 @@ import java.util.Date;
* *
* @author Matt Tucker * @author Matt Tucker
*/ */
public class PublishedItem { public class PublishedItem implements Serializable {
private static final Logger log = LoggerFactory.getLogger(PublishedItem.class);
private static final int POOL_SIZE = 50;
/**
* Pool of SAX Readers. SAXReader is not thread safe so we need to have a pool of readers.
*/
private static BlockingQueue<SAXReader> xmlReaders = new LinkedBlockingQueue<SAXReader>(POOL_SIZE);
private static final long serialVersionUID = 7012925993623144574L;
static {
// Initialize the pool of sax readers
for (int i=0; i<POOL_SIZE; i++) {
SAXReader xmlReader = new SAXReader();
xmlReader.setEncoding("UTF-8");
xmlReaders.add(xmlReader);
}
}
/** /**
* JID of the entity that published the item to the node. This is the full JID * JID of the entity that published the item to the node. This is the full JID
...@@ -47,7 +75,15 @@ public class PublishedItem { ...@@ -47,7 +75,15 @@ public class PublishedItem {
/** /**
* The node where the item was published. * The node where the item was published.
*/ */
private LeafNode node; private transient LeafNode node;
/**
* The id for the node where the item was published.
*/
private String nodeId;
/**
* The id for the service hosting the node for this item
*/
private String serviceId;
/** /**
* ID that uniquely identifies the published item in the node. * ID that uniquely identifies the published item in the node.
*/ */
...@@ -57,28 +93,48 @@ public class PublishedItem { ...@@ -57,28 +93,48 @@ public class PublishedItem {
*/ */
private Date creationDate; private Date creationDate;
/** /**
* The payload included when publishing the item. * The optional payload is included when publishing the item. This value
* is created from the payload XML and cached as/when needed.
*/ */
private Element payload; private transient Element payload;
/** /**
* XML representation of the payload. This is actually a cache that avoids * XML representation of the payload (for serialization)
* doing Element#asXML.
*/ */
private String payloadXML; private String payloadXML;
PublishedItem(LeafNode node, JID publisher, String id, Date creationDate) { PublishedItem(LeafNode node, JID publisher, String id, Date creationDate) {
this.node = node; this.node = node;
this.nodeId = node.getNodeID();
this.serviceId = node.getService().getServiceID();
this.publisher = publisher; this.publisher = publisher;
this.id = id; this.id = id;
this.creationDate = creationDate; this.creationDate = creationDate;
} }
/**
* Returns the id for the {@link LeafNode} where this item was published.
*
* @return the ID for the leaf node where this item was published.
*/
public String getNodeID() {
return nodeId;
}
/** /**
* Returns the {@link LeafNode} where this item was published. * Returns the {@link LeafNode} where this item was published.
* *
* @return the leaf node where this item was published. * @return the leaf node where this item was published.
*/ */
public LeafNode getNode() { public LeafNode getNode() {
if (node == null) {
if (Node.PUBSUB_SVC_ID.equals(serviceId)) {
node = (LeafNode) XMPPServer.getInstance().getPubSubModule().getNode(nodeId);
} else {
PEPServiceManager serviceMgr = XMPPServer.getInstance().getIQPEPHandler().getServiceManager();
node = serviceMgr.hasCachedService(new JID(serviceId)) ?
(LeafNode) serviceMgr.getPEPService(serviceId).getNode(nodeId) : null;
}
}
return node; return node;
} }
...@@ -117,6 +173,20 @@ public class PublishedItem { ...@@ -117,6 +173,20 @@ public class PublishedItem {
* @return the payload included when publishing the item or <tt>null</tt> if none was found. * @return the payload included when publishing the item or <tt>null</tt> if none was found.
*/ */
public Element getPayload() { public Element getPayload() {
if (payload == null && payloadXML != null) {
// payload initialized as XML string from DB
SAXReader xmlReader = null;
try {
xmlReader = xmlReaders.take();
payload = xmlReader.read(new StringReader(payloadXML)).getRootElement();
} catch (Exception ex) {
log.error("Failed to parse payload XML", ex);
} finally {
if (xmlReader != null) {
xmlReaders.add(xmlReader);
}
}
}
return payload; return payload;
} }
...@@ -131,6 +201,19 @@ public class PublishedItem { ...@@ -131,6 +201,19 @@ public class PublishedItem {
return payloadXML; return payloadXML;
} }
/**
* Sets the payload included when publishing the item. A published item may or may not
* have a payload. Transient nodes that are configured to not broadcast payloads may allow
* published items to have no payload.
*
* @param payloadXML the payload included when publishing the item or <tt>null</tt>
* if none was found.
*/
void setPayloadXML(String payloadXML) {
this.payloadXML = payloadXML;
this.payload = null; // will be recreated only if needed
}
/** /**
* Sets the payload included when publishing the item. A published item may or may not * Sets the payload included when publishing the item. A published item may or may not
* have a payload. Transient nodes that are configured to not broadcast payloads may allow * have a payload. Transient nodes that are configured to not broadcast payloads may allow
...@@ -144,8 +227,7 @@ public class PublishedItem { ...@@ -144,8 +227,7 @@ public class PublishedItem {
// Update XML representation of the payload // Update XML representation of the payload
if (payload == null) { if (payload == null) {
payloadXML = null; payloadXML = null;
} } else {
else {
payloadXML = payload.asXML(); payloadXML = payload.asXML();
} }
} }
...@@ -158,7 +240,7 @@ public class PublishedItem { ...@@ -158,7 +240,7 @@ public class PublishedItem {
* @return true if payload contains the specified keyword. * @return true if payload contains the specified keyword.
*/ */
boolean containsKeyword(String keyword) { boolean containsKeyword(String keyword) {
if (payloadXML == null || keyword == null) { if (getPayloadXML() == null || keyword == null) {
return true; return true;
} }
return payloadXML.contains(keyword); return payloadXML.contains(keyword);
...@@ -173,9 +255,41 @@ public class PublishedItem { ...@@ -173,9 +255,41 @@ public class PublishedItem {
*/ */
public boolean canDelete(JID user) { public boolean canDelete(JID user) {
if (publisher.equals(user) || publisher.toBareJID().equals(user.toBareJID()) || if (publisher.equals(user) || publisher.toBareJID().equals(user.toBareJID()) ||
node.isAdmin(user)) { getNode().isAdmin(user)) {
return true; return true;
} }
return false; return false;
} }
/**
* Returns a string that uniquely identifies this published item
* in the following format: <i>nodeId:itemId</i>
* @return Unique identifier for this item
*/
public String getItemKey() {
return getItemKey(nodeId,id);
}
/**
* Returns a string that uniquely identifies this published item
* in the following format: <i>nodeId:itemId</i>
* @param node Node for the published item
* @param itemId Id for the published item (unique within the node)
* @return Unique identifier for this item
*/
public static String getItemKey(LeafNode node, String itemId) {
return getItemKey(node.getNodeID(), itemId);
}
/**
* Returns a string that uniquely identifies this published item
* in the following format: <i>nodeId:itemId</i>
* @param nodeId Node id for the published item
* @param itemId Id for the published item (unique within the node)
* @return Unique identifier for this item
*/
public static String getItemKey(String nodeId, String itemId) {
return new StringBuilder(nodeId)
.append(":").append(itemId).toString();
}
} }
...@@ -17,7 +17,7 @@ public class FlushTask implements ClusterTask ...@@ -17,7 +17,7 @@ public class FlushTask implements ClusterTask
@Override @Override
public void run() public void run()
{ {
PubSubPersistenceManager.flushItems(false); // just this member PubSubPersistenceManager.flushPendingItems(false); // just this member
} }
@Override @Override
......
...@@ -128,6 +128,7 @@ public class CacheFactory { ...@@ -128,6 +128,7 @@ public class CacheFactory {
cacheNames.put("Entity Capabilities Users", "entityCapabilitiesUsers"); cacheNames.put("Entity Capabilities Users", "entityCapabilitiesUsers");
cacheNames.put("Clearspace SSO Nonce", "clearspaceSSONonce"); cacheNames.put("Clearspace SSO Nonce", "clearspaceSSONonce");
cacheNames.put("PEPServiceManager", "pepServiceManager"); cacheNames.put("PEPServiceManager", "pepServiceManager");
cacheNames.put("Published Items", "publishedItems");
cacheProps.put("cache.fileTransfer.size", 128 * 1024l); cacheProps.put("cache.fileTransfer.size", 128 * 1024l);
cacheProps.put("cache.fileTransfer.maxLifetime", 1000 * 60 * 10l); cacheProps.put("cache.fileTransfer.maxLifetime", 1000 * 60 * 10l);
...@@ -199,6 +200,8 @@ public class CacheFactory { ...@@ -199,6 +200,8 @@ public class CacheFactory {
cacheProps.put("cache.clearspaceSSONonce.maxLifetime", JiveConstants.MINUTE * 2); cacheProps.put("cache.clearspaceSSONonce.maxLifetime", JiveConstants.MINUTE * 2);
cacheProps.put("cache.pepServiceManager.size", 1024l * 1024 * 10); cacheProps.put("cache.pepServiceManager.size", 1024l * 1024 * 10);
cacheProps.put("cache.pepServiceManager.maxLifetime", JiveConstants.MINUTE * 30); cacheProps.put("cache.pepServiceManager.maxLifetime", JiveConstants.MINUTE * 30);
cacheProps.put("cache.publishedItems.size", 1024l * 1024 * 10);
cacheProps.put("cache.publishedItems.maxLifetime", JiveConstants.MINUTE * 15);
} }
private CacheFactory() { private CacheFactory() {
......
...@@ -523,6 +523,25 @@ http://www.tangosol.com/UserGuide-Reference-CacheConfig.jsp ...@@ -523,6 +523,25 @@ http://www.tangosol.com/UserGuide-Reference-CacheConfig.jsp
</init-params> </init-params>
</cache-mapping> </cache-mapping>
<cache-mapping>
<cache-name>Published Items</cache-name>
<scheme-name>replicated</scheme-name>
<init-params>
<init-param>
<param-name>back-size-high</param-name>
<param-value>10485760</param-value>
</init-param>
<init-param>
<param-name>back-expiry</param-name>
<param-value>15m</param-value>
</init-param>
<init-param>
<param-name>back-size-low</param-name>
<param-value>9437180</param-value>
</init-param>
</init-params>
</cache-mapping>
<!-- partitioned caches --> <!-- partitioned caches -->
<cache-mapping> <cache-mapping>
......
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