本章敘述如何在OSGi容器中提供必要的Web Application環(huán)境,其中包括Servlet 2.4、Jsp 2.0和Commons-Logging相關的package,使得其他在OSGi容器中的bundle可以import。
為了在OSGi容器中提供export的package,一般有三種方式:
- 一個常規(guī)的bundle,自身包含必要的class,同時在Export-Package中聲明。
- 一個Host為System Bundle的Fragment Bundle,同樣也可以在Export-Package中聲明導出的package,只要這個package中的class在System Bundle的ClassLoader中能load到。
- 通過啟動Framework的配置項:org.osgi.framework.system.packages和org.osgi.framework.system.packages.extra。OSGi 4.2規(guī)范中描述了這兩個標準的配置項。在這兩個配置項中描述的package都等同于在System Bundle中聲明了export。
對于在Web Application中運行的OSGi容器,一些必要的環(huán)境是通過Web Container提供的,我們最好不要,也不應該用自己的類來替換,這包括了j2ee相關的jar,如servlet和jsp相關的jar等等。在一些WebServer的實現(xiàn)中,會自動屏蔽Web Application的classpath中的j2ee相關的jar。
除了j2ee相關的jar之外,還有一些使用非常普遍的jar,比如說Apache commons一類,其中最常用的大概就是commons-lang.jar、commons-io.jar和commons-logging.jar了,這些jar最好也有Web Container來提供,或者有必要的話,在Web Application中提供,而不是在OSGi容器中提供,這涉及到一些JVM層次的單例類,或者希望能由Web Application級別來統(tǒng)一實現(xiàn)和配置的環(huán)境,最常見的應用就是日志配置了。通過由Web Application提供的commons-logging來給OSGi容器中的環(huán)境使用,而commons-logging通過何種方式來實現(xiàn),不需要讓OSGi內部知道。
至于導出package到OSGi的方式中,是采用第二種還是第三種,主要區(qū)別在于:第三種方式是加載framework時指定的,在其后的生命周期中不可更改,而第二種方式則更符合OSGi動態(tài)加載的特性。
我采用第二種方式來給OSGi容器增加環(huán)境支持,具體操作很簡單,以Servlet為例,首先編寫一個文本文件,名字為:MANIFEST.MF,內容如下:
1 Manifest-Version: 1.0
2 Bundle-ManifestVersion: 2
3 Bundle-Name: Servlet Extension Fragment
4 Bundle-SymbolicName: javax.servlet_extension;singleton:=true
5 Bundle-Version: 2.4.0
6 Fragment-Host: system.bundle; extension:=framework
7 Bundle-RequiredExecutionEnvironment: J2SE-1.5
8 Export-Package: javax.servlet;version="2.4.0",
9 javax.servlet.http;version="2.4.0",
10 javax.servlet.resources;version="2.4.0"
11 Bundle-Vendor: dbstar
注意其中關鍵的header屬性,F(xiàn)ragment-Host: system.bundle; extension:=framework
這樣寫才能保證這個Fragment Bundle在各種OSGi Framework實現(xiàn)中都能兼容。
保存以后,將這個文件放置到一個名字為META-INF的目錄中,然后用jar命令打包成一個jar即可(或者用winrar打包,記得選擇壓縮方式為zip,在打包后將zip后綴名改成jar,我通常都是這么干的)。
Jsp的MANIFEST.MF:
1 Manifest-Version: 1.0
2 Bundle-ManifestVersion: 2
3 Bundle-Name: Jsp Extension Fragment
4 Bundle-SymbolicName: javax.servlet.jsp_extension;singleton:=true
5 Bundle-Version: 2.0.0
6 Bundle-Vendor: dbstar
7 Fragment-Host: system.bundle; extension:=framework
8 Bundle-RequiredExecutionEnvironment: J2SE-1.5
9 Export-Package: javax.servlet.jsp;version="2.0.0",
10 javax.servlet.jsp.el;version="2.0.0",
11 javax.servlet.jsp.resources;version="2.0.0",
12 javax.servlet.jsp.tagext;version="2.0.0"
commons-logging的MANIFEST.MF
1 Manifest-Version: 1.0
2 Bundle-ManifestVersion: 2
3 Bundle-Name: Commons Logging Extension Fragment
4 Bundle-SymbolicName: org.apache.commons.logging_extension;singleton:=true
5 Bundle-Version: 1.1.1
6 Bundle-Vendor: dbstar
7 Fragment-Host: system.bundle; extension:=framework
8 Bundle-RequiredExecutionEnvironment: J2SE-1.5
9 Export-Package: org.apache.commons.logging;version="1.1.1",
10 org.apache.commons.logging.impl;version="1.1.1"
因為我用的是commons-logging-1.1.1.jar,所以version寫的是1.1.1,大家可以修改成自己所使用的jar的版本。
將上面生成的三個jar放到OSGi-Web項目的WEB-INF/osgi/plugins目錄下面。還記得我在上一章創(chuàng)建的那個Tomcat Server么,clean一次,新的jar會部署到Tomcat中去,然后就可以運行Server了。
至于為什么是clean而不是publish,區(qū)別在于clean會清除所有OSGi容器創(chuàng)建出來的文件,這樣下次啟動OSGi時就會做一個install bundle的事情,而publish不會自動install新加進去的bundle。
如果你使用的是equinox,那么你可以在控制臺中看到Syetem Bundle現(xiàn)在多了幾個Fragments,查看一下Servlet Bundle,會顯示下列信息,表示servlet 2.4的package在OSGi容器中已經(jīng)可用了:
osgi> bundle 2
javax.servlet_extension_2.4.0 [2]
Id=2, Status=RESOLVED Data Root=D:\dbstar\workspaces\OSGi\.metadata\.plugins\org.eclipse.wst.server.core\tmp0\wtpwebapps\OSGi-Web\WEB-INF\osgi\configuration\org.eclipse.osgi\bundles\2\data
No registered services.
No services in use.
Exported packages
javax.servlet; version="2.4.0"[exported]
javax.servlet.http; version="2.4.0"[exported]
javax.servlet.resources; version="2.4.0"[exported]
No imported packages
Host bundles
org.eclipse.osgi_3.6.0.v20100128-1430 [0]
No named class spaces
No required bundles
最后提供幾個本章提到的bundle給大家下載,大家就不用自己再起生成一個了。
javax.servlet_extension_2.4.0.jar
javax.servlet.jsp_extension_2.0.0.jar
org.apache.commons.logging_extension_1.1.1.jar
系統(tǒng)不讓傳擴展名為.jar的文件,大家下載后把擴展名改改吧,阿門。