/* * JBoss, Home of Professional Open Source * Copyright 2010 Red Hat Inc. and/or its affiliates and other * contributors as indicated by the @author tags. All rights reserved. * See the copyright.txt in the distribution for a full listing of * individual contributors. * * This is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This software is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this software; if not, write to the Free * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA, or see the FSF site: http://www.fsf.org. */ package org.infinispan.statetransfer; import org.infinispan.Cache; import org.infinispan.config.Configuration; import org.infinispan.config.GlobalConfiguration; import org.infinispan.manager.EmbeddedCacheManager; import org.infinispan.test.MultipleCacheManagersTest; import org.infinispan.test.TestingUtil; import org.infinispan.util.logging.Log; import org.infinispan.util.logging.LogFactory; import org.testng.annotations.Test; import javax.transaction.TransactionManager; import java.io.Externalizable; import java.io.IOException; import java.io.ObjectInput; import java.io.ObjectOutput; import java.lang.reflect.Method; import java.util.Properties; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicBoolean; /** * State transfer and replication queue test verifying that the interaction between them two works in fine. * * @author Galder Zamarre�o * @since 4.1 */ @Test(groups = "functional", testName = "statetransfer.StateTransferReplicationQueueTest") public class StateTransferReplicationQueueTest extends MultipleCacheManagersTest { public static final String A_B_NAME = "a_b_name"; public static final String A_C_NAME = "a_c_name"; public static final String A_B_AGE = "a_b_age"; public static final String A_C_AGE = "a_c_age"; public static final String JOE = "JOE"; public static final String BOB = "BOB"; public static final Integer TWENTY = 20; public static final Integer FORTY = 40; private final String cacheName = "nbst-replqueue"; Configuration config; protected void createCacheManagers() throws Throwable { // This impl only really sets up a configuration for use later. config = getDefaultClusteredConfig(Configuration.CacheMode.REPL_ASYNC, true); config.setUseReplQueue(true); config.setReplQueueInterval(100, TimeUnit.MILLISECONDS); config.setReplQueueMaxElements(100); config.setUseAsyncMarshalling(false); config.setFetchInMemoryState(true); config.setUseLockStriping(false); // reduces the odd chance of a key collision and deadlock } protected EmbeddedCacheManager createCacheManager() { EmbeddedCacheManager cm = addClusterEnabledCacheManager(); GlobalConfiguration gc = cm.getGlobalConfiguration(); Properties p = new Properties(); p.setProperty("maxThreads", "25"); gc.setAsyncTransportExecutorProperties(p); cm.defineConfiguration(cacheName, config.clone()); return cm; } protected void writeInitialData(final Cache<Object, Object> c) { c.put(A_B_NAME, JOE); c.put(A_B_AGE, TWENTY); c.put(A_C_NAME, BOB); c.put(A_C_AGE, FORTY); } protected void verifyInitialData(Cache<Object, Object> c) { assert JOE.equals(c.get(A_B_NAME)) : "Incorrect value for key " + A_B_NAME; assert TWENTY.equals(c.get(A_B_AGE)) : "Incorrect value for key " + A_B_AGE; assert BOB.equals(c.get(A_C_NAME)) : "Incorrect value for key " + A_C_NAME; assert FORTY.equals(c.get(A_C_AGE)) : "Incorrect value for key " + A_C_AGE; } /** * In particular, this test focuses on checking that ordering is maintained when multiple operations are executed * on the same key in a asynchronous environment with async marshalling turned off. */ public void testStateTransferWithNodeRestartedAndBusy(Method m) throws Exception { log.info(m.getName() + " start"); thirdWritingCacheTest(false); log.info(m.getName() + "end"); } private void thirdWritingCacheTest(boolean tx) throws InterruptedException { Cache<Object, Object> cache1, cache3; cache1 = createCacheManager().getCache(cacheName); EmbeddedCacheManager manager3 = createCacheManager(); cache3 = manager3.getCache(cacheName); writeInitialData(cache1); WritingThread writerThread = new WritingThread(cache1, tx); writerThread.start(); manager3.stop(); // Pause for view to update TestingUtil.blockUntilViewsReceived(60000, false, cache1); cache3 = createCacheManager().getCache(cacheName); // Pause to give caches time to see each other TestingUtil.blockUntilViewsReceived(60000, cache1, cache3); writerThread.stopThread(); writerThread.join(60000); verifyInitialData(cache3); int count = writerThread.result(); // Since this is async, sleep a bit to allow any ongoing repls to go through TestingUtil.sleepThread(5000); for (int c = 0; c < count; c++) { Object o = cache3.get("test" + c); // Nothing should be left after a put/remove on a key assert o == null; } } private static class WritingThread extends Thread { private final Cache<Object, Object> cache; private final boolean tx; private volatile boolean stop; private volatile int result; private TransactionManager tm; WritingThread(Cache<Object, Object> cache, boolean tx) { super("WriterThread"); this.cache = cache; this.tx = tx; if (tx) tm = TestingUtil.getTransactionManager(cache); setDaemon(true); } public int result() { return result; } public void run() { int c = 0; while (!stop) { try { if (tx) tm.begin(); cache.put("test" + c, new PojoValue(c)); cache.remove("test" + c); c++; if (tx) tm.commit(); if (c % 1000 == 0) TestingUtil.sleepThread(1); // Slow it down a bit } catch (Exception e) { stopThread(); } } result = c; } public void stopThread() { stop = true; } } public static class PojoValue implements Externalizable { Log log = LogFactory.getLog(PojoValue.class); static AtomicBoolean holdUp = new AtomicBoolean(); volatile int value; public PojoValue() { } public PojoValue(int value) { this.value = value; } @Override public void writeExternal(ObjectOutput out) throws IOException { String threadName = Thread.currentThread().getName(); if (!holdUp.get() && threadName.contains("STREAMING_STATE_TRANSFER-sender")) { log.debug("In streaming..."); holdUp.compareAndSet(false, true); log.debug("Holding up..."); TestingUtil.sleepThread(2000); // Sleep for 2 seconds to hold up state transfer } out.writeInt(value); } @Override public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { value = in.readInt(); } @Override public int hashCode() { return value + 31; } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; PojoValue pojo = (PojoValue) o; if (value != pojo.value) return false; return true; } } }