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.ManagementMBeanAssembler;
027    import org.apache.camel.spi.Registry;
028    import org.apache.camel.spring.spi.ApplicationContextRegistry;
029    import org.apache.camel.spring.spi.SpringInjector;
030    import org.apache.camel.spring.spi.SpringManagementMBeanAssembler;
031    import org.slf4j.Logger;
032    import org.slf4j.LoggerFactory;
033    import org.springframework.beans.BeansException;
034    import org.springframework.beans.factory.DisposableBean;
035    import org.springframework.beans.factory.InitializingBean;
036    import org.springframework.context.ApplicationContext;
037    import org.springframework.context.ApplicationContextAware;
038    import org.springframework.context.ApplicationEvent;
039    import org.springframework.context.ConfigurableApplicationContext;
040    import org.springframework.context.event.ContextClosedEvent;
041    import org.springframework.context.event.ContextRefreshedEvent;
042    import org.springframework.context.event.ContextStoppedEvent;
043    import org.springframework.context.support.ClassPathXmlApplicationContext;
044    
045    import static org.apache.camel.util.ObjectHelper.wrapRuntimeCamelException;
046    
047    /**
048     * A Spring aware implementation of {@link org.apache.camel.CamelContext} which
049     * will automatically register itself with Springs lifecycle methods plus allows
050     * spring to be used to customize a any <a
051     * href="http://camel.apache.org/type-converter.html">Type Converters</a>
052     * as well as supporting accessing components and beans via the Spring
053     * {@link ApplicationContext}
054     *
055     * @version 
056     */
057    public class SpringCamelContext extends DefaultCamelContext implements InitializingBean, DisposableBean,
058            ApplicationContextAware {
059    
060        private static final Logger LOG = LoggerFactory.getLogger(SpringCamelContext.class);
061        private static final ThreadLocal<Boolean> NO_START = new ThreadLocal<Boolean>();
062        private ApplicationContext applicationContext;
063        private EventComponent eventComponent;
064        private boolean shutdownEager = true;
065    
066        public SpringCamelContext() {
067        }
068    
069        public SpringCamelContext(ApplicationContext applicationContext) {
070            setApplicationContext(applicationContext);
071        }
072    
073        public static void setNoStart(boolean b) {
074            if (b) {
075                NO_START.set(b);
076            } else {
077                NO_START.remove();
078            }
079        }
080        
081        public static SpringCamelContext springCamelContext(ApplicationContext applicationContext) throws Exception {
082            return springCamelContext(applicationContext, true);
083        }
084        
085        public static SpringCamelContext springCamelContext(ApplicationContext applicationContext, boolean maybeStart) throws Exception {
086            if (applicationContext != null) {
087                // lets try and look up a configured camel context in the context
088                String[] names = applicationContext.getBeanNamesForType(SpringCamelContext.class);
089                if (names.length == 1) {
090                    return applicationContext.getBean(names[0], SpringCamelContext.class);
091                }
092            }
093            SpringCamelContext answer = new SpringCamelContext();
094            answer.setApplicationContext(applicationContext);
095            if (maybeStart) {
096                answer.afterPropertiesSet();
097            }
098            return answer;
099        }
100    
101        public static SpringCamelContext springCamelContext(String configLocations) throws Exception {
102            return springCamelContext(new ClassPathXmlApplicationContext(configLocations));
103        }
104    
105        public void afterPropertiesSet() throws Exception {
106            maybeStart();
107        }
108    
109        public void destroy() throws Exception {
110            stop();
111        }
112    
113        public void onApplicationEvent(ApplicationEvent event) {
114            LOG.debug("onApplicationEvent: {}", event);
115    
116            if (event instanceof ContextRefreshedEvent) {
117                // now lets start the CamelContext so that all its possible
118                // dependencies are initialized
119                try {
120                    maybeStart();
121                } catch (Exception e) {
122                    throw wrapRuntimeCamelException(e);
123                }
124            } else if (event instanceof ContextClosedEvent) {
125                // ContextClosedEvent is emitted when Spring is about to be shutdown
126                if (isShutdownEager()) {
127                    try {
128                        maybeStop();
129                    } catch (Exception e) {
130                        throw wrapRuntimeCamelException(e);
131                    }
132                }
133            } else if (event instanceof ContextStoppedEvent) {
134                // ContextStoppedEvent is emitted when Spring is end of shutdown
135                try {
136                    maybeStop();
137                } catch (Exception e) {
138                    throw wrapRuntimeCamelException(e);
139                }
140            }
141    
142            if (eventComponent != null) {
143                eventComponent.onApplicationEvent(event);
144            }
145        }
146    
147        // Properties
148        // -----------------------------------------------------------------------
149    
150        public ApplicationContext getApplicationContext() {
151            return applicationContext;
152        }
153    
154        public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
155            this.applicationContext = applicationContext;
156            ClassLoader cl;
157    
158            // set the application context classloader
159            if (applicationContext != null && applicationContext.getClassLoader() != null) {
160                cl = applicationContext.getClassLoader();
161            } else {
162                LOG.warn("Cannot find the class loader from application context, using the thread context class loader instead");
163                cl = Thread.currentThread().getContextClassLoader();
164            }
165            LOG.debug("Set the application context classloader to: {}", cl);
166            this.setApplicationContextClassLoader(cl);
167    
168            if (applicationContext instanceof ConfigurableApplicationContext) {
169                // only add if not already added
170                if (hasComponent("spring-event") == null) {
171                    eventComponent = new EventComponent(applicationContext);
172                    addComponent("spring-event", eventComponent);
173                }
174            }
175        }
176    
177        @Deprecated
178        public EventEndpoint getEventEndpoint() {
179            return null;
180        }
181    
182        @Deprecated
183        public void setEventEndpoint(EventEndpoint eventEndpoint) {
184            // noop
185        }
186    
187        /**
188         * Whether to shutdown this {@link org.apache.camel.spring.SpringCamelContext} eager (first)
189         * when Spring {@link org.springframework.context.ApplicationContext} is being stopped.
190         * <p/>
191         * <b>Important:</b> This option is default <tt>true</tt> which ensures we shutdown Camel
192         * before other beans. Setting this to <tt>false</tt> restores old behavior in earlier
193         * Camel releases, which can be used for special cases to behave as before.
194         *
195         * @return <tt>true</tt> to shutdown eager (first), <tt>false</tt> to shutdown last
196         */
197        public boolean isShutdownEager() {
198            return shutdownEager;
199        }
200    
201        /**
202         * @see #isShutdownEager()
203         */
204        public void setShutdownEager(boolean shutdownEager) {
205            this.shutdownEager = shutdownEager;
206        }
207    
208        // Implementation methods
209        // -----------------------------------------------------------------------
210    
211        @Override
212        protected Injector createInjector() {
213            if (applicationContext instanceof ConfigurableApplicationContext) {
214                return new SpringInjector((ConfigurableApplicationContext)applicationContext);
215            } else {
216                LOG.warn("Cannot use SpringInjector as applicationContext is not a ConfigurableApplicationContext as its: "
217                          + applicationContext);
218                return super.createInjector();
219            }
220        }
221    
222        @Override
223        protected ManagementMBeanAssembler createManagementMBeanAssembler() {
224            // use a spring mbean assembler
225            return new SpringManagementMBeanAssembler(this);
226        }
227    
228        protected EventEndpoint createEventEndpoint() {
229            return getEndpoint("spring-event:default", EventEndpoint.class);
230        }
231    
232        protected Endpoint convertBeanToEndpoint(String uri, Object bean) {
233            // We will use the type convert to build the endpoint first
234            Endpoint endpoint = getTypeConverter().convertTo(Endpoint.class, bean);
235            if (endpoint != null) {
236                endpoint.setCamelContext(this);
237                return endpoint;
238            }
239    
240            return new ProcessorEndpoint(uri, this, new BeanProcessor(bean, this));
241        }
242    
243        @Override
244        protected Registry createRegistry() {
245            return new ApplicationContextRegistry(getApplicationContext());
246        }
247    
248        private void maybeStart() throws Exception {
249            // for example from unit testing we want to start Camel later and not when Spring framework
250            // publish a ContextRefreshedEvent
251    
252            if (NO_START.get() == null) {
253                if (!isStarted() && !isStarting()) {
254                    start();
255                } else {
256                    // ignore as Camel is already started
257                    LOG.trace("Ignoring maybeStart() as Apache Camel is already started");
258                }
259            } else {
260                LOG.trace("Ignoring maybeStart() as NO_START is false");
261            }
262        }
263    
264        private void maybeStop() throws Exception {
265            if (!isStopping() && !isStopped()) {
266                stop();
267            } else {
268                // ignore as Camel is already stopped
269                LOG.trace("Ignoring maybeStop() as Apache Camel is already stopped");
270            }
271        }
272    
273        @Override
274        public String toString() {
275            StringBuilder sb = new StringBuilder();
276            sb.append("SpringCamelContext(").append(getName()).append(")");
277            if (applicationContext != null) {
278                sb.append(" with spring id ").append(applicationContext.getId());
279            }
280            return sb.toString();
281        }
282    
283    }