org.apache.camel.spring.SpringCamelContext 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.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);
getCamelContextExtension().setManagementMBeanAssembler(new SpringManagementMBeanAssembler(this));
}
public SpringCamelContext(ApplicationContext applicationContext) {
this();
setApplicationContext(applicationContext);
// force creating injector that works with spring
getInjector();
}
/**
* @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(128);
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