Hibernate 4 Multi-Tenancy и Spring 3 Hibernate

Все

Я оценивал функцию Multi-Tenancy, присутствующую в Hibernate 4 (4.1.0), вместе с Spring 3 (3.1.0), но не смог заставить ее работать с настройкой HibernateTransaction. Я определил настройки следующим образом.

LocalSessionFactoryBean:

@org.springframework.context.annotation.Configuration
public class Configuration {

    @Inject private DataSource dataSource;
    @Inject private MultiTenantConnectionProvider multiTenantConnectionProvider;

    @Bean
    public LocalSessionFactoryBean sessionFactory() throws IOException{
        LocalSessionFactoryBean bean = new LocalSessionFactoryBean();
        bean.setDataSource(dataSource);
        bean.setPackagesToScan("com");
        bean.getHibernateProperties().put("hibernate.multi_tenant_connection_provider", multiTenantConnectionProvider);
        bean.getHibernateProperties().put("hibernate.multiTenancy", "SCHEMA");
        bean.getHibernateProperties().put("hibernate.tenant_identifier_resolver", new CurrentTenantIdentifierResolverImpl());
        bean.setConfigLocation(new ClassPathResource("/hibernate.cfg.xml"));
        return bean;
    }

}

Configuration.xml:

<context:component-scan base-package="com.green" />


    <context:annotation-config />

    <!-- Enable annotation style of managing transactions -->
    <tx:annotation-driven transaction-manager="transactionManager" />

    <!-- Declare a datasource that has pooling capabilities -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
        destroy-method="close" p:driverClass="${app.jdbc.driverClassName}"
        p:jdbcUrl="${app.jdbc.url}" p:user="${app.jdbc.username}" p:password="${app.jdbc.password}"
        p:acquireIncrement="5" p:idleConnectionTestPeriod="60" p:maxPoolSize="100"
        p:maxStatements="50" p:minPoolSize="10" />

    <!-- Declare a transaction manager -->
    <bean id="transactionManager"
        class="com.green.saas.hibernate.SaasHibernateTransactionManager"
        depends-on="sessionFactory" >
          <property name="sessionFactory" ref="sessionFactory"></property>
          <property name="dataSource" ref="dataSource"></property>
    </bean>

Если я использую простой HibernateTransactionManager, предоставленный Spring 3, я не задаю идентификатор идентификатора ошибки, который не задан, причина для этого существа, он открывает сеанс следующим образом

  • Session newSession = SessionFactoryUtils.openSession(getSessionFactory( ));
  • (Session) ReflectionUtils.invokeMethod(openSessionMethod, sessionFactory) в openSession метод
  • Method openSessionMethod = ClassUtils.getMethod(SessionFactory.class, "openSession") выше аргумент openSessionMethod определяется как.

мы видим, что нет крюка, где вы можете предоставить идентификатор арендатора, как ожидалось, при открытии сеанса с идентификатором арендатора, например,

Session newSession = getSessionFactory().withOptions().tenantIdentifier ( "abc" ).openSession();

или

class instance provided by hibernate property "hibernate.tenant_identifier_resolver" is called by which session is provided with tenant identifier.

Чтобы преодолеть это, я расширил класс HibernateTransactionManager и переопределил метод doBegin и где открыт новый сеанс, я открыл его

getSessionFactory().withOptions().tenantIdentifier ( "abc" ).openSession();    

это заставляет щелкнуть и работать.

Я просто хотел знать, этот подход слишком хорош или есть некоторые настройки, о которых я не знаю, из-за чего он работает из коробки.

Спасибо заранее.

Спящий режим - 4.1.0 Spring - 3.1.0

Ответ 1

Похоже, эта функция была ошибочной на стороне Hibernate с используемой вами версией: SPR-9222, HHH-7306.

Начиная с Hibernate version 4.1.4 Вы должны использовать CurrentTenantIdentifierResolver, чтобы передать текущий идентификатор арендатора SessionFactory.