Commit d9d6d947 authored by Gaston Dombiak's avatar Gaston Dombiak Committed by gato

More and more and more work on clustering events.

git-svn-id: http://svn.igniterealtime.org/svn/repos/openfire/trunk@8553 b35dd754-fafc-0310-a699-88a17e54d16e
parent 20ffb1c4
...@@ -233,6 +233,14 @@ public interface RoutingTable { ...@@ -233,6 +233,14 @@ public interface RoutingTable {
*/ */
Collection<ClientSession> getClientsRoutes(boolean onlyLocal); Collection<ClientSession> getClientsRoutes(boolean onlyLocal);
/**
* Returns the ID of the node that is hosting the specified client session.
*
* @param jid the address of the specified client session.
* @return the ID of the node that is hosting the specified client session or null if not found.
*/
byte[] getNodeIDForClientRoute(JID jid);
/** /**
* Returns the outgoing server session associated to the specified XMPP address or <tt>null</tt> * Returns the outgoing server session associated to the specified XMPP address or <tt>null</tt>
* if none was found. When running inside of a cluster and a remote node is hosting * if none was found. When running inside of a cluster and a remote node is hosting
......
...@@ -1398,94 +1398,30 @@ public class SessionManager extends BasicModule implements ClusterEventListener ...@@ -1398,94 +1398,30 @@ public class SessionManager extends BasicModule implements ClusterEventListener
} }
} }
public void joinedCluster(byte[] oldNodeID) { public void joinedCluster() {
restoreCacheContent(); restoreCacheContent();
} }
public void leavingCluster() { public void joinedCluster(byte[] nodeID) {
if (XMPPServer.getInstance().isShuttingDown()) { // Do nothing
// Do nothing since local sessions will be closed. Local session manager
// and local routing table will be correctly updated thus updating the
// other cluster nodes correctly
}
else {
// This JVM is leaving the cluster but will continue to work. That means
// that clients connected to this JVM will be able to keep talking.
// In other words, their sessions will not be closed (and not removed from
// the routing table or the session manager). However, other nodes should
// get their session managers correctly updated.
// Remove external component sessions hosted locally to the cache (using new nodeID)
for (Session session : localSessionManager.getComponentsSessions()) {
componentSessionsCache.remove(session.getAddress().toString());
}
// Remove connection multiplexer sessions hosted locally to the cache (using new nodeID)
for (String address : localSessionManager.getConnnectionManagerSessions().keySet()) {
multiplexerSessionsCache.remove(address);
}
// Remove incoming server sessions hosted locally to the cache (using new nodeID)
for (LocalIncomingServerSession session : localSessionManager.getIncomingServerSessions()) {
String streamID = session.getStreamID().getID();
incomingServerSessionsCache.remove(streamID);
for (String hostname : session.getValidatedDomains()) {
// Update list of sockets/sessions coming from the same remote hostname
Lock lock = LockManager.getLock(hostname);
try {
lock.lock();
List<String> streamIDs = hostnameSessionsCache.get(hostname);
streamIDs.remove(streamID);
if (streamIDs.isEmpty()) {
hostnameSessionsCache.remove(hostname);
}
else {
hostnameSessionsCache.put(hostname, streamIDs);
}
}
finally {
lock.unlock();
}
// Remove from clustered cache
lock = LockManager.getLock(streamID);
try {
lock.lock();
Set<String> validatedDomains = validatedDomainsCache.get(streamID);
if (validatedDomains == null) {
validatedDomains = new HashSet<String>();
}
validatedDomains.remove(hostname);
if (!validatedDomains.isEmpty()) {
validatedDomainsCache.put(streamID, validatedDomains);
}
else {
validatedDomainsCache.remove(streamID);
}
} finally {
lock.unlock();
}
}
}
// Update counters of client sessions
for (ClientSession session : routingTable.getClientsRoutes(true)) {
// Increment the counter of user sessions
decrementCounter("conncounter");
if (session.getStatus() == Session.STATUS_AUTHENTICATED) {
// Increment counter of authenticated sessions
decrementCounter("usercounter");
}
}
}
} }
public void leftCluster() { public void leftCluster() {
// TODO Send unavailable presence TO roster contacts not hosted in this JVM (type=FROM)
// TODO Send unavailable presence FROM roster contacts not hosted in this JVM (type=TO)
// TODO Send unavailable presence FROM & TO roster contacts not hosted in this JVM (type=BOTH)
// TODO Send unavailable presence TO other resources of the user not hosted in this JVM
if (!XMPPServer.getInstance().isShuttingDown()) { if (!XMPPServer.getInstance().isShuttingDown()) {
// Add local sessions to caches // Add local sessions to caches
restoreCacheContent(); restoreCacheContent();
} }
} }
public void leftCluster(byte[] nodeID) {
// Do nothing
}
public void markedAsSeniorClusterMember() { public void markedAsSeniorClusterMember() {
// Do nothing // Do nothing
} }
......
...@@ -29,29 +29,24 @@ public interface ClusterEventListener { ...@@ -29,29 +29,24 @@ public interface ClusterEventListener {
* At this point the CacheFactory holds clustered caches. That means that modifications * At this point the CacheFactory holds clustered caches. That means that modifications
* to the caches will be reflected in the cluster. The clustered caches were just * to the caches will be reflected in the cluster. The clustered caches were just
* obtained from the cluster and no local cached data was automatically moved.<p> * obtained from the cluster and no local cached data was automatically moved.<p>
*
* @param oldNodeID nodeID used by this JVM before joining the cluster.
*/ */
void joinedCluster(byte[] oldNodeID); void joinedCluster();
/** /**
* Notification event indicating that this JVM is about to leave the cluster. This could * Notification event indicating that another JVM is now part of a cluster.<p>
* happen when disabling clustering support, removing the enterprise plugin that provides
* clustering support or even shutdown the server.<p>
* *
* At this point the CacheFactory is still holding clustered caches. That means that * At this point the CacheFactory of the new node holds clustered caches. That means
* modifications to the caches will be reflected in the cluster.<p> * that modifications to the caches of this JVM will be reflected in the cluster and
* in particular in the new node.
* *
* Use {@link org.jivesoftware.openfire.XMPPServer#isShuttingDown()} to figure out if the * @param nodeID ID of the node that joined the cluster.
* server is being shutdown.
*/ */
void leavingCluster(); void joinedCluster(byte[] nodeID);
/** /**
* Notification event indicating that this JVM is no longer part of the cluster. This could * Notification event indicating that this JVM is no longer part of the cluster. This could
* happen when disabling clustering support, removing the enterprise plugin that provides * happen when disabling clustering support, removing the enterprise plugin that provides
* clustering support or connection to cluster got lost. If connection to cluster was lost * clustering support or connection to cluster got lost.<p>
* then this event will not be predated by the {@link #leavingCluster()} event.<p>
* *
* Moreover, if we were in a "split brain" scenario (ie. separated cluster islands) and the * Moreover, if we were in a "split brain" scenario (ie. separated cluster islands) and the
* island were this JVM belonged was marked as "old" then all nodes of that island will * island were this JVM belonged was marked as "old" then all nodes of that island will
...@@ -65,6 +60,25 @@ public interface ClusterEventListener { ...@@ -65,6 +60,25 @@ public interface ClusterEventListener {
*/ */
void leftCluster(); void leftCluster();
/**
* Notification event indicating that another JVM is no longer part of the cluster. This could
* happen when disabling clustering support, removing the enterprise plugin that provides
* clustering support or connection to cluster got lost.<p>
*
* Moreover, if we were in a "split brain" scenario (ie. separated cluster islands) and the
* island were the other JVM belonged was marked as "old" then all nodes of that island will
* get the <tt>left cluster event</tt> and <tt>joined cluster events</tt>. That means that
* caches will be reset and thus will need to be repopulated again with fresh data from this JVM.
* This also includes the case where the other JVM was the senior cluster member and when the islands
* met again then the other JVM stopped being the senior member.<p>
*
* At this point the CacheFactory of the leaving node holds local caches. That means that modifications to
* the caches of this JVM will not affect the leaving node but other cluster members.
*
* @param nodeID ID of the node that is left the cluster.
*/
void leftCluster(byte[] nodeID);
/** /**
* Notification event indicating that this JVM is now the senior cluster member. This * Notification event indicating that this JVM is now the senior cluster member. This
* could either happen when initially joining the cluster or when the senior cluster * could either happen when initially joining the cluster or when the senior cluster
...@@ -74,7 +88,7 @@ public interface ClusterEventListener { ...@@ -74,7 +88,7 @@ public interface ClusterEventListener {
* island will have its own senior cluster member. However, when the islands meet again there * island will have its own senior cluster member. However, when the islands meet again there
* could only be one senior cluster member so one of the senior cluster members will stop playing * could only be one senior cluster member so one of the senior cluster members will stop playing
* that role. When that happens the JVM no longer playing that role will receive the * that role. When that happens the JVM no longer playing that role will receive the
* {@link #leftCluster()} and {@link #joinedCluster(byte[])} events. * {@link #leftCluster()} and {@link #joinedCluster()} events.
*/ */
void markedAsSeniorClusterMember(); void markedAsSeniorClusterMember();
} }
...@@ -43,11 +43,11 @@ public class ClusterManager { ...@@ -43,11 +43,11 @@ public class ClusterManager {
Event event = events.take(); Event event = events.take();
EventType eventType = event.getType(); EventType eventType = event.getType();
// Make sure that CacheFactory is getting this events first (to update cache structure) // Make sure that CacheFactory is getting this events first (to update cache structure)
if (eventType == EventType.joined_cluster) { if (eventType == EventType.joined_cluster && event.getNodeID() == null) {
// Replace standalone caches with clustered caches. Local cached data is not moved. // Replace standalone caches with clustered caches. Local cached data is not moved.
CacheFactory.joinedCluster(); CacheFactory.joinedCluster();
} }
else if (eventType == EventType.left_cluster) { else if (eventType == EventType.left_cluster && event.getNodeID() == null) {
// Replace clustered caches with standalone caches. Cached data is not moved to new cache. // Replace clustered caches with standalone caches. Cached data is not moved to new cache.
CacheFactory.leftCluster(); CacheFactory.leftCluster();
} }
...@@ -56,15 +56,21 @@ public class ClusterManager { ...@@ -56,15 +56,21 @@ public class ClusterManager {
try { try {
switch (eventType) { switch (eventType) {
case joined_cluster: { case joined_cluster: {
listener.joinedCluster(event.getOldNodeID()); if (event.getNodeID() == null) {
break; listener.joinedCluster();
}
else {
listener.joinedCluster(event.getNodeID());
} }
case leaving_cluster: {
listener.leavingCluster();
break; break;
} }
case left_cluster: { case left_cluster: {
if (event.getNodeID() == null) {
listener.leftCluster(); listener.leftCluster();
}
else {
listener.leftCluster(event.getNodeID());
}
break; break;
} }
case marked_senior_cluster_member: { case marked_senior_cluster_member: {
...@@ -126,12 +132,11 @@ public class ClusterManager { ...@@ -126,12 +132,11 @@ public class ClusterManager {
* This event will be triggered in another thread. This will avoid potential deadlocks * This event will be triggered in another thread. This will avoid potential deadlocks
* in Coherence. * in Coherence.
* *
* @param oldNodeID nodeID used by this JVM before joining the cluster.
* @param asynchronous true if event will be triggered in background * @param asynchronous true if event will be triggered in background
*/ */
public static void fireJoinedCluster(byte[] oldNodeID, boolean asynchronous) { public static void fireJoinedCluster(boolean asynchronous) {
try { try {
Event event = new Event(EventType.joined_cluster, oldNodeID); Event event = new Event(EventType.joined_cluster, null);
events.put(event); events.put(event);
if (!asynchronous) { if (!asynchronous) {
while (!event.isProcessed()) { while (!event.isProcessed()) {
...@@ -144,21 +149,23 @@ public class ClusterManager { ...@@ -144,21 +149,23 @@ public class ClusterManager {
} }
/** /**
* Triggers event indicating that this JVM is about to leave the cluster. This could * Triggers event indicating that another JVM is now part of a cluster.<p>
* happen when disabling clustering support, removing the enterprise plugin that provides *
* clustering support or even shutdown the server.<p> * This event will be triggered in another thread. This will avoid potential deadlocks
* <p/> * in Coherence.
* This event will be triggered in another thread but won't return until all listeners have *
* been alerted. This will give listeners the chance to use the cluster for any clean up * @param nodeID nodeID assigned to the JVM when joining the cluster.
* operation before the node actually leaves the cluster. * @param asynchronous true if event will be triggered in background
*/ */
public static void fireLeavingCluster() { public static void fireJoinedCluster(byte[] nodeID, boolean asynchronous) {
try { try {
Event event = new Event(EventType.leaving_cluster, null); Event event = new Event(EventType.joined_cluster, nodeID);
events.put(event); events.put(event);
if (!asynchronous) {
while (!event.isProcessed()) { while (!event.isProcessed()) {
Thread.sleep(50); Thread.sleep(50);
} }
}
} catch (InterruptedException e) { } catch (InterruptedException e) {
// Should never happen // Should never happen
} }
...@@ -204,7 +211,7 @@ public class ClusterManager { ...@@ -204,7 +211,7 @@ public class ClusterManager {
* island will have its own senior cluster member. However, when the islands meet again there * island will have its own senior cluster member. However, when the islands meet again there
* could only be one senior cluster member so one of the senior cluster members will stop playing * could only be one senior cluster member so one of the senior cluster members will stop playing
* that role. When that happens the JVM no longer playing that role will receive the * that role. When that happens the JVM no longer playing that role will receive the
* {@link #fireLeftCluster(boolean)} and {@link #fireJoinedCluster(byte[],boolean)} events.<p> * {@link #fireLeftCluster(boolean)} and {@link #fireJoinedCluster(boolean)} events.<p>
* <p/> * <p/>
* This event will be triggered in another thread. This will avoid potential deadlocks * This event will be triggered in another thread. This will avoid potential deadlocks
* in Coherence. * in Coherence.
...@@ -305,20 +312,20 @@ public class ClusterManager { ...@@ -305,20 +312,20 @@ public class ClusterManager {
private static class Event { private static class Event {
private EventType type; private EventType type;
private byte[] oldNodeID; private byte[] nodeID;
private boolean processed; private boolean processed;
public Event(EventType type, byte[] oldNodeID) { public Event(EventType type, byte[] oldNodeID) {
this.type = type; this.type = type;
this.oldNodeID = oldNodeID; this.nodeID = oldNodeID;
} }
public EventType getType() { public EventType getType() {
return type; return type;
} }
public byte[] getOldNodeID() { public byte[] getNodeID() {
return oldNodeID; return nodeID;
} }
public boolean isProcessed() { public boolean isProcessed() {
...@@ -344,11 +351,6 @@ public class ClusterManager { ...@@ -344,11 +351,6 @@ public class ClusterManager {
*/ */
joined_cluster, joined_cluster,
/**
* This JVM is about to leave the cluster.
*/
leaving_cluster,
/** /**
* This JVM is no longer part of the cluster. * This JVM is no longer part of the cluster.
*/ */
......
...@@ -12,15 +12,15 @@ ...@@ -12,15 +12,15 @@
package org.jivesoftware.openfire.muc.spi; package org.jivesoftware.openfire.muc.spi;
import org.dom4j.Element; import org.dom4j.Element;
import org.jivesoftware.util.LocaleUtils;
import org.jivesoftware.util.Log;
import org.jivesoftware.util.NotFoundException;
import org.jivesoftware.openfire.PacketException; import org.jivesoftware.openfire.PacketException;
import org.jivesoftware.openfire.PacketRouter; import org.jivesoftware.openfire.PacketRouter;
import org.jivesoftware.openfire.auth.UnauthorizedException; import org.jivesoftware.openfire.auth.UnauthorizedException;
import org.jivesoftware.openfire.muc.*; import org.jivesoftware.openfire.muc.*;
import org.jivesoftware.openfire.user.UserAlreadyExistsException; import org.jivesoftware.openfire.user.UserAlreadyExistsException;
import org.jivesoftware.openfire.user.UserNotFoundException; import org.jivesoftware.openfire.user.UserNotFoundException;
import org.jivesoftware.util.LocaleUtils;
import org.jivesoftware.util.Log;
import org.jivesoftware.util.NotFoundException;
import org.xmpp.packet.*; import org.xmpp.packet.*;
import java.util.*; import java.util.*;
...@@ -476,6 +476,7 @@ public class MUCUserImpl implements MUCUser { ...@@ -476,6 +476,7 @@ public class MUCUserImpl implements MUCUser {
else { else {
if (Presence.Type.unavailable == packet.getType()) { if (Presence.Type.unavailable == packet.getType()) {
try { try {
// TODO Consider that different nodes can be creating and processing this presence at the same time (when remote node went down)
removeRole(group); removeRole(group);
role.getChatRoom().leaveRoom(role.getNickname()); role.getChatRoom().leaveRoom(role.getNickname());
} }
......
...@@ -923,18 +923,30 @@ public class MultiUserChatServerImpl extends BasicModule implements MultiUserCha ...@@ -923,18 +923,30 @@ public class MultiUserChatServerImpl extends BasicModule implements MultiUserCha
outMessages.addAndGet(numOccupants); outMessages.addAndGet(numOccupants);
} }
public void joinedCluster(byte[] oldNodeID) { public void joinedCluster() {
// Disable the service until we know that we are the senior cluster member // Disable the service until we know that we are the senior cluster member
enableService(false, false); enableService(false, false);
//TODO Do not disable the service. All nodes will host the service (unless it was disabled before)
//TODO Merge rooms existing in the cluster with the ones of the new node
//TODO For rooms that exist in cluster and in new node then send presences of remote occupants to LOCAL occupants
} }
public void leavingCluster() { public void joinedCluster(byte[] nodeID) {
// Do nothing //TODO Merge rooms existing in the cluster with the ones of the new node
//TODO For rooms that exist in cluster and in new node then send presences of new remote occupants to LOCAL occupants
} }
public void leftCluster() { public void leftCluster() {
// Offer the service when not running in a cluster // Offer the service when not running in a cluster
enableService(true, false); enableService(true, false);
//TODO Do not mess with service enablement! :)
//TODO Send unavailable presences of leaving remote occupants to LOCAL occupants
//TODO Remove rooms with no occupants (should happen with previous step)?
}
public void leftCluster(byte[] nodeID) {
//TODO Send unavailable presences of leaving remote occupants to LOCAL occupants
//TODO Remove rooms with no occupants (should happen with previous step)?
} }
public void markedAsSeniorClusterMember() { public void markedAsSeniorClusterMember() {
......
...@@ -439,12 +439,12 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -439,12 +439,12 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
} }
} }
public void joinedCluster(byte[] oldNodeID) { public void joinedCluster() {
// Disable the service until we know that we are the senior cluster member // Disable the service until we know that we are the senior cluster member
enableService(false); enableService(false);
} }
public void leavingCluster() { public void joinedCluster(byte[] nodeID) {
// Do nothing // Do nothing
} }
...@@ -453,6 +453,10 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di ...@@ -453,6 +453,10 @@ public class PubSubModule extends BasicModule implements ServerItemsProvider, Di
enableService(true); enableService(true);
} }
public void leftCluster(byte[] nodeID) {
// Do nothing
}
public void markedAsSeniorClusterMember() { public void markedAsSeniorClusterMember() {
// Offer the service since we are the senior cluster member // Offer the service since we are the senior cluster member
enableService(true); enableService(true);
......
...@@ -527,11 +527,22 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust ...@@ -527,11 +527,22 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust
} }
public boolean hasClientRoute(JID jid) { public boolean hasClientRoute(JID jid) {
return usersCache.get(jid.toString()) != null || isAnonymousRoute(jid); return usersCache.containsKey(jid.toString()) || isAnonymousRoute(jid);
}
public byte[] getNodeIDForClientRoute(JID jid) {
ClientRoute clientRoute = usersCache.get(jid.toString());
if (clientRoute == null) {
clientRoute = anonymousUsersCache.get(jid.toString());
}
if (clientRoute != null) {
return clientRoute.getNodeID();
}
return null;
} }
public boolean isAnonymousRoute(JID jid) { public boolean isAnonymousRoute(JID jid) {
return anonymousUsersCache.get(jid.toString()) != null; return anonymousUsersCache.containsKey(jid.toString());
} }
public boolean isLocalRoute(JID jid) { public boolean isLocalRoute(JID jid) {
...@@ -539,11 +550,11 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust ...@@ -539,11 +550,11 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust
} }
public boolean hasServerRoute(JID jid) { public boolean hasServerRoute(JID jid) {
return serversCache.get(jid.getDomain()) != null; return serversCache.containsKey(jid.getDomain());
} }
public boolean hasComponentRoute(JID jid) { public boolean hasComponentRoute(JID jid) {
return componentsCache.get(jid.getDomain()) != null; return componentsCache.containsKey(jid.getDomain());
} }
public List<JID> getRoutes(JID route) { public List<JID> getRoutes(JID route) {
...@@ -581,7 +592,7 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust ...@@ -581,7 +592,7 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust
else if (route.getDomain().contains(serverName)) { else if (route.getDomain().contains(serverName)) {
// Packet sent to component hosted in this server // Packet sent to component hosted in this server
if (componentsCache.containsKey(route.getDomain())) { if (componentsCache.containsKey(route.getDomain())) {
jids.add(route); jids.add(new JID(route.getDomain()));
} }
} }
else { else {
...@@ -686,7 +697,7 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust ...@@ -686,7 +697,7 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust
localRoutingTable.stop(); localRoutingTable.stop();
} }
public void joinedCluster(byte[] oldNodeID) { public void joinedCluster() {
restoreCacheContent(); restoreCacheContent();
// Broadcast presence of local sessions to remote sessions when subscribed to presence // Broadcast presence of local sessions to remote sessions when subscribed to presence
...@@ -703,89 +714,8 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust ...@@ -703,89 +714,8 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust
} }
} }
public void leavingCluster() { public void joinedCluster(byte[] nodeID) {
if (XMPPServer.getInstance().isShuttingDown()) { // Do nothing
// Do nothing since local sessions will be closed. Local session manager
// and local routing table will be correctly updated thus updating the
// other cluster nodes correctly
}
else {
// This JVM is leaving the cluster but will continue to work. That means
// that clients connected to this JVM will be able to keep talking.
// In other words, their sessions will not be closed (and not removed from
// the routing table or the session manager). However, other nodes should
// get their routing tables correctly updated so we need to temporarily
// remove the content from the cache so other cluster nodes are correctly
// updated. Local content will be restored to cache in #leftCluster
// In the case of an abnormal disconnection from the cluster this event will
// not be triggered so it is up to the cluster nodes to know how to clean up
// their caches from the local data added by this JVM
// Remove outgoing server sessions hosted locally from the cache (using new nodeID)
for (LocalOutgoingServerSession session : localRoutingTable.getServerRoutes()) {
String address = session.getAddress().getDomain();
serversCache.remove(address);
}
// Remove component sessions hosted locally from the cache (using new nodeID) and remove traces to old nodeID
for (RoutableChannelHandler componentRoute : localRoutingTable.getComponentRoute()) {
JID route = componentRoute.getAddress();
String address = route.getDomain();
Lock lock = LockManager.getLock(address + "rt");
try {
lock.lock();
Set<byte[]> nodes = componentsCache.get(address);
if (nodes != null) {
nodes.remove(server.getNodeID());
if (nodes.isEmpty()) {
componentsCache.remove(address);
}
else {
componentsCache.put(address, nodes);
}
}
} finally {
lock.unlock();
}
}
// Remove client sessions hosted locally from the cache (using new nodeID)
for (LocalClientSession session : localRoutingTable.getClientRoutes()) {
boolean anonymous = false;
JID route = session.getAddress();
String address = route.toString();
ClientRoute clientRoute = usersCache.remove(address);
if (clientRoute == null) {
clientRoute = anonymousUsersCache.remove(address);
anonymous = true;
}
if (clientRoute != null && route.getResource() != null) {
Lock lock = LockManager.getLock(route.toBareJID());
try {
lock.lock();
if (anonymous) {
usersSessions.remove(route.toBareJID());
}
else {
Collection<String> jids = usersSessions.get(route.toBareJID());
if (jids != null) {
jids.remove(route.toString());
if (!jids.isEmpty()) {
usersSessions.put(route.toBareJID(), jids);
}
else {
usersSessions.remove(route.toBareJID());
}
}
}
}
finally {
lock.unlock();
}
}
}
}
} }
public void leftCluster() { public void leftCluster() {
...@@ -795,6 +725,10 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust ...@@ -795,6 +725,10 @@ public class RoutingTableImpl extends BasicModule implements RoutingTable, Clust
} }
} }
public void leftCluster(byte[] nodeID) {
// Do nothing
}
public void markedAsSeniorClusterMember() { public void markedAsSeniorClusterMember() {
// Do nothing // Do nothing
} }
......
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