我正在使用 Tomcat9 部署共享相同数据源连接池的多个应用程序。我通过以下方式实现了目的:
- 在 conf/server.xml 中的 GlobalNamingResources 下添加数据源作为 Resource 元素,
- 使用 conf/context.xml 中的 ResourceLink 元素链接全局 JNDI 资源,使其对所有 Web 应用程序可用,并且
- 从 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/