package org.infinispan.distribution.rehash;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import org.infinispan.Cache;
import org.infinispan.distribution.BaseDistFunctionalTest;
import org.infinispan.distribution.MagicKey;
import org.infinispan.distribution.ch.ConsistentHash;
import org.infinispan.distribution.ch.KeyPartitioner;
import org.infinispan.distribution.ch.impl.DefaultConsistentHash;
import org.infinispan.distribution.ch.impl.DefaultConsistentHashFactory;
import org.infinispan.manager.EmbeddedCacheManager;
import org.infinispan.remoting.transport.Address;
import org.infinispan.test.TestingUtil;
import org.testng.annotations.Test;
/**
* Tests performing some work on the joiner during a JOIN
*
* @author Manik Surtani
* @since 4.0
*/
// TODO This test makes no sense anymore, now that a joiner blocks until the join completes, before returning from cache.start(). This test needs to be re-thought and redesigned to test the eventual consistency (UnsureResponse) of a remote GET properly.
@Test(groups = "unstable", testName = "distribution.rehash.WorkDuringJoinTest", description = "original group: functional")
public class WorkDuringJoinTest extends BaseDistFunctionalTest<Object, String> {
EmbeddedCacheManager joinerManager;
Cache<Object, String> joiner;
public WorkDuringJoinTest() {
INIT_CLUSTER_SIZE = 2;
performRehashing = true;
}
private List<MagicKey> init() {
List<MagicKey> keys = new ArrayList<>(Arrays.asList(
new MagicKey("k1", c1), new MagicKey("k2", c2),
new MagicKey("k3", c1), new MagicKey("k4", c2)
));
int i = 0;
for (Cache<Object, String> c : caches) c.put(keys.get(i++), "v" + i);
log.infof("Initialized with keys %s", keys);
return keys;
}
Address startNewMember() {
joinerManager = addClusterEnabledCacheManager();
joinerManager.defineConfiguration(cacheName, configuration.build());
joiner = joinerManager.getCache(cacheName);
return manager(joiner).getAddress();
}
public void testJoinAndGet() {
List<MagicKey> keys = init();
KeyPartitioner keyPartitioner = TestingUtil.extractComponent(c1, KeyPartitioner.class);
ConsistentHash chOld = getCacheTopology(c1).getWriteConsistentHash();
Address joinerAddress = startNewMember();
List<Address> newMembers = new ArrayList<>(chOld.getMembers());
newMembers.add(joinerAddress);
DefaultConsistentHashFactory chf = new DefaultConsistentHashFactory();
ConsistentHash chNew = chf.rebalance(chf.updateMembers((DefaultConsistentHash) chOld, newMembers, null));
// which key should me mapped to the joiner?
MagicKey keyToTest = null;
for (MagicKey k: keys) {
int segment = keyPartitioner.getSegment(k);
if (chNew.isSegmentLocalToNode(joinerAddress, segment)) {
keyToTest = k;
break;
}
}
if (keyToTest == null) throw new NullPointerException("Couldn't find a key mapped to J!");
assert joiner.get(keyToTest) != null;
}
}