Mysql 与 Hibernate AbstractMultiTenantConnectionProvider 与 C3P0ConnectionProvider 抛出 GenericJDBCException : Could not open connection

标签 mysql hibernate multi-tenant c3p0

我正在致力于实现 multitenancy="DATABASE",即每个租户数据库/架构的单个连接池(mysql 数据库和架构是同义词)。

我有 MultiTenantConnectionProviderImpl.java

public class MultiTenantConnectionProviderImpl extends AbstractMultiTenantConnectionProvider implements
    ServiceRegistryAwareService {

private static final long serialVersionUID = 1234567890L;

private final HashMap<String, C3P0ConnectionProvider> connProviderMap = new HashMap<>();
private Map<String, String> originalSettings;
private ServiceRegistryImplementor serviceRegistry;

public MultiTenantConnectionProviderImpl() {
}

@Override
protected C3P0ConnectionProvider getAnyConnectionProvider() {
    return selectConnectionProvider(TenantContext.getTenantId());
}

@Override
protected C3P0ConnectionProvider selectConnectionProvider(String tenantId) {
    if (tenantId == null || !tenantId.equals(TenantContext.getTenantId())) {
        throw new TenantStateException(
                "Cannot get connection. Cause: tenant_id is not defined.");
    }

    C3P0ConnectionProvider connectionProvider = connProviderMap.get(tenantId);
    if (connectionProvider == null) {
        // create the new connection and register it
        Map<String, String> settings = new HashMap<>(originalSettings);
        // alter connection by changing user / password of the connection
        Properties properties = new PropertiesBuilder().withScope("database").build();
        settings.put("hibernate.connection.user", DatabaseConnectionProperties.getUser());
        settings.put("hibernate.connection.password", DatabaseConnectionProperties.getPassword());
        settings.put("hibernate.connection.url", DatabaseConnectionProperties.getConnectionUrl());
        settings.put("hibernate.connection.driver_class", properties.getProperty("hibernate.connection.driver_class"));

        connectionProvider = new C3P0ConnectionProvider();
        connectionProvider.injectServices(serviceRegistry);
        connectionProvider.configure(settings);
        connProviderMap.put(tenantId, connectionProvider);
    }

    return connectionProvider;
}

@Override
public void injectServices(ServiceRegistryImplementor serviceRegistry) {
    this.serviceRegistry = serviceRegistry;
    originalSettings = serviceRegistry.getService(ConfigurationService.class).getSettings();
    C3P0ConnectionProvider connectionProvider = new C3P0ConnectionProvider();
    connectionProvider.injectServices(serviceRegistry);
    connectionProvider.configure(originalSettings);
    connProviderMap.put(TenantContext.getTenantId(), connectionProvider);
}
}

连接用户如下

String.format("jdbc:mysql://%s/%s", databaseHost, databaseName)

这失败了

rg.hibernate.exception.GenericJDBCException: Could not open connection
    at org.hibernate.exception.internal.StandardSQLExceptionConverter.convert(StandardSQLExceptionConverter.java:54)
...

  at org.hibernate.engine.jdbc.internal.LogicalConnectionImpl.obtainConnection(LogicalConnectionImpl.java:297) ~[hibernate-core-4.1.12.Final.jar:4.1.12.Final]
    ... 43 common frames omitted
   Caused by:     com.mchange.v2.resourcepool.CannotAcquireResourceException: A ResourcePool  could not acquire a resource from its primary factory or source.

我尝试重写 getConnection(StringtenantId) 来执行“usetenantId”来获取正确的数据库。但是它没有帮助,我恢复到使用连接 url 指定数据库的原始方式。

还有其他人遇到过这个问题吗?

最佳答案

我最终实现了 MultiTenantConnectionProvider,首先连接到默认架构,然后对租户架构进行架构更改

public class MultiTenantConnectionProviderImpl implements MultiTenantConnectionProvider,
    ServiceRegistryAwareService {

private static final long serialVersionUID = 12345567890;

C3P0ConnectionProvider connectionProvider = null;

@Override
public void injectServices(ServiceRegistryImplementor serviceRegistry) {
    Map<String, String> originalSettings = serviceRegistry
        .getService(ConfigurationService.class).getSettings();
    connectionProvider = new C3P0ConnectionProvider();
    connectionProvider.injectServices(serviceRegistry);
    connectionProvider.configure(originalSettings);
}

@Override
public Connection getAnyConnection() throws SQLException {
    try {
        Class.forName("com.mysql.jdbc.Driver");
        final Connection connection = DriverManager.getConnection(
            DatabaseConnectionProperties.getConnectionUrl(),
            DatabaseConnectionProperties.getUser(), DatabaseConnectionProperties.getPassword());
        return connection;
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
    return connectionProvider.getConnection();
}

@Override
public void releaseAnyConnection(Connection connection) throws SQLException {
    try {
        connection.createStatement().execute("use default_tenant");
    }
    catch (SQLException e) {
        throw new HibernateException("Could not alter JDBC connection to specified schema [public]", e);
    }
    connectionProvider.closeConnection(connection);
}

@Override
public Connection getConnection(String tenantIdentifier) throws SQLException {
    final Connection connection = getAnyConnection();
    try {
        connection.createStatement().execute("use " + tenantIdentifier);
    }
    catch (SQLException e) {
        throw new HibernateException("Could not alter JDBC connection to specified schema ["
            + tenantIdentifier + "]", e);
    }
    return connection;
}

@Override
public void releaseConnection(String tenantIdentifier, Connection connection)
        throws SQLException {
    releaseAnyConnection(connection);
}

@Override
public boolean supportsAggressiveRelease() {
    return false;
}

@Override
public boolean isUnwrappableAs(Class unwrapType) {
    return false;
}

@Override
public <T> T unwrap(Class<T> unwrapType) {
    return null;
}
}

关于Mysql 与 Hibernate AbstractMultiTenantConnectionProvider 与 C3P0ConnectionProvider 抛出 GenericJDBCException : Could not open connection,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42369710/

相关文章:

MySQL 导入在 Mac 上的 Vagrant CoreOS 盒子上挂起

mysql - Microsoft Mysql Manager 中的截断错误

java - 组织.hibernate.HibernateException : Access to DialectResolutionInfo cannot be null when 'hibernate.dialect' not set

java - Hibernate HQL - query.list() 返回对象数组的对象数组

mysql - 第一个带有注释的 Hibernate 类

angular - keycloak-angular : Multi-tenant application

Mysql:如何实时显示表的新插入和更新?

postgresql - 是否可以在 PostgreSQL 中限制所有带有 tenant_id 的查询?

c# - Multi-Tenancy Asp.net Core网站中基于参数的JWT认证

mysql - 您可以使用 MySQL 连接到独立数据库吗? "mydata.db"?