Hibernate 4 Multi-Tenancy и Spring 3 Hibernate - PullRequest
11 голосов
/ 14 февраля 2012

Все

Я оценивал функцию 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. У меня не указан идентификатор клиента ошибки, причина этого в том, что он открывает сессию следующим образом

  1. Session newSession = SessionFactoryUtils.openSession(getSessionFactory( ));
  2. (Session) ReflectionUtils.invokeMethod(openSessionMethod, sessionFactory) in openSession method
  3. 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();    

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

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

Заранее спасибо.

Hibernate - 4.1.0 Spring - 3.1.0

1 Ответ

5 голосов
/ 11 сентября 2012

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

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

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...