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

org.springframework.transaction.support.TransactionSynchronizationManager Maven / Gradle / Ivy

There is a newer version: 5.3.34
Show newest version
/*
 * Copyright 2002-2005 the original author or authors.
 * 
 * Licensed 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.springframework.transaction.support;

import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import org.springframework.core.OrderComparator;

/**
 * Internal class that manages resources and transaction synchronizations per thread.
 * Supports one resource per key without overwriting, i.e. a resource needs to
 * be removed before a new one can be set for the same key.
 * Supports a list of transaction synchronizations if synchronization is active.
 *
 * 

Resource management code should check for thread-bound resources, e.g. JDBC * Connections or Hibernate Sessions, via getResource. Such code is * normally not supposed to bind resources to threads, as this is the responsiblity * of transaction managers. A further option is to lazily bind on first use if * transaction synchronization is active, for performing transactions that span * an arbitrary number of resources. * *

Transaction synchronization must be activated and deactivated by a transaction * manager via initSynchronization and clearSynchronization. * This is automatically supported by AbstractPlatformTransactionManager, and thus * by all standard Spring transaction managers, like DataSourceTransactionManager * and JtaTransactionManager. * *

Resource management code should only register synchronizations when this * manager is active, which can be checked via isSynchronizationActive; * it should perform immediate resource cleanup else. If transaction synchronization * isn't active, there is either no current transaction, or the transaction manager * doesn't support transaction synchronizations. * *

Synchronization is for example used to always return the same resources like * JDBC Connections or Hibernate Sessions within a JTA transaction, for any given * DataSource or SessionFactory. In the Hibernate case, the afterCompletion Session * close calls allow for proper transactional JVM-level caching even without a * custom TransactionManagerLookup in Hibernate configuration. * * @author Juergen Hoeller * @since 02.06.2003 * @see #isSynchronizationActive * @see #registerSynchronization * @see TransactionSynchronization * @see AbstractPlatformTransactionManager * @see org.springframework.jdbc.datasource.DataSourceTransactionManager * @see org.springframework.transaction.jta.JtaTransactionManager * @see org.springframework.jdbc.datasource.DataSourceUtils#getConnection * @see org.springframework.orm.hibernate.SessionFactoryUtils#getSession * @see org.springframework.orm.jdo.PersistenceManagerFactoryUtils#getPersistenceManager */ public abstract class TransactionSynchronizationManager { private static final Log logger = LogFactory.getLog(TransactionSynchronizationManager.class); private static final ThreadLocal resources = new ThreadLocal(); private static final ThreadLocal synchronizations = new ThreadLocal(); private static final Comparator synchronizationComparator = new OrderComparator(); private static final ThreadLocal currentTransactionReadOnly = new ThreadLocal(); //------------------------------------------------------------------------- // Management of transaction-associated resource handles //------------------------------------------------------------------------- /** * Return all resources that are bound to the current thread. *

Mainly for debugging purposes. Resource managers should always invoke * hasResource for a specific resource key that they are interested in. * @return Map with resource keys and resource objects, * or empty Map if currently none bound * @see #hasResource */ public static Map getResourceMap() { Map map = (Map) resources.get(); if (map == null) { map = new HashMap(); } return Collections.unmodifiableMap(map); } /** * Check if there is a resource for the given key bound to the current thread. * @param key key to check * @return if there is a value bound to the current thread */ public static boolean hasResource(Object key) { Map map = (Map) resources.get(); return (map != null && map.containsKey(key)); } /** * Retrieve a resource for the given key that is bound to the current thread. * @param key key to check * @return a value bound to the current thread, or null if none */ public static Object getResource(Object key) { Map map = (Map) resources.get(); if (map == null) { return null; } Object value = map.get(key); if (value != null && logger.isDebugEnabled()) { logger.debug("Retrieved value [" + value + "] for key [" + key + "] bound to thread [" + Thread.currentThread().getName() + "]"); } return value; } /** * Bind the given resource for the given key to the current thread. * @param key key to bind the value to * @param value value to bind * @throws IllegalStateException if there is already a value bound to the thread */ public static void bindResource(Object key, Object value) throws IllegalStateException { Map map = (Map) resources.get(); // set ThreadLocal Map if none found if (map == null) { map = new HashMap(); resources.set(map); } if (map.containsKey(key)) { throw new IllegalStateException("Already value [" + map.get(key) + "] for key [" + key + "] bound to thread [" + Thread.currentThread().getName() + "]"); } map.put(key, value); if (logger.isDebugEnabled()) { logger.debug("Bound value [" + value + "] for key [" + key + "] to thread [" + Thread.currentThread().getName() + "]"); } } /** * Unbind a resource for the given key from the current thread. * @param key key to check * @return the previously bound value * @throws IllegalStateException if there is no value bound to the thread */ public static Object unbindResource(Object key) throws IllegalStateException { Map map = (Map) resources.get(); if (map == null || !map.containsKey(key)) { throw new IllegalStateException( "No value for key [" + key + "] bound to thread [" + Thread.currentThread().getName() + "]"); } Object value = map.remove(key); // remove entire ThreadLocal if empty if (map.isEmpty()) { resources.set(null); } if (logger.isDebugEnabled()) { logger.debug("Removed value [" + value + "] for key [" + key + "] from thread [" + Thread.currentThread().getName() + "]"); } return value; } //------------------------------------------------------------------------- // Management of transaction synchronizations //------------------------------------------------------------------------- /** * Return if transaction synchronization is active for the current thread. * Can be called before register to avoid unnecessary instance creation. * @see #registerSynchronization */ public static boolean isSynchronizationActive() { return (synchronizations.get() != null); } /** * Activate transaction synchronization for the current thread. * Called by a transaction manager on transaction begin. * @throws IllegalStateException if synchronization is already active */ public static void initSynchronization() throws IllegalStateException { if (isSynchronizationActive()) { throw new IllegalStateException("Cannot activate transaction synchronization - already active"); } logger.debug("Initializing transaction synchronization"); synchronizations.set(new LinkedList()); } /** * Register a new transaction synchronization for the current thread. * Typically called by resource management code. *

Note that synchronizations can implemented the Ordered interface. * They will be executed in an order according to their order value (if any). * @throws IllegalStateException if synchronization is not active * @see org.springframework.core.Ordered */ public static void registerSynchronization(TransactionSynchronization synchronization) throws IllegalStateException { if (!isSynchronizationActive()) { throw new IllegalStateException("Transaction synchronization is not active"); } List synchs = (List) synchronizations.get(); synchs.add(synchronization); Collections.sort(synchs, synchronizationComparator); } /** * Return an unmodifiable snapshot list of all registered synchronizations * for the current thread. * @return unmodifiable List of TransactionSynchronization instances * @throws IllegalStateException if synchronization is not active * @see TransactionSynchronization */ public static List getSynchronizations() throws IllegalStateException { if (!isSynchronizationActive()) { throw new IllegalStateException("Transaction synchronization is not active"); } List synchs = (List) synchronizations.get(); return Collections.unmodifiableList(new LinkedList(synchs)); } /** * Deactivate transaction synchronization for the current thread. * Called by transaction manager on transaction cleanup. * @throws IllegalStateException if synchronization is not active */ public static void clearSynchronization() throws IllegalStateException { if (!isSynchronizationActive()) { throw new IllegalStateException("Cannot deactivate transaction synchronization - not active"); } logger.debug("Clearing transaction synchronization"); synchronizations.set(null); } /** * Expose a read-only flag for the current transaction. * Called by transaction manager on transaction begin and on cleanup. * @param readOnly true to mark the current transaction as read-only; * false to reset such a read-only marker * @see org.springframework.transaction.TransactionDefinition#isReadOnly */ public static void setCurrentTransactionReadOnly(boolean readOnly) { currentTransactionReadOnly.set(readOnly ? Boolean.TRUE : null); } /** * Return whether the current transaction is marked as read-only. * To be called by resource management code when preparing a newly * created resource (for example, a Hibernate Session). *

Note that transaction synchronizations receive the read-only flag * as argument for the beforeCommit callback, to be able * to suppress change detection on commit. The present method is meant * to be used for earlier read-only checks, for example to set the * flush mode of a Hibernate Session to FlushMode.NEVER upfront. * @see TransactionSynchronization#beforeCommit(boolean) * @see org.hibernate.Session#flush * @see org.hibernate.Session#setFlushMode * @see org.hibernate.FlushMode#NEVER */ public static boolean isCurrentTransactionReadOnly() { return (currentTransactionReadOnly.get() != null); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy