You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
TransactionalTestExecutionListener uses cached TransactionManager (TM), even after ApplicationContext that created the TM has been marked 'dirty' and closed. [SPR-4638]
#9315
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.
publicfinalclassRaptorTestRunnerextendsSpringJUnit4ClassRunner {
publicRaptorTestRunner(ClasstestClass) throwsInitializationError {
super(testClass);
}
protectedTestContextManagercreateTestContextManager(Class<?> clazz) {
TestContextManagercontextManager = newTestContextManager(clazz) {
publicTestInstancePreparerpreparer;
publicvoidprepareTestInstance(ObjecttestInstance) throwsException {
TestContexttestContext = getTestContext();
ApplicationContextspringApplicationContext = testContext.getApplicationContext();
Mapmap = springApplicationContext.getBeansOfType(Context.class);
ContextraptorContext = (Context) toArray(map.values().iterator())[0];
preparer = newTestInstancePreparer(testInstance, testContext, raptorContext);
preparer.prepareSubject();
}
publicvoidafterTestMethod(ObjecttestInstance, MethodtestMethod, Throwableexception) throwsException {
super.afterTestMethod(testInstance, testMethod, exception);
preparer.discardApplicationContextIfInjectWasUsed();
forceSpringTransactionTestListenerToUseTheLatestDatasource();
}
privatevoidforceSpringTransactionTestListenerToUseTheLatestDatasource() throwsNoSuchFieldException, IllegalAccessException {
List<TestExecutionListener> listeners = getTestExecutionListeners();
for (TestExecutionListenerlistener : 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 (listenerinstanceofTransactionalTestExecutionListener) {
Fieldfield = listener.getClass().getDeclaredField("transactionManager");
field.setAccessible(true);
field.set(listener, null);
}
}
}
};
returncontextManager;
}
}
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.
Jason Zaugg opened SPR-4638 and commented
TransactionalTestExecutionListener:378
I suggest that you do the following instead:
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.
Affects: 2.5.1
Issue Links:
The text was updated successfully, but these errors were encountered: