mirror of https://github.com/apache/activemq.git
fix and test for: https://issues.apache.org/jira/browse/AMQ-3982
git-svn-id: https://svn.apache.org/repos/asf/activemq/trunk@1375140 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
2027c87791
commit
030d09b816
|
@ -25,6 +25,7 @@ import java.util.NoSuchElementException;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
import org.apache.kahadb.index.ListIndex;
|
import org.apache.kahadb.index.ListIndex;
|
||||||
import org.apache.kahadb.journal.Location;
|
import org.apache.kahadb.journal.Location;
|
||||||
import org.apache.kahadb.page.Transaction;
|
import org.apache.kahadb.page.Transaction;
|
||||||
|
@ -201,7 +202,9 @@ public class PList extends ListIndex<String, Location> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean hasNext() {
|
public boolean hasNext() {
|
||||||
return iterator.hasNext();
|
synchronized (indexLock) {
|
||||||
|
return iterator.hasNext();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -47,7 +47,7 @@ public class PListTest {
|
||||||
final String idSeed = new String("Seed" + new byte[1024]);
|
final String idSeed = new String("Seed" + new byte[1024]);
|
||||||
final Vector<Throwable> exceptions = new Vector<Throwable>();
|
final Vector<Throwable> exceptions = new Vector<Throwable>();
|
||||||
ExecutorService executor;
|
ExecutorService executor;
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testAddLast() throws Exception {
|
public void testAddLast() throws Exception {
|
||||||
|
@ -97,8 +97,8 @@ public class PListTest {
|
||||||
public void testRemove() throws IOException {
|
public void testRemove() throws IOException {
|
||||||
doTestRemove(2000);
|
doTestRemove(2000);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void doTestRemove(final int COUNT) throws IOException {
|
protected void doTestRemove(final int COUNT) throws IOException {
|
||||||
Map<String, ByteSequence> map = new LinkedHashMap<String, ByteSequence>();
|
Map<String, ByteSequence> map = new LinkedHashMap<String, ByteSequence>();
|
||||||
for (int i = 0; i < COUNT; i++) {
|
for (int i = 0; i < COUNT; i++) {
|
||||||
String test = new String("test" + i);
|
String test = new String("test" + i);
|
||||||
|
@ -122,7 +122,7 @@ public class PListTest {
|
||||||
plist.destroy();
|
plist.destroy();
|
||||||
assertEquals(0,plist.size());
|
assertEquals(0,plist.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testDestroyNonEmpty() throws Exception {
|
public void testDestroyNonEmpty() throws Exception {
|
||||||
final int COUNT = 1000;
|
final int COUNT = 1000;
|
||||||
|
@ -136,17 +136,17 @@ public class PListTest {
|
||||||
plist.destroy();
|
plist.destroy();
|
||||||
assertEquals(0,plist.size());
|
assertEquals(0,plist.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testRemoveSecond() throws Exception {
|
public void testRemoveSecond() throws Exception {
|
||||||
plist.addLast("First", new ByteSequence("A".getBytes()));
|
plist.addLast("First", new ByteSequence("A".getBytes()));
|
||||||
plist.addLast("Second", new ByteSequence("B".getBytes()));
|
plist.addLast("Second", new ByteSequence("B".getBytes()));
|
||||||
|
|
||||||
assertTrue(plist.remove("Second"));
|
assertTrue(plist.remove("Second"));
|
||||||
assertTrue(plist.remove("First"));
|
assertTrue(plist.remove("First"));
|
||||||
assertFalse(plist.remove("doesNotExist"));
|
assertFalse(plist.remove("doesNotExist"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testRemoveSingleEntry() throws Exception {
|
public void testRemoveSingleEntry() throws Exception {
|
||||||
|
@ -154,7 +154,7 @@ public class PListTest {
|
||||||
|
|
||||||
Iterator<PListEntry> iterator = plist.iterator();
|
Iterator<PListEntry> iterator = plist.iterator();
|
||||||
while (iterator.hasNext()) {
|
while (iterator.hasNext()) {
|
||||||
PListEntry v = iterator.next();
|
iterator.next();
|
||||||
iterator.remove();
|
iterator.remove();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -163,7 +163,7 @@ public class PListTest {
|
||||||
public void testRemoveSecondPosition() throws Exception {
|
public void testRemoveSecondPosition() throws Exception {
|
||||||
plist.addLast("First", new ByteSequence("A".getBytes()));
|
plist.addLast("First", new ByteSequence("A".getBytes()));
|
||||||
plist.addLast("Second", new ByteSequence("B".getBytes()));
|
plist.addLast("Second", new ByteSequence("B".getBytes()));
|
||||||
|
|
||||||
assertTrue(plist.remove(1));
|
assertTrue(plist.remove(1));
|
||||||
assertTrue(plist.remove(0));
|
assertTrue(plist.remove(0));
|
||||||
assertFalse(plist.remove(0));
|
assertFalse(plist.remove(0));
|
||||||
|
@ -209,7 +209,7 @@ public class PListTest {
|
||||||
Thread.currentThread().setName("ALRF:"+candidate.getName());
|
Thread.currentThread().setName("ALRF:"+candidate.getName());
|
||||||
synchronized (plistLocks(candidate)) {
|
synchronized (plistLocks(candidate)) {
|
||||||
candidate.addLast(String.valueOf(i), payload);
|
candidate.addLast(String.valueOf(i), payload);
|
||||||
PListEntry entry = candidate.getFirst();
|
candidate.getFirst();
|
||||||
assertTrue(candidate.remove(String.valueOf(i)));
|
assertTrue(candidate.remove(String.valueOf(i)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -233,7 +233,7 @@ public class PListTest {
|
||||||
Thread.currentThread().setName("ALRF:"+candidate.getName());
|
Thread.currentThread().setName("ALRF:"+candidate.getName());
|
||||||
synchronized (plistLocks(candidate)) {
|
synchronized (plistLocks(candidate)) {
|
||||||
candidate.addLast(String.valueOf(i), payload);
|
candidate.addLast(String.valueOf(i), payload);
|
||||||
PListEntry entry = candidate.getFirst();
|
candidate.getFirst();
|
||||||
assertTrue(candidate.remove(String.valueOf(i)));
|
assertTrue(candidate.remove(String.valueOf(i)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -321,6 +321,7 @@ public class PListTest {
|
||||||
store.setDirectory(directory);
|
store.setDirectory(directory);
|
||||||
store.setJournalMaxFileLength(1024*5);
|
store.setJournalMaxFileLength(1024*5);
|
||||||
store.setCleanupInterval(5000);
|
store.setCleanupInterval(5000);
|
||||||
|
store.setIndexWriteBatchSize(500);
|
||||||
store.start();
|
store.start();
|
||||||
|
|
||||||
final int iterations = 500;
|
final int iterations = 500;
|
||||||
|
@ -441,6 +442,87 @@ public class PListTest {
|
||||||
assertTrue("test did not timeout ", shutdown);
|
assertTrue("test did not timeout ", shutdown);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSerialAddIterate() throws Exception {
|
||||||
|
File directory = store.getDirectory();
|
||||||
|
store.stop();
|
||||||
|
IOHelper.mkdirs(directory);
|
||||||
|
IOHelper.deleteChildren(directory);
|
||||||
|
store = new PListStore();
|
||||||
|
store.setIndexPageSize(512);
|
||||||
|
store.setJournalMaxFileLength(100*1024);
|
||||||
|
store.setDirectory(directory);
|
||||||
|
store.setCleanupInterval(-1);
|
||||||
|
store.setIndexEnablePageCaching(false);
|
||||||
|
store.setIndexWriteBatchSize(2000);
|
||||||
|
store.setEnableIndexWriteAsync(false);
|
||||||
|
store.start();
|
||||||
|
|
||||||
|
final int iterations = 1000;
|
||||||
|
final int numLists = 1;
|
||||||
|
|
||||||
|
LOG.info("create");
|
||||||
|
for (int i=0; i<numLists;i++) {
|
||||||
|
new Job(i, PListTest.TaskType.CREATE, iterations).run();
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG.info("serial add and iterate");
|
||||||
|
for (int i=0; i<iterations; i++) {
|
||||||
|
new Job(0, TaskType.ADD, i).run();
|
||||||
|
new Job(0, TaskType.ITERATE, 0).run();
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue("no exceptions: " + exceptions, exceptions.isEmpty());
|
||||||
|
LOG.info("Num dataFiles:" + store.getJournal().getFiles().size());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testConcurrentAddIterate() throws Exception {
|
||||||
|
File directory = store.getDirectory();
|
||||||
|
store.stop();
|
||||||
|
IOHelper.mkdirs(directory);
|
||||||
|
IOHelper.deleteChildren(directory);
|
||||||
|
store = new PListStore();
|
||||||
|
store.setIndexPageSize(2*1024);
|
||||||
|
store.setJournalMaxFileLength(1024*1024);
|
||||||
|
store.setDirectory(directory);
|
||||||
|
store.setCleanupInterval(-1);
|
||||||
|
store.setIndexEnablePageCaching(false);
|
||||||
|
store.setIndexWriteBatchSize(100);
|
||||||
|
store.start();
|
||||||
|
|
||||||
|
final int iterations = 250;
|
||||||
|
final int numLists = 10;
|
||||||
|
|
||||||
|
LOG.info("create");
|
||||||
|
for (int i=0; i<numLists;i++) {
|
||||||
|
new Job(i, PListTest.TaskType.CREATE, iterations).run();
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG.info("parallel add and iterate");
|
||||||
|
// We want a lot of adds occurring so that new free pages get created along
|
||||||
|
// with overlapping seeks from the iterators so that we are likely to seek into
|
||||||
|
// some bad area in the page file.
|
||||||
|
executor = Executors.newFixedThreadPool(400);
|
||||||
|
final int numProducer = 300;
|
||||||
|
final int numConsumer = 100;
|
||||||
|
for (int i=0; i<numLists; i++) {
|
||||||
|
for (int j=0; j<numProducer; j++) {
|
||||||
|
executor.execute(new Job(i, PListTest.TaskType.ADD, iterations));
|
||||||
|
}
|
||||||
|
for (int k=0;k<numConsumer; k++) {
|
||||||
|
executor.execute(new Job(i, TaskType.ITERATE, iterations*2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
executor.shutdown();
|
||||||
|
LOG.info("wait for parallel work to complete");
|
||||||
|
boolean shutdown = executor.awaitTermination(60*60, TimeUnit.SECONDS);
|
||||||
|
assertTrue("no exceptions: " + exceptions, exceptions.isEmpty());
|
||||||
|
assertTrue("test did not timeout ", shutdown);
|
||||||
|
LOG.info("Num dataFiles:" + store.getJournal().getFiles().size());
|
||||||
|
}
|
||||||
|
|
||||||
enum TaskType {CREATE, DELETE, ADD, REMOVE, ITERATE, ITERATE_REMOVE}
|
enum TaskType {CREATE, DELETE, ADD, REMOVE, ITERATE, ITERATE_REMOVE}
|
||||||
|
|
||||||
class Job implements Runnable {
|
class Job implements Runnable {
|
||||||
|
@ -476,10 +558,17 @@ public class PListTest {
|
||||||
|
|
||||||
for (int j = 0; j < iterations; j++) {
|
for (int j = 0; j < iterations; j++) {
|
||||||
synchronized (plistLocks(plist)) {
|
synchronized (plistLocks(plist)) {
|
||||||
plist.addLast ("PL>" + id + idSeed + "-" + j, payload);
|
if (exceptions.isEmpty()) {
|
||||||
|
plist.addLast ("PL>" + id + idSeed + "-" + j, payload);
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
LOG.info("Job-" + id + ", Add, done: " + iterations);
|
|
||||||
|
if (exceptions.isEmpty()) {
|
||||||
|
LOG.info("Job-" + id + ", Add, done: " + iterations);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case REMOVE:
|
case REMOVE:
|
||||||
Thread.currentThread().setName("R:"+id);
|
Thread.currentThread().setName("R:"+id);
|
||||||
|
@ -497,12 +586,20 @@ public class PListTest {
|
||||||
case ITERATE:
|
case ITERATE:
|
||||||
Thread.currentThread().setName("I:"+id);
|
Thread.currentThread().setName("I:"+id);
|
||||||
plist = store.getPList(String.valueOf(id));
|
plist = store.getPList(String.valueOf(id));
|
||||||
|
int iterateCount = 0;
|
||||||
synchronized (plistLocks(plist)) {
|
synchronized (plistLocks(plist)) {
|
||||||
Iterator<PListEntry> iterator = plist.iterator();
|
if (exceptions.isEmpty()) {
|
||||||
PListEntry element = null;
|
Iterator<PListEntry> iterator = plist.iterator();
|
||||||
while (iterator.hasNext()) {
|
while (iterator.hasNext() && exceptions.isEmpty()) {
|
||||||
element = iterator.next();
|
iterator.next();
|
||||||
|
iterateCount++;
|
||||||
|
}
|
||||||
|
|
||||||
|
//LOG.info("Job-" + id + " Done iterate: it=" + iterator + ", count:" + iterateCount + ", size:" + plist.size());
|
||||||
|
if (plist.size() != iterateCount) {
|
||||||
|
System.err.println("Count Wrong: " + iterator);
|
||||||
|
}
|
||||||
|
assertEquals("iterate got all " + id + " iterator:" + iterator , plist.size(), iterateCount);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -515,10 +612,9 @@ public class PListTest {
|
||||||
synchronized (plistLocks(plist)) {
|
synchronized (plistLocks(plist)) {
|
||||||
|
|
||||||
Iterator<PListEntry> removeIterator = plist.iterator();
|
Iterator<PListEntry> removeIterator = plist.iterator();
|
||||||
PListEntry v = null;
|
|
||||||
|
|
||||||
while (removeIterator.hasNext()) {
|
while (removeIterator.hasNext()) {
|
||||||
v = removeIterator.next();
|
removeIterator.next();
|
||||||
removeIterator.remove();
|
removeIterator.remove();
|
||||||
if (removeCount++ > iterations) {
|
if (removeCount++ > iterations) {
|
||||||
break;
|
break;
|
||||||
|
@ -532,9 +628,12 @@ public class PListTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
LOG.warn("Job["+id+"] caught exception: " + e.getMessage());
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
exceptions.add(e);
|
exceptions.add(e);
|
||||||
executor.shutdownNow();
|
if (executor != null) {
|
||||||
|
executor.shutdownNow();
|
||||||
|
}
|
||||||
} finally {
|
} finally {
|
||||||
Thread.currentThread().setName(threadName);
|
Thread.currentThread().setName(threadName);
|
||||||
}
|
}
|
||||||
|
@ -557,7 +656,7 @@ public class PListTest {
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
public void setUp() throws Exception {
|
public void setUp() throws Exception {
|
||||||
File directory = new File("target/test/PlistDB");
|
File directory = new File("/tmp/target/test/PlistDB");
|
||||||
IOHelper.mkdirs(directory);
|
IOHelper.mkdirs(directory);
|
||||||
IOHelper.deleteChildren(directory);
|
IOHelper.deleteChildren(directory);
|
||||||
startStore(directory);
|
startStore(directory);
|
||||||
|
|
Loading…
Reference in New Issue