Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

TransactionalTestExecutionListener uses cached TransactionManager (TM), even after ApplicationContext that created the TM has been marked 'dirty' and closed. [SPR-4638] #9315

Closed
spring-projects-issues opened this issue Mar 27, 2008 · 2 comments
Assignees
Labels
type: bug A general bug
Milestone

Comments

@spring-projects-issues
Copy link
Collaborator

spring-projects-issues commented Mar 27, 2008

Jason Zaugg opened SPR-4638 and commented

TransactionalTestExecutionListener:378

	protected final PlatformTransactionManager getTransactionManager(final TestContext testContext) throws Exception {

		if (this.transactionManager != null) {
			return this.transactionManager;
		}

		// else...

		final TransactionConfigurationAttributes configAttributes = retrieveTransactionConfigurationAttributes(testContext.getTestClass());
		final String transactionManagerName = configAttributes.getTransactionManagerName();

		try {
			this.transactionManager = (PlatformTransactionManager) testContext.getApplicationContext().getBean(
					transactionManagerName, PlatformTransactionManager.class);
		}

I suggest that you do the following instead:

		if (this.transactionManager != null && this.applicationContext = testContext.getApplicationContext()) {
			return this.transactionManager;
		}

		// else...

		final TransactionConfigurationAttributes configAttributes = retrieveTransactionConfigurationAttributes(testContext.getTestClass());
		final String transactionManagerName = configAttributes.getTransactionManagerName();

		try {
			this.transactionManager = (PlatformTransactionManager) testContext.getApplicationContext().getBean(
					transactionManagerName, PlatformTransactionManager.class);
                        this.applicationContext = testContext.getApplicationContext();
		}

I experience this when I am using a custom TestRunner and TestContextManager, who are responsible for injecting Mock objects into the ApplicationWiring if the Test Case contains suitably annotated fields. As a workaround, my custom TestContextManager reflectively sets the TransactionalTestExecutionListener.transactionManager to null after each test.

public final class RaptorTestRunner extends SpringJUnit4ClassRunner {
    public RaptorTestRunner(Class testClass) throws InitializationError {
        super(testClass);
    }

    protected TestContextManager createTestContextManager(Class<?> clazz) {
        TestContextManager contextManager = new TestContextManager(clazz) {
            public TestInstancePreparer preparer;

            public void prepareTestInstance(Object testInstance) throws Exception {
                TestContext testContext = getTestContext();
                ApplicationContext springApplicationContext = testContext.getApplicationContext();
                Map map = springApplicationContext.getBeansOfType(Context.class);
                Context raptorContext = (Context) toArray(map.values().iterator())[0];
                preparer = new TestInstancePreparer(testInstance, testContext, raptorContext);
                preparer.prepareSubject();
            }

            public void afterTestMethod(Object testInstance, Method testMethod, Throwable exception) throws Exception {
                super.afterTestMethod(testInstance, testMethod, exception);
                preparer.discardApplicationContextIfInjectWasUsed();
                forceSpringTransactionTestListenerToUseTheLatestDatasource();
            }

            private void forceSpringTransactionTestListenerToUseTheLatestDatasource() throws NoSuchFieldException, IllegalAccessException {
                List<TestExecutionListener> listeners = getTestExecutionListeners();
                for (TestExecutionListener listener : listeners) {
                    // TODO Raise line 378 of TransactionalTestExecutionListener returns a transaction
                    //      manager that can point to a datasource that has been closed, if the spring application context
                    //      for the first test has been marked as dirty and closed.
                    if (listener instanceof TransactionalTestExecutionListener) {
                        Field field = listener.getClass().getDeclaredField("transactionManager");
                        field.setAccessible(true);
                        field.set(listener, null);
                    }
                }
            }


        };
        return contextManager;
    }
}

Affects: 2.5.1

Issue Links:

@spring-projects-issues
Copy link
Collaborator Author

Juergen Hoeller commented

Thanks for the suggestion! TransactionalTestExecutionListener is currently under heavy revision for other reasons already, so I've sneaked this in right away - although not as ApplicationContext comparison condition but rather as freshly obtaining the transaction manager for every transaction. We cache the parsed TransactionConfigurationAttributes now so the actual transaction manager getBean call is essentially just a hash lookup.

Juergen

@spring-projects-issues
Copy link
Collaborator Author

Sam Brannen commented

Reopened in order to format code.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
type: bug A general bug
Projects
None yet
Development

No branches or pull requests

2 participants