java - Tomcat多个数据源与解密密码逻辑

标签 java datasource tomcat8.5

我在 Tomcat 环境中使用多个数据源时遇到一些问题。请在下面找到详细信息。 我的 tomcat/conf/server.xml 文件中有以下 2 个数据源

数据来源1:

<Resource name="myds1" 
          global="myds1"
          auth="Container"
          type="javax.sql.DataSource"
          factory="com.tomcat.datasorceEncrypt.EncryptedDataSourceFactory"
          driverClassName="oracle.jdbc.driver.OracleDriver"
          singleton = "false"/>

数据来源2:

<Resource name="myds2" 
          global="myds2"
          auth="Container"
          type="javax.sql.DataSource"
          factory="com.tomcat.datasorceEncrypt.EncryptedDataSourceFactory"
          driverClassName="com.ibm.db2.jcc.DB2Driver"
          singleton = "false"/>

这是我的EncryptedDataSourceFactory扩展 DataSourceFactory 的文件:

package com.tomcat.datasorceEncrypt;

import java.io.InputStream;
import java.util.Hashtable;
import java.util.Properties;
import java.util.stream.Stream;

import javax.naming.Context;
import javax.naming.Name;
import javax.naming.RefAddr;
import javax.naming.Reference;
import javax.sql.DataSource;

import org.apache.juli.logging.Log;
import org.apache.juli.logging.LogFactory;
import org.apache.tomcat.jdbc.pool.DataSourceFactory;
import org.apache.tomcat.jdbc.pool.PoolConfiguration;
import org.apache.tomcat.jdbc.pool.XADataSource;

public class EncryptedDataSourceFactory extends DataSourceFactory {

    private static final Log log = LogFactory.getLog(EncryptedDataSourceFactory.class);

    private static final String PROP_DIALECT = "dialect";
    private static final String[] CUSTOM_PROPERTIES = new String[]{PROP_DIALECT};
    private static final String[] PROPERTIES = Stream.of(ALL_PROPERTIES, CUSTOM_PROPERTIES).flatMap(Stream::of).toArray(String[]::new);


    @Override
    public Object getObjectInstance(Object obj, Name name, Context nameCtx, Hashtable<?, ?> environment)
            throws Exception {
         if (obj != null && obj instanceof Reference) {
             Reference ref = (Reference) obj;
             Properties properties = new Properties();

             for (int i = 0; i < PROPERTIES.length; ++i) {
                 String propertyName = PROPERTIES[i];
                 RefAddr ra = ref.get(propertyName);
                 if (ra != null) {
                     String propertyValue = ra.getContent().toString();
                     properties.setProperty(propertyName, propertyValue);
                 }
             }

             return this.createDataSource(properties, nameCtx,false);
         } else {
             return null;
         }
    }

    @Override
    public DataSource createDataSource(Properties properties, Context context, boolean XA) throws Exception {
        // Here we decrypt our password.
        PoolConfiguration poolProperties = parsePoolProperties(properties);
        Properties dbProperties = loadProperties();
        poolProperties.setPassword(CryptoUtility.decryptDBPass(dbProperties.getProperty("DB_key"), dbProperties.getProperty("DB_password")));
        poolProperties.setUsername(dbProperties.getProperty("DB_username"));
        poolProperties.setUrl(dbProperties.getProperty("DB_url"));
        System.out.println(poolProperties.getPoolName() + "****-------*****" + poolProperties.getName() );
        System.out.println(poolProperties.getDataSourceJNDI() + "****------***" + poolProperties.getDataSource());
        // The rest of the code is copied from Tomcat's DataSourceFactory.
        if (poolProperties.getDataSourceJNDI() != null && poolProperties.getDataSource() == null) {
            performJNDILookup(context, poolProperties);
        }
        org.apache.tomcat.jdbc.pool.DataSource dataSource = XA ? new XADataSource(poolProperties)
                : new org.apache.tomcat.jdbc.pool.DataSource(poolProperties);
        dataSource.createPool();

        return dataSource;
    }

    private Properties loadProperties() {
        Properties prop = new Properties();

        try {
            InputStream inputStream = this.getClass().getClassLoader().getResourceAsStream("DBPassword.properties");
            prop.load(inputStream);
        }catch (Exception e) {
            log.fatal("Error Loading the properties.", e);
            throw new RuntimeException(e);
        }
        return prop;
    }

}

我试图从 poolProperties.getDataSourceJNDI() 中识别数据源 JNDI 名称这样我就可以通过我的属性应用适当的凭据,但我收到 poolProperties.getDataSourceJNDI()为空。

创建数据源时我是否错过了资源中的任何属性?

注意:在使用资源时,我可以设置用户名和密码,尽管我有 poolProperties.getDataSourceJNDI()为空。

最佳答案

对于 timebeeing,我通过扩展第二个数据源的 DatasourceFactory 又创建了一个类,它正在工作,但我不认为这是一个空闲的解决方案。

关于java - Tomcat多个数据源与解密密码逻辑,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59371174/

相关文章:

java - if 语句表现不同

java - 类似于 Applaud 的 Phonegap Eclipse 插件

Java 协方差

java - 在 Spring 中使用 Hibernate 的后备数据源

Java `InvocationTargetException` 通过反射进行类实例化

.net - 如何将两个组合框的所选项目与单个DataSource分开?

java - 使用 Windows 集成身份验证将 JBoss AS 7 数据源连接到 SQL Server

java - 使用 tomcat7-maven-plugin 部署 Tomcat 失败,错误为 "Cannot invoke Tomcat manager: Connection reset by peer: socket write error"

linux - 在 Redhat Linux 2.6.32 上使用带有 HTTP/2 的 Tomcat 的损坏页面

java-8 - 支持 Java 8 并在 Tomcat 8.5 上运行的 Oracle Jdbc 驱动程序版本