/*
* JBoss, Home of Professional Open Source
* Copyright 2009 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.notifications;
import org.infinispan.Cache;
import org.infinispan.manager.CacheContainer;
import org.infinispan.notifications.cachelistener.annotation.CacheEntryCreated;
import org.infinispan.notifications.cachelistener.annotation.CacheEntryModified;
import org.infinispan.notifications.cachelistener.annotation.CacheEntryRemoved;
import org.infinispan.notifications.cachelistener.annotation.CacheEntryVisited;
import org.infinispan.notifications.cachelistener.event.Event;
import org.infinispan.test.AbstractInfinispanTest;
import org.infinispan.test.TestingUtil;
import org.infinispan.test.fwk.TestCacheManagerFactory;
import org.infinispan.util.logging.Log;
import org.infinispan.util.logging.LogFactory;
import org.testng.annotations.AfterMethod;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.Test;
import java.util.LinkedList;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicInteger;
@Test(groups = "functional", testName = "notifications.ConcurrentNotificationTest")
public class ConcurrentNotificationTest extends AbstractInfinispanTest {
Cache<String, String> cache;
CacheContainer cm;
CacheListener listener;
Log log = LogFactory.getLog(ConcurrentNotificationTest.class);
@BeforeMethod
public void setUp() {
cm = TestCacheManagerFactory.createLocalCacheManager(false);
cache = cm.getCache();
listener = new CacheListener();
cache.addListener(listener);
}
@AfterMethod
public void tearDown() {
TestingUtil.killCacheManagers(cm);
cm = null;
cache = null;
listener = null;
}
public void testThreads() throws Exception {
Thread workers[] = new Thread[20];
final List<Exception> exceptions = new LinkedList<Exception>();
final int loops = 100;
final CountDownLatch latch = new CountDownLatch(1);
for (int i = 0; i < workers.length; i++) {
workers[i] = new Thread() {
public void run() {
try {
latch.await();
}
catch (InterruptedException e) {
}
for (int j = 0; j < loops; j++) {
try {
cache.put("key", "value");
}
catch (Exception e) {
exceptions.add(new Exception("Caused on thread " + getName() + " in loop " + j + " when doing a put()", e));
}
try {
cache.remove("key");
}
catch (Exception e) {
exceptions.add(new Exception("Caused on thread " + getName() + " in loop " + j + " when doing a remove()", e));
}
try {
cache.get("key");
}
catch (Exception e) {
log.error("Exception received!", e);
exceptions.add(new Exception("Caused on thread " + getName() + " in loop " + j + " when doing a get()", e));
}
}
}
};
workers[i].start();
}
latch.countDown();
for (Thread t : workers)
t.join();
for (Exception e : exceptions)
throw e;
// we cannot ascertain the exact number of invocations on the replListener since some removes would mean that other
// gets would miss. And this would cause no notification to fire for that get. And we cannot be sure of the
// timing between removes and gets, so we just make sure *some* of these have got through, and no exceptions
// were thrown due to concurrent access.
assert loops * workers.length < listener.counter.get();
}
@Listener
static public class CacheListener {
private AtomicInteger counter = new AtomicInteger(0);
@CacheEntryModified
@CacheEntryRemoved
@CacheEntryVisited
@CacheEntryCreated
public void catchEvent(Event e) {
if (e.isPre())
counter.getAndIncrement();
}
}
}