我的客户希望在运行时具有动态模式。我现在所做的就是这样:
对于我的web.xml(仅重要部分):
<servlet>
<servlet-name>servlet1</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>servlet1</servlet-name>
<url-pattern>/myservlet/sample1/</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>servlet2</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>2</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>servlet2</servlet-name>
<url-pattern>/myservlet/sample2/</url-pattern>
</servlet-mapping>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
/WEB-INF/servlet1-servlet.xml
/WEB-INF/servlet2-servlet.xml
</param-value>
</context-param>
... some properties ...
<resource-ref>
<res-ref-name>DATASOURCE_1</res-ref-name>
<res-type>javax.sql.DataSource</res-type>
<res-auth>Container</res-auth>
<res-sharing-scope>Shareable</res-sharing-scope>
</resource-ref>
<resource-ref>
<res-ref-name>DATASOURCE_2</res-ref-name>
<res-type>javax.sql.DataSource</res-type>
<res-auth>Container</res-auth>
<res-sharing-scope>Shareable</res-sharing-scope>
</resource-ref>
对于我的servlet1-servlet.xml(除了JNDI名称及其默认架构,servlet2-servlet.xml的内容相同)
<bean id="dataSource" class="org.springframework.jndi.JndiObjectFactoryBean">
<property name="jndiName" value="DATASOURCE_1" />
<property name="lookupOnStartup" value="false" />
<property name="cache" value="true" />
<property name="proxyInterface" value="javax.sql.DataSource" />
</bean>
<bean id="entityManagerFactory"
class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
<property name="dataSource" ref="dataSource" />
<property name="jpaVendorAdapter" ref="jpaVendorAdapter" />
<property name="jpaDialect" ref="jpaDialect" />
<property name="packagesToScan">
<array>
<value>ph.prj.domain.models</value>
</array>
</property>
<property name="jpaProperties">
<props>
<prop key="hibernate.default_schema">SCHEMA1</prop>
</props>
</property>
</bean>
<bean id="jpaVendorAdapter"
class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
<property name="database" value="${jdbc.databaseName}" />
<property name="databasePlatform" value="${hibernate.dialect}" />
<property name="generateDdl" value="${init-db}" />
<property name="showSql" value="${hibernate.show_sql}" />
</bean>
<bean id="jpaDialect" class="org.springframework.orm.jpa.vendor.HibernateJpaDialect" />
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
<property name="entityManagerFactory" ref="entityManagerFactory" />
<property name="dataSource" ref="dataSource" />
<property name="jpaDialect" ref="jpaDialect" />
</bean>
最后是我在运行时处理动态jndi名称的方法:
protected void setCurrentSchema(String identifier) throws Exception {
if (StringUtils.equalsIgnoreCase(IdentifierEnum.SCHEMA_1.getValue(), identifier) ||
StringUtils.equalsIgnoreCase(IdentifierEnum.SCHEMA_2.getValue(), identifier)) {
// WEBSPHERE
JndiObjectFactoryBean dataSource = context.getBean(JndiObjectFactoryBean.class);
dataSource.setJndiName(ApplicationConstants.JNDI_ROOT_NAME + identifier);
} else {
logger.debug(String.format("INVALID identifier SPECIFIED - %s", identifier));
throw new RequestParamException(String.format("INVALID identifier SPECIFIED - %s", identifier));
}
}
为了访问我的api,我需要使用ff进行调用。上下文路径:
/ myservlet / sample1 /
/ myservlet / sample2 /
不幸的是,我的客户不想在上下文路径中使用sample1 / sample2部分。
客户想要访问
/ myservlet /
仅,与架构名称无关。
我正计划利用此link中的 AbstractRoutingDataSource 但是然后,该示例使用的是 DriverManagerDataSource 而不是JndiObjectFactoryBean,那我找不到如何动态设置模式的方法。
我觉得这些方法可以帮助我实现所需的输出 setJndiEnvironment / setJndiTemplate ,但是,他们缺少一些使用它的示例,我找不到一些使用这些方法更改其架构的资源。
希望你们能帮我这个忙。
答案 0 :(得分:1)
您可以在运行时在 JpaTransactionManager 和 LocalContainerEntityManagerManagerBean bean上动态设置 jpaProperties 。
protected void setCurrentSchema(String identifier) throws Exception {
if (StringUtils.equalsIgnoreCase(IdentifierEnum.SCHEMA_1.getValue(), identifier) ||
StringUtils.equalsIgnoreCase(IdentifierEnum.SCHEMA_2.getValue(), identifier)) {
// WEBSPHERE
Properties jpaProperties = new Properties();
jpaProperties.put("hibernate.default_schema", identifier);
LocalContainerEntityManagerFactoryBean entityManagerFactory = context.getBean(LocalContainerEntityManagerFactoryBean.class);
entityManagerFactory.setJpaProperties(jpaProperties);
entityManagerFactory.afterPropertiesSet();
JpaTransactionManager transactionManager = context.getBean(JpaTransactionManager.class);
transactionManager.setJpaProperties(jpaProperties);
transactionManager.afterPropertiesSet();
} else {
logger.debug(String.format("INVALID identifier SPECIFIED - %s", identifier));
throw new RequestParamException(String.format("INVALID identifier SPECIFIED - %s", identifier));
}
}
有了这个,您可以只定义一个数据源,并且不需要默认模式,因为它将在运行时设置。您还可以摆脱映射到每个数据库模式的多个* servlet.xml文件,并且大多数情况下,可以消除应用程序上下文路径中的多余部分