This closes #3568
This commit is contained in:
commit
969901a5ff
|
@ -19,7 +19,6 @@ package org.apache.activemq.artemis.protocol.amqp.broker;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
|
|
||||||
import org.apache.activemq.artemis.Closeable;
|
|
||||||
import org.apache.activemq.artemis.api.config.ActiveMQDefaultConfiguration;
|
import org.apache.activemq.artemis.api.config.ActiveMQDefaultConfiguration;
|
||||||
import org.apache.activemq.artemis.api.core.ActiveMQAddressExistsException;
|
import org.apache.activemq.artemis.api.core.ActiveMQAddressExistsException;
|
||||||
import org.apache.activemq.artemis.api.core.ActiveMQException;
|
import org.apache.activemq.artemis.api.core.ActiveMQException;
|
||||||
|
@ -160,9 +159,6 @@ public class AMQPSessionCallback implements SessionCallback {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void addCloseable(Closeable closeable) {
|
|
||||||
serverSession.addCloseable(closeable);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void withinContext(Runnable run) throws Exception {
|
public void withinContext(Runnable run) throws Exception {
|
||||||
OperationContext context = recoverContext();
|
OperationContext context = recoverContext();
|
||||||
|
@ -434,6 +430,13 @@ public class AMQPSessionCallback implements SessionCallback {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void close(boolean failed) {
|
||||||
|
if (protonSession != null) {
|
||||||
|
protonSession.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void ack(Transaction transaction, Object brokerConsumer, Message message) throws Exception {
|
public void ack(Transaction transaction, Object brokerConsumer, Message message) throws Exception {
|
||||||
if (transaction == null) {
|
if (transaction == null) {
|
||||||
transaction = serverSession.getCurrentTransaction();
|
transaction = serverSession.getCurrentTransaction();
|
||||||
|
|
|
@ -88,9 +88,6 @@ public abstract class ProtonAbstractReceiver extends ProtonInitializable impleme
|
||||||
this.creditRunnable = createCreditRunnable(amqpCredits, minCreditRefresh, receiver, connection, this);
|
this.creditRunnable = createCreditRunnable(amqpCredits, minCreditRefresh, receiver, connection, this);
|
||||||
useModified = this.connection.getProtocolManager().isUseModifiedForTransientDeliveryErrors();
|
useModified = this.connection.getProtocolManager().isUseModifiedForTransientDeliveryErrors();
|
||||||
this.routingContext = new RoutingContextImpl(null).setDuplicateDetection(connection.getProtocolManager().isAmqpDuplicateDetection());
|
this.routingContext = new RoutingContextImpl(null).setDuplicateDetection(connection.getProtocolManager().isAmqpDuplicateDetection());
|
||||||
if (sessionSPI != null) {
|
|
||||||
sessionSPI.addCloseable((boolean failed) -> clearLargeMessage());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void recoverContext() {
|
protected void recoverContext() {
|
||||||
|
@ -137,8 +134,8 @@ public abstract class ProtonAbstractReceiver extends ProtonInitializable impleme
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* The reason why we use the AtomicRunnable here
|
* The reason why we use the AtomicRunnable here
|
||||||
* is because PagingManager will call Runnables in case it was blocked.
|
* is because PagingManager will call Runnable in case it was blocked.
|
||||||
* however it could call many Runnables
|
* however it could call many Runnable
|
||||||
* and this serves as a control to avoid duplicated calls
|
* and this serves as a control to avoid duplicated calls
|
||||||
* */
|
* */
|
||||||
static class FlowControlRunner implements Runnable {
|
static class FlowControlRunner implements Runnable {
|
||||||
|
@ -178,10 +175,10 @@ public abstract class ProtonAbstractReceiver extends ProtonInitializable impleme
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public int incrementSettle() {
|
public void incrementSettle() {
|
||||||
assert pendingSettles >= 0;
|
assert pendingSettles >= 0;
|
||||||
connection.requireInHandler();
|
connection.requireInHandler();
|
||||||
return pendingSettles++;
|
pendingSettles++;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void settle(Delivery settlement) {
|
public void settle(Delivery settlement) {
|
||||||
|
@ -289,13 +286,13 @@ public abstract class ProtonAbstractReceiver extends ProtonInitializable impleme
|
||||||
@Override
|
@Override
|
||||||
public void close(boolean remoteLinkClose) throws ActiveMQAMQPException {
|
public void close(boolean remoteLinkClose) throws ActiveMQAMQPException {
|
||||||
protonSession.removeReceiver(receiver);
|
protonSession.removeReceiver(receiver);
|
||||||
|
clearLargeMessage();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void close(ErrorCondition condition) throws ActiveMQAMQPException {
|
public void close(ErrorCondition condition) throws ActiveMQAMQPException {
|
||||||
receiver.setCondition(condition);
|
receiver.setCondition(condition);
|
||||||
close(false);
|
close(false);
|
||||||
clearLargeMessage();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract void actualDelivery(AMQPMessage message, Delivery delivery, Receiver receiver, Transaction tx);
|
protected abstract void actualDelivery(AMQPMessage message, Delivery delivery, Receiver receiver, Transaction tx);
|
||||||
|
|
|
@ -24,6 +24,7 @@ import org.apache.activemq.artemis.core.paging.PagingStore;
|
||||||
import org.apache.activemq.artemis.core.persistence.OperationContext;
|
import org.apache.activemq.artemis.core.persistence.OperationContext;
|
||||||
import org.apache.activemq.artemis.core.server.ActiveMQServer;
|
import org.apache.activemq.artemis.core.server.ActiveMQServer;
|
||||||
import org.apache.activemq.artemis.protocol.amqp.proton.AMQPConnectionContext;
|
import org.apache.activemq.artemis.protocol.amqp.proton.AMQPConnectionContext;
|
||||||
|
import org.apache.activemq.artemis.protocol.amqp.proton.AMQPSessionContext;
|
||||||
import org.apache.activemq.artemis.protocol.amqp.proton.ProtonServerReceiverContext;
|
import org.apache.activemq.artemis.protocol.amqp.proton.ProtonServerReceiverContext;
|
||||||
import org.apache.activemq.artemis.spi.core.remoting.Connection;
|
import org.apache.activemq.artemis.spi.core.remoting.Connection;
|
||||||
import org.apache.qpid.proton.engine.Receiver;
|
import org.apache.qpid.proton.engine.Receiver;
|
||||||
|
@ -258,4 +259,20 @@ public class AMQPSessionCallbackTest {
|
||||||
// Credit runnable should not grant what would be negative credit here
|
// Credit runnable should not grant what would be negative credit here
|
||||||
Mockito.verify(receiver, never()).flow(anyInt());
|
Mockito.verify(receiver, never()).flow(anyInt());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCloseBoolCallsProtonSessionClose() throws Exception {
|
||||||
|
Mockito.reset(connection);
|
||||||
|
Mockito.when(manager.getServer()).thenReturn(server);
|
||||||
|
|
||||||
|
// Capture credit runnable and invoke to trigger credit top off
|
||||||
|
ArgumentCaptor<Runnable> argument = ArgumentCaptor.forClass(Runnable.class);
|
||||||
|
AMQPSessionCallback session = new AMQPSessionCallback(protonSPI, manager, connection, transportConnection, executor, operationContext);
|
||||||
|
|
||||||
|
AMQPSessionContext protonSession = Mockito.mock(AMQPSessionContext.class);
|
||||||
|
session.init(protonSession, null);
|
||||||
|
session.close(false);
|
||||||
|
|
||||||
|
Mockito.verify(protonSession).close();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,6 +35,7 @@ import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import org.apache.activemq.artemis.api.core.ActiveMQAddressFullException;
|
import org.apache.activemq.artemis.api.core.ActiveMQAddressFullException;
|
||||||
import org.apache.activemq.artemis.api.core.ActiveMQException;
|
import org.apache.activemq.artemis.api.core.ActiveMQException;
|
||||||
import org.apache.activemq.artemis.api.core.SimpleString;
|
import org.apache.activemq.artemis.api.core.SimpleString;
|
||||||
|
import org.apache.activemq.artemis.core.persistence.impl.nullpm.NullStorageManager;
|
||||||
import org.apache.activemq.artemis.core.server.RoutingContext;
|
import org.apache.activemq.artemis.core.server.RoutingContext;
|
||||||
import org.apache.activemq.artemis.core.transaction.Transaction;
|
import org.apache.activemq.artemis.core.transaction.Transaction;
|
||||||
import org.apache.activemq.artemis.protocol.amqp.broker.AMQPMessage;
|
import org.apache.activemq.artemis.protocol.amqp.broker.AMQPMessage;
|
||||||
|
@ -89,6 +90,49 @@ public class ProtonServerReceiverContextTest {
|
||||||
doOnMessageWithDeliveryException(singletonList(Accepted.DESCRIPTOR_SYMBOL), null, new ActiveMQException(), Rejected.class);
|
doOnMessageWithDeliveryException(singletonList(Accepted.DESCRIPTOR_SYMBOL), null, new ActiveMQException(), Rejected.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testClearLargeOnClose() throws Exception {
|
||||||
|
Receiver mockReceiver = mock(Receiver.class);
|
||||||
|
AMQPConnectionContext mockConnContext = mock(AMQPConnectionContext.class);
|
||||||
|
|
||||||
|
when(mockConnContext.getAmqpCredits()).thenReturn(100);
|
||||||
|
when(mockConnContext.getAmqpLowCredits()).thenReturn(30);
|
||||||
|
|
||||||
|
when(mockConnContext.getProtocolManager()).thenReturn(mock(ProtonProtocolManager.class));
|
||||||
|
|
||||||
|
AMQPSessionCallback mockSessionSpi = mock(AMQPSessionCallback.class);
|
||||||
|
when(mockSessionSpi.getStorageManager()).thenReturn(new NullStorageManager());
|
||||||
|
|
||||||
|
AMQPSessionContext mockProtonContext = mock(AMQPSessionContext.class);
|
||||||
|
|
||||||
|
AtomicInteger clearLargeMessage = new AtomicInteger(0);
|
||||||
|
ProtonServerReceiverContext rc = new ProtonServerReceiverContext(mockSessionSpi, mockConnContext, mockProtonContext, mockReceiver) {
|
||||||
|
@Override
|
||||||
|
protected void clearLargeMessage() {
|
||||||
|
super.clearLargeMessage();
|
||||||
|
clearLargeMessage.incrementAndGet();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Delivery mockDelivery = mock(Delivery.class);
|
||||||
|
when(mockDelivery.isAborted()).thenReturn(false);
|
||||||
|
when(mockDelivery.isPartial()).thenReturn(false);
|
||||||
|
when(mockDelivery.getLink()).thenReturn(mockReceiver);
|
||||||
|
|
||||||
|
when(mockReceiver.current()).thenReturn(mockDelivery);
|
||||||
|
|
||||||
|
rc.onMessage(mockDelivery);
|
||||||
|
|
||||||
|
rc.close(true);
|
||||||
|
|
||||||
|
verify(mockReceiver, times(1)).current();
|
||||||
|
verify(mockReceiver, times(1)).advance();
|
||||||
|
|
||||||
|
Assert.assertTrue(clearLargeMessage.get() > 0);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
private void doOnMessageWithAbortedDeliveryTestImpl(boolean drain) throws ActiveMQAMQPException {
|
private void doOnMessageWithAbortedDeliveryTestImpl(boolean drain) throws ActiveMQAMQPException {
|
||||||
Receiver mockReceiver = mock(Receiver.class);
|
Receiver mockReceiver = mock(Receiver.class);
|
||||||
AMQPConnectionContext mockConnContext = mock(AMQPConnectionContext.class);
|
AMQPConnectionContext mockConnContext = mock(AMQPConnectionContext.class);
|
||||||
|
|
|
@ -305,6 +305,11 @@ public class ServerSessionImpl implements ServerSession, FailureListener {
|
||||||
this.closeables.add(closeable);
|
this.closeables.add(closeable);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// for testing
|
||||||
|
public final Set<Closeable> getCloseables() {
|
||||||
|
return closeables;
|
||||||
|
}
|
||||||
|
|
||||||
public Map<SimpleString, TempQueueCleanerUpper> getTempQueueCleanUppers() {
|
public Map<SimpleString, TempQueueCleanerUpper> getTempQueueCleanUppers() {
|
||||||
return tempQueueCleannerUppers;
|
return tempQueueCleannerUppers;
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,13 +16,17 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.activemq.artemis.tests.integration.amqp;
|
package org.apache.activemq.artemis.tests.integration.amqp;
|
||||||
|
|
||||||
|
import org.apache.activemq.artemis.core.server.ServerSession;
|
||||||
|
import org.apache.activemq.artemis.core.server.impl.ServerSessionImpl;
|
||||||
import org.apache.activemq.transport.amqp.client.AmqpClient;
|
import org.apache.activemq.transport.amqp.client.AmqpClient;
|
||||||
import org.apache.activemq.transport.amqp.client.AmqpConnection;
|
import org.apache.activemq.transport.amqp.client.AmqpConnection;
|
||||||
import org.apache.activemq.transport.amqp.client.AmqpReceiver;
|
import org.apache.activemq.transport.amqp.client.AmqpReceiver;
|
||||||
|
import org.apache.activemq.transport.amqp.client.AmqpSender;
|
||||||
import org.apache.activemq.transport.amqp.client.AmqpSession;
|
import org.apache.activemq.transport.amqp.client.AmqpSession;
|
||||||
import org.apache.activemq.transport.amqp.client.AmqpValidator;
|
import org.apache.activemq.transport.amqp.client.AmqpValidator;
|
||||||
import org.apache.qpid.proton.engine.Receiver;
|
import org.apache.qpid.proton.engine.Receiver;
|
||||||
import org.apache.qpid.proton.engine.Session;
|
import org.apache.qpid.proton.engine.Session;
|
||||||
|
import org.junit.Assert;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
public class AmqpSessionTest extends AmqpClientTestSupport {
|
public class AmqpSessionTest extends AmqpClientTestSupport {
|
||||||
|
@ -71,4 +75,28 @@ public class AmqpSessionTest extends AmqpClientTestSupport {
|
||||||
connection.getStateInspector().assertValid();
|
connection.getStateInspector().assertValid();
|
||||||
connection.close();
|
connection.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test(timeout = 60000)
|
||||||
|
public void testCreateSessionProducerConsumerDoesNotLeakClosable() throws Exception {
|
||||||
|
AmqpClient client = createAmqpClient();
|
||||||
|
AmqpConnection connection = addConnection(client.connect());
|
||||||
|
AmqpSession session = connection.createSession();
|
||||||
|
assertNotNull(session);
|
||||||
|
|
||||||
|
for (int i = 0; i < 10; i++) {
|
||||||
|
AmqpReceiver receiver = session.createReceiver(getQueueName());
|
||||||
|
AmqpSender sender = session.createSender(getQueueName());
|
||||||
|
receiver.close();
|
||||||
|
sender.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEquals(1, server.getSessions().size());
|
||||||
|
for (ServerSession serverSession : server.getSessions()) {
|
||||||
|
Assert.assertNull( ((ServerSessionImpl) serverSession).getCloseables());
|
||||||
|
}
|
||||||
|
|
||||||
|
connection.close();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue