SqlSessionUtils.java

  1. /*
  2.  * Copyright 2010-2024 the original author or authors.
  3.  *
  4.  * Licensed under the Apache License, Version 2.0 (the "License");
  5.  * you may not use this file except in compliance with the License.
  6.  * You may obtain a copy of the License at
  7.  *
  8.  *    https://www.apache.org/licenses/LICENSE-2.0
  9.  *
  10.  * Unless required by applicable law or agreed to in writing, software
  11.  * distributed under the License is distributed on an "AS IS" BASIS,
  12.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13.  * See the License for the specific language governing permissions and
  14.  * limitations under the License.
  15.  */
  16. package org.mybatis.spring;

  17. import static org.springframework.util.Assert.notNull;

  18. import org.apache.ibatis.exceptions.PersistenceException;
  19. import org.apache.ibatis.session.ExecutorType;
  20. import org.apache.ibatis.session.SqlSession;
  21. import org.apache.ibatis.session.SqlSessionFactory;
  22. import org.mybatis.logging.Logger;
  23. import org.mybatis.logging.LoggerFactory;
  24. import org.mybatis.spring.transaction.SpringManagedTransactionFactory;
  25. import org.springframework.dao.TransientDataAccessResourceException;
  26. import org.springframework.dao.support.PersistenceExceptionTranslator;
  27. import org.springframework.jdbc.datasource.DataSourceUtils;
  28. import org.springframework.transaction.support.TransactionSynchronization;
  29. import org.springframework.transaction.support.TransactionSynchronizationManager;

  30. /**
  31.  * Handles MyBatis SqlSession life cycle. It can register and get SqlSessions from Spring
  32.  * {@code TransactionSynchronizationManager}. Also works if no transaction is active.
  33.  *
  34.  * @author Hunter Presnall
  35.  * @author Eduardo Macarron
  36.  */
  37. public final class SqlSessionUtils {

  38.   private static final Logger LOGGER = LoggerFactory.getLogger(SqlSessionUtils.class);

  39.   private static final String NO_EXECUTOR_TYPE_SPECIFIED = "No ExecutorType specified";
  40.   private static final String NO_SQL_SESSION_FACTORY_SPECIFIED = "No SqlSessionFactory specified";
  41.   private static final String NO_SQL_SESSION_SPECIFIED = "No SqlSession specified";

  42.   /**
  43.    * This class can't be instantiated, exposes static utility methods only.
  44.    */
  45.   private SqlSessionUtils() {
  46.     // do nothing
  47.   }

  48.   /**
  49.    * Creates a new MyBatis {@code SqlSession} from the {@code SqlSessionFactory} provided as a parameter and using its
  50.    * {@code DataSource} and {@code ExecutorType}
  51.    *
  52.    * @param sessionFactory
  53.    *          a MyBatis {@code SqlSessionFactory} to create new sessions
  54.    *
  55.    * @return a MyBatis {@code SqlSession}
  56.    *
  57.    * @throws TransientDataAccessResourceException
  58.    *           if a transaction is active and the {@code SqlSessionFactory} is not using a
  59.    *           {@code SpringManagedTransactionFactory}
  60.    */
  61.   public static SqlSession getSqlSession(SqlSessionFactory sessionFactory) {
  62.     var executorType = sessionFactory.getConfiguration().getDefaultExecutorType();
  63.     return getSqlSession(sessionFactory, executorType, null);
  64.   }

  65.   /**
  66.    * Gets an SqlSession from Spring Transaction Manager or creates a new one if needed. Tries to get a SqlSession out of
  67.    * current transaction. If there is not any, it creates a new one. Then, it synchronizes the SqlSession with the
  68.    * transaction if Spring TX is active and <code>SpringManagedTransactionFactory</code> is configured as a transaction
  69.    * manager.
  70.    *
  71.    * @param sessionFactory
  72.    *          a MyBatis {@code SqlSessionFactory} to create new sessions
  73.    * @param executorType
  74.    *          The executor type of the SqlSession to create
  75.    * @param exceptionTranslator
  76.    *          Optional. Translates SqlSession.commit() exceptions to Spring exceptions.
  77.    *
  78.    * @return an SqlSession managed by Spring Transaction Manager
  79.    *
  80.    * @throws TransientDataAccessResourceException
  81.    *           if a transaction is active and the {@code SqlSessionFactory} is not using a
  82.    *           {@code SpringManagedTransactionFactory}
  83.    *
  84.    * @see SpringManagedTransactionFactory
  85.    */
  86.   public static SqlSession getSqlSession(SqlSessionFactory sessionFactory, ExecutorType executorType,
  87.       PersistenceExceptionTranslator exceptionTranslator) {

  88.     notNull(sessionFactory, NO_SQL_SESSION_FACTORY_SPECIFIED);
  89.     notNull(executorType, NO_EXECUTOR_TYPE_SPECIFIED);

  90.     var holder = (SqlSessionHolder) TransactionSynchronizationManager.getResource(sessionFactory);

  91.     var session = sessionHolder(executorType, holder);
  92.     if (session != null) {
  93.       return session;
  94.     }

  95.     LOGGER.debug(() -> "Creating a new SqlSession");
  96.     session = sessionFactory.openSession(executorType);

  97.     registerSessionHolder(sessionFactory, executorType, exceptionTranslator, session);

  98.     return session;
  99.   }

  100.   /**
  101.    * Register session holder if synchronization is active (i.e. a Spring TX is active).
  102.    * <p>
  103.    * Note: The DataSource used by the Environment should be synchronized with the transaction either through
  104.    * DataSourceTxMgr or another tx synchronization. Further assume that if an exception is thrown, whatever started the
  105.    * transaction will handle closing / rolling back the Connection associated with the SqlSession.
  106.    *
  107.    * @param sessionFactory
  108.    *          sqlSessionFactory used for registration.
  109.    * @param executorType
  110.    *          executorType used for registration.
  111.    * @param exceptionTranslator
  112.    *          persistenceExceptionTranslator used for registration.
  113.    * @param session
  114.    *          sqlSession used for registration.
  115.    */
  116.   private static void registerSessionHolder(SqlSessionFactory sessionFactory, ExecutorType executorType,
  117.       PersistenceExceptionTranslator exceptionTranslator, SqlSession session) {
  118.     SqlSessionHolder holder;
  119.     if (TransactionSynchronizationManager.isSynchronizationActive()) {
  120.       var environment = sessionFactory.getConfiguration().getEnvironment();

  121.       if (environment.getTransactionFactory() instanceof SpringManagedTransactionFactory) {
  122.         LOGGER.debug(() -> "Registering transaction synchronization for SqlSession [" + session + "]");

  123.         holder = new SqlSessionHolder(session, executorType, exceptionTranslator);
  124.         TransactionSynchronizationManager.bindResource(sessionFactory, holder);
  125.         TransactionSynchronizationManager
  126.             .registerSynchronization(new SqlSessionSynchronization(holder, sessionFactory));
  127.         holder.setSynchronizedWithTransaction(true);
  128.         holder.requested();
  129.       } else if (TransactionSynchronizationManager.getResource(environment.getDataSource()) == null) {
  130.         LOGGER.debug(() -> "SqlSession [" + session
  131.             + "] was not registered for synchronization because DataSource is not transactional");
  132.       } else {
  133.         throw new TransientDataAccessResourceException(
  134.             "SqlSessionFactory must be using a SpringManagedTransactionFactory in order to use Spring transaction synchronization");
  135.       }
  136.     } else {
  137.       LOGGER.debug(() -> "SqlSession [" + session
  138.           + "] was not registered for synchronization because synchronization is not active");
  139.     }

  140.   }

  141.   private static SqlSession sessionHolder(ExecutorType executorType, SqlSessionHolder holder) {
  142.     SqlSession session = null;
  143.     if (holder != null && holder.isSynchronizedWithTransaction()) {
  144.       if (holder.getExecutorType() != executorType) {
  145.         throw new TransientDataAccessResourceException(
  146.             "Cannot change the ExecutorType when there is an existing transaction");
  147.       }

  148.       holder.requested();

  149.       LOGGER.debug(() -> "Fetched SqlSession [" + holder.getSqlSession() + "] from current transaction");
  150.       session = holder.getSqlSession();
  151.     }
  152.     return session;
  153.   }

  154.   /**
  155.    * Checks if {@code SqlSession} passed as an argument is managed by Spring {@code TransactionSynchronizationManager}
  156.    * If it is not, it closes it, otherwise it just updates the reference counter and lets Spring call the close callback
  157.    * when the managed transaction ends
  158.    *
  159.    * @param session
  160.    *          a target SqlSession
  161.    * @param sessionFactory
  162.    *          a factory of SqlSession
  163.    */
  164.   public static void closeSqlSession(SqlSession session, SqlSessionFactory sessionFactory) {
  165.     notNull(session, NO_SQL_SESSION_SPECIFIED);
  166.     notNull(sessionFactory, NO_SQL_SESSION_FACTORY_SPECIFIED);

  167.     var holder = (SqlSessionHolder) TransactionSynchronizationManager.getResource(sessionFactory);
  168.     if (holder != null && holder.getSqlSession() == session) {
  169.       LOGGER.debug(() -> "Releasing transactional SqlSession [" + session + "]");
  170.       holder.released();
  171.     } else {
  172.       LOGGER.debug(() -> "Closing non transactional SqlSession [" + session + "]");
  173.       session.close();
  174.     }
  175.   }

  176.   /**
  177.    * Returns if the {@code SqlSession} passed as an argument is being managed by Spring
  178.    *
  179.    * @param session
  180.    *          a MyBatis SqlSession to check
  181.    * @param sessionFactory
  182.    *          the SqlSessionFactory which the SqlSession was built with
  183.    *
  184.    * @return true if session is transactional, otherwise false
  185.    */
  186.   public static boolean isSqlSessionTransactional(SqlSession session, SqlSessionFactory sessionFactory) {
  187.     notNull(session, NO_SQL_SESSION_SPECIFIED);
  188.     notNull(sessionFactory, NO_SQL_SESSION_FACTORY_SPECIFIED);

  189.     var holder = (SqlSessionHolder) TransactionSynchronizationManager.getResource(sessionFactory);

  190.     return holder != null && holder.getSqlSession() == session;
  191.   }

  192.   /**
  193.    * Callback for cleaning up resources. It cleans TransactionSynchronizationManager and also commits and closes the
  194.    * {@code SqlSession}. It assumes that {@code Connection} life cycle will be managed by
  195.    * {@code DataSourceTransactionManager} or {@code JtaTransactionManager}
  196.    */
  197.   private static final class SqlSessionSynchronization implements TransactionSynchronization {

  198.     private final SqlSessionHolder holder;

  199.     private final SqlSessionFactory sessionFactory;

  200.     private boolean holderActive = true;

  201.     public SqlSessionSynchronization(SqlSessionHolder holder, SqlSessionFactory sessionFactory) {
  202.       notNull(holder, "Parameter 'holder' must be not null");
  203.       notNull(sessionFactory, "Parameter 'sessionFactory' must be not null");

  204.       this.holder = holder;
  205.       this.sessionFactory = sessionFactory;
  206.     }

  207.     @Override
  208.     public int getOrder() {
  209.       // order right before any Connection synchronization
  210.       return DataSourceUtils.CONNECTION_SYNCHRONIZATION_ORDER - 1;
  211.     }

  212.     @Override
  213.     public void suspend() {
  214.       if (this.holderActive) {
  215.         LOGGER.debug(() -> "Transaction synchronization suspending SqlSession [" + this.holder.getSqlSession() + "]");
  216.         TransactionSynchronizationManager.unbindResource(this.sessionFactory);
  217.       }
  218.     }

  219.     @Override
  220.     public void resume() {
  221.       if (this.holderActive) {
  222.         LOGGER.debug(() -> "Transaction synchronization resuming SqlSession [" + this.holder.getSqlSession() + "]");
  223.         TransactionSynchronizationManager.bindResource(this.sessionFactory, this.holder);
  224.       }
  225.     }

  226.     @Override
  227.     public void beforeCommit(boolean readOnly) {
  228.       // Connection commit or rollback will be handled by ConnectionSynchronization or
  229.       // DataSourceTransactionManager.
  230.       // But, do cleanup the SqlSession / Executor, including flushing BATCH statements so
  231.       // they are actually executed.
  232.       // SpringManagedTransaction will no-op the commit over the jdbc connection
  233.       // TODO This updates 2nd level caches but the tx may be rolledback later on!
  234.       if (TransactionSynchronizationManager.isActualTransactionActive()) {
  235.         try {
  236.           LOGGER.debug(() -> "Transaction synchronization committing SqlSession [" + this.holder.getSqlSession() + "]");
  237.           this.holder.getSqlSession().commit();
  238.         } catch (PersistenceException p) {
  239.           if (this.holder.getPersistenceExceptionTranslator() != null) {
  240.             var translated = this.holder.getPersistenceExceptionTranslator().translateExceptionIfPossible(p);
  241.             if (translated != null) {
  242.               throw translated;
  243.             }
  244.           }
  245.           throw p;
  246.         }
  247.       }
  248.     }

  249.     @Override
  250.     public void beforeCompletion() {
  251.       // Issue #18 Close SqlSession and deregister it now
  252.       // because afterCompletion may be called from a different thread
  253.       if (!this.holder.isOpen()) {
  254.         LOGGER
  255.             .debug(() -> "Transaction synchronization deregistering SqlSession [" + this.holder.getSqlSession() + "]");
  256.         TransactionSynchronizationManager.unbindResource(sessionFactory);
  257.         this.holderActive = false;
  258.         LOGGER.debug(() -> "Transaction synchronization closing SqlSession [" + this.holder.getSqlSession() + "]");
  259.         this.holder.getSqlSession().close();
  260.       }
  261.     }

  262.     @Override
  263.     public void afterCompletion(int status) {
  264.       if (this.holderActive) {
  265.         // afterCompletion may have been called from a different thread
  266.         // so avoid failing if there is nothing in this one
  267.         LOGGER
  268.             .debug(() -> "Transaction synchronization deregistering SqlSession [" + this.holder.getSqlSession() + "]");
  269.         TransactionSynchronizationManager.unbindResourceIfPossible(sessionFactory);
  270.         this.holderActive = false;
  271.         LOGGER.debug(() -> "Transaction synchronization closing SqlSession [" + this.holder.getSqlSession() + "]");
  272.         this.holder.getSqlSession().close();
  273.       }
  274.       this.holder.reset();
  275.     }
  276.   }

  277. }