org.apache.activemq.artemis.ra.ActiveMQRASessionFactoryImpl Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.activemq.artemis.ra;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;
import jakarta.jms.ConnectionConsumer;
import jakarta.jms.ConnectionMetaData;
import jakarta.jms.Destination;
import jakarta.jms.ExceptionListener;
import jakarta.jms.IllegalStateException;
import jakarta.jms.JMSContext;
import jakarta.jms.JMSException;
import jakarta.jms.Queue;
import jakarta.jms.QueueSession;
import jakarta.jms.ServerSessionPool;
import jakarta.jms.Session;
import jakarta.jms.TemporaryQueue;
import jakarta.jms.TemporaryTopic;
import jakarta.jms.Topic;
import jakarta.jms.TopicSession;
import jakarta.jms.XAJMSContext;
import jakarta.jms.XAQueueSession;
import jakarta.jms.XASession;
import jakarta.jms.XATopicSession;
import javax.naming.Reference;
import jakarta.resource.Referenceable;
import jakarta.resource.spi.ConnectionManager;
import jakarta.transaction.Status;
import jakarta.transaction.TransactionSynchronizationRegistry;
import org.apache.activemq.artemis.api.jms.ActiveMQJMSConstants;
import org.apache.activemq.artemis.jms.client.ActiveMQConnectionForContext;
import org.apache.activemq.artemis.jms.client.ActiveMQConnectionForContextImpl;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.lang.invoke.MethodHandles;
/**
* Implements the JMS Connection API and produces {@link ActiveMQRASession} objects.
*/
public final class ActiveMQRASessionFactoryImpl extends ActiveMQConnectionForContextImpl implements ActiveMQRASessionFactory, ActiveMQConnectionForContext, Referenceable {
private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
/**
* Are we closed?
*/
private boolean closed = false;
/**
* The naming reference
*/
private Reference reference;
/**
* The user name
*/
private String userName;
/**
* The password
*/
private String password;
/**
* The client ID
*/
private String clientID;
/**
* The connection type
*/
private final int type;
/**
* Whether we are started
*/
private boolean started = false;
/**
* The managed connection factory
*/
private final ActiveMQRAManagedConnectionFactory mcf;
private final TransactionSynchronizationRegistry tsr;
/**
* The connection manager
*/
private ConnectionManager cm;
/**
* The sessions
*/
private final Set sessions = new HashSet<>();
/**
* The temporary queues
*/
private final Set tempQueues = new HashSet<>();
/**
* The temporary topics
*/
private final Set tempTopics = new HashSet<>();
private boolean allowLocalTransaction;
/**
* Constructor
*
* @param mcf The managed connection factory
* @param cm The connection manager
* @param type The connection type
*/
public ActiveMQRASessionFactoryImpl(final ActiveMQRAManagedConnectionFactory mcf,
final ConnectionManager cm,
final TransactionSynchronizationRegistry tsr,
final int type) {
this.mcf = mcf;
this.tsr = tsr;
if (cm == null) {
this.cm = new ActiveMQRAConnectionManager();
} else {
this.cm = cm;
}
this.type = type;
if (logger.isTraceEnabled()) {
logger.trace("constructor({}, {}, {})", mcf, cm, type);
}
}
@Override
public JMSContext createContext(int sessionMode) {
boolean inJtaTx = inJtaTransaction();
int sessionModeToUse;
switch (sessionMode) {
case Session.AUTO_ACKNOWLEDGE:
case Session.DUPS_OK_ACKNOWLEDGE:
case ActiveMQJMSConstants.INDIVIDUAL_ACKNOWLEDGE:
case ActiveMQJMSConstants.PRE_ACKNOWLEDGE:
sessionModeToUse = sessionMode;
break;
//these are prohibited in JEE unless not in a JTA tx where they should be ignored and auto_ack used
case Session.CLIENT_ACKNOWLEDGE:
if (!inJtaTx) {
throw ActiveMQRABundle.BUNDLE.invalidSessionTransactedModeRuntime();
}
sessionModeToUse = Session.AUTO_ACKNOWLEDGE;
break;
case Session.SESSION_TRANSACTED:
if (!inJtaTx) {
throw ActiveMQRABundle.BUNDLE.invalidClientAcknowledgeModeRuntime();
}
sessionModeToUse = Session.AUTO_ACKNOWLEDGE;
break;
default:
throw ActiveMQRABundle.BUNDLE.invalidAcknowledgeMode(sessionMode);
}
incrementRefCounter();
return new ActiveMQRAJMSContext(this, sessionModeToUse, threadAwareContext);
}
@Override
public XAJMSContext createXAContext() {
incrementRefCounter();
return new ActiveMQRAXAJMSContext(this, threadAwareContext);
}
/**
* Set the naming reference
*
* @param reference The reference
*/
@Override
public void setReference(final Reference reference) {
logger.trace("setReference({})", reference);
this.reference = reference;
}
/**
* Get the naming reference
*
* @return The reference
*/
@Override
public Reference getReference() {
logger.trace("getReference()");
return reference;
}
/**
* Set the user name
*
* @param name The user name
*/
public void setUserName(final String name) {
logger.trace("setUserName({})", name);
userName = name;
}
/**
* Set the password
*
* @param password The password
*/
public void setPassword(final String password) {
logger.trace("setPassword(****)");
this.password = password;
}
/**
* Get the client ID
*
* @return The client ID
* @throws JMSException Thrown if an error occurs
*/
@Override
public String getClientID() throws JMSException {
logger.trace("getClientID()");
checkClosed();
if (clientID == null) {
return ((ActiveMQResourceAdapter) mcf.getResourceAdapter()).getProperties().getClientID();
}
return clientID;
}
/**
* Set the client ID -- throws IllegalStateException
*
* @param cID The client ID
* @throws JMSException Thrown if an error occurs
*/
@Override
public void setClientID(final String cID) throws JMSException {
logger.trace("setClientID({})", cID);
throw new IllegalStateException(ISE);
}
/**
* Create a queue session
*
* @param transacted Use transactions
* @param acknowledgeMode The acknowledge mode
* @return The queue session
* @throws JMSException Thrown if an error occurs
*/
@Override
public QueueSession createQueueSession(final boolean transacted, final int acknowledgeMode) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createQueueSession({}, {})", transacted, acknowledgeMode);
}
checkClosed();
if (type == ActiveMQRAConnectionFactory.TOPIC_CONNECTION || type == ActiveMQRAConnectionFactory.XA_TOPIC_CONNECTION) {
throw new IllegalStateException("Can not get a queue session from a topic connection");
}
return allocateConnection(transacted, acknowledgeMode, type);
}
/**
* Create a XA queue session
*
* @return The XA queue session
* @throws JMSException Thrown if an error occurs
*/
@Override
public XAQueueSession createXAQueueSession() throws JMSException {
logger.trace("createXAQueueSession()");
checkClosed();
if (type == ActiveMQRAConnectionFactory.CONNECTION || type == ActiveMQRAConnectionFactory.TOPIC_CONNECTION ||
type == ActiveMQRAConnectionFactory.XA_TOPIC_CONNECTION) {
throw new IllegalStateException("Can not get a topic session from a queue connection");
}
return allocateConnection(type);
}
/**
* Create a connection consumer -- throws IllegalStateException
*
* @param queue The queue
* @param messageSelector The message selector
* @param sessionPool The session pool
* @param maxMessages The number of max messages
* @return The connection consumer
* @throws JMSException Thrown if an error occurs
*/
@Override
public ConnectionConsumer createConnectionConsumer(final Queue queue,
final String messageSelector,
final ServerSessionPool sessionPool,
final int maxMessages) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createConnectionConsumer({}, {}, {}, {})", queue, messageSelector, sessionPool, maxMessages);
}
throw new IllegalStateException(ISE);
}
/**
* Create a topic session
*
* @param transacted Use transactions
* @param acknowledgeMode The acknowledge mode
* @return The topic session
* @throws JMSException Thrown if an error occurs
*/
@Override
public TopicSession createTopicSession(final boolean transacted, final int acknowledgeMode) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createTopicSession({}, {})", transacted, acknowledgeMode);
}
checkClosed();
if (type == ActiveMQRAConnectionFactory.QUEUE_CONNECTION || type == ActiveMQRAConnectionFactory.XA_QUEUE_CONNECTION) {
throw new IllegalStateException("Can not get a topic session from a queue connection");
}
return allocateConnection(transacted, acknowledgeMode, type);
}
/**
* Create a XA topic session
*
* @return The XA topic session
* @throws JMSException Thrown if an error occurs
*/
@Override
public XATopicSession createXATopicSession() throws JMSException {
logger.trace("createXATopicSession()");
checkClosed();
if (type == ActiveMQRAConnectionFactory.CONNECTION || type == ActiveMQRAConnectionFactory.QUEUE_CONNECTION ||
type == ActiveMQRAConnectionFactory.XA_QUEUE_CONNECTION) {
throw new IllegalStateException("Can not get a topic session from a queue connection");
}
return allocateConnection(type);
}
/**
* Create a connection consumer -- throws IllegalStateException
*
* @param topic The topic
* @param messageSelector The message selector
* @param sessionPool The session pool
* @param maxMessages The number of max messages
* @return The connection consumer
* @throws JMSException Thrown if an error occurs
*/
@Override
public ConnectionConsumer createConnectionConsumer(final Topic topic,
final String messageSelector,
final ServerSessionPool sessionPool,
final int maxMessages) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createConnectionConsumer({}, {}, {}, {})", topic, messageSelector, sessionPool, maxMessages);
}
throw new IllegalStateException(ISE);
}
/**
* Create a durable connection consumer -- throws IllegalStateException
*
* @param topic The topic
* @param subscriptionName The subscription name
* @param messageSelector The message selector
* @param sessionPool The session pool
* @param maxMessages The number of max messages
* @return The connection consumer
* @throws JMSException Thrown if an error occurs
*/
@Override
public ConnectionConsumer createDurableConnectionConsumer(final Topic topic,
final String subscriptionName,
final String messageSelector,
final ServerSessionPool sessionPool,
final int maxMessages) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createConnectionConsumer({}, {}, {}, {}, {})",
topic, subscriptionName, messageSelector, sessionPool, maxMessages);
}
throw new IllegalStateException(ISE);
}
/**
* Create a connection consumer -- throws IllegalStateException
*
* @param destination The destination
* @param pool The session pool
* @param maxMessages The number of max messages
* @return The connection consumer
* @throws JMSException Thrown if an error occurs
*/
public ConnectionConsumer createConnectionConsumer(final Destination destination,
final ServerSessionPool pool,
final int maxMessages) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createConnectionConsumer({}, {}, {})", destination, pool, maxMessages);
}
throw new IllegalStateException(ISE);
}
/**
* Create a connection consumer -- throws IllegalStateException
*
* @param destination The destination
* @param name The name
* @param pool The session pool
* @param maxMessages The number of max messages
* @return The connection consumer
* @throws JMSException Thrown if an error occurs
*/
@Override
public ConnectionConsumer createConnectionConsumer(final Destination destination,
final String name,
final ServerSessionPool pool,
final int maxMessages) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createConnectionConsumer({}, {}, {}, {})", destination, name, pool, maxMessages);
}
throw new IllegalStateException(ISE);
}
/**
* Create a session
*
* @param transacted Use transactions
* @param acknowledgeMode The acknowledge mode
* @return The session
* @throws JMSException Thrown if an error occurs
*/
@Override
public Session createSession(final boolean transacted, final int acknowledgeMode) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createSession({}, {})", transacted, acknowledgeMode);
}
checkClosed();
return allocateConnection(transacted, acknowledgeMode, type);
}
/**
* Create a XA session
*
* @return The XA session
* @throws JMSException Thrown if an error occurs
*/
@Override
public XASession createXASession() throws JMSException {
logger.trace("createXASession()");
checkClosed();
return allocateConnection(type);
}
/**
* Get the connection metadata
*
* @return The connection metadata
* @throws JMSException Thrown if an error occurs
*/
@Override
public ConnectionMetaData getMetaData() throws JMSException {
logger.trace("getMetaData()");
checkClosed();
return mcf.getMetaData();
}
/**
* Get the exception listener -- throws IllegalStateException
*
* @return The exception listener
* @throws JMSException Thrown if an error occurs
*/
@Override
public ExceptionListener getExceptionListener() throws JMSException {
logger.trace("getExceptionListener()");
throw new IllegalStateException(ISE);
}
/**
* Set the exception listener -- throws IllegalStateException
*
* @param listener The exception listener
* @throws JMSException Thrown if an error occurs
*/
@Override
public void setExceptionListener(final ExceptionListener listener) throws JMSException {
logger.trace("setExceptionListener({})", listener);
throw new IllegalStateException(ISE);
}
/**
* Start
*
* @throws JMSException Thrown if an error occurs
*/
@Override
public void start() throws JMSException {
checkClosed();
logger.trace("start() {}", this);
synchronized (sessions) {
if (started) {
return;
}
started = true;
for (ActiveMQRASession session : sessions) {
session.start();
}
}
}
/**
* Stop
*
* @throws IllegalStateException
* @throws JMSException Thrown if an error occurs
*/
@Override
public void stop() throws JMSException {
logger.trace("stop() {}", this);
throw new IllegalStateException(ISE);
}
/**
* Close
*
* @throws JMSException Thrown if an error occurs
*/
@Override
public void close() throws JMSException {
logger.trace("close() {}", this);
if (closed) {
return;
}
closed = true;
synchronized (tempQueues) {
for (Iterator i = tempQueues.iterator(); i.hasNext(); ) {
TemporaryQueue temp = i.next();
try {
logger.trace("Closing temporary queue {} for {}", temp, this);
temp.delete();
} catch (Throwable t) {
logger.trace("Error deleting temporary queue", t);
}
i.remove();
}
}
synchronized (tempTopics) {
for (Iterator i = tempTopics.iterator(); i.hasNext(); ) {
TemporaryTopic temp = i.next();
try {
logger.trace("Closing temporary topic {} for {}", temp, this);
temp.delete();
} catch (Throwable t) {
logger.trace("Error deleting temporary queue", t);
}
i.remove();
}
}
synchronized (sessions) {
for (Iterator i = sessions.iterator(); i.hasNext(); ) {
ActiveMQRASession session = i.next();
try {
session.closeSession();
} catch (Throwable t) {
logger.trace("Error closing session", t);
}
i.remove();
}
}
}
/**
* Close session
*
* @param session The session
* @throws JMSException Thrown if an error occurs
*/
@Override
public void closeSession(final ActiveMQRASession session) throws JMSException {
logger.trace("closeSession({})", session);
synchronized (sessions) {
sessions.remove(session);
}
}
/**
* Add temporary queue
*
* @param temp The temporary queue
*/
@Override
public void addTemporaryQueue(final TemporaryQueue temp) {
logger.trace("addTemporaryQueue({})", temp);
synchronized (tempQueues) {
tempQueues.add(temp);
}
}
/**
* Add temporary topic
*
* @param temp The temporary topic
*/
@Override
public void addTemporaryTopic(final TemporaryTopic temp) {
logger.trace("addTemporaryTopic({})", temp);
synchronized (tempTopics) {
tempTopics.add(temp);
}
}
@Override
public Session createSession(int sessionMode) throws JMSException {
return createSession(sessionMode == Session.SESSION_TRANSACTED, sessionMode);
}
@Override
public Session createSession() throws JMSException {
return createSession(Session.AUTO_ACKNOWLEDGE);
}
@Override
public ConnectionConsumer createSharedConnectionConsumer(Topic topic,
String subscriptionName,
String messageSelector,
ServerSessionPool sessionPool,
int maxMessages) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createSharedConnectionConsumer({}, {}, {}, {}, {})",
topic, subscriptionName, messageSelector, sessionPool, maxMessages);
}
throw new IllegalStateException(ISE);
}
@Override
public ConnectionConsumer createSharedDurableConnectionConsumer(Topic topic,
String subscriptionName,
String messageSelector,
ServerSessionPool sessionPool,
int maxMessages) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("createSharedDurableConnectionConsumer({}, {}, {}, {}, {})",
topic, subscriptionName, messageSelector, sessionPool, maxMessages);
}
throw new IllegalStateException(ISE);
}
/**
* Allocation a connection
*
* @param sessionType The session type
* @return The session
* @throws JMSException Thrown if an error occurs
*/
protected ActiveMQRASession allocateConnection(final int sessionType) throws JMSException {
return allocateConnection(false, Session.AUTO_ACKNOWLEDGE, sessionType);
}
/**
* Allocate a connection
*
* @param transacted Use transactions
* @param acknowledgeMode The acknowledge mode
* @param sessionType The session type
* @return The session
* @throws JMSException Thrown if an error occurs
*/
protected ActiveMQRASession allocateConnection(boolean transacted,
int acknowledgeMode,
final int sessionType) throws JMSException {
if (logger.isTraceEnabled()) {
logger.trace("allocateConnection({}, {}, {})", transacted, acknowledgeMode, sessionType);
}
try {
synchronized (sessions) {
if (sessions.isEmpty() == false) {
throw new IllegalStateException("Only allowed one session per connection. See the J2EE spec, e.g. J2EE1.4 Section 6.6");
}
//from createSession
// In a Java EE web or EJB container, when there is an active JTA transaction in progress:
//Both arguments {@code transacted} and {@code acknowledgeMode} are ignored.
// fix of ARTEMIS-1669 - when a JMSConnectionFactoryDefinition annotation with the transactional attribute set to false="false" is set
// then it should not be included in any JTA transaction and behave like that there is no JTA transaction.
if (!mcf.isIgnoreJTA() && (inJtaTransaction() || mcf.isInJtaTransaction())) {
transacted = true;
//from getAcknowledgeMode
// If the session is transacted, returns SESSION_TRANSACTED.
acknowledgeMode = Session.SESSION_TRANSACTED;
} else {
//In the Java EE web or EJB container, when there is no active JTA transaction in progress
// The argument {@code transacted} is ignored.
//The session will always be non-transacted, unless allow-local-transactions is true
if (transacted && mcf.isAllowLocalTransactions()) {
acknowledgeMode = Session.SESSION_TRANSACTED;
} else {
transacted = false;
switch (acknowledgeMode) {
//using one of the two acknowledgement modes AUTO_ACKNOWLEDGE and DUPS_OK_ACKNOWLEDGE.
case Session.AUTO_ACKNOWLEDGE:
case Session.DUPS_OK_ACKNOWLEDGE:
//plus our own
case ActiveMQJMSConstants.INDIVIDUAL_ACKNOWLEDGE:
case ActiveMQJMSConstants.PRE_ACKNOWLEDGE:
break;
//The value {@code Session.CLIENT_ACKNOWLEDGE} may not be used.
case Session.CLIENT_ACKNOWLEDGE:
throw ActiveMQRABundle.BUNDLE.invalidClientAcknowledgeModeRuntime();
//same with this although the spec doesn't explicitly say
case Session.SESSION_TRANSACTED:
if (!mcf.isAllowLocalTransactions()) {
throw ActiveMQRABundle.BUNDLE.invalidSessionTransactedModeRuntimeAllowLocal();
}
transacted = true;
break;
default:
throw ActiveMQRABundle.BUNDLE.invalidAcknowledgeMode(acknowledgeMode);
}
}
}
ActiveMQRAConnectionRequestInfo info = new ActiveMQRAConnectionRequestInfo(transacted, acknowledgeMode, sessionType);
info.setUserName(userName);
info.setPassword(password);
info.setClientID(clientID);
info.setDefaults(((ActiveMQResourceAdapter) mcf.getResourceAdapter()).getProperties());
logger.trace("Allocating session for {} with request info={}", this, info);
ActiveMQRASession session = (ActiveMQRASession) cm.allocateConnection(mcf, info);
try {
logger.trace("Allocated {} session={}", this, session);
session.setActiveMQSessionFactory(this);
if (started) {
session.start();
}
sessions.add(session);
return session;
} catch (Throwable t) {
try {
session.close();
} catch (Throwable ignored) {
}
if (t instanceof Exception) {
throw (Exception) t;
} else {
throw new RuntimeException("Unexpected error: ", t);
}
}
}
} catch (Exception e) {
Throwable current = e;
while (current != null && !(current instanceof JMSException)) {
current = current.getCause();
}
if (current != null && current instanceof JMSException) {
throw (JMSException) current;
} else {
JMSException je = new JMSException("Could not create a session: " + e.getMessage());
je.setLinkedException(e);
je.initCause(e);
throw je;
}
}
}
/**
* Check if we are closed
*
* @throws IllegalStateException Thrown if closed
*/
protected void checkClosed() throws IllegalStateException {
logger.trace("checkClosed() {}", this);
if (closed) {
throw new IllegalStateException("The connection is closed");
}
}
private boolean inJtaTransaction() {
boolean inJtaTx = false;
if (tsr != null) {
inJtaTx = tsr.getTransactionStatus() != Status.STATUS_NO_TRANSACTION;
}
return inJtaTx;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy