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

org.apache.camel.test.spring.junit5.CamelSpringTestContextLoader Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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.apache.camel.test.spring.junit5;

import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.Map;

import org.apache.camel.spring.SpringCamelContext;
import org.apache.camel.test.ExcludingPackageScanClassResolver;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.support.RootBeanDefinition;
import org.springframework.beans.factory.xml.XmlBeanDefinitionReader;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotatedBeanDefinitionReader;
import org.springframework.context.annotation.AnnotationConfigUtils;
import org.springframework.context.support.GenericApplicationContext;
import org.springframework.core.io.Resource;
import org.springframework.test.context.MergedContextConfiguration;
import org.springframework.test.context.support.AbstractContextLoader;
import org.springframework.test.context.support.AbstractGenericContextLoader;
import org.springframework.test.context.support.GenericXmlContextLoader;
import org.springframework.util.StringUtils;

/**
 * Replacement for the default {@link GenericXmlContextLoader} that provides hooks for processing some class level Camel
 * related test annotations.
 */
public class CamelSpringTestContextLoader extends AbstractContextLoader {

    private static final Logger LOG = LoggerFactory.getLogger(CamelSpringTestContextLoader.class);

    /**
     * Modeled after the Spring implementation in {@link AbstractGenericContextLoader}, this method creates and
     * refreshes the application context while providing for processing of additional Camel specific post-refresh
     * actions. We do not provide the pre-post hooks for customization seen in {@link AbstractGenericContextLoader}
     * because they probably are unnecessary for 90+% of users.
     * 

* For some functionality, we cannot use {@link org.springframework.test.context.TestExecutionListener} because we * need to both produce the desired outcome during application context loading, and also cleanup after ourselves * even if the test class never executes. Thus the listeners, which only run if the application context is * successfully initialized are insufficient to provide the behavior described above. */ @Override public ApplicationContext loadContext(MergedContextConfiguration mergedConfig) throws Exception { Class testClass = getTestClass(); if (LOG.isDebugEnabled()) { LOG.debug("Loading ApplicationContext for merged context configuration [{}].", mergedConfig); } try { GenericApplicationContext context = createContext(testClass, mergedConfig); prepareContext(context, mergedConfig); loadBeanDefinitions(context, mergedConfig); AnnotationConfigUtils.registerAnnotationConfigProcessors(context); customizeContext(context, mergedConfig); return loadContext(context, testClass); } finally { CamelAnnotationsHandler.cleanup(); } } /** * Modeled after the Spring implementation in {@link AbstractGenericContextLoader}, this method creates and * refreshes the application context while providing for processing of additional Camel specific post-refresh * actions. We do not provide the pre-post hooks for customization seen in {@link AbstractGenericContextLoader} * because they probably are unnecessary for 90+% of users. *

* For some functionality, we cannot use {@link org.springframework.test.context.TestExecutionListener} because we * need to both produce the desired outcome during application context loading, and also cleanup after ourselves * even if the test class never executes. Thus the listeners, which only run if the application context is * successfully initialized are insufficient to provide the behavior described above. */ @Override public ApplicationContext loadContext(String... locations) throws Exception { Class testClass = getTestClass(); if (LOG.isDebugEnabled()) { LOG.debug("Loading ApplicationContext for locations [{}].", StringUtils.arrayToCommaDelimitedString(locations)); } try { GenericApplicationContext context = createContext(testClass, null); loadBeanDefinitions(context, testClass, locations); return loadContext(context, testClass); } finally { CamelAnnotationsHandler.cleanup(); } } /** * Returns "-context.xml". */ @Override public String getResourceSuffix() { return "-context.xml"; } /** * Performs the bulk of the Spring application context loading/customization. * * @param context the partially configured context. The context should have the bean definitions loaded, but * nothing else. * @param testClass the test class being executed * @return the initialized (refreshed) Spring application context * * @throws Exception if there is an error during initialization/customization */ protected ApplicationContext loadContext(GenericApplicationContext context, Class testClass) throws Exception { AnnotationConfigUtils.registerAnnotationConfigProcessors(context); // Pre CamelContext(s) instantiation setup CamelAnnotationsHandler.handleDisableJmx(testClass); CamelAnnotationsHandler.handleExcludeRoutes(testClass); CamelAnnotationsHandler.handleUseOverridePropertiesWithPropertiesComponent(context, testClass); // Temporarily disable CamelContext start while the contexts are instantiated. SpringCamelContext.setNoStart(true); try { context.refresh(); context.registerShutdownHook(); } finally { // Turn CamelContext startup back on since the context's have now been instantiated. SpringCamelContext.setNoStart(false); } // Post CamelContext(s) instantiation but pre CamelContext(s) start setup CamelAnnotationsHandler.handleRouteCoverage(context, testClass, s -> getTestMethod().getName()); CamelAnnotationsHandler.handleProvidesBreakpoint(context, testClass); CamelAnnotationsHandler.handleShutdownTimeout(context, testClass); CamelAnnotationsHandler.handleMockEndpoints(context, testClass); CamelAnnotationsHandler.handleMockEndpointsAndSkip(context, testClass); // CamelContext(s) startup CamelAnnotationsHandler.handleCamelContextStartup(context, testClass); return context; } protected void loadBeanDefinitions(GenericApplicationContext context, MergedContextConfiguration mergedConfig) { loadBeanDefinitions(context, mergedConfig.getTestClass(), mergedConfig.getLocations()); new AnnotatedBeanDefinitionReader(context).register(mergedConfig.getClasses()); } protected void loadBeanDefinitions(GenericApplicationContext context, Class clazz, String... locations) { Map props = CamelSpringTestSupport.getTranslationProperties(clazz); XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(context); for (String location : locations) { Resource r = reader.getResourceLoader().getResource(location); Resource t = new CamelSpringTestSupport.TranslatedResource(r, props); reader.loadBeanDefinitions(t); } } /** * Creates and starts the Spring context while optionally starting any loaded Camel contexts. * * @param testClass the test class that is being executed * @return the loaded Spring context */ protected GenericApplicationContext createContext(Class testClass, MergedContextConfiguration mergedConfig) { ApplicationContext parentContext = null; GenericApplicationContext routeExcludingContext = null; if (mergedConfig != null) { parentContext = mergedConfig.getParentApplicationContext(); } if (testClass.isAnnotationPresent(ExcludeRoutes.class)) { Class[] excludedClasses = testClass.getAnnotation(ExcludeRoutes.class).value(); if (excludedClasses.length > 0) { if (LOG.isDebugEnabled()) { LOG.debug( "Setting up package scanning excluded classes as ExcludeRoutes annotation was found. Excluding [{}]", StringUtils.arrayToCommaDelimitedString(excludedClasses)); } if (parentContext == null) { routeExcludingContext = new GenericApplicationContext(); } else { routeExcludingContext = new GenericApplicationContext(parentContext); } routeExcludingContext.registerBeanDefinition("excludingResolver", new RootBeanDefinition(ExcludingPackageScanClassResolver.class)); routeExcludingContext.refresh(); ExcludingPackageScanClassResolver excludingResolver = routeExcludingContext.getBean("excludingResolver", ExcludingPackageScanClassResolver.class); List> excluded = Arrays.asList(excludedClasses); excludingResolver.setExcludedClasses(new HashSet<>(excluded)); } else { if (LOG.isDebugEnabled()) { LOG.debug("Not enabling package scanning excluded classes as ExcludeRoutes " + "annotation was found but no classes were excluded."); } } } GenericApplicationContext context; if (routeExcludingContext != null) { context = new GenericApplicationContext(routeExcludingContext); } else { if (parentContext != null) { context = new GenericApplicationContext(parentContext); } else { context = new GenericApplicationContext(); } } return context; } /** * Returns the class under test in order to enable inspection of annotations while the Spring context is being * created. * * @return the test class that is being executed * @see CamelSpringTestHelper */ protected Class getTestClass() { return CamelSpringTestHelper.getTestClass(); } /** * Returns the test method under test. * * @return the method that is being executed * @see CamelSpringTestHelper */ protected Method getTestMethod() { return CamelSpringTestHelper.getTestMethod(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy