Tomcat:如何以编程方式在 GlobalNamingResources 中定义数据源资源并将其链接到应用程序

标签 tomcat datasource tomcat9

我正在使用 Tomcat9 部署共享相同数据源连接池的多个应用程序。我通过以下方式实现了目的:

  1. 在 conf/server.xml 中的 GlobalNamingResources 下添加数据源作为 Resource 元素,
  2. 使用 conf/context.xml 中的 ResourceLink 元素链接全局 JNDI 资源,使其对所有 Web 应用程序可用,并且
  3. 从 Web 应用程序使用 JNDI 访问数据源。

它按预期工作。

但是我需要避免编辑任何 xml(server.xml 或 conf/context.xml 或特定于应用程序的 context.xml)并尝试以编程方式实现步骤 1 和 2。如果是这样,优点是,只需要修改一个需要添加到/lib 中的 jar 以支持新的数据源,而绝对不需要对任何 xml 进行修改(或者可能在 server.xml 中进行最小配置)。

有什么方法可以实现将资源定义为 GlobalNamingResources 并通过 ResourceLink 将其链接到 Web 应用程序的要求,全部通过 Java?

最佳答案

通过 tomcat javadocs 后,我已经达到了我的要求。详情如下:

<强>1。添加数据源作为 GlobalNamingResource

为此,创建一个服务器生命周期监听器,创建一个数据源并将其添加到 JNDI GlobalNamingContext。

public class GlobalDatasourceCreator implements LifecycleListener {
    @Override
    public void lifecycleEvent(LifecycleEvent event) {
        if (event.getSource() instanceof Server) {
            Context namingContext = ((Server) event.getSource()).getGlobalNamingContext();
            if (Lifecycle.START_EVENT.equals(event.getType())) {
                bindDatasources(namingContext);
            } else if (Lifecycle.STOP_EVENT.equals(event.getType())) {
                unbindDatasources(namingContext);
            }
        }
    } 

    private void bindDatasources(Context namingContext) {
        if (createSubContext(namingContext)) {
            try {
                DataSource ds = getDatasource(); //TODO: Implement it 
                namingContext.rebind("jdbc/myds_global", ds);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    private boolean createSubContext(Context namingContext) {
        try {
            namingContext.createSubcontext("jdbc");
        } catch (NameAlreadyBoundException e) {
        } catch (NamingException e) {
            return false;
        }
        return true;
    }

    private void unbindDatasources(Context namingContext) {
        try {
            namingContext.unbind("jdbc/myds_global");
        } catch (NamingException e) {
            e.printStackTrace();
        }
    }

然后将这个类作为监听器添加到conf/server.xml

<Listener className="com.test.GlobalDatasourceCreator" />

<强>2。通过 ResourceLink 向所有 Web 应用程序公开数据源

创建上下文 LifecycleListener。在 START 事件中,创建 ResourceLink 并将其附加到上下文。

注意:由于这是上下文级别的监听器,因此将为所有应用程序创建 ResourceLink。我的要求是将它暴露给所有应用程序,因为它是一个受控环境。如果只需要为选定的应用程序创建 ResourceLink,则可以应用基于上下文名称的过滤。

public class AppDatasourceLinkCreator implements LifecycleListener {

    @Override
    public void lifecycleEvent(LifecycleEvent event) {
        if (event.getSource() instanceof Context) {
            Context ctx = (Context) event.getSource();
            if (Lifecycle.START_EVENT.equals(event.getType())) {
                addResourceLink(ctx);
            } else if (Lifecycle.STOP_EVENT.equals(event.getType())) {
                removeResourceLink(ctx);
            }
        } 
    }

    private void removeResourceLink(Context ctx) {
        ctx.getNamingResources().removeResourceLink("jdbc/myds");
    }

    private void addResourceLink(Context ctx) {
        ContextResourceLink resourceLink = new ContextResourceLink();
        resourceLink.setGlobal("jdbc/myds_global");
        resourceLink.setName("jdbc/myds");
        resourceLink.setType("javax.sql.DataSource");
        ctx.getNamingResources().addResourceLink(resourceLink);
    }
}

然后将这个类作为监听器添加到conf/context.xml

    <Listener className="com.test.AppDatasourceLinkCreator" />

创建一个包含这两个类的 jar 并将其放在/lib 文件夹中。

优点:无需进一步修改 xml 即可添加任意数量的数据源。只需修改java代码添加新数据源,更新lib文件夹中的jar并重启服务器,这完全符合我的项目要求。这也解决了在 xml 中以纯文本形式公开数据源凭据的问题(尽管不是 100% 的风险证明)。

关于Tomcat:如何以编程方式在 GlobalNamingResources 中定义数据源资源并将其链接到应用程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58407349/

相关文章:

java - Spring 启动 : How to skip datasource configuration running tests

performance - 引入更多过滤器是否会降低Tomcat的性能?

apache - 不要记录 tomcat localhost_access_log 的特定 url 模式

spring - Spring Boot 中的多个数据源和模式创建

rest - JasperServer 使用 REST 来运行在运行时指定数据源的报告

java - 无需指定端口 8080 即可运行 Tomcat

java - 如何修复 Tomcat 9 上的部署错误 - Java 动态 Web 项目中的 java.lang.IllegalArgumentException?

java - Tomcat 9 中的编码问题 "The valid characters are defined in RFC 7230 and RFC 3986"

java - 带有 Tomcat 插件的简单 Gradle 项目不起作用

tomcat - 部署 Grails 应用程序时出错 : java. lang.NoClassDefFoundError: org/codehaus/groovy/runtime/BytecodeInterface8