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

org.apache.camel.spring.SpringCamelContext Maven / Gradle / Ivy

There is a newer version: 4.9.0
Show newest version
/*
 * 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.spring;

import java.util.Optional;

import org.apache.camel.Endpoint;
import org.apache.camel.Processor;
import org.apache.camel.RuntimeCamelException;
import org.apache.camel.component.event.EventComponent;
import org.apache.camel.component.event.EventEndpoint;
import org.apache.camel.impl.DefaultCamelContext;
import org.apache.camel.spi.BeanProcessorFactory;
import org.apache.camel.spi.BeanRepository;
import org.apache.camel.spi.Injector;
import org.apache.camel.spi.ModelJAXBContextFactory;
import org.apache.camel.spi.Registry;
import org.apache.camel.spring.spi.ApplicationContextBeanRepository;
import org.apache.camel.spring.spi.SpringInjector;
import org.apache.camel.spring.spi.SpringManagementMBeanAssembler;
import org.apache.camel.support.DefaultRegistry;
import org.apache.camel.support.PluginHelper;
import org.apache.camel.support.ProcessorEndpoint;
import org.apache.camel.support.ResolverHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.context.ApplicationEvent;
import org.springframework.context.ApplicationListener;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.Lifecycle;
import org.springframework.context.Phased;
import org.springframework.context.event.ContextRefreshedEvent;
import org.springframework.core.Ordered;

import static org.apache.camel.RuntimeCamelException.wrapRuntimeCamelException;

/**
 * A Spring aware implementation of {@link org.apache.camel.CamelContext} which will automatically register itself with
 * Springs lifecycle methods plus allows spring to be used to customize a any
 * Type Converters as well as supporting accessing components
 * and beans via the Spring {@link ApplicationContext}
 */
public class SpringCamelContext extends DefaultCamelContext
        implements Lifecycle, ApplicationContextAware, Phased,
        ApplicationListener, Ordered {

    public static final String EXCLUDE_ROUTES = "CamelTestSpringExcludeRoutes";

    private static final Logger LOG = LoggerFactory.getLogger(SpringCamelContext.class);
    private ApplicationContext applicationContext;
    private EventComponent eventComponent;
    private boolean shutdownEager = true;

    public SpringCamelContext() {
        super(false);
        setManagementMBeanAssembler(new SpringManagementMBeanAssembler(this));
    }

    public SpringCamelContext(ApplicationContext applicationContext) {
        this();
        setApplicationContext(applicationContext);
    }

    /**
     * @deprecated its better to create and boot Spring the standard Spring way and to get hold of CamelContext using
     *             the Spring API.
     */
    @Deprecated
    public static SpringCamelContext springCamelContext(ApplicationContext applicationContext, boolean maybeStart)
            throws Exception {
        if (applicationContext != null) {
            // lets try and look up a configured camel context in the context
            String[] names = applicationContext.getBeanNamesForType(SpringCamelContext.class);
            if (names.length == 1) {
                return applicationContext.getBean(names[0], SpringCamelContext.class);
            }
        }
        SpringCamelContext answer = new SpringCamelContext();
        answer.setApplicationContext(applicationContext);
        answer.build();
        if (maybeStart) {
            answer.start();
        }
        return answer;
    }

    @Override
    public void doBuild() throws Exception {
        super.doBuild();
        if (applicationContext instanceof ConfigurableApplicationContext) {
            // only add if not already added
            if (hasComponent("spring-event") == null) {
                eventComponent = new EventComponent(applicationContext);
                addComponent("spring-event", eventComponent);
            }
        }
    }

    @Override
    public void start() {
        try {
            super.start();
        } catch (Exception e) {
            throw wrapRuntimeCamelException(e);
        }
    }

    @Override
    public void stop() {
        try {
            super.stop();
        } catch (Exception e) {
            throw wrapRuntimeCamelException(e);
        }
    }

    @Override
    public void onApplicationEvent(ApplicationEvent event) {
        LOG.debug("onApplicationEvent: {}", event);

        if (event instanceof ContextRefreshedEvent
                && ((ContextRefreshedEvent) event).getApplicationContext() == this.applicationContext) {
            // nominally we would prefer to use Lifecycle interface that
            // would invoke start() method, but in order to do that
            // SpringCamelContext needs to implement SmartLifecycle
            // (look at DefaultLifecycleProcessor::startBeans), but it
            // cannot implement it as it already implements
            // RuntimeConfiguration, and both SmartLifecycle and
            // RuntimeConfiguration declare isAutoStartup method but
            // with boolean and Boolean return types, and covariant
            // methods with primitive types are not allowed by the JLS
            // so we need to listen for ContextRefreshedEvent and start
            // on its reception
            start();
        }

        if (eventComponent != null) {
            eventComponent.onApplicationEvent(event);
        }
    }

    @Override
    public int getOrder() {
        // SpringCamelContext implements Ordered so that it's the last
        // in ApplicationListener to receive events, this is important
        // for startup as we want all resources to be ready and all
        // routes added to the context (see
        // org.apache.camel.spring.boot.RoutesCollector)
        // and we need to be after CamelContextFactoryBean
        return LOWEST_PRECEDENCE;
    }

    // Properties
    // -----------------------------------------------------------------------

    public ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
        ClassLoader cl;

        // set the application context classloader
        if (applicationContext != null && applicationContext.getClassLoader() != null) {
            cl = applicationContext.getClassLoader();
        } else {
            LOG.warn("Cannot find the class loader from application context, using the thread context class loader instead");
            cl = Thread.currentThread().getContextClassLoader();
        }
        LOG.debug("Set the application context classloader to: {}", cl);
        this.setApplicationContextClassLoader(cl);
    }

    /**
     * Whether to shutdown this {@link org.apache.camel.spring.SpringCamelContext} eager (first) when Spring
     * {@link org.springframework.context.ApplicationContext} is being stopped.
     * 

* Important: This option is default true which ensures we shutdown Camel before other beans. * Setting this to false restores old behavior in earlier Camel releases, which can be used for special * cases to behave as before. * * @return true to shutdown eager (first), false to shutdown last */ public boolean isShutdownEager() { return shutdownEager; } /** * @see #isShutdownEager() */ public void setShutdownEager(boolean shutdownEager) { this.shutdownEager = shutdownEager; } // Implementation methods // ----------------------------------------------------------------------- @Override protected Injector createInjector() { if (applicationContext instanceof ConfigurableApplicationContext) { return new SpringInjector((ConfigurableApplicationContext) applicationContext); } else { LOG.warn("Cannot use SpringInjector as applicationContext is not a ConfigurableApplicationContext as its: {}", applicationContext); return super.createInjector(); } } protected EventEndpoint createEventEndpoint() { return getEndpoint("spring-event:default", EventEndpoint.class); } @Override protected Endpoint convertBeanToEndpoint(String uri, Object bean) { // We will use the type convert to build the endpoint first Endpoint endpoint = getTypeConverter().convertTo(Endpoint.class, bean); if (endpoint != null) { endpoint.setCamelContext(this); return endpoint; } BeanProcessorFactory bpf = PluginHelper.getBeanProcessorFactory(getCamelContextExtension()); try { Processor bp = bpf.createBeanProcessor(this, bean, null); return new ProcessorEndpoint(uri, this, bp); } catch (Exception e) { throw RuntimeCamelException.wrapRuntimeCamelException(e); } } @Override protected Registry createRegistry() { BeanRepository repository = new ApplicationContextBeanRepository(getApplicationContext()); return new DefaultRegistry(repository); } @Override protected ModelJAXBContextFactory createModelJAXBContextFactory() { Optional result = ResolverHelper.resolveService( getCamelContextReference(), getCamelContextExtension().getBootstrapFactoryFinder(), ModelJAXBContextFactory.FACTORY + "-spring", ModelJAXBContextFactory.class); if (result.isPresent()) { return result.get(); } else { throw new IllegalArgumentException( "Cannot find ModelJAXBContextFactory on classpath. Add camel-spring-xml to classpath."); } } @Override public String toString() { StringBuilder sb = new StringBuilder(); sb.append("SpringCamelContext(").append(getName()).append(")"); if (applicationContext != null) { sb.append(" with spring id ").append(applicationContext.getId()); } return sb.toString(); } @Override public int getPhase() { // the context is started by invoking start method which // happens either on ContextRefreshedEvent or explicitly // invoking the method, for instance CamelContextFactoryBean // is using that to start the context, _so_ here we want to // have maximum priority as the getPhase() will be used only // for stopping, in order to be used for starting we would // need to implement SmartLifecycle which we cannot // (explained in comment in the onApplicationEvent method) // we use LOWEST_PRECEDENCE here as this is taken into account // only when stopping and then in reversed order return Integer.MAX_VALUE - 2049; // we need to be less than max value as spring-boot comes with // graceful shutdown services (the http server in spring boot) // that must shutdown before camel, and they have max value - 2048, // so we use 2049 to have a higher gap } @Override public boolean isRunning() { return !isStopping() && !isStopped(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy