001 /** 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017 package org.apache.camel.spring; 018 019 import org.apache.camel.Endpoint; 020 import org.apache.camel.component.bean.BeanProcessor; 021 import org.apache.camel.component.event.EventComponent; 022 import org.apache.camel.component.event.EventEndpoint; 023 import org.apache.camel.impl.DefaultCamelContext; 024 import org.apache.camel.impl.ProcessorEndpoint; 025 import org.apache.camel.spi.Injector; 026 import org.apache.camel.spi.Registry; 027 import org.apache.camel.spring.spi.ApplicationContextRegistry; 028 import org.apache.camel.spring.spi.SpringInjector; 029 import org.apache.camel.util.ObjectHelper; 030 import org.apache.commons.logging.Log; 031 import org.apache.commons.logging.LogFactory; 032 import org.springframework.beans.BeansException; 033 import org.springframework.beans.factory.DisposableBean; 034 import org.springframework.beans.factory.InitializingBean; 035 import org.springframework.context.ApplicationContext; 036 import org.springframework.context.ApplicationContextAware; 037 import org.springframework.context.ApplicationEvent; 038 import org.springframework.context.ApplicationListener; 039 import org.springframework.context.ConfigurableApplicationContext; 040 import org.springframework.context.event.ContextRefreshedEvent; 041 import org.springframework.context.support.ClassPathXmlApplicationContext; 042 043 import static org.apache.camel.util.ObjectHelper.wrapRuntimeCamelException; 044 045 /** 046 * A Spring aware implementation of {@link org.apache.camel.CamelContext} which 047 * will automatically register itself with Springs lifecycle methods plus allows 048 * spring to be used to customize a any <a 049 * href="http://camel.apache.org/type-converter.html">Type Converters</a> 050 * as well as supporting accessing components and beans via the Spring 051 * {@link ApplicationContext} 052 * 053 * @version $Revision: 787928 $ 054 */ 055 public class SpringCamelContext extends DefaultCamelContext implements InitializingBean, DisposableBean, 056 ApplicationContextAware, ApplicationListener { 057 058 private static final transient Log LOG = LogFactory.getLog(SpringCamelContext.class); 059 private ApplicationContext applicationContext; 060 private EventEndpoint eventEndpoint; 061 private boolean shouldStartContext = ObjectHelper.getSystemProperty("shouldStartContext", Boolean.TRUE); 062 063 public SpringCamelContext() { 064 } 065 066 public SpringCamelContext(ApplicationContext applicationContext) { 067 setApplicationContext(applicationContext); 068 } 069 070 public static SpringCamelContext springCamelContext(ApplicationContext applicationContext) throws Exception { 071 // lets try and look up a configured camel context in the context 072 String[] names = applicationContext.getBeanNamesForType(SpringCamelContext.class); 073 if (names.length == 1) { 074 return (SpringCamelContext)applicationContext.getBean(names[0], SpringCamelContext.class); 075 } 076 SpringCamelContext answer = new SpringCamelContext(); 077 answer.setApplicationContext(applicationContext); 078 answer.afterPropertiesSet(); 079 return answer; 080 } 081 082 public static SpringCamelContext springCamelContext(String configLocations) throws Exception { 083 return springCamelContext(new ClassPathXmlApplicationContext(configLocations)); 084 } 085 086 public void afterPropertiesSet() throws Exception { 087 maybeStart(); 088 } 089 090 private void maybeStart() throws Exception { 091 if (!getShouldStartContext()) { 092 LOG.info("Not starting Apache Camel as property ShouldStartContext is false"); 093 return; 094 } 095 096 if (!isStarted() && !isStarting()) { 097 // Make sure we will not get into the endless loop of calling star 098 LOG.info("Starting Apache Camel as property ShouldStartContext is true"); 099 start(); 100 } else { 101 // ignore as Camel is already started 102 LOG.trace("Ignoring maybeStart() as Apache Camel is already started"); 103 } 104 } 105 106 public void destroy() throws Exception { 107 stop(); 108 } 109 110 public void onApplicationEvent(ApplicationEvent event) { 111 if (LOG.isDebugEnabled()) { 112 LOG.debug("onApplicationEvent: " + event); 113 } 114 115 if (event instanceof ContextRefreshedEvent) { 116 // now lets start the CamelContext so that all its possible 117 // dependencies are initialized 118 try { 119 maybeStart(); 120 } catch (Exception e) { 121 throw wrapRuntimeCamelException(e); 122 } 123 } 124 125 if (eventEndpoint != null) { 126 eventEndpoint.onApplicationEvent(event); 127 } else { 128 LOG.warn("No spring-event endpoint enabled to handle event: " + event); 129 } 130 } 131 132 // Properties 133 // ----------------------------------------------------------------------- 134 135 public ApplicationContext getApplicationContext() { 136 return applicationContext; 137 } 138 139 public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { 140 this.applicationContext = applicationContext; 141 142 if (applicationContext instanceof ConfigurableApplicationContext) { 143 addComponent("spring-event", new EventComponent(applicationContext)); 144 } 145 } 146 147 public EventEndpoint getEventEndpoint() { 148 return eventEndpoint; 149 } 150 151 public void setEventEndpoint(EventEndpoint eventEndpoint) { 152 this.eventEndpoint = eventEndpoint; 153 } 154 155 // Implementation methods 156 // ----------------------------------------------------------------------- 157 158 @Override 159 protected void doStart() throws Exception { 160 maybeDoStart(); 161 } 162 163 protected void maybeDoStart() throws Exception { 164 if (getShouldStartContext()) { 165 super.doStart(); 166 if (eventEndpoint == null) { 167 eventEndpoint = createEventEndpoint(); 168 } 169 } 170 } 171 172 @Override 173 protected Injector createInjector() { 174 if (applicationContext instanceof ConfigurableApplicationContext) { 175 return new SpringInjector((ConfigurableApplicationContext)applicationContext); 176 } else { 177 LOG.warn("Cannot use SpringInjector as applicationContext is not a ConfigurableApplicationContext as its: " 178 + applicationContext); 179 return super.createInjector(); 180 } 181 } 182 183 protected EventEndpoint createEventEndpoint() { 184 return getEndpoint("spring-event:default", EventEndpoint.class); 185 } 186 187 protected Endpoint convertBeanToEndpoint(String uri, Object bean) { 188 // We will use the type convert to build the endpoint first 189 Endpoint endpoint = getTypeConverter().convertTo(Endpoint.class, bean); 190 if (endpoint != null) { 191 endpoint.setCamelContext(this); 192 return endpoint; 193 } 194 195 return new ProcessorEndpoint(uri, this, new BeanProcessor(bean, this)); 196 } 197 198 @Override 199 protected Registry createRegistry() { 200 return new ApplicationContextRegistry(getApplicationContext()); 201 } 202 203 public void setShouldStartContext(boolean shouldStartContext) { 204 this.shouldStartContext = shouldStartContext; 205 } 206 207 public boolean getShouldStartContext() { 208 return shouldStartContext; 209 } 210 211 @Override 212 public String toString() { 213 StringBuilder sb = new StringBuilder(); 214 sb.append("SpringCamelContext(").append(getName()).append(")"); 215 if (applicationContext != null) { 216 sb.append(" with spring id ").append(applicationContext.getId()); 217 } 218 return sb.toString(); 219 } 220 221 }