All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.activemq.artemis.ra.ActiveMQRAManagedConnection 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 jakarta.jms.ExceptionListener;
import jakarta.jms.JMSException;
import jakarta.jms.ResourceAllocationException;
import jakarta.jms.Session;
import jakarta.jms.XASession;
import jakarta.resource.ResourceException;
import jakarta.resource.spi.ConnectionEvent;
import jakarta.resource.spi.ConnectionEventListener;
import jakarta.resource.spi.ConnectionRequestInfo;
import jakarta.resource.spi.IllegalStateException;
import jakarta.resource.spi.LocalTransaction;
import jakarta.resource.spi.ManagedConnection;
import jakarta.resource.spi.ManagedConnectionMetaData;
import jakarta.resource.spi.SecurityException;
import javax.security.auth.Subject;
import jakarta.transaction.Status;
import jakarta.transaction.TransactionSynchronizationRegistry;
import javax.transaction.xa.XAResource;
import java.io.PrintWriter;
import java.lang.invoke.MethodHandles;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.locks.ReentrantLock;

import org.apache.activemq.artemis.core.client.impl.ClientSessionInternal;
import org.apache.activemq.artemis.jms.client.ActiveMQConnection;
import org.apache.activemq.artemis.jms.client.ActiveMQConnectionFactory;
import org.apache.activemq.artemis.jms.client.ActiveMQXAConnection;
import org.apache.activemq.artemis.service.extensions.ServiceUtils;
import org.apache.activemq.artemis.service.extensions.xa.ActiveMQXAResourceWrapper;
import org.apache.activemq.artemis.utils.VersionLoader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * The managed connection
 */
public final class ActiveMQRAManagedConnection implements ManagedConnection, ExceptionListener {

   private static final Logger logger = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());

   /**
    * The managed connection factory
    */
   private final ActiveMQRAManagedConnectionFactory mcf;

   /**
    * The connection request information
    */
   private final ActiveMQRAConnectionRequestInfo cri;

   /**
    * The resource adapter
    */
   private final ActiveMQResourceAdapter ra;

   /**
    * The user name
    */
   private final String userName;

   /**
    * The password
    */
   private final String password;

   /**
    * Has the connection been destroyed
    */
   private final AtomicBoolean isDestroyed = new AtomicBoolean(false);

   /**
    * Event listeners
    */
   private final List eventListeners;

   /**
    * Handles
    */
   private final Set handles;

   /**
    * Lock
    */
   private ReentrantLock lock = new ReentrantLock();

   // Physical connection stuff
   private ActiveMQConnectionFactory connectionFactory;

   private ActiveMQXAConnection connection;

   // The ManagedConnection will play with a XA and a NonXASession to couple with
   // cases where a commit is called on a non-XAed (or non-enlisted) case.
   private Session nonXAsession;

   private XASession xaSession;

   private XAResource xaResource;

   private final TransactionSynchronizationRegistry tsr;

   private boolean inManagedTx;

   /**
    * Constructor
    *
    * @param mcf      The managed connection factory
    * @param cri      The connection request information
    * @param userName The user name
    * @param password The password
    */
   public ActiveMQRAManagedConnection(final ActiveMQRAManagedConnectionFactory mcf,
                                      final ActiveMQRAConnectionRequestInfo cri,
                                      final ActiveMQResourceAdapter ra,
                                      final String userName,
                                      final String password) throws ResourceException {
      if (logger.isTraceEnabled()) {
         logger.trace("constructor({}, {}, {}, ****))", mcf, cri, userName);
      }

      this.mcf = mcf;
      this.cri = cri;
      this.tsr = ra.getTSR();
      this.ra = ra;
      this.userName = userName;
      this.password = password;
      eventListeners = Collections.synchronizedList(new ArrayList<>());
      handles = Collections.synchronizedSet(new HashSet<>());

      connection = null;
      nonXAsession = null;
      xaSession = null;
      xaResource = null;

      try {
         setup();
      } catch (ResourceException e) {
         try {
            destroy();
         } catch (Throwable ignored) {
         }

         throw e;
      } catch (Throwable t) {
         try {
            destroy();
         } catch (Throwable ignored) {
         }
         throw new ResourceException("Error during setup", t);
      }
   }

   /**
    * Get a connection
    *
    * @param subject       The security subject
    * @param cxRequestInfo The request info
    * @return The connection
    * @throws ResourceException Thrown if an error occurs
    */
   @Override
   public synchronized Object getConnection(final Subject subject,
                                            final ConnectionRequestInfo cxRequestInfo) throws ResourceException {
      logger.trace("getConnection({}, {})", subject, cxRequestInfo);

      // Check user first
      ActiveMQRACredential credential = ActiveMQRACredential.getCredential(mcf, subject, cxRequestInfo);

      // Null users are allowed!
      if (userName != null && !userName.equals(credential.getUserName())) {
         throw new SecurityException("Password credentials not the same, reauthentication not allowed");
      }

      if (userName == null && credential.getUserName() != null) {
         throw new SecurityException("Password credentials not the same, reauthentication not allowed");
      }

      if (isDestroyed.get()) {
         throw new IllegalStateException("The managed connection is already destroyed");
      }

      ActiveMQRASession session = new ActiveMQRASession(this, (ActiveMQRAConnectionRequestInfo) cxRequestInfo);
      handles.add(session);
      return session;
   }

   /**
    * Destroy all handles.
    *
    * @throws ResourceException Failed to close one or more handles.
    */
   private void destroyHandles() throws ResourceException {
      logger.trace("destroyHandles()");

      for (ActiveMQRASession session : handles) {
         session.destroy();
      }

      handles.clear();
   }

   /**
    * Destroy the physical connection.
    *
    * @throws ResourceException Could not property close the session and connection.
    */
   @Override
   public void destroy() throws ResourceException {
      logger.trace("destroy()");

      if (isDestroyed.get() || connection == null) {
         return;
      }

      isDestroyed.set(true);

      try {
         connection.setExceptionListener(null);
      } catch (JMSException e) {
         logger.debug("Error unsetting the exception listener {}", this, e);
      }

      connection.signalStopToAllSessions();

      try {
         // we must close the ActiveMQConnectionFactory because it contains a ServerLocator
         if (connectionFactory != null) {
            ra.closeConnectionFactory(mcf.getProperties());
         }
      } catch (Exception e) {
         logger.debug(e.getMessage(), e);
      }

      destroyHandles();

      try {
         // The following calls should not be necessary, as the connection should close the
         // ClientSessionFactory, which will close the sessions.
         try {
            /**
             * (xa|nonXA)Session.close() may NOT be called BEFORE connection.close()
             * 

* If the ClientSessionFactory is trying to fail-over or reconnect with -1 attempts, and * one calls session.close() it may effectively dead-lock. *

* connection close will close the ClientSessionFactory which will close all sessions. */ connection.close(); if (nonXAsession != null) { nonXAsession.close(); } if (xaSession != null) { xaSession.close(); } } catch (JMSException e) { logger.debug("Error closing session {}", this, e); } } catch (Throwable e) { throw new ResourceException("Could not properly close the session and connection", e); } } /** * Cleanup * * @throws ResourceException Thrown if an error occurs */ @Override public void cleanup() throws ResourceException { logger.trace("cleanup()"); if (isDestroyed.get()) { throw new IllegalStateException("ManagedConnection already destroyed"); } destroyHandles(); inManagedTx = false; inManagedTx = false; // I'm recreating the lock object when we return to the pool // because it looks too nasty to expect the connection handle // to unlock properly in certain race conditions // where the dissociation of the managed connection is "random". lock = new ReentrantLock(); } /** * Move a handler from one mc to this one. * * @param obj An object of type ActiveMQSession. * @throws ResourceException Failed to associate connection. * @throws IllegalStateException ManagedConnection in an illegal state. */ @Override public void associateConnection(final Object obj) throws ResourceException { logger.trace("associateConnection({})", obj); if (!isDestroyed.get() && obj instanceof ActiveMQRASession) { ActiveMQRASession h = (ActiveMQRASession) obj; h.setManagedConnection(this); handles.add(h); } else { throw new IllegalStateException("ManagedConnection in an illegal state"); } } public void checkTransactionActive() throws JMSException { // don't bother looking at the transaction if there's an active XID if (!inManagedTx && tsr != null) { int status = tsr.getTransactionStatus(); // Only allow states that will actually succeed if (status == Status.STATUS_COMMITTED || status == Status.STATUS_MARKED_ROLLBACK || status == Status.STATUS_ROLLEDBACK || status == Status.STATUS_ROLLING_BACK) { String statusMessage = ""; if (status == Status.STATUS_COMMITTED) { statusMessage = "committed"; } else if (status == Status.STATUS_MARKED_ROLLBACK) { statusMessage = "marked for rollback"; } else if (status == Status.STATUS_ROLLEDBACK) { statusMessage = "rolled back"; } else if (status == Status.STATUS_ROLLING_BACK) { statusMessage = "rolling back"; } throw new jakarta.jms.IllegalStateException("Transaction is " + statusMessage); } } } /** * Aqquire a lock on the managed connection */ protected void lock() { logger.trace("lock()"); lock.lock(); } /** * Aqquire a lock on the managed connection within the specified period * * @throws JMSException Thrown if an error occurs */ protected void tryLock() throws JMSException { logger.trace("tryLock()"); Integer tryLock = mcf.getUseTryLock(); if (tryLock == null || tryLock.intValue() <= 0) { lock(); return; } try { if (lock.tryLock(tryLock.intValue(), TimeUnit.SECONDS) == false) { throw new ResourceAllocationException("Unable to obtain lock in " + tryLock + " seconds: " + this); } } catch (InterruptedException e) { throw new ResourceAllocationException("Interrupted attempting lock: " + this); } } /** * Unlock the managed connection */ protected void unlock() { logger.trace("unlock()"); lock.unlock(); } /** * Add a connection event listener. * * @param l The connection event listener to be added. */ @Override public void addConnectionEventListener(final ConnectionEventListener l) { logger.trace("addConnectionEventListener({})", l); eventListeners.add(l); } /** * Remove a connection event listener. * * @param l The connection event listener to be removed. */ @Override public void removeConnectionEventListener(final ConnectionEventListener l) { logger.trace("removeConnectionEventListener({})", l); eventListeners.remove(l); } /** * Get the XAResource for the connection. * * @return The XAResource for the connection. * @throws ResourceException XA transaction not supported */ @Override public XAResource getXAResource() throws ResourceException { logger.trace("getXAResource()"); // // Spec says a mc must always return the same XA resource, // so we cache it. // if (xaResource == null) { ClientSessionInternal csi = (ClientSessionInternal) xaSession.getXAResource(); ActiveMQRAXAResource activeMQRAXAResource = new ActiveMQRAXAResource(this, xaSession.getXAResource()); Map xaResourceProperties = new HashMap<>(); xaResourceProperties.put(ActiveMQXAResourceWrapper.ACTIVEMQ_JNDI_NAME, ra.getJndiName()); xaResourceProperties.put(ActiveMQXAResourceWrapper.ACTIVEMQ_NODE_ID, csi.getNodeId()); xaResourceProperties.put(ActiveMQXAResourceWrapper.ACTIVEMQ_PRODUCT_NAME, ActiveMQResourceAdapter.PRODUCT_NAME); xaResourceProperties.put(ActiveMQXAResourceWrapper.ACTIVEMQ_PRODUCT_VERSION, VersionLoader.getVersion().getFullVersion()); xaResource = ServiceUtils.wrapXAResource(activeMQRAXAResource, xaResourceProperties); } logger.trace("XAResource={}", xaResource); return xaResource; } /** * Get the location transaction for the connection. * * @return The local transaction for the connection. * @throws ResourceException Thrown if operation fails. */ @Override public LocalTransaction getLocalTransaction() throws ResourceException { logger.trace("getLocalTransaction()"); LocalTransaction tx = new ActiveMQRALocalTransaction(this); logger.trace("LocalTransaction={}", tx); return tx; } /** * Get the meta data for the connection. * * @return The meta data for the connection. * @throws ResourceException Thrown if the operation fails. * @throws IllegalStateException Thrown if the managed connection already is destroyed. */ @Override public ManagedConnectionMetaData getMetaData() throws ResourceException { logger.trace("getMetaData()"); if (isDestroyed.get()) { throw new IllegalStateException("The managed connection is already destroyed"); } return new ActiveMQRAMetaData(this); } /** * Set the log writer -- NOT SUPPORTED * * @param out The log writer * @throws ResourceException If operation fails */ @Override public void setLogWriter(final PrintWriter out) throws ResourceException { logger.trace("setLogWriter({})", out); } /** * Get the log writer -- NOT SUPPORTED * * @return Always null * @throws ResourceException If operation fails */ @Override public PrintWriter getLogWriter() throws ResourceException { logger.trace("getLogWriter()"); return null; } /** * Notifies user of a JMS exception. * * @param exception The JMS exception */ @Override public void onException(final JMSException exception) { if (ActiveMQConnection.EXCEPTION_FAILOVER.equals(exception.getErrorCode())) { return; } if (logger.isTraceEnabled()) { logger.trace("onException()", exception); } if (isDestroyed.get()) { if (logger.isTraceEnabled()) { logger.trace("Ignoring error on already destroyed connection {}", this, exception); } return; } ActiveMQRALogger.LOGGER.handlingJMSFailure(exception); try { connection.setExceptionListener(null); } catch (JMSException e) { logger.debug("Unable to unset exception listener", e); } ConnectionEvent event = new ConnectionEvent(this, ConnectionEvent.CONNECTION_ERROR_OCCURRED, exception); sendEvent(event); } /** * Get the session for this connection. * * @return The session * @throws JMSException */ protected Session getSession() throws JMSException { if (xaResource != null && inManagedTx) { logger.trace("getSession() -> XA session {}", xaSession.getSession()); return xaSession.getSession(); } else { logger.trace("getSession() -> non XA session {}", nonXAsession); return nonXAsession; } } /** * Send an event. * * @param event The event to send. */ protected void sendEvent(final ConnectionEvent event) { logger.trace("sendEvent({})", event); int type = event.getId(); // convert to an array to avoid concurrent modification exceptions ConnectionEventListener[] list = eventListeners.toArray(new ConnectionEventListener[eventListeners.size()]); for (ConnectionEventListener l : list) { switch (type) { case ConnectionEvent.CONNECTION_CLOSED: l.connectionClosed(event); break; case ConnectionEvent.LOCAL_TRANSACTION_STARTED: l.localTransactionStarted(event); break; case ConnectionEvent.LOCAL_TRANSACTION_COMMITTED: l.localTransactionCommitted(event); break; case ConnectionEvent.LOCAL_TRANSACTION_ROLLEDBACK: l.localTransactionRolledback(event); break; case ConnectionEvent.CONNECTION_ERROR_OCCURRED: l.connectionErrorOccurred(event); break; default: throw new IllegalArgumentException("Illegal eventType: " + type); } } } /** * Remove a handle from the handle map. * * @param handle The handle to remove. */ protected void removeHandle(final ActiveMQRASession handle) { logger.trace("removeHandle({})", handle); handles.remove(handle); } /** * Get the request info for this connection. * * @return The connection request info for this connection. */ protected ActiveMQRAConnectionRequestInfo getCRI() { logger.trace("getCRI()"); return cri; } /** * Get the connection factory for this connection. * * @return The connection factory for this connection. */ protected ActiveMQRAManagedConnectionFactory getManagedConnectionFactory() { logger.trace("getManagedConnectionFactory()"); return mcf; } /** * Start the connection * * @throws JMSException Thrown if the connection can't be started */ void start() throws JMSException { logger.trace("start()"); if (connection != null) { connection.start(); } } /** * Stop the connection * * @throws JMSException Thrown if the connection can't be stopped */ void stop() throws JMSException { logger.trace("stop()"); if (connection != null) { connection.stop(); } } /** * Get the user name * * @return The user name */ protected String getUserName() { logger.trace("getUserName()"); return userName; } /** * Setup the connection. * * @throws ResourceException Thrown if a connection couldn't be created */ private void setup() throws ResourceException { logger.trace("setup()"); try { createCF(); boolean transacted = cri.isTransacted(); int acknowledgeMode = Session.AUTO_ACKNOWLEDGE; if (cri.getType() == ActiveMQRAConnectionFactory.TOPIC_CONNECTION) { if (userName != null && password != null) { connection = (ActiveMQXAConnection) connectionFactory.createXATopicConnection(userName, password); } else { connection = (ActiveMQXAConnection) connectionFactory.createXATopicConnection(); } connection.setExceptionListener(this); xaSession = connection.createXATopicSession(); nonXAsession = connection.createNonXATopicSession(transacted, acknowledgeMode); } else if (cri.getType() == ActiveMQRAConnectionFactory.QUEUE_CONNECTION) { if (userName != null && password != null) { connection = (ActiveMQXAConnection) connectionFactory.createXAQueueConnection(userName, password); } else { connection = (ActiveMQXAConnection) connectionFactory.createXAQueueConnection(); } connection.setExceptionListener(this); xaSession = connection.createXAQueueSession(); nonXAsession = connection.createNonXAQueueSession(transacted, acknowledgeMode); } else { if (userName != null && password != null) { connection = (ActiveMQXAConnection) connectionFactory.createXAConnection(userName, password); } else { connection = (ActiveMQXAConnection) connectionFactory.createXAConnection(); } connection.setExceptionListener(this); xaSession = connection.createXASession(); nonXAsession = connection.createNonXASession(transacted, acknowledgeMode); } } catch (JMSException je) { throw new ResourceException(je.getMessage(), je); } } private void createCF() { if (connectionFactory == null) { connectionFactory = ra.getConnectionFactory(mcf.getProperties()); } } protected void setInManagedTx(boolean inManagedTx) { this.inManagedTx = inManagedTx; } public ActiveMQConnectionFactory getConnectionFactory() { return connectionFactory; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy