mirror of https://github.com/apache/activemq.git
https://issues.apache.org/jira/browse/AMQ-3374 - first stab at fixing long kahadb tx oom problem
git-svn-id: https://svn.apache.org/repos/asf/activemq/trunk@1138442 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
37384a1ad1
commit
40ae055f37
|
@ -19,6 +19,7 @@ package org.apache.activemq.usecases;
|
||||||
import java.lang.management.ManagementFactory;
|
import java.lang.management.ManagementFactory;
|
||||||
|
|
||||||
import javax.jms.Connection;
|
import javax.jms.Connection;
|
||||||
|
import javax.jms.MessageProducer;
|
||||||
import javax.jms.Session;
|
import javax.jms.Session;
|
||||||
import javax.management.MBeanServer;
|
import javax.management.MBeanServer;
|
||||||
import javax.management.ObjectName;
|
import javax.management.ObjectName;
|
||||||
|
@ -43,7 +44,15 @@ public class DurableUnsubscribeTest extends org.apache.activemq.TestSupport {
|
||||||
Destination d = broker.getDestination(topic);
|
Destination d = broker.getDestination(topic);
|
||||||
assertEquals("Subscription is missing.", 1, d.getConsumers().size());
|
assertEquals("Subscription is missing.", 1, d.getConsumers().size());
|
||||||
|
|
||||||
|
|
||||||
session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
|
session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
|
||||||
|
MessageProducer producer = session.createProducer(topic);
|
||||||
|
for (int i = 0; i < 1000; i++) {
|
||||||
|
producer.send(session.createTextMessage("text"));
|
||||||
|
}
|
||||||
|
|
||||||
|
Thread.sleep(1000);
|
||||||
|
|
||||||
session.unsubscribe("SubsId");
|
session.unsubscribe("SubsId");
|
||||||
session.close();
|
session.close();
|
||||||
|
|
||||||
|
@ -92,7 +101,7 @@ public class DurableUnsubscribeTest extends org.apache.activemq.TestSupport {
|
||||||
|
|
||||||
private void createBroker() throws Exception {
|
private void createBroker() throws Exception {
|
||||||
broker = BrokerFactory.createBroker("broker:(vm://localhost)");
|
broker = BrokerFactory.createBroker("broker:(vm://localhost)");
|
||||||
broker.setPersistent(false);
|
//broker.setPersistent(false);
|
||||||
broker.setUseJmx(true);
|
broker.setUseJmx(true);
|
||||||
broker.setBrokerName(getName());
|
broker.setBrokerName(getName());
|
||||||
broker.start();
|
broker.start();
|
||||||
|
|
|
@ -62,6 +62,7 @@ import org.slf4j.LoggerFactory;
|
||||||
* requesting many consumers TODO configure consumers with small prefetch.
|
* requesting many consumers TODO configure consumers with small prefetch.
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
|
*
|
||||||
*/
|
*/
|
||||||
public class WebClient implements HttpSessionActivationListener, HttpSessionBindingListener, Externalizable {
|
public class WebClient implements HttpSessionActivationListener, HttpSessionBindingListener, Externalizable {
|
||||||
|
|
||||||
|
|
|
@ -138,15 +138,35 @@ public class PageFile {
|
||||||
Page page;
|
Page page;
|
||||||
byte[] current;
|
byte[] current;
|
||||||
byte[] diskBound;
|
byte[] diskBound;
|
||||||
|
int currentLocation = -1;
|
||||||
|
int diskBoundLocation = -1;
|
||||||
|
File tmpFile;
|
||||||
|
int length;
|
||||||
|
|
||||||
public PageWrite(Page page, byte[] data) {
|
public PageWrite(Page page, byte[] data) {
|
||||||
this.page=page;
|
this.page=page;
|
||||||
current=data;
|
current=data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PageWrite(Page page, int currentLocation, int length, File tmpFile) {
|
||||||
|
this.page = page;
|
||||||
|
this.currentLocation = currentLocation;
|
||||||
|
this.tmpFile = tmpFile;
|
||||||
|
this.length = length;
|
||||||
|
}
|
||||||
|
|
||||||
public void setCurrent(Page page, byte[] data) {
|
public void setCurrent(Page page, byte[] data) {
|
||||||
this.page=page;
|
this.page=page;
|
||||||
current=data;
|
current=data;
|
||||||
|
currentLocation = -1;
|
||||||
|
diskBoundLocation = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCurrentLocation(Page page, int location, int length) {
|
||||||
|
this.page = page;
|
||||||
|
this.currentLocation = location;
|
||||||
|
this.length = length;
|
||||||
|
this.current = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -159,21 +179,41 @@ public class PageFile {
|
||||||
return page;
|
return page;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public byte[] getDiskBound() throws IOException {
|
||||||
|
if (diskBound == null && diskBoundLocation != -1) {
|
||||||
|
diskBound = new byte[length];
|
||||||
|
RandomAccessFile file = new RandomAccessFile(tmpFile, "r");
|
||||||
|
file.seek(diskBoundLocation);
|
||||||
|
int readNum = file.read(diskBound);
|
||||||
|
file.close();
|
||||||
|
diskBoundLocation = -1;
|
||||||
|
}
|
||||||
|
return diskBound;
|
||||||
|
}
|
||||||
|
|
||||||
void begin() {
|
void begin() {
|
||||||
|
if (currentLocation != -1) {
|
||||||
|
diskBoundLocation = currentLocation;
|
||||||
|
currentLocation = -1;
|
||||||
|
current = null;
|
||||||
|
} else {
|
||||||
diskBound = current;
|
diskBound = current;
|
||||||
current = null;
|
current = null;
|
||||||
|
currentLocation = -1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return true if there is no pending writes to do.
|
* @return true if there is no pending writes to do.
|
||||||
*/
|
*/
|
||||||
boolean done() {
|
boolean done() {
|
||||||
|
diskBoundLocation = -1;
|
||||||
diskBound=null;
|
diskBound=null;
|
||||||
return current == null;
|
return current == null || currentLocation == -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean isDone() {
|
boolean isDone() {
|
||||||
return diskBound == null && current == null;
|
return diskBound == null && diskBoundLocation == -1 && current == null && currentLocation == -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -470,7 +510,7 @@ public class PageFile {
|
||||||
return new File(directory, IOHelper.toFileSystemSafeName(name)+RECOVERY_FILE_SUFFIX);
|
return new File(directory, IOHelper.toFileSystemSafeName(name)+RECOVERY_FILE_SUFFIX);
|
||||||
}
|
}
|
||||||
|
|
||||||
private long toOffset(long pageId) {
|
public long toOffset(long pageId) {
|
||||||
return PAGE_FILE_HEADER_SIZE+(pageId*pageSize);
|
return PAGE_FILE_HEADER_SIZE+(pageId*pageSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -823,19 +863,29 @@ public class PageFile {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
boolean longTx = false;
|
||||||
|
|
||||||
for (Map.Entry<Long, PageWrite> entry : updates) {
|
for (Map.Entry<Long, PageWrite> entry : updates) {
|
||||||
Long key = entry.getKey();
|
Long key = entry.getKey();
|
||||||
PageWrite value = entry.getValue();
|
PageWrite value = entry.getValue();
|
||||||
PageWrite write = writes.get(key);
|
PageWrite write = writes.get(key);
|
||||||
if( write==null ) {
|
if( write==null ) {
|
||||||
writes.put(key, value);
|
writes.put(key, value);
|
||||||
|
} else {
|
||||||
|
if (value.currentLocation != -1) {
|
||||||
|
write.setCurrentLocation(value.page, value.currentLocation, value.length);
|
||||||
|
write.tmpFile = value.tmpFile;
|
||||||
|
longTx = true;
|
||||||
} else {
|
} else {
|
||||||
write.setCurrent(value.page, value.current);
|
write.setCurrent(value.page, value.current);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Once we start approaching capacity, notify the writer to start writing
|
// Once we start approaching capacity, notify the writer to start writing
|
||||||
if( canStartWriteBatch() ) {
|
// sync immediately for long txs
|
||||||
|
if( longTx || canStartWriteBatch() ) {
|
||||||
|
|
||||||
if( enabledWriteThread ) {
|
if( enabledWriteThread ) {
|
||||||
writes.notify();
|
writes.notify();
|
||||||
} else {
|
} else {
|
||||||
|
@ -919,12 +969,6 @@ public class PageFile {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @return true if there are still pending writes to do.
|
|
||||||
* @throws InterruptedException
|
|
||||||
* @throws IOException
|
|
||||||
*/
|
|
||||||
private void writeBatch() throws IOException {
|
private void writeBatch() throws IOException {
|
||||||
|
|
||||||
CountDownLatch checkpointLatch;
|
CountDownLatch checkpointLatch;
|
||||||
|
@ -939,7 +983,7 @@ public class PageFile {
|
||||||
// Move the current write to the diskBound write, this lets folks update the
|
// Move the current write to the diskBound write, this lets folks update the
|
||||||
// page again without blocking for this write.
|
// page again without blocking for this write.
|
||||||
write.begin();
|
write.begin();
|
||||||
if (write.diskBound == null) {
|
if (write.diskBound == null && write.diskBoundLocation == -1) {
|
||||||
batch.remove(write);
|
batch.remove(write);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -950,24 +994,26 @@ public class PageFile {
|
||||||
this.checkpointLatch=null;
|
this.checkpointLatch=null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Checksum checksum = new Adler32();
|
||||||
|
recoveryFile.seek(RECOVERY_FILE_HEADER_SIZE);
|
||||||
|
for (PageWrite w : batch) {
|
||||||
|
if (enableRecoveryFile) {
|
||||||
|
try {
|
||||||
|
checksum.update(w.getDiskBound(), 0, pageSize);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
throw IOExceptionSupport.create("Cannot create recovery file. Reason: " + t, t);
|
||||||
|
}
|
||||||
|
recoveryFile.writeLong(w.page.getPageId());
|
||||||
|
recoveryFile.write(w.getDiskBound(), 0, pageSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
writeFile.seek(toOffset(w.page.getPageId()));
|
||||||
|
writeFile.write(w.getDiskBound(), 0, pageSize);
|
||||||
|
w.done();
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (enableRecoveryFile) {
|
if (enableRecoveryFile) {
|
||||||
|
|
||||||
// Using Adler-32 instead of CRC-32 because it's much faster and
|
|
||||||
// it's
|
|
||||||
// weakness for short messages with few hundred bytes is not a
|
|
||||||
// factor in this case since we know
|
|
||||||
// our write batches are going to much larger.
|
|
||||||
Checksum checksum = new Adler32();
|
|
||||||
for (PageWrite w : batch) {
|
|
||||||
try {
|
|
||||||
checksum.update(w.diskBound, 0, pageSize);
|
|
||||||
} catch (Throwable t) {
|
|
||||||
throw IOExceptionSupport.create(
|
|
||||||
"Cannot create recovery file. Reason: " + t, t);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Can we shrink the recovery buffer??
|
// Can we shrink the recovery buffer??
|
||||||
if (recoveryPageCount > recoveryFileMaxPageCount) {
|
if (recoveryPageCount > recoveryFileMaxPageCount) {
|
||||||
int t = Math.max(recoveryFileMinPageCount, batch.size());
|
int t = Math.max(recoveryFileMinPageCount, batch.size());
|
||||||
|
@ -984,34 +1030,13 @@ public class PageFile {
|
||||||
recoveryFile.writeLong(checksum.getValue());
|
recoveryFile.writeLong(checksum.getValue());
|
||||||
// Write the # of pages that will follow
|
// Write the # of pages that will follow
|
||||||
recoveryFile.writeInt(batch.size());
|
recoveryFile.writeInt(batch.size());
|
||||||
|
|
||||||
// Write the pages.
|
|
||||||
recoveryFile.seek(RECOVERY_FILE_HEADER_SIZE);
|
|
||||||
|
|
||||||
for (PageWrite w : batch) {
|
|
||||||
recoveryFile.writeLong(w.page.getPageId());
|
|
||||||
recoveryFile.write(w.diskBound, 0, pageSize);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (enableDiskSyncs) {
|
if (enableDiskSyncs) {
|
||||||
// Sync to make sure recovery buffer writes land on disk..
|
// Sync to make sure recovery buffer writes land on disk..
|
||||||
recoveryFile.getFD().sync();
|
recoveryFile.getFD().sync();
|
||||||
}
|
|
||||||
|
|
||||||
recoveryPageCount = batch.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
for (PageWrite w : batch) {
|
|
||||||
writeFile.seek(toOffset(w.page.getPageId()));
|
|
||||||
writeFile.write(w.diskBound, 0, pageSize);
|
|
||||||
w.done();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sync again
|
|
||||||
if (enableDiskSyncs) {
|
|
||||||
writeFile.getFD().sync();
|
writeFile.getFD().sync();
|
||||||
}
|
}
|
||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
synchronized (writes) {
|
synchronized (writes) {
|
||||||
for (PageWrite w : batch) {
|
for (PageWrite w : batch) {
|
||||||
|
@ -1135,4 +1160,7 @@ public class PageFile {
|
||||||
return getMainPageFile();
|
return getMainPageFile();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public File getDirectory() {
|
||||||
|
return directory;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,22 +16,11 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.kahadb.page;
|
package org.apache.kahadb.page;
|
||||||
|
|
||||||
import java.io.DataInputStream;
|
import java.io.*;
|
||||||
import java.io.EOFException;
|
import java.util.*;
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.io.OutputStream;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.NoSuchElementException;
|
|
||||||
|
|
||||||
import org.apache.kahadb.page.PageFile.PageWrite;
|
import org.apache.kahadb.page.PageFile.PageWrite;
|
||||||
import org.apache.kahadb.util.ByteSequence;
|
import org.apache.kahadb.util.*;
|
||||||
import org.apache.kahadb.util.DataByteArrayInputStream;
|
|
||||||
import org.apache.kahadb.util.DataByteArrayOutputStream;
|
|
||||||
import org.apache.kahadb.util.Marshaller;
|
|
||||||
import org.apache.kahadb.util.Sequence;
|
|
||||||
import org.apache.kahadb.util.SequenceSet;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The class used to read/update a PageFile object. Using a transaction allows you to
|
* The class used to read/update a PageFile object. Using a transaction allows you to
|
||||||
|
@ -39,6 +28,11 @@ import org.apache.kahadb.util.SequenceSet;
|
||||||
*/
|
*/
|
||||||
public class Transaction implements Iterable<Page> {
|
public class Transaction implements Iterable<Page> {
|
||||||
|
|
||||||
|
|
||||||
|
private RandomAccessFile tmpFile;
|
||||||
|
private File txfFile;
|
||||||
|
private int nextLocation = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The PageOverflowIOException occurs when a page write is requested
|
* The PageOverflowIOException occurs when a page write is requested
|
||||||
* and it's data is larger than what would fit into a single page.
|
* and it's data is larger than what would fit into a single page.
|
||||||
|
@ -91,12 +85,16 @@ public class Transaction implements Iterable<Page> {
|
||||||
// If this transaction is updating stuff.. this is the tx of
|
// If this transaction is updating stuff.. this is the tx of
|
||||||
private long writeTransactionId=-1;
|
private long writeTransactionId=-1;
|
||||||
// List of pages that this transaction has modified.
|
// List of pages that this transaction has modified.
|
||||||
private HashMap<Long, PageWrite> writes=new HashMap<Long, PageWrite>();
|
private TreeMap<Long, PageWrite> writes=new TreeMap<Long, PageWrite>();
|
||||||
// List of pages allocated in this transaction
|
// List of pages allocated in this transaction
|
||||||
private final SequenceSet allocateList = new SequenceSet();
|
private final SequenceSet allocateList = new SequenceSet();
|
||||||
// List of pages freed in this transaction
|
// List of pages freed in this transaction
|
||||||
private final SequenceSet freeList = new SequenceSet();
|
private final SequenceSet freeList = new SequenceSet();
|
||||||
|
|
||||||
|
private long maxTransactionSize = 10485760;
|
||||||
|
|
||||||
|
private long size = 0;
|
||||||
|
|
||||||
Transaction(PageFile pageFile) {
|
Transaction(PageFile pageFile) {
|
||||||
this.pageFile = pageFile;
|
this.pageFile = pageFile;
|
||||||
}
|
}
|
||||||
|
@ -650,7 +648,16 @@ public class Transaction implements Iterable<Page> {
|
||||||
allocateList.clear();
|
allocateList.clear();
|
||||||
writes.clear();
|
writes.clear();
|
||||||
writeTransactionId = -1;
|
writeTransactionId = -1;
|
||||||
|
if (tmpFile != null) {
|
||||||
|
tmpFile.close();
|
||||||
|
if (!getTempFile().delete()) {
|
||||||
|
throw new IOException("Can't delete temporary KahaDB transaction file:" + getTempFile());
|
||||||
}
|
}
|
||||||
|
tmpFile = null;
|
||||||
|
txfFile = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
size = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -665,7 +672,16 @@ public class Transaction implements Iterable<Page> {
|
||||||
allocateList.clear();
|
allocateList.clear();
|
||||||
writes.clear();
|
writes.clear();
|
||||||
writeTransactionId = -1;
|
writeTransactionId = -1;
|
||||||
|
if (tmpFile != null) {
|
||||||
|
tmpFile.close();
|
||||||
|
if (getTempFile().delete()) {
|
||||||
|
throw new IOException("Can't delete temporary KahaDB transaction file:" + getTempFile());
|
||||||
}
|
}
|
||||||
|
tmpFile = null;
|
||||||
|
txfFile = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
size = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private long getWriteTransactionId() {
|
private long getWriteTransactionId() {
|
||||||
|
@ -675,16 +691,36 @@ public class Transaction implements Iterable<Page> {
|
||||||
return writeTransactionId;
|
return writeTransactionId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
protected File getTempFile() {
|
||||||
|
if (txfFile == null) {
|
||||||
|
txfFile = new File(getPageFile().getDirectory(), IOHelper.toFileSystemSafeName(Long.toString(getWriteTransactionId())) + ".tmp");
|
||||||
|
}
|
||||||
|
return txfFile;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Queues up a page write that should get done when commit() gets called.
|
* Queues up a page write that should get done when commit() gets called.
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
private void write(final Page page, byte[] data) throws IOException {
|
private void write(final Page page, byte[] data) throws IOException {
|
||||||
Long key = page.getPageId();
|
Long key = page.getPageId();
|
||||||
// TODO: if a large update transaction is in progress, we may want to move
|
size += data.length;
|
||||||
// all the current updates to a temp file so that we don't keep using
|
|
||||||
// up memory.
|
PageWrite write;
|
||||||
writes.put(key, new PageWrite(page, data));
|
if (size > maxTransactionSize) {
|
||||||
|
if (tmpFile == null) {
|
||||||
|
tmpFile = new RandomAccessFile(getTempFile(), "rw");
|
||||||
|
}
|
||||||
|
int location = nextLocation;
|
||||||
|
tmpFile.seek(nextLocation);
|
||||||
|
tmpFile.write(data);
|
||||||
|
nextLocation = location + data.length;
|
||||||
|
write = new PageWrite(page, location, data.length, getTempFile());
|
||||||
|
} else {
|
||||||
|
write = new PageWrite(page, data);
|
||||||
|
}
|
||||||
|
writes.put(key, write);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
Loading…
Reference in New Issue