io.mats3.spring.test.TestSpringMatsFactoryProvider Maven / Gradle / Ivy
package io.mats3.spring.test;
import java.sql.Connection;
import java.util.concurrent.atomic.AtomicInteger;
import javax.jms.ConnectionFactory;
import javax.sql.DataSource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.transaction.PlatformTransactionManager;
import io.mats3.MatsFactory;
import io.mats3.MatsFactory.FactoryConfig;
import io.mats3.MatsFactory.MatsFactoryWrapper;
import io.mats3.impl.jms.JmsMatsFactory;
import io.mats3.impl.jms.JmsMatsJmsSessionHandler;
import io.mats3.impl.jms.JmsMatsJmsSessionHandler_Pooling;
import io.mats3.impl.jms.JmsMatsTransactionManager;
import io.mats3.impl.jms.JmsMatsTransactionManager_Jms;
import io.mats3.localinspect.LocalStatsMatsInterceptor;
import io.mats3.serial.MatsSerializer;
import io.mats3.spring.jms.factories.SpringJmsMatsFactoryWrapper;
import io.mats3.spring.jms.tx.JmsMatsTransactionManager_JmsAndSpringManagedSqlTx;
import io.mats3.test.broker.MatsTestBroker;
/**
* A testing-oriented {@link MatsFactory}-provider which utilizes the {@link MatsTestBroker} for the produced
* MatsFactory to connect to - this is for the scenarios where you do NOT have your test load the entire
* application's Spring configuration, but instead "piece together" the relevant Spring
* {@literal @Components}
containing test-relevant Mats endpoints and other beans from your application
* along with test-specific mocked-out endpoints: You will then probably not have the MatsFactory present in the Spring
* context. You can then use this class to get a "quick and dirty" MatsFactory (with an in-vm MQ Broker backing it) on
* which your application's endpoints, and mocked-out endpoints in the test, can run.
*
* If your requirements aren't all that exotic, you may get this indirectly invoked by using the
* {@literal @}{@link MatsTestContext} annotation directly on the Test-class. If your requirements are slightly more
* involved, check out the {@link MatsTestInfrastructureConfiguration} and {@link MatsTestInfrastructureDbConfiguration}
* Spring Configuration classes (loaded in a test with {@literal @}{@link ContextConfiguration}), which employs the
* methods in this class.
*
* @see MatsTestContext
* @see MatsTestInfrastructureConfiguration
* @see MatsTestInfrastructureDbConfiguration
* @author Endre Stølsvik 2019-06-17 21:26 - http://stolsvik.com/, [email protected]
*/
public class TestSpringMatsFactoryProvider {
private static final Logger log = LoggerFactory.getLogger(TestSpringMatsFactoryProvider.class);
private static final String LOG_PREFIX = "#SPRINGJMATS(test)# ";
private static final AtomicInteger _sequence = new AtomicInteger(0);
/**
* If you need a {@link MatsFactory} employing Spring's DataSourceTransactionManager (which you probably do in a
* Spring environment utilizing SQL), this is your factory method. If you need to make y
*
* Usage: In the test, make a @Bean-annotated method which returns the result of this method - or employ the
* convenience {@link MatsTestInfrastructureDbConfiguration}.
*
* @param concurrency
* The concurrency of the created {@link MatsFactory}, set with
* {@link FactoryConfig#setConcurrency(int)}.
* @param sqlDataSource
* the SQL DataSource which to stash into a Spring {@link DataSourceTransactionManager}, and from which
* SQL {@link Connection}s are fetched from, using {@link DataSource#getConnection()}. It is assumed that
* if username and password is needed, you have configured that on the DataSource.
* @param matsSerializer
* the MatsSerializer to use. If you employ the standard, use MatsSerializerJson.create()
.
*
* @return the produced {@link MatsFactory}
*/
public static MatsFactory createSpringDataSourceTxTestMatsFactory(int concurrency, DataSource sqlDataSource,
MatsSerializer> matsSerializer) {
// JMS + Spring's DataSourceTransactionManager-based MatsTransactionManager
JmsMatsTransactionManager springSqlTxMgr = JmsMatsTransactionManager_JmsAndSpringManagedSqlTx.create(
sqlDataSource);
return getMatsFactoryStopLocalVmBrokerWrapper(concurrency, matsSerializer, springSqlTxMgr);
}
/**
* Convenience variant of {@link #createSpringDataSourceTxTestMatsFactory(int, DataSource, MatsSerializer)} where
* concurrency is 1, which should be adequate for most testing - unless you explicitly want to test concurrency.
*
* @param sqlDataSource
* the SQL DataSource which to stash into a Spring {@link DataSourceTransactionManager}, and from which
* SQL {@link Connection}s are fetched from, using {@link DataSource#getConnection()}. It is assumed that
* if username and password is needed, you have configured that on the DataSource.
* @param matsSerializer
* the MatsSerializer to use. If you employ the standard, use MatsSerializerJson.create()
.
*
* @return the produced {@link MatsFactory}
*/
public static MatsFactory createSpringDataSourceTxTestMatsFactory(DataSource sqlDataSource,
MatsSerializer> matsSerializer) {
return createSpringDataSourceTxTestMatsFactory(1, sqlDataSource, matsSerializer);
}
/**
* If you need a {@link MatsFactory}, but you need to make your own {@link PlatformTransactionManager}, this is your
* factory method - please note that the contained {@link DataSource} should have been wrapped using the static
* method {@link JmsMatsTransactionManager_JmsAndSpringManagedSqlTx#wrapLazyConnectionDatasource(DataSource)}.
*
* Usage: In the test, make a @Bean-annotated method which returns the result of this method - or employ the
* convenience {@link MatsTestInfrastructureDbConfiguration}.
*
* @param concurrency
* The concurrency of the created {@link MatsFactory}, set with
* {@link FactoryConfig#setConcurrency(int)}.
* @param platformTransactionManager
* the {@link PlatformTransactionManager} that the SpringJmsMats transaction manager should employ. From
* this, the {@link DataSource} is gotten using refelction to find a method 'getDataSource()'. please
* note that the contained {@link DataSource} should have been wrapped using the static * method
* {@link JmsMatsTransactionManager_JmsAndSpringManagedSqlTx#wrapLazyConnectionDatasource(DataSource)}.
* It is assumed that if username and password is needed, you have configured that on the DataSource.
* @param matsSerializer
* the MatsSerializer to use. If you employ the standard, use MatsSerializerJson.create()
.
*
* @return the produced {@link MatsFactory}
*/
public static MatsFactory createSpringDataSourceTxTestMatsFactory(int concurrency,
PlatformTransactionManager platformTransactionManager, MatsSerializer> matsSerializer) {
// JMS + Spring's DataSourceTransactionManager-based MatsTransactionManager
JmsMatsTransactionManager springSqlTxMgr = JmsMatsTransactionManager_JmsAndSpringManagedSqlTx.create(
platformTransactionManager);
return getMatsFactoryStopLocalVmBrokerWrapper(concurrency, matsSerializer, springSqlTxMgr);
}
/**
* Convenience variant of
* {@link #createSpringDataSourceTxTestMatsFactory(int, PlatformTransactionManager, MatsSerializer)} where
* concurrency is 2, which should be adequate for most testing - unless you explicitly want to test concurrency.
*
* @param platformTransactionManager
* the {@link PlatformTransactionManager} that the SpringJmsMats transaction manager should employ. From
* this, the {@link DataSource} is gotten using reflection to find a method 'getDataSource()'. please
* note that the contained {@link DataSource} should have been wrapped using the static method
* {@link JmsMatsTransactionManager_JmsAndSpringManagedSqlTx#wrapLazyConnectionDatasource(DataSource)}.
* It is assumed that if username and password is needed, you have configured that on the DataSource.
* @param matsSerializer
* the MatsSerializer to use. If you employ the standard, use MatsSerializerJson.create()
.
*
* @return the produced {@link MatsFactory}
*/
public static MatsFactory createSpringDataSourceTxTestMatsFactory(
PlatformTransactionManager platformTransactionManager, MatsSerializer> matsSerializer) {
return createSpringDataSourceTxTestMatsFactory(2, platformTransactionManager, matsSerializer);
}
/**
* If you need a {@link MatsFactory} that only handles the JMS transactions, this is your factory method - but if
* you DO make any database calls within any Mats endpoint lambda, you will now have no or poor transactional
* demarcation, use {@link #createSpringDataSourceTxTestMatsFactory(int, DataSource, MatsSerializer)
* createSpringDataSourceTxTestMatsFactory(..)} instead.
*
* Usage: In the test, make a @Bean-annotated method which returns the result of this method - or employ the
* convenience {@link MatsTestInfrastructureConfiguration}.
*
* @param concurrency
* The concurrency of the created {@link MatsFactory}, set with
* {@link FactoryConfig#setConcurrency(int)}.
* @param matsSerializer
* the MatsSerializer to use. If you employ the standard, use MatsSerializerJson.create()
.
*
* @return the produced {@link MatsFactory}
*/
public static MatsFactory createJmsTxOnlyTestMatsFactory(int concurrency,
MatsSerializer> matsSerializer) {
// JMS only MatsTransactionManager
JmsMatsTransactionManager jmsOnlyTxMgr = JmsMatsTransactionManager_Jms.create();
return getMatsFactoryStopLocalVmBrokerWrapper(concurrency, matsSerializer, jmsOnlyTxMgr);
}
/**
* Convenience variant of {@link #createJmsTxOnlyTestMatsFactory(int, MatsSerializer)} where concurrency is 2, which
* should be adequate for most testing - unless you explicitly want to test concurrency.
*
* @param matsSerializer
* the MatsSerializer to use. If you employ the standard, use MatsSerializerJson.create()
.
*
* @return the produced {@link MatsFactory}
*/
public static MatsFactory createJmsTxOnlyTestMatsFactory(MatsSerializer> matsSerializer) {
return createJmsTxOnlyTestMatsFactory(2, matsSerializer);
}
private static SpringJmsMatsFactoryWrapper getMatsFactoryStopLocalVmBrokerWrapper(int concurrency,
MatsSerializer> matsSerializer, JmsMatsTransactionManager springSqlTxMgr) {
// Naming broker as calling class, performing replacement of illegal chars according to ActiveMQ rules.
String appName = getAppNamePrefix().replaceAll("[^a-zA-Z0-9._\\-:]", ".")
+ "_" + _sequence.getAndIncrement();
MatsTestBroker inVmActiveMq = MatsTestBroker.create();
ConnectionFactory jmsConnectionFactory = inVmActiveMq.getConnectionFactory();
// :: Create the JMS and Spring DataSourceTransactionManager-backed JMS MatsFactory.
// JmsSessionHandler (pooler)
JmsMatsJmsSessionHandler jmsSessionHandler = JmsMatsJmsSessionHandler_Pooling.create(jmsConnectionFactory);
// The MatsFactory itself, supplying the JmsSessionHandler and MatsTransactionManager.
JmsMatsFactory> matsFactory = JmsMatsFactory
.createMatsFactory(appName, "#testing#", jmsSessionHandler, springSqlTxMgr, matsSerializer);
// Set concurrency.
matsFactory.getFactoryConfig().setConcurrency(concurrency);
// TEMPORARY? Install the LocalStatsMatsInterceptor, to get it tested.
LocalStatsMatsInterceptor.install(matsFactory);
// Now wrap this in a wrapper that will close the LocalVM ActiveMQ broker upon matsFactory.stop().
MatsFactoryStopLocalVmBrokerWrapper matsFactoryStopLocalVmBrokerWrapper = new MatsFactoryStopLocalVmBrokerWrapper(
matsFactory, inVmActiveMq);
// And then finally wrap this in the Spring wrapper
return new SpringJmsMatsFactoryWrapper(inVmActiveMq.getConnectionFactory(),
matsFactoryStopLocalVmBrokerWrapper);
}
private static class MatsFactoryStopLocalVmBrokerWrapper extends MatsFactoryWrapper {
private final MatsTestBroker _matsTestBroker;
public MatsFactoryStopLocalVmBrokerWrapper(JmsMatsFactory> targetMatsFactory,
MatsTestBroker matsTestBroker) {
super(targetMatsFactory);
_matsTestBroker = matsTestBroker;
}
@Override
public boolean stop(int gracefulShutdownMillis) {
log.info(LOG_PREFIX + "Stopping test JmsMatsFactory.");
boolean stopped = super.stop(gracefulShutdownMillis);
log.info(LOG_PREFIX + "Stopping MatsTestBroker (stopping any in-vm broker).");
_matsTestBroker.close();
return stopped;
}
}
/**
* from Stackoverflow - Denys Séguret.
*/
private static String getAppNamePrefix() {
StackTraceElement[] stElements = Thread.currentThread().getStackTrace();
for (int i = 1; i < stElements.length; i++) {
StackTraceElement ste = stElements[i];
if (!ste.getClassName().equals(TestSpringMatsFactoryProvider.class.getName())
&& !ste.getClassName().startsWith(MatsTestInfrastructureConfiguration.class.getName())
&& !ste.getClassName().startsWith(MatsTestInfrastructureDbConfiguration.class.getName())
&& !ste.getClassName().startsWith("org.springframework.")
&& !ste.getClassName().startsWith("sun.")
&& !ste.getClassName().startsWith("java.")
&& !ste.getClassName().startsWith("org.junit.")
&& !ste.getClassName().startsWith("com.intellij.")) {
return ste.getClassName();
}
}
// E-> Could not determine a reasonable suggestion for "app name" based on stack frames.
return "Testing_" + TestSpringMatsFactoryProvider.class.getSimpleName();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy