打造一个基于OSGi的Web Application——使用Tomcat原生API来动态管理Web元素:原理

Tomcat的org.apache.catalina.Context接口提供了动态管理注入到Catalina Web Container中的Web元素的API。在基于OSGi的Web Application中,可以利用这个接口来实现在OSGi容器中动态管理Web元素的目的。为了达到这个目的,我们还需要做一些额外的配置。请注意,以下方法仅适用于Tomcat,并非通用的实现,而且只针对5.5.28版和6.0.24版的Tomcat做过简单的测试。


首先我们要做的事情,就是将Tomcat的org.apache.catalina.Context实现类作为Service注入到OSGi容器中去。在OSGi-Web工程的WebContent/META-INF目录中,增加一个context.xml文件,内容如下:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> --> 1   <? xml version="1.0" encoding="UTF-8" ?>
2   < Context  privileged ="true" />

这样我们就可以使用org.apache.catalina.ContainerServlet这个接口类了,通过它可以访问Catalina的内部功能,它有Catalina被类加载器加载,而不是我们的WebApplication类加载器。它的 Setter方法在这个Servlet的新的实例被放进Service时被执行。
接下来我们写一个Servlet,这个Servlet将实现ContainerServlet接口,请注意它是怎么工作的:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->  1   package  org.dbstar.osgi.web.launcher.tomcat;
 2  
 3   import  java.util.Properties;
 4  
 5   import  javax.servlet.ServletException;
 6   import  javax.servlet.UnavailableException;
 7   import  javax.servlet.http.HttpServlet;
 8  
 9   import  org.apache.catalina.ContainerServlet;
10   import  org.apache.catalina.Context;
11   import  org.apache.catalina.Wrapper;
12   import  org.dbstar.osgi.web.launcher.FrameworkConfigListener;
13   import  org.osgi.framework.BundleContext;
14   import  org.osgi.framework.ServiceRegistration;
15   import  org.osgi.framework.launch.Framework;
16  
17   public   final   class  TomcatContextServlet  extends  HttpServlet  implements  ContainerServlet {
18        private   static   final   long  serialVersionUID  =   - 3977062987005392657L ;
19  
20        private  Wrapper wrapper;
21        private  Context context;
22  
23        private  ServiceRegistration registration;
24  
25        public  Wrapper getWrapper() {
26            return  wrapper;
27       }
28  
29        public   void  setWrapper(Wrapper wrapper) {
30            this .wrapper  =  wrapper;
31            if  (wrapper  ==   null ) context  =   null ;
32            else  context  =  (Context) wrapper.getParent();
33       }
34  
35       @Override
36        public   void  init()  throws  ServletException {
37            //  Ensure that our ContainerServlet properties have been set
38            if  ((wrapper  ==   null ||  (context  ==   null ))  throw   new  UnavailableException( " Wrapper not set. " );
39  
40            //  Ensure that Framework have been set
41           Framework framework  =  FrameworkConfigListener.getFramework();
42            if  (framework  ==   null throw   new  UnavailableException( " Framework not set. " );
43  
44            //  将context注册为服务
45           registration  =  registerContext(framework.getBundleContext(), context);
46       }
47  
48        private   static  ServiceRegistration registerContext(BundleContext bundleContext, Context context) {
49           Properties properties  =   new  Properties();
50           properties.setProperty( " DisplayName " , context.getDisplayName());
51           properties.setProperty( " ContextPath " , context.getPath());
52            return  bundleContext.registerService(Context. class .getName(), context, properties);
53       }
54  
55       @Override
56        public   void  destroy() {
57            if  (registration  ==   null return ;
58  
59           Framework framework  =  FrameworkConfigListener.getFramework();
60            if  (framework  ==   null return ;
61  
62            if  (framework.getState()  ==  Framework.ACTIVE) registration.unregister();
63           registration  =   null ;
64       }
65   }

通过ContainerServlet接口提供的setWrapper方法,我们获得了一个Wrapper实例,这个实例对应于TomcatContextServlet部署到Tomcat中的封装类,通过其getParent方法我们就可以获得Servlet所在的Context了。
接下来在init方法中,我们将获得的Context实例,通过Framework注册到OSGi容器中去。在destroy方法中,注销Context的注册,这样形成了一个完整的生命周期。
然后,将这个TomcatContextServlet部署到web.xml中去:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> --> 1        <!--  register a org.apache.catalina.Context to OSGi Container     -->
2        < servlet >
3            < servlet-name > TomcatContextServlet </ servlet-name >
4            < servlet-class > org.dbstar.osgi.web.launcher.tomcat.TomcatContextServlet </ servlet-class >
5            < load-on-startup > 1 </ load-on-startup >
6        </ servlet >

设置<load-on-startup>使这个Servlet在WebContainer初始化时加载,否则它将没有加载的机会,因为我们在应用中不会直接使用到这个Servlet。

最后还有一件事情不要忘记了,我们需要将org.apache.catalina及其相关的package export到OSGi容器中去,这样才能在OSGi容器中供给bundle来import。参照《打造一个基于OSGi的Web Application——为OSGi容器提供Web Application环境 》一文中提到的方式,我们将catalina.jar作为extension Fragment的方式,引入到OSGi容器中去。
Catalina的MANIFEST.MF:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->  1   Manifest-Version:  1.0
 2   Bundle-ManifestVersion:  2
 3   Bundle-Name: Catalina Extension Fragment
 4   Bundle-SymbolicName: org.apache.catalina_extension ; singleton:=true
 5   Bundle-Version:  5.5.28
 6   Bundle-Vendor: dbstar
 7   Fragment-Host: system.bundle ;  extension:=framework
 8   Bundle-RequiredExecutionEnvironment: J2SE- 1.5
 9   Export-Package: org.apache.catalina , org.apache.catalina.authenticator ,
10    org.apache.catalina.connector , org.apache.catalina.core , org.apache.cat
11    alina.deploy , org.apache.catalina.loader , org.apache.catalina.mbeans , or
12    g.apache.catalina.realm , org.apache.catalina.security , org.apache.catal
13    ina.session , org.apache.catalina.startup , org.apache.catalina.users , org
14    .apache.catalina.util , org.apache.catalina.valves


在接下来的章节中,我会逐一描述如何在基于Tomcat的OSGi容器中,如何实现各种Web元素的动态管理,尽请期待哦:)

最后提供几个本章提到的bundle给大家下载,大家就不用自己再起生成一个了。
org.apache.catalina_extension_5.5.28.jar

你可能感兴趣的:(apache,tomcat,Web,配置管理,osgi)