Tomcat目录部署与Context描述文件context.xml

在Web应用开发完毕后,一些项目有时候采用目录部署的形式,而且是在server.xml中增加Context配置的形式,例如下面这种形式:

<Context path="/test" docBase="/home/abc/test"/>

但是官方并不鼓励这样配置,可以通过两种在外部文件配置的形式,不影响Tomcat主配置来实现同样的效果。

  1. $CATALINA_BASE/conf/[enginename]/[hostname]/[webappname].xml

  2. $CATALINA_BASE/webapps/[webappname]/META-INF/context.xml

前面曾写过Tomcat的应用部署(WEB应用是怎么被部署的?如何在Tomcat中部署应用的多个版本),当时没有详细介绍context.xml。

关于上面提到的这两种形式,内容一样,只是在不同的位置进行配置,例如上面在server.xml中配置的内容,可以在创建一个名为test.xml的文件,将其存放在Tomcat的conf/localhost/test.xml,即上面两项的第一种情况,这里注意应用的名称会直接使用文件名,path的配置会被忽略。

严格来说,第二种并不能直接替换前面server.xml里的情况,所能解决的是对于webapps目录内部署应用的额外配置,例如manager这个应用,要增加远程访问限制(为什么你的Manager登录不成功?),就是通过第二种,在META-INF目录内增加context.xml来实现的。

而实质上这两种统一称为Context的描述文件(Context Descriptor)。这些描述文件里包含一些应用对于Context的特定配置,例如配置Session Manager,指定naming Resrouce,定义SessionCookie名称等。使用描述文件,相当于给Context做了自定义,没提供描述文件的应用,Tomcat就会直接使用默认值初始化应用。

而且还有一点,使用描述文件定义的应用,会先部署。

下面来看源代码

源码

在HostConfig中,所有的部署从这儿开始。

protected void deployApps() {

    File appBase = host.getAppBaseFile();
    File configBase = host.getConfigBaseFile();
    String[] filteredAppPaths = filterAppPaths(appBase.list());
    // Deploy XML descriptors from configBase
    deployDescriptors(configBase, configBase.list());
    // Deploy WARs
    deployWARs(appBase, filteredAppPaths);
    // Deploy expanded folders
    deployDirectories(appBase, filteredAppPaths);

}

我们看到部署顺序:

  1. 部署XML描述文件

  2. 部署WAR应用

  3. 部署解压的Web应用

其中读取应用的目录就是各个Host对应的appBase对应的配置,我们常用的webapps目录是虚拟主机localhost默认对应的位置。

部署描述符,即第一种webappname.xml的时候,会从configBase中遍历所有以xml描述文件,然后部署之。

for (int i = 0; i < files.length; i++) {
    File contextXml = new File(configBase, files[i]);

    if (files[i].toLowerCase(Locale.ENGLISH).endsWith(".xml")) {
        ContextName cn = new ContextName(files[i], true);// 看这里

        if (isServiced(cn.getName()) || deploymentExists(cn.getName()))
            continue;

        results.add(
                es.submit(new DeployDescriptor(this, cn, contextXml)));
    }
}

for (Future<?> result : results) {
    try {
        result.get();
    } catch (Exception e) {
        log.error(sm.getString(
                "hostConfig.deployDescriptor.threaded.error"), e);
    }
}

其中注意ContextName生成那行代码,是直接使用了文件名进行Context的设置,ContextName我们前面文章看过源代码,这里配置的就是path,即实际请求的应用名称。

然后实际部署时,先通过Digester解析文件获取Context对象,再进行属性配置,之后的部署流程和其它的一致。

在StandardContext的初始化流程中,会判断获取之前设置的configFile对象,不为空会解析文件内容。

而我们上面的第二种方式:为应用提供context.xml,在Tomcat代码内,称之为ApplicationContextXml。

这个文件是在第二和第三个部署的时候,解析META-INF目录下是否包含context.xml,然后设置Context的configFile属性。初始化Context的时候,会有如下逻辑

if (context.getConfigFile() != null) {
    processContextConfig(digester, context.getConfigFile());
}

这个就是前面webappname.xml解析时设置configFile之后走的流程,在ContextConfig中。Digester的解析请看前面的文章:Tomcat配置文件解析与Digester

这样一来,我们的自定义配置文件就在Tomcat中生效了。

微信内代码展示不太好,可以点击阅读原文查看网页内容。

关注Tomcat那些事儿,发现更多精彩文章!了解各种常见问题背后的原理与答案。深入源码,分析细节,内容原创,欢迎关注。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值