使用Spring MVC和Hibernate 4.2.0.Final的多租户webapp

时间:2013-12-17 07:06:55

标签: spring hibernate spring-mvc spring-security multi-tenant

我使用SpringMVC(3.1.3.RELEASE)和Hibernate 4.2.0.Final开发了一个小型webapp。

我正在尝试将其转换为多租户应用程序。

其他主题已涵盖类似的主题,但我无法找到解决问题的明确方法。

我想要实现的是设计一个能够:

的网络应用程序
  1. 在启动时读取数据源配置(包含多个数据源定义的XML文件,它位于WAR文件之外,而不是应用程序上下文或hibernate配置文件)

  2. 为每一个创建会话工厂(考虑到每个数据源都是具有不同架构的数据库)。

  3. 如何将会话工厂范围设置为会话? (或者我可以重用相同的会话工厂吗?)。

  4. 示例:

     Url for client a - URL: http://project.com/a/login.html
     Url for client b - URL: http://project.com/b/login.html
    

    如果客户端“a”发出请求,请阅读数据源配置文件并使用该XML文件为客户端“a”创建会话工厂。

    如果客户“b”将发送请求,则会重复相同的过程。

    我在寻找,如何在不编辑Spring配置文件的情况下在客户订阅时实现数据源创建。它需要自动化。

    这是我的代码,到目前为止我已经完成了。

    请有人告诉我,我需要做哪些修改?

    请用一些示例代码给出答案。我在春天和休眠世界都很新。

    Spring.xml

      <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"
            destroy-method="close" p:driverClassName="${jdbc.driverClassName}"
    
            p:url="${jdbc.databaseurl}" 
    p:username="${jdbc.username}" p:password="${jdbc.password}" />
    
        <bean id="sessionFactory"
            class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
            <property name="dataSource" ref="dataSource" />
            <property name="configLocation">
                <value>classpath:hibernate.cfg.xml</value>
            </property>
    
            <property name="hibernateProperties">
                <props>
                    <prop key="hibernate.dialect">${jdbc.dialect}</prop>
                    <prop key="hibernate.show_sql">true</prop>
                </props>
            </property>
        </bean>
    
    <bean id="transactionManager"
            class="org.springframework.orm.hibernate4.HibernateTransactionManager">
            <property name="sessionFactory" ref="sessionFactory" />
    

    JDBC.properties文件

    jdbc.driverClassName=com.mysql.jdbc.Driver
    jdbc.dialect=org.hibernate.dialect.MySQLDialect
    jdbc.databaseurl=jdbc:mysql://localhost:3306/Logistics
    jdbc.username=root
    jdbc.password=rot@pspl#12
    

    hibernate.cfg.xml文件

    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE hibernate-configuration PUBLIC
        "-//Hibernate/Hibernate Configuration DTD//EN"
        "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
    
    <hibernate-configuration>
        <session-factory>
        <mapping class="pepper.logis.organizations.model.Organizaions" />
        <mapping class="pepper.logis.assets.model.Assets" />
    
    </session-factory>
    </hibernate-configuration>
    

    谢谢,

3 个答案:

答案 0 :(得分:1)

首先使用tenant_id为Tenant创建一个表,并将其与所有用户关联。现在,您可以在用户登录时获取此详细信息并将其设置为会话。

答案 1 :(得分:0)

想知道你最终得到了什么。

以下是一些想法。

选项1)单个应用程序实例。

使用你实际想要实现的目标来实现这一目标有点雄心勃勃 要点是在同一个JVM上简单地使用不同的上下文根部署相同的应用程序。如果您拥有真正的多租户应用程序,您仍然可以像整体调整JVM一样。但这是以重复类,上下文,本地缓存,启动时间等为代价的。

但是截至今天,Spring Framework 4.0并没有提供太多的多租户支持(Hot Swapable目标/数据源除外)等等。我正在寻找一个好的框架,但它可能是一个摆脱Spring在这个时候对我来说。

选项2)同一应用程序的多个部署(比今天更实用)

将相同的应用程序部署到同一个应用程序服务器JVM实例,甚至是不同的。

如果您使用相同的实例,您现在可能需要根据实例应提供的内容来引导您的应用程序以获取DataSource,例如client =一个属性足以拾取** a ** DataSource&#34;或者** b ** DataSource我自己最终采用了这种方法。

如果您有不同的应用程序服务器实例,则可以配置不同的JNDI路径并对其进行一般处理。不需要客户=&#34; a&#34;属性,因为您可以使用相同的名称以不同方式定义数据源。

答案 2 :(得分:0)

我们正在使用AbstractRoutingDataSourceSpring Boot上的每个请求切换DataSource。我认为上面的@bhantol提到了Hot Swapable targets/datasource

它解决了我们的问题,但我不认为这是合理的解决方案。我猜JNDI可能比AbstractRoutingDataSource更好。