@Transactional qualifiers work in unit tests as well (SPR-6892)
This commit is contained in:
parent
23cb161fb3
commit
bb75662a7e
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2009 the original author or authors.
|
* Copyright 2002-2010 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -29,6 +29,7 @@ import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
|
|
||||||
import org.springframework.beans.BeansException;
|
import org.springframework.beans.BeansException;
|
||||||
|
import org.springframework.beans.factory.BeanFactory;
|
||||||
import org.springframework.core.annotation.AnnotationUtils;
|
import org.springframework.core.annotation.AnnotationUtils;
|
||||||
import org.springframework.test.annotation.NotTransactional;
|
import org.springframework.test.annotation.NotTransactional;
|
||||||
import org.springframework.test.annotation.Rollback;
|
import org.springframework.test.annotation.Rollback;
|
||||||
|
|
@ -40,10 +41,12 @@ import org.springframework.transaction.TransactionException;
|
||||||
import org.springframework.transaction.TransactionStatus;
|
import org.springframework.transaction.TransactionStatus;
|
||||||
import org.springframework.transaction.annotation.AnnotationTransactionAttributeSource;
|
import org.springframework.transaction.annotation.AnnotationTransactionAttributeSource;
|
||||||
import org.springframework.transaction.interceptor.DelegatingTransactionAttribute;
|
import org.springframework.transaction.interceptor.DelegatingTransactionAttribute;
|
||||||
|
import org.springframework.transaction.interceptor.TransactionAspectUtils;
|
||||||
import org.springframework.transaction.interceptor.TransactionAttribute;
|
import org.springframework.transaction.interceptor.TransactionAttribute;
|
||||||
import org.springframework.transaction.interceptor.TransactionAttributeSource;
|
import org.springframework.transaction.interceptor.TransactionAttributeSource;
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.util.Assert;
|
||||||
import org.springframework.util.ReflectionUtils;
|
import org.springframework.util.ReflectionUtils;
|
||||||
|
import org.springframework.util.StringUtils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>
|
* <p>
|
||||||
|
|
@ -94,14 +97,13 @@ public class TransactionalTestExecutionListener extends AbstractTestExecutionLis
|
||||||
|
|
||||||
protected final TransactionAttributeSource attributeSource = new AnnotationTransactionAttributeSource();
|
protected final TransactionAttributeSource attributeSource = new AnnotationTransactionAttributeSource();
|
||||||
|
|
||||||
private TransactionConfigurationAttributes configAttributes;
|
private TransactionConfigurationAttributes configurationAttributes;
|
||||||
|
|
||||||
private volatile int transactionsStarted = 0;
|
private volatile int transactionsStarted = 0;
|
||||||
|
|
||||||
private final Map<Method, TransactionContext> transactionContextCache =
|
private final Map<Method, TransactionContext> transactionContextCache =
|
||||||
Collections.synchronizedMap(new IdentityHashMap<Method, TransactionContext>());
|
Collections.synchronizedMap(new IdentityHashMap<Method, TransactionContext>());
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If the test method of the supplied {@link TestContext test context} is
|
* If the test method of the supplied {@link TestContext test context} is
|
||||||
* configured to run within a transaction, this method will run
|
* configured to run within a transaction, this method will run
|
||||||
|
|
@ -144,8 +146,18 @@ public class TransactionalTestExecutionListener extends AbstractTestExecutionLis
|
||||||
logger.debug("Explicit transaction definition [" + transactionDefinition +
|
logger.debug("Explicit transaction definition [" + transactionDefinition +
|
||||||
"] found for test context [" + testContext + "]");
|
"] found for test context [" + testContext + "]");
|
||||||
}
|
}
|
||||||
TransactionContext txContext =
|
String qualifier = transactionAttribute.getQualifier();
|
||||||
new TransactionContext(getTransactionManager(testContext), transactionDefinition);
|
PlatformTransactionManager tm;
|
||||||
|
if (StringUtils.hasLength(qualifier)) {
|
||||||
|
// Use autowire-capable factory in order to support extended qualifier matching
|
||||||
|
// (only exposed on the internal BeanFactory, not on the ApplicationContext).
|
||||||
|
BeanFactory bf = testContext.getApplicationContext().getAutowireCapableBeanFactory();
|
||||||
|
tm = TransactionAspectUtils.getTransactionManager(bf, qualifier);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
tm = getTransactionManager(testContext);
|
||||||
|
}
|
||||||
|
TransactionContext txContext = new TransactionContext(tm, transactionDefinition);
|
||||||
runBeforeTransactionMethods(testContext);
|
runBeforeTransactionMethods(testContext);
|
||||||
startNewTransaction(testContext, txContext);
|
startNewTransaction(testContext, txContext);
|
||||||
this.transactionContextCache.put(testMethod, txContext);
|
this.transactionContextCache.put(testMethod, txContext);
|
||||||
|
|
@ -291,18 +303,14 @@ public class TransactionalTestExecutionListener extends AbstractTestExecutionLis
|
||||||
* @throws BeansException if an error occurs while retrieving the transaction manager
|
* @throws BeansException if an error occurs while retrieving the transaction manager
|
||||||
*/
|
*/
|
||||||
protected final PlatformTransactionManager getTransactionManager(TestContext testContext) {
|
protected final PlatformTransactionManager getTransactionManager(TestContext testContext) {
|
||||||
if (this.configAttributes == null) {
|
String tmName = retrieveConfigurationAttributes(testContext).getTransactionManagerName();
|
||||||
this.configAttributes = retrieveTransactionConfigurationAttributes(testContext.getTestClass());
|
|
||||||
}
|
|
||||||
String transactionManagerName = this.configAttributes.getTransactionManagerName();
|
|
||||||
try {
|
try {
|
||||||
return (PlatformTransactionManager) testContext.getApplicationContext().getBean(
|
return testContext.getApplicationContext().getBean(tmName, PlatformTransactionManager.class);
|
||||||
transactionManagerName, PlatformTransactionManager.class);
|
|
||||||
}
|
}
|
||||||
catch (BeansException ex) {
|
catch (BeansException ex) {
|
||||||
if (logger.isWarnEnabled()) {
|
if (logger.isWarnEnabled()) {
|
||||||
logger.warn("Caught exception while retrieving transaction manager with bean name [" +
|
logger.warn("Caught exception while retrieving transaction manager with bean name [" +
|
||||||
transactionManagerName + "] for test context [" + testContext + "]", ex);
|
tmName + "] for test context [" + testContext + "]", ex);
|
||||||
}
|
}
|
||||||
throw ex;
|
throw ex;
|
||||||
}
|
}
|
||||||
|
|
@ -317,7 +325,7 @@ public class TransactionalTestExecutionListener extends AbstractTestExecutionLis
|
||||||
* @throws Exception if an error occurs while determining the default rollback flag
|
* @throws Exception if an error occurs while determining the default rollback flag
|
||||||
*/
|
*/
|
||||||
protected final boolean isDefaultRollback(TestContext testContext) throws Exception {
|
protected final boolean isDefaultRollback(TestContext testContext) throws Exception {
|
||||||
return retrieveTransactionConfigurationAttributes(testContext.getTestClass()).isDefaultRollback();
|
return retrieveConfigurationAttributes(testContext).isDefaultRollback();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -439,7 +447,6 @@ public class TransactionalTestExecutionListener extends AbstractTestExecutionLis
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>
|
|
||||||
* Retrieves the {@link TransactionConfigurationAttributes} for the
|
* Retrieves the {@link TransactionConfigurationAttributes} for the
|
||||||
* specified {@link Class class} which may optionally declare or inherit a
|
* specified {@link Class class} which may optionally declare or inherit a
|
||||||
* {@link TransactionConfiguration @TransactionConfiguration}. If a
|
* {@link TransactionConfiguration @TransactionConfiguration}. If a
|
||||||
|
|
@ -450,33 +457,36 @@ public class TransactionalTestExecutionListener extends AbstractTestExecutionLis
|
||||||
* the configuration attributes should be retrieved
|
* the configuration attributes should be retrieved
|
||||||
* @return a new TransactionConfigurationAttributes instance
|
* @return a new TransactionConfigurationAttributes instance
|
||||||
*/
|
*/
|
||||||
private TransactionConfigurationAttributes retrieveTransactionConfigurationAttributes(Class<?> clazz) {
|
private TransactionConfigurationAttributes retrieveConfigurationAttributes(TestContext testContext) {
|
||||||
Class<TransactionConfiguration> annotationType = TransactionConfiguration.class;
|
if (this.configurationAttributes == null) {
|
||||||
TransactionConfiguration config = clazz.getAnnotation(annotationType);
|
Class<?> clazz = testContext.getTestClass();
|
||||||
if (logger.isDebugEnabled()) {
|
Class<TransactionConfiguration> annotationType = TransactionConfiguration.class;
|
||||||
logger.debug("Retrieved @TransactionConfiguration [" + config + "] for test class [" + clazz + "]");
|
TransactionConfiguration config = clazz.getAnnotation(annotationType);
|
||||||
}
|
if (logger.isDebugEnabled()) {
|
||||||
|
logger.debug("Retrieved @TransactionConfiguration [" + config + "] for test class [" + clazz + "]");
|
||||||
|
}
|
||||||
|
|
||||||
String transactionManagerName;
|
String transactionManagerName;
|
||||||
boolean defaultRollback;
|
boolean defaultRollback;
|
||||||
if (config != null) {
|
if (config != null) {
|
||||||
transactionManagerName = config.transactionManager();
|
transactionManagerName = config.transactionManager();
|
||||||
defaultRollback = config.defaultRollback();
|
defaultRollback = config.defaultRollback();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
transactionManagerName = (String) AnnotationUtils.getDefaultValue(annotationType, "transactionManager");
|
transactionManagerName = (String) AnnotationUtils.getDefaultValue(annotationType, "transactionManager");
|
||||||
defaultRollback = (Boolean) AnnotationUtils.getDefaultValue(annotationType, "defaultRollback");
|
defaultRollback = (Boolean) AnnotationUtils.getDefaultValue(annotationType, "defaultRollback");
|
||||||
}
|
}
|
||||||
|
|
||||||
TransactionConfigurationAttributes configAttributes =
|
TransactionConfigurationAttributes configAttributes =
|
||||||
new TransactionConfigurationAttributes(transactionManagerName, defaultRollback);
|
new TransactionConfigurationAttributes(transactionManagerName, defaultRollback);
|
||||||
if (logger.isDebugEnabled()) {
|
if (logger.isDebugEnabled()) {
|
||||||
logger.debug("Retrieved TransactionConfigurationAttributes [" + configAttributes + "] for class [" + clazz + "]");
|
logger.debug("Retrieved TransactionConfigurationAttributes [" + configAttributes + "] for class [" + clazz + "]");
|
||||||
|
}
|
||||||
|
this.configurationAttributes = configAttributes;
|
||||||
}
|
}
|
||||||
return configAttributes;
|
return this.configurationAttributes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Internal context holder for a specific test method.
|
* Internal context holder for a specific test method.
|
||||||
*/
|
*/
|
||||||
|
|
|
||||||
|
|
@ -16,13 +16,11 @@
|
||||||
|
|
||||||
package org.springframework.test.context.junit4;
|
package org.springframework.test.context.junit4;
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
|
||||||
import static org.springframework.test.transaction.TransactionTestUtils.assertInTransaction;
|
|
||||||
|
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
import javax.sql.DataSource;
|
import javax.sql.DataSource;
|
||||||
|
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
|
import static org.junit.Assert.*;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.junit.runner.RunWith;
|
import org.junit.runner.RunWith;
|
||||||
|
|
@ -34,6 +32,8 @@ import org.springframework.test.context.TestExecutionListeners;
|
||||||
import org.springframework.test.context.support.DependencyInjectionTestExecutionListener;
|
import org.springframework.test.context.support.DependencyInjectionTestExecutionListener;
|
||||||
import org.springframework.test.context.support.DirtiesContextTestExecutionListener;
|
import org.springframework.test.context.support.DirtiesContextTestExecutionListener;
|
||||||
import org.springframework.test.context.transaction.TransactionalTestExecutionListener;
|
import org.springframework.test.context.transaction.TransactionalTestExecutionListener;
|
||||||
|
import org.springframework.test.context.transaction.TransactionConfiguration;
|
||||||
|
import static org.springframework.test.transaction.TransactionTestUtils.*;
|
||||||
import org.springframework.transaction.annotation.Transactional;
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -85,7 +85,7 @@ public class MethodLevelTransactionalSpringRunnerTests extends AbstractTransacti
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@Transactional
|
@Transactional("transactionManager2")
|
||||||
public void modifyTestDataWithinTransaction() {
|
public void modifyTestDataWithinTransaction() {
|
||||||
assertInTransaction(true);
|
assertInTransaction(true);
|
||||||
assertEquals("Deleting bob", 1, deletePerson(simpleJdbcTemplate, BOB));
|
assertEquals("Deleting bob", 1, deletePerson(simpleJdbcTemplate, BOB));
|
||||||
|
|
@ -109,7 +109,7 @@ public class MethodLevelTransactionalSpringRunnerTests extends AbstractTransacti
|
||||||
public static class DatabaseSetup {
|
public static class DatabaseSetup {
|
||||||
|
|
||||||
@Resource
|
@Resource
|
||||||
public void setDataSource(DataSource dataSource) {
|
public void setDataSource2(DataSource dataSource) {
|
||||||
simpleJdbcTemplate = new SimpleJdbcTemplate(dataSource);
|
simpleJdbcTemplate = new SimpleJdbcTemplate(dataSource);
|
||||||
createPersonTable(simpleJdbcTemplate);
|
createPersonTable(simpleJdbcTemplate);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -4,12 +4,16 @@
|
||||||
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">
|
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">
|
||||||
|
|
||||||
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"
|
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource"
|
||||||
p:driverClassName="org.hsqldb.jdbcDriver" p:url="jdbc:hsqldb:mem:transactional_tests" p:username="sa" p:password="" />
|
p:driverClassName="org.hsqldb.jdbcDriver" p:url="jdbc:hsqldb:mem:transactional_tests" p:username="sa" p:password=""/>
|
||||||
|
|
||||||
<bean id="dataSource2" class="org.springframework.jdbc.datasource.DriverManagerDataSource"
|
<bean id="dataSource2" class="org.springframework.jdbc.datasource.DriverManagerDataSource"
|
||||||
p:driverClassName="org.hsqldb.jdbcDriver" p:url="jdbc:hsqldb:mem:transactional_tests" p:username="sa" p:password="" />
|
p:driverClassName="org.hsqldb.jdbcDriver" p:url="jdbc:hsqldb:mem:transactional_tests" p:username="sa" p:password=""/>
|
||||||
|
|
||||||
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"
|
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"
|
||||||
p:data-source-ref="dataSource" />
|
p:dataSource-ref="dataSource" p:transactionSynchronizationName="SYNCHRONIZATION_NEVER"/>
|
||||||
|
|
||||||
|
<bean id="transactionManager2" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"
|
||||||
|
p:dataSource-ref="dataSource2">
|
||||||
|
</bean>
|
||||||
|
|
||||||
</beans>
|
</beans>
|
||||||
|
|
|
||||||
|
|
@ -17,7 +17,6 @@
|
||||||
package org.springframework.transaction.interceptor;
|
package org.springframework.transaction.interceptor;
|
||||||
|
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
|
|
@ -28,18 +27,13 @@ import org.springframework.beans.factory.BeanFactoryAware;
|
||||||
import org.springframework.beans.factory.BeanFactoryUtils;
|
import org.springframework.beans.factory.BeanFactoryUtils;
|
||||||
import org.springframework.beans.factory.InitializingBean;
|
import org.springframework.beans.factory.InitializingBean;
|
||||||
import org.springframework.beans.factory.ListableBeanFactory;
|
import org.springframework.beans.factory.ListableBeanFactory;
|
||||||
import org.springframework.beans.factory.annotation.Qualifier;
|
|
||||||
import org.springframework.beans.factory.config.BeanDefinition;
|
|
||||||
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
|
|
||||||
import org.springframework.beans.factory.support.AbstractBeanDefinition;
|
|
||||||
import org.springframework.beans.factory.support.AutowireCandidateQualifier;
|
|
||||||
import org.springframework.core.NamedThreadLocal;
|
import org.springframework.core.NamedThreadLocal;
|
||||||
import org.springframework.transaction.NoTransactionException;
|
import org.springframework.transaction.NoTransactionException;
|
||||||
import org.springframework.transaction.PlatformTransactionManager;
|
import org.springframework.transaction.PlatformTransactionManager;
|
||||||
import org.springframework.transaction.TransactionStatus;
|
import org.springframework.transaction.TransactionStatus;
|
||||||
import org.springframework.transaction.TransactionSystemException;
|
import org.springframework.transaction.TransactionSystemException;
|
||||||
|
import org.springframework.transaction.interceptor.TransactionAspectUtils;
|
||||||
import org.springframework.util.ClassUtils;
|
import org.springframework.util.ClassUtils;
|
||||||
import org.springframework.util.ObjectUtils;
|
|
||||||
import org.springframework.util.StringUtils;
|
import org.springframework.util.StringUtils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -249,38 +243,7 @@ public abstract class TransactionAspectSupport implements BeanFactoryAware, Init
|
||||||
}
|
}
|
||||||
String qualifier = txAttr.getQualifier();
|
String qualifier = txAttr.getQualifier();
|
||||||
if (StringUtils.hasLength(qualifier)) {
|
if (StringUtils.hasLength(qualifier)) {
|
||||||
if (!(this.beanFactory instanceof ConfigurableListableBeanFactory)) {
|
return TransactionAspectUtils.getTransactionManager(this.beanFactory, qualifier);
|
||||||
throw new IllegalStateException("BeanFactory required to be a ConfigurableListableBeanFactory " +
|
|
||||||
"for resolution of qualifier '" + qualifier + "': " + this.beanFactory.getClass());
|
|
||||||
}
|
|
||||||
ConfigurableListableBeanFactory bf = (ConfigurableListableBeanFactory) this.beanFactory;
|
|
||||||
Map<String, PlatformTransactionManager> tms =
|
|
||||||
BeanFactoryUtils.beansOfTypeIncludingAncestors(bf, PlatformTransactionManager.class);
|
|
||||||
PlatformTransactionManager chosen = null;
|
|
||||||
for (String beanName : tms.keySet()) {
|
|
||||||
if (bf.containsBeanDefinition(beanName)) {
|
|
||||||
BeanDefinition bd = bf.getBeanDefinition(beanName);
|
|
||||||
if (bd instanceof AbstractBeanDefinition) {
|
|
||||||
AbstractBeanDefinition abd = (AbstractBeanDefinition) bd;
|
|
||||||
AutowireCandidateQualifier candidate = abd.getQualifier(Qualifier.class.getName());
|
|
||||||
if ((candidate != null && qualifier.equals(candidate.getAttribute(AutowireCandidateQualifier.VALUE_KEY))) ||
|
|
||||||
qualifier.equals(beanName) || ObjectUtils.containsElement(bf.getAliases(beanName), qualifier)) {
|
|
||||||
if (chosen != null) {
|
|
||||||
throw new IllegalStateException("No unique PlatformTransactionManager bean found " +
|
|
||||||
"for qualifier '" + qualifier + "'");
|
|
||||||
}
|
|
||||||
chosen = tms.get(beanName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (chosen != null) {
|
|
||||||
return chosen;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
throw new IllegalStateException(
|
|
||||||
"No matching PlatformTransactionManager bean found for qualifier '" + qualifier + "'");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (this.transactionManagerBeanName != null) {
|
else if (this.transactionManagerBeanName != null) {
|
||||||
return this.beanFactory.getBean(this.transactionManagerBeanName, PlatformTransactionManager.class);
|
return this.beanFactory.getBean(this.transactionManagerBeanName, PlatformTransactionManager.class);
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,102 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2010 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.interceptor;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.springframework.beans.factory.BeanFactory;
|
||||||
|
import org.springframework.beans.factory.BeanFactoryUtils;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
import org.springframework.beans.factory.config.BeanDefinition;
|
||||||
|
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
|
||||||
|
import org.springframework.beans.factory.support.AbstractBeanDefinition;
|
||||||
|
import org.springframework.beans.factory.support.AutowireCandidateQualifier;
|
||||||
|
import org.springframework.transaction.PlatformTransactionManager;
|
||||||
|
import org.springframework.util.ObjectUtils;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Utility methods for obtaining a PlatformTransactionManager by
|
||||||
|
* {@link TransactionAttribute#getQualifier() qualifier value}.
|
||||||
|
*
|
||||||
|
* @author Juergen Hoeller
|
||||||
|
* @since 3.0.2
|
||||||
|
*/
|
||||||
|
public abstract class TransactionAspectUtils {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Obtain a PlatformTransactionManager from the given BeanFactory,
|
||||||
|
* matching the given qualifier.
|
||||||
|
* @param beanFactory the BeanFactory to get the PlatformTransactionManager bean from
|
||||||
|
* @param qualifier the qualifier for selecting between multiple PlatformTransactionManager matches
|
||||||
|
* @return the chosen PlatformTransactionManager (never <code>null</code>)
|
||||||
|
* @throws IllegalStateException if no matching PlatformTransactionManager bean found
|
||||||
|
*/
|
||||||
|
public static PlatformTransactionManager getTransactionManager(BeanFactory beanFactory, String qualifier) {
|
||||||
|
if (beanFactory instanceof ConfigurableListableBeanFactory) {
|
||||||
|
// Full qualifier matching supported.
|
||||||
|
return getTransactionManager((ConfigurableListableBeanFactory) beanFactory, qualifier);
|
||||||
|
}
|
||||||
|
else if (beanFactory.containsBean(qualifier)) {
|
||||||
|
// Fallback: PlatformTransactionManager at least found by bean name.
|
||||||
|
return beanFactory.getBean(qualifier, PlatformTransactionManager.class);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new IllegalStateException("No matching PlatformTransactionManager bean found for bean name '" +
|
||||||
|
qualifier + "'! (Note: Qualifier matching not supported because given BeanFactory does not " +
|
||||||
|
"implement ConfigurableListableBeanFactory.)");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Obtain a PlatformTransactionManager from the given BeanFactory,
|
||||||
|
* matching the given qualifier.
|
||||||
|
* @param beanFactory the BeanFactory to get the PlatformTransactionManager bean from
|
||||||
|
* @param qualifier the qualifier for selecting between multiple PlatformTransactionManager matches
|
||||||
|
* @return the chosen PlatformTransactionManager (never <code>null</code>)
|
||||||
|
* @throws IllegalStateException if no matching PlatformTransactionManager bean found
|
||||||
|
*/
|
||||||
|
public static PlatformTransactionManager getTransactionManager(ConfigurableListableBeanFactory bf, String qualifier) {
|
||||||
|
Map<String, PlatformTransactionManager> tms =
|
||||||
|
BeanFactoryUtils.beansOfTypeIncludingAncestors(bf, PlatformTransactionManager.class);
|
||||||
|
PlatformTransactionManager chosen = null;
|
||||||
|
for (String beanName : tms.keySet()) {
|
||||||
|
if (bf.containsBeanDefinition(beanName)) {
|
||||||
|
BeanDefinition bd = bf.getBeanDefinition(beanName);
|
||||||
|
if (bd instanceof AbstractBeanDefinition) {
|
||||||
|
AbstractBeanDefinition abd = (AbstractBeanDefinition) bd;
|
||||||
|
AutowireCandidateQualifier candidate = abd.getQualifier(Qualifier.class.getName());
|
||||||
|
if ((candidate != null && qualifier.equals(candidate.getAttribute(AutowireCandidateQualifier.VALUE_KEY))) ||
|
||||||
|
qualifier.equals(beanName) || ObjectUtils.containsElement(bf.getAliases(beanName), qualifier)) {
|
||||||
|
if (chosen != null) {
|
||||||
|
throw new IllegalStateException("No unique PlatformTransactionManager bean found " +
|
||||||
|
"for qualifier '" + qualifier + "'");
|
||||||
|
}
|
||||||
|
chosen = tms.get(beanName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (chosen != null) {
|
||||||
|
return chosen;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new IllegalStateException("No matching PlatformTransactionManager bean found for qualifier '" +
|
||||||
|
qualifier + "' - neither qualifier match nor bean name match!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue