diff --git a/src/main/java/net/spy/memcached/ConnectionFactoryBuilder.java b/src/main/java/net/spy/memcached/ConnectionFactoryBuilder.java index b3ece2eec..18a303ca1 100644 --- a/src/main/java/net/spy/memcached/ConnectionFactoryBuilder.java +++ b/src/main/java/net/spy/memcached/ConnectionFactoryBuilder.java @@ -357,6 +357,15 @@ public NodeLocator createLocator(List nodes) { return new ArrayModNodeLocator(nodes, getHashAlg()); case CONSISTENT: return new KetamaNodeLocator(nodes, getHashAlg()); + case ROUND_ROBIN: + if (getFailureMode() != FailureMode.Cancel + && getFailureMode() != FailureMode.Retry) + { + throw new IllegalStateException( + "The round-robin locator is only supported for the 'cancel' " + + "and 'retry' failure modes."); + } + return new RoundRobinLocator(nodes); default: throw new IllegalStateException("Unhandled locator type: " + locator); } @@ -493,6 +502,10 @@ public static enum Locator { * algorithm. */ CONSISTENT, + /** + * Round robin algorithm. + */ + ROUND_ROBIN, /** * VBucket support. */ diff --git a/src/main/java/net/spy/memcached/RoundRobinLocator.java b/src/main/java/net/spy/memcached/RoundRobinLocator.java new file mode 100644 index 000000000..e42deaf4b --- /dev/null +++ b/src/main/java/net/spy/memcached/RoundRobinLocator.java @@ -0,0 +1,112 @@ +package net.spy.memcached; + +import java.util.Arrays; +import java.util.Collection; +import java.util.Iterator; +import java.util.List; + +import net.spy.memcached.MemcachedNode; +import net.spy.memcached.MemcachedNodeROImpl; +import net.spy.memcached.NodeLocator; + + +/** + * NodeLocator implementation that round-robins across active nodes. + */ +public final class RoundRobinLocator implements NodeLocator { + + private int nodeIndex; + private MemcachedNode[] nodes; + + public RoundRobinLocator(List n) { + super(); + nodes = n.toArray(new MemcachedNode[n.size()]); + } + + private RoundRobinLocator(MemcachedNode[] n) { + super(); + nodes = n; + } + + /** + * @param k Input key (which is ignored in round robin) + * @return Next active node. If none of the nodes are active, the next node in + * the list is returned. Never returns null. + */ + @Override + public synchronized MemcachedNode getPrimary(String k) { + int i; + for (i = nodeIndex; !nodes[i % nodes.length].isActive() + && i < nodeIndex + nodes.length; i++) {} + + nodeIndex = (i + 1) % nodes.length; + return nodes[i % nodes.length]; + } + + @Override + public Iterator getSequence(String k) { + return new NodeIterator(nodeIndex); + } + + @Override + public Collection getAll() { + return Arrays.asList(nodes); + } + + @Override + public NodeLocator getReadonlyCopy() { + MemcachedNode[] n = new MemcachedNode[nodes.length]; + for (int i = 0; i < nodes.length; i++) { + n[i] = new MemcachedNodeROImpl(nodes[i]); + } + return new RoundRobinLocator(n); + } + + @Override + public void updateLocator(List newNodes) { + this.nodes = newNodes.toArray(new MemcachedNode[newNodes.size()]); + } + + class NodeIterator implements Iterator { + + private final int start; + private int next = 0; + + public NodeIterator(int keyStart) { + start = keyStart; + next = start; + computeNext(); + assert next >= 0 || nodes.length == 1 : "Starting sequence at " + + start + " of " + nodes.length + " next is " + next; + } + + @Override + public boolean hasNext() { + return next >= 0; + } + + private void computeNext() { + if (++next >= nodes.length) { + next = 0; + } + if (next == start) { + next = -1; + } + } + + @Override + public MemcachedNode next() { + try { + return nodes[next]; + } finally { + computeNext(); + } + } + + @Override + public void remove() { + throw new UnsupportedOperationException("Can't remove a node"); + } + } + +} \ No newline at end of file diff --git a/src/test/java/net/spy/memcached/AbstractNodeLocationCase.java b/src/test/java/net/spy/memcached/AbstractNodeLocationCase.java index f973da489..9ab177d7d 100644 --- a/src/test/java/net/spy/memcached/AbstractNodeLocationCase.java +++ b/src/test/java/net/spy/memcached/AbstractNodeLocationCase.java @@ -57,7 +57,7 @@ private void runSequenceAssertion(NodeLocator l, String k, int... seq) { assertEquals("Incorrect sequence size for " + k, seq.length, pos); } - public final void testCloningGetPrimary() { + public void testCloningGetPrimary() { setupNodes(5); assertTrue(locator.getReadonlyCopy().getPrimary("hi") instanceof MemcachedNodeROImpl); diff --git a/src/test/java/net/spy/memcached/RoundRobinLocatorTest.java b/src/test/java/net/spy/memcached/RoundRobinLocatorTest.java new file mode 100644 index 000000000..ab0ce8311 --- /dev/null +++ b/src/test/java/net/spy/memcached/RoundRobinLocatorTest.java @@ -0,0 +1,121 @@ +package net.spy.memcached; + +import java.util.Arrays; +import java.util.Collection; + + +/** + * Test the RoundRobinLocatorTest. + */ +public class RoundRobinLocatorTest extends AbstractNodeLocationCase { + + private void setActive(boolean value, int ... nodes) { + for (int n : nodes) { + nodeMocks[n].expects(atLeastOnce()).method("isActive") + .will(returnValue(value)); + } + } + + @Override + protected void setupNodes(int n) { + super.setupNodes(n); + locator = new RoundRobinLocator(Arrays.asList(nodes)); + } + + public void testPrimarySingleNodeActive() throws Exception { + setupNodes(1); + setActive(true, 0); + assertSame(nodes[0], locator.getPrimary("a")); + assertSame(nodes[0], locator.getPrimary("b")); + assertSame(nodes[0], locator.getPrimary("c")); + } + + public void testPrimarySingleNodeDown() throws Exception { + setupNodes(1); + setActive(false, 0); + assertSame(nodes[0], locator.getPrimary("a")); + assertSame(nodes[0], locator.getPrimary("b")); + assertSame(nodes[0], locator.getPrimary("c")); + } + + public void testPrimaryMultiNodeOneDown() throws Exception { + setupNodes(2); + setActive(false, 0); + setActive(true, 1); + assertSame(nodes[1], locator.getPrimary("a")); + assertSame(nodes[1], locator.getPrimary("b")); + assertSame(nodes[1], locator.getPrimary("c")); + } + + public void testPrimaryMultiMixedDown() throws Exception { + setupNodes(4); + setActive(false, 0, 2); + setActive(true, 1, 3); + assertSame(nodes[1], locator.getPrimary("a")); + assertSame(nodes[3], locator.getPrimary("b")); + assertSame(nodes[1], locator.getPrimary("c")); + assertSame(nodes[3], locator.getPrimary("a")); + } + + public void testPrimaryMultiNodeAllActive() throws Exception { + setupNodes(4); + setActive(true, 0, 1, 2, 3); + assertSame(nodes[0], locator.getPrimary("a")); + assertSame(nodes[1], locator.getPrimary("b")); + assertSame(nodes[2], locator.getPrimary("c")); + assertSame(nodes[3], locator.getPrimary("d")); + assertSame(nodes[0], locator.getPrimary("e")); + assertSame(nodes[1], locator.getPrimary("f")); + assertSame(nodes[2], locator.getPrimary("g")); + assertSame(nodes[3], locator.getPrimary("h")); + assertSame(nodes[0], locator.getPrimary("i")); + } + + public void testPrimaryMultiNodeAllDown() throws Exception { + setupNodes(4); + setActive(false, 0, 1, 2, 3); + assertSame(nodes[0], locator.getPrimary("a")); + assertSame(nodes[1], locator.getPrimary("b")); + assertSame(nodes[2], locator.getPrimary("c")); + assertSame(nodes[3], locator.getPrimary("d")); + assertSame(nodes[0], locator.getPrimary("e")); + assertSame(nodes[1], locator.getPrimary("f")); + assertSame(nodes[2], locator.getPrimary("g")); + assertSame(nodes[3], locator.getPrimary("h")); + assertSame(nodes[0], locator.getPrimary("i")); + } + + public void testPrimaryClone() throws Exception { + setupNodes(2); + setActive(true, 0); + assertEquals(nodes[0].toString(), + locator.getReadonlyCopy().getPrimary("a").toString()); + assertEquals(nodes[0].toString(), + locator.getReadonlyCopy().getPrimary("b").toString()); + } + + public void testAll() throws Exception { + setupNodes(4); + Collection all = locator.getAll(); + assertEquals(4, all.size()); + assertTrue(all.contains(nodes[0])); + assertTrue(all.contains(nodes[1])); + assertTrue(all.contains(nodes[2])); + assertTrue(all.contains(nodes[3])); + } + + public void testAllClone() throws Exception { + setupNodes(4); + Collection all = locator.getReadonlyCopy().getAll(); + assertEquals(4, all.size()); + } + + @Override + public final void testCloningGetPrimary() { + setupNodes(5); + setActive(true, 0); + assertTrue(locator.getReadonlyCopy().getPrimary("hi") + instanceof MemcachedNodeROImpl); + } + +}