com.couchbase.client.CouchbaseMemcachedConnection Maven / Gradle / Ivy
/**
* Copyright (C) 2009-2013 Couchbase, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALING
* IN THE SOFTWARE.
*/
package com.couchbase.client;
import com.couchbase.client.vbucket.Reconfigurable;
import com.couchbase.client.vbucket.VBucketNodeLocator;
import com.couchbase.client.vbucket.config.Bucket;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.channels.CancelledKeyException;
import java.nio.channels.ClosedSelectorException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.ConcurrentModificationException;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import net.spy.memcached.ConnectionObserver;
import net.spy.memcached.FailureMode;
import net.spy.memcached.MemcachedConnection;
import net.spy.memcached.MemcachedNode;
import net.spy.memcached.OperationFactory;
import net.spy.memcached.ops.Operation;
/**
* Couchbase implementation of CouchbaseConnection.
*
* The behavior of a CouchbaseMemcached connection extends spy's
* MemcachedConnection by handling reconfiguration events. In a Couchbase
* deployment scenario, reconfiguration updates may notify the client of
* nodes to be added to or removed from the cluster.
*
* This class provides that functionality by extending the MemcachedConnection
* and adding a method to handle reconfiguration of a bucket.
*/
public class CouchbaseMemcachedConnection extends MemcachedConnection implements
Reconfigurable {
protected volatile boolean reconfiguring = false;
private final CouchbaseConnectionFactory cf;
public CouchbaseMemcachedConnection(int bufSize, CouchbaseConnectionFactory f,
List a, Collection obs,
FailureMode fm, OperationFactory opfactory) throws IOException {
super(bufSize, f, a, obs, fm, opfactory);
this.cf = f;
}
@Override
public void reconfigure(Bucket bucket) {
if(reconfiguring) {
getLogger().debug("Suppressing attempt to reconfigure again while "
+ "reconfiguring.");
return;
}
reconfiguring = true;
try {
// get a new collection of addresses from the received config
List servers = bucket.getConfig().getServers();
HashSet newServerAddresses = new HashSet();
ArrayList newServers =
new ArrayList();
for (String server : servers) {
int finalColon = server.lastIndexOf(':');
if (finalColon < 1) {
throw new IllegalArgumentException("Invalid server ``" + server
+ "'' in vbucket's server list");
}
String hostPart = server.substring(0, finalColon);
String portNum = server.substring(finalColon + 1);
InetSocketAddress address =
new InetSocketAddress(hostPart, Integer.parseInt(portNum));
// add parsed address to our collections
newServerAddresses.add(address);
newServers.add(address);
}
// split current nodes to "odd nodes" and "stay nodes"
ArrayList oddNodes = new ArrayList();
ArrayList stayNodes = new ArrayList();
ArrayList stayServers =
new ArrayList();
for (MemcachedNode current : locator.getAll()) {
if (newServerAddresses.contains(current.getSocketAddress())) {
stayNodes.add(current);
stayServers.add((InetSocketAddress) current.getSocketAddress());
} else {
oddNodes.add(current);
}
}
// prepare a collection of addresses for new nodes
newServers.removeAll(stayServers);
// create a collection of new nodes
List newNodes = createConnections(newServers);
// merge stay nodes with new nodes
List mergedNodes = new ArrayList();
mergedNodes.addAll(stayNodes);
mergedNodes.addAll(newNodes);
for(MemcachedNode keepingNode : mergedNodes) {
getLogger().debug("Node " + keepingNode.getSocketAddress()
+ " will stay in cluster config after reconfiguration.");
}
// call update locator with new nodes list and vbucket config
if (locator instanceof VBucketNodeLocator) {
((VBucketNodeLocator)locator).updateLocator(mergedNodes,
bucket.getConfig());
} else {
// We update the locator with the merged nodes
// before initiating a reconnect on the queue
locator.updateLocator(mergedNodes);
}
// schedule shutdown for the oddNodes
for(MemcachedNode shutDownNode : oddNodes) {
getLogger().info("Scheduling Node "
+ shutDownNode.getSocketAddress() + "for shutdown.");
}
nodesToShutdown.addAll(oddNodes);
} catch (IOException e) {
getLogger().error("Connection reconfiguration failed", e);
} finally {
reconfiguring = false;
}
}
@Override
protected void addOperation(final String key, final Operation o) {
MemcachedNode placeIn = null;
MemcachedNode primary = locator.getPrimary(key);
if (primary == null) {
o.cancel();
cf.checkConfigUpdate();
return;
}
if(!primary.isActive()) {
cf.checkConfigUpdate();
}
if (primary.isActive() || failureMode == FailureMode.Retry) {
placeIn = primary;
} else if (failureMode == FailureMode.Cancel) {
o.cancel();
} else {
// Look for another node in sequence that is ready.
for (Iterator i = locator.getSequence(key); placeIn == null
&& i.hasNext();) {
MemcachedNode n = i.next();
if (n.isActive()) {
placeIn = n;
}
}
// If we didn't find an active node, queue it in the primary node
// and wait for it to come back online.
if (placeIn == null) {
placeIn = primary;
this.getLogger().warn(
"Could not redistribute "
+ "to another node, retrying primary node for %s.", key);
}
}
assert o.isCancelled() || placeIn != null : "No node found for key " + key;
if (placeIn != null) {
addOperation(placeIn, o);
} else {
assert o.isCancelled() : "No node found for " + key
+ " (and not immediately cancelled)";
}
}
/**
* Infinitely loop processing IO.
*/
@Override
public void run() {
while (running) {
if (!reconfiguring) {
try {
handleIO();
} catch (IOException e) {
logRunException(e);
} catch (CancelledKeyException e) {
logRunException(e);
} catch (ClosedSelectorException e) {
logRunException(e);
} catch (IllegalStateException e) {
logRunException(e);
} catch (ConcurrentModificationException e) {
logRunException(e);
}
}
}
getLogger().info("Shut down Couchbase client");
}
private void logRunException(Exception e) {
if (shutDown) {
// There are a couple types of errors that occur during the
// shutdown sequence that are considered OK. Log at debug.
getLogger().debug("Exception occurred during shutdown", e);
} else {
getLogger().warn("Problem handling Couchbase IO", e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy