具有动态数据源的Spring Boot JPA多租户

我正在尝试创建一个多租户Web应用程序,我找到了一个很好的教程here。这解释了如何配置MVC以找到新的租户(使用CurrentTenantIdentifierResolver和一个扩展HandlerInterceptorAdapter的MultiTenancyInterceptor),如何为三个不同的租户配置三个不同的数据源,以及如何通过扩展AbstractDataSourceBasedMultiTenantConnectionProviderImpl

在运行时为服务器提供正确的数据源

现在,这是一个很好的起点,让我了解在春季和休眠中多租户是如何工作的,但我想更进一步,我想让它成为完全动态的租户,也就是说,我不会假设应用程序可以有多少租户。

我是这么想的:

  • 应用程序配置为在引导时扫描路径(不在类路径中,例如/usr/data/config),并在各种目录(每个租户一个目录)下找到各种应用程序属性文件,例如tenantA、tenantB、tenantC...
  • 对于每个应用程序.properties,Spring Boot将基于该文件(该文件将只有引导属性spring.datource.url)创建一个数据源。请注意,使用Spring Boot的属性会很好,因为它为我提供了从单个URL所需的所有信息,如JDBC类等。
  • 我将在HashMap中注册每个数据源(如上一个链接所示)

之后,前面的链接中已经描述了基本的多租户结构:每次最终用户向浏览器发出请求时,服务器都会详细说明租户并返回正确的数据源,以查找要使用的数据库。

任何人都可以给我一些资源,如果这是以前做过的(我搜索了很多次,但没有什么能让我开始),或者给出一些关于使用哪些Spring类/配置来实现这一点的建议?

提前谢谢


解决方案

如果有人有这种需要,我最终就是这么做的。 欢迎对此进行任何进一步扩展,或对侵犯最佳实践的行为发表评论。

扩展AbstractDataSourceBasedMultiTenantConnectionProviderImplDataSourceProvider必须重写两个方法

  • selectAnyDataSource,它返回一个@Autowired DataSource,该@Autowired DataSource是由Spring使用为应用程序实例化数据源的常用方法实例化的。
  • selectDataSource(String tenant)执行以下操作:
    • 获取租户的配置文件夹的路径
    • 实例化DataSourceProperties,其属性取自租户的配置文件夹中的应用程序.properties文件
    • 通过DataSourceBuilder创建并返回一个新的数据源,将先前实例化的DataSourceProperties中的字段用作属性(这很有用,因为Spring从数据库URL动态提供驱动程序类名)

此处提供的代码,请随意使用:

String configPath = [...]; // Instantiate your configuration path
File file = new File(realPath);
DataSourceProperties dsProp = new DataSourceProperties();
Properties properties = new Properties();
try {
    properties.load(new FileInputStream(file));
} catch (IOException e) {
    throw new TenantNotConfiguredException(tenant); // Custom exception
}

PropertiesConfigurationFactory<DataSourceProperties> pcf = new PropertiesConfigurationFactory<>(dsProp);
pcf.setTargetName(DataSourceProperties.PREFIX);
pcf.setProperties(properties);

try {
    dsProp = pcf.getObject();
} catch (Exception e) {
    e.printStackTrace();
}

return DataSourceBuilder.create()
            .url(dsProp.getUrl())
            .driverClassName(dsProp.getDriverClassName())
            .username(dsProp.getUsername())
            .password(dsProp.getPassword())
            .build();

相关文章