Jpa/JdoTransactionManager passes resolved timeout into Jpa/JdoDialect's beginTransaction; HibernateJpaDialect applies timeout onto native Hibernate Transaction before begin call (SPR-5195)
This commit is contained in:
parent
414d6633ea
commit
45dc856993
|
|
@ -35,6 +35,7 @@ import org.springframework.transaction.TransactionException;
|
||||||
import org.springframework.transaction.TransactionSystemException;
|
import org.springframework.transaction.TransactionSystemException;
|
||||||
import org.springframework.transaction.support.AbstractPlatformTransactionManager;
|
import org.springframework.transaction.support.AbstractPlatformTransactionManager;
|
||||||
import org.springframework.transaction.support.DefaultTransactionStatus;
|
import org.springframework.transaction.support.DefaultTransactionStatus;
|
||||||
|
import org.springframework.transaction.support.DelegatingTransactionDefinition;
|
||||||
import org.springframework.transaction.support.ResourceTransactionManager;
|
import org.springframework.transaction.support.ResourceTransactionManager;
|
||||||
import org.springframework.transaction.support.TransactionSynchronizationManager;
|
import org.springframework.transaction.support.TransactionSynchronizationManager;
|
||||||
|
|
||||||
|
|
@ -299,7 +300,7 @@ public class JdoTransactionManager extends AbstractPlatformTransactionManager
|
||||||
"on a single DataSource, no matter whether JDO or JDBC access.");
|
"on a single DataSource, no matter whether JDO or JDBC access.");
|
||||||
}
|
}
|
||||||
|
|
||||||
PersistenceManager pm = null;
|
PersistenceManager pm;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (txObject.getPersistenceManagerHolder() == null ||
|
if (txObject.getPersistenceManagerHolder() == null ||
|
||||||
|
|
@ -314,13 +315,19 @@ public class JdoTransactionManager extends AbstractPlatformTransactionManager
|
||||||
pm = txObject.getPersistenceManagerHolder().getPersistenceManager();
|
pm = txObject.getPersistenceManagerHolder().getPersistenceManager();
|
||||||
|
|
||||||
// Delegate to JdoDialect for actual transaction begin.
|
// Delegate to JdoDialect for actual transaction begin.
|
||||||
Object transactionData = getJdoDialect().beginTransaction(pm.currentTransaction(), definition);
|
final int timeoutToUse = determineTimeout(definition);
|
||||||
|
Object transactionData = getJdoDialect().beginTransaction(pm.currentTransaction(),
|
||||||
|
new DelegatingTransactionDefinition(definition) {
|
||||||
|
@Override
|
||||||
|
public int getTimeout() {
|
||||||
|
return timeoutToUse;
|
||||||
|
}
|
||||||
|
});
|
||||||
txObject.setTransactionData(transactionData);
|
txObject.setTransactionData(transactionData);
|
||||||
|
|
||||||
// Register transaction timeout.
|
// Register transaction timeout.
|
||||||
int timeout = determineTimeout(definition);
|
if (timeoutToUse != TransactionDefinition.TIMEOUT_DEFAULT) {
|
||||||
if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
|
txObject.getPersistenceManagerHolder().setTimeoutInSeconds(timeoutToUse);
|
||||||
txObject.getPersistenceManagerHolder().setTimeoutInSeconds(timeout);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Register the JDO PersistenceManager's JDBC Connection for the DataSource, if set.
|
// Register the JDO PersistenceManager's JDBC Connection for the DataSource, if set.
|
||||||
|
|
@ -328,8 +335,8 @@ public class JdoTransactionManager extends AbstractPlatformTransactionManager
|
||||||
ConnectionHandle conHandle = getJdoDialect().getJdbcConnection(pm, definition.isReadOnly());
|
ConnectionHandle conHandle = getJdoDialect().getJdbcConnection(pm, definition.isReadOnly());
|
||||||
if (conHandle != null) {
|
if (conHandle != null) {
|
||||||
ConnectionHolder conHolder = new ConnectionHolder(conHandle);
|
ConnectionHolder conHolder = new ConnectionHolder(conHandle);
|
||||||
if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
|
if (timeoutToUse != TransactionDefinition.TIMEOUT_DEFAULT) {
|
||||||
conHolder.setTimeoutInSeconds(timeout);
|
conHolder.setTimeoutInSeconds(timeoutToUse);
|
||||||
}
|
}
|
||||||
if (logger.isDebugEnabled()) {
|
if (logger.isDebugEnabled()) {
|
||||||
logger.debug("Exposing JDO transaction as JDBC transaction [" + conHolder.getConnectionHandle() + "]");
|
logger.debug("Exposing JDO transaction as JDBC transaction [" + conHolder.getConnectionHandle() + "]");
|
||||||
|
|
|
||||||
|
|
@ -42,6 +42,7 @@ import org.springframework.transaction.TransactionException;
|
||||||
import org.springframework.transaction.TransactionSystemException;
|
import org.springframework.transaction.TransactionSystemException;
|
||||||
import org.springframework.transaction.support.AbstractPlatformTransactionManager;
|
import org.springframework.transaction.support.AbstractPlatformTransactionManager;
|
||||||
import org.springframework.transaction.support.DefaultTransactionStatus;
|
import org.springframework.transaction.support.DefaultTransactionStatus;
|
||||||
|
import org.springframework.transaction.support.DelegatingTransactionDefinition;
|
||||||
import org.springframework.transaction.support.ResourceTransactionManager;
|
import org.springframework.transaction.support.ResourceTransactionManager;
|
||||||
import org.springframework.transaction.support.TransactionSynchronizationManager;
|
import org.springframework.transaction.support.TransactionSynchronizationManager;
|
||||||
import org.springframework.util.CollectionUtils;
|
import org.springframework.util.CollectionUtils;
|
||||||
|
|
@ -327,13 +328,19 @@ public class JpaTransactionManager extends AbstractPlatformTransactionManager
|
||||||
EntityManager em = txObject.getEntityManagerHolder().getEntityManager();
|
EntityManager em = txObject.getEntityManagerHolder().getEntityManager();
|
||||||
|
|
||||||
// Delegate to JpaDialect for actual transaction begin.
|
// Delegate to JpaDialect for actual transaction begin.
|
||||||
Object transactionData = getJpaDialect().beginTransaction(em, definition);
|
final int timeoutToUse = determineTimeout(definition);
|
||||||
|
Object transactionData = getJpaDialect().beginTransaction(em,
|
||||||
|
new DelegatingTransactionDefinition(definition) {
|
||||||
|
@Override
|
||||||
|
public int getTimeout() {
|
||||||
|
return timeoutToUse;
|
||||||
|
}
|
||||||
|
});
|
||||||
txObject.setTransactionData(transactionData);
|
txObject.setTransactionData(transactionData);
|
||||||
|
|
||||||
// Register transaction timeout.
|
// Register transaction timeout.
|
||||||
int timeout = determineTimeout(definition);
|
if (timeoutToUse != TransactionDefinition.TIMEOUT_DEFAULT) {
|
||||||
if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
|
txObject.getEntityManagerHolder().setTimeoutInSeconds(timeoutToUse);
|
||||||
txObject.getEntityManagerHolder().setTimeoutInSeconds(timeout);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Register the JPA EntityManager's JDBC Connection for the DataSource, if set.
|
// Register the JPA EntityManager's JDBC Connection for the DataSource, if set.
|
||||||
|
|
@ -341,8 +348,8 @@ public class JpaTransactionManager extends AbstractPlatformTransactionManager
|
||||||
ConnectionHandle conHandle = getJpaDialect().getJdbcConnection(em, definition.isReadOnly());
|
ConnectionHandle conHandle = getJpaDialect().getJdbcConnection(em, definition.isReadOnly());
|
||||||
if (conHandle != null) {
|
if (conHandle != null) {
|
||||||
ConnectionHolder conHolder = new ConnectionHolder(conHandle);
|
ConnectionHolder conHolder = new ConnectionHolder(conHandle);
|
||||||
if (timeout != TransactionDefinition.TIMEOUT_DEFAULT) {
|
if (timeoutToUse != TransactionDefinition.TIMEOUT_DEFAULT) {
|
||||||
conHolder.setTimeoutInSeconds(timeout);
|
conHolder.setTimeoutInSeconds(timeoutToUse);
|
||||||
}
|
}
|
||||||
if (logger.isDebugEnabled()) {
|
if (logger.isDebugEnabled()) {
|
||||||
logger.debug("Exposing JPA transaction as JDBC transaction [" + conHolder.getConnectionHandle() + "]");
|
logger.debug("Exposing JPA transaction as JDBC transaction [" + conHolder.getConnectionHandle() + "]");
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2008 the original author or authors.
|
* Copyright 2002-2009 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -18,7 +18,6 @@ package org.springframework.orm.jpa.vendor;
|
||||||
|
|
||||||
import java.sql.Connection;
|
import java.sql.Connection;
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
|
|
||||||
import javax.persistence.EntityManager;
|
import javax.persistence.EntityManager;
|
||||||
import javax.persistence.PersistenceException;
|
import javax.persistence.PersistenceException;
|
||||||
|
|
||||||
|
|
@ -38,7 +37,7 @@ import org.springframework.transaction.TransactionException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@link org.springframework.orm.jpa.JpaDialect} implementation for
|
* {@link org.springframework.orm.jpa.JpaDialect} implementation for
|
||||||
* Hibernate EntityManager. Developed and tested against Hibernate 3.2.
|
* Hibernate EntityManager. Developed and tested against Hibernate 3.3.
|
||||||
*
|
*
|
||||||
* @author Costin Leau
|
* @author Costin Leau
|
||||||
* @author Juergen Hoeller
|
* @author Juergen Hoeller
|
||||||
|
|
@ -50,6 +49,9 @@ public class HibernateJpaDialect extends DefaultJpaDialect {
|
||||||
public Object beginTransaction(EntityManager entityManager, TransactionDefinition definition)
|
public Object beginTransaction(EntityManager entityManager, TransactionDefinition definition)
|
||||||
throws PersistenceException, SQLException, TransactionException {
|
throws PersistenceException, SQLException, TransactionException {
|
||||||
|
|
||||||
|
if (definition.getTimeout() != TransactionDefinition.TIMEOUT_DEFAULT) {
|
||||||
|
getSession(entityManager).getTransaction().setTimeout(definition.getTimeout());
|
||||||
|
}
|
||||||
super.beginTransaction(entityManager, definition);
|
super.beginTransaction(entityManager, definition);
|
||||||
return prepareTransaction(entityManager, definition.isReadOnly(), definition.getName());
|
return prepareTransaction(entityManager, definition.isReadOnly(), definition.getName());
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2008 the original author or authors.
|
* Copyright 2002-2009 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -18,7 +18,7 @@ package org.springframework.transaction.interceptor;
|
||||||
|
|
||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
|
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.transaction.support.DelegatingTransactionDefinition;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@link TransactionAttribute} implementation that delegates all calls to a given target
|
* {@link TransactionAttribute} implementation that delegates all calls to a given target
|
||||||
|
|
@ -29,7 +29,8 @@ import org.springframework.util.Assert;
|
||||||
* @author Juergen Hoeller
|
* @author Juergen Hoeller
|
||||||
* @since 1.2
|
* @since 1.2
|
||||||
*/
|
*/
|
||||||
public abstract class DelegatingTransactionAttribute implements TransactionAttribute, Serializable {
|
public abstract class DelegatingTransactionAttribute extends DelegatingTransactionDefinition
|
||||||
|
implements TransactionAttribute, Serializable {
|
||||||
|
|
||||||
private final TransactionAttribute targetAttribute;
|
private final TransactionAttribute targetAttribute;
|
||||||
|
|
||||||
|
|
@ -39,31 +40,11 @@ public abstract class DelegatingTransactionAttribute implements TransactionAttri
|
||||||
* @param targetAttribute the target TransactionAttribute to delegate to
|
* @param targetAttribute the target TransactionAttribute to delegate to
|
||||||
*/
|
*/
|
||||||
public DelegatingTransactionAttribute(TransactionAttribute targetAttribute) {
|
public DelegatingTransactionAttribute(TransactionAttribute targetAttribute) {
|
||||||
Assert.notNull(targetAttribute, "Target attribute must not be null");
|
super(targetAttribute);
|
||||||
this.targetAttribute = targetAttribute;
|
this.targetAttribute = targetAttribute;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public int getPropagationBehavior() {
|
|
||||||
return this.targetAttribute.getPropagationBehavior();
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getIsolationLevel() {
|
|
||||||
return this.targetAttribute.getIsolationLevel();
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getTimeout() {
|
|
||||||
return this.targetAttribute.getTimeout();
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isReadOnly() {
|
|
||||||
return this.targetAttribute.isReadOnly();
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getName() {
|
|
||||||
return this.targetAttribute.getName();
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getQualifier() {
|
public String getQualifier() {
|
||||||
return this.targetAttribute.getQualifier();
|
return this.targetAttribute.getQualifier();
|
||||||
}
|
}
|
||||||
|
|
@ -72,20 +53,4 @@ public abstract class DelegatingTransactionAttribute implements TransactionAttri
|
||||||
return this.targetAttribute.rollbackOn(ex);
|
return this.targetAttribute.rollbackOn(ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean equals(Object obj) {
|
|
||||||
return this.targetAttribute.equals(obj);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int hashCode() {
|
|
||||||
return this.targetAttribute.hashCode();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return this.targetAttribute.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,84 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2009 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.springframework.transaction.support;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
import org.springframework.transaction.TransactionDefinition;
|
||||||
|
import org.springframework.util.Assert;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link TransactionDefinition} implementation that delegates all calls to a given target
|
||||||
|
* {@link TransactionDefinition} instance. Abstract because it is meant to be subclassed,
|
||||||
|
* with subclasses overriding specific methods that are not supposed to simply delegate
|
||||||
|
* to the target instance.
|
||||||
|
*
|
||||||
|
* @author Juergen Hoeller
|
||||||
|
* @since 3.0
|
||||||
|
*/
|
||||||
|
public abstract class DelegatingTransactionDefinition implements TransactionDefinition, Serializable {
|
||||||
|
|
||||||
|
private final TransactionDefinition targetDefinition;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a DelegatingTransactionAttribute for the given target attribute.
|
||||||
|
* @param targetAttribute the target TransactionAttribute to delegate to
|
||||||
|
*/
|
||||||
|
public DelegatingTransactionDefinition(TransactionDefinition targetDefinition) {
|
||||||
|
Assert.notNull(targetDefinition, "Target definition must not be null");
|
||||||
|
this.targetDefinition = targetDefinition;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public int getPropagationBehavior() {
|
||||||
|
return this.targetDefinition.getPropagationBehavior();
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getIsolationLevel() {
|
||||||
|
return this.targetDefinition.getIsolationLevel();
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getTimeout() {
|
||||||
|
return this.targetDefinition.getTimeout();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isReadOnly() {
|
||||||
|
return this.targetDefinition.isReadOnly();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return this.targetDefinition.getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object obj) {
|
||||||
|
return this.targetDefinition.equals(obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return this.targetDefinition.hashCode();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return this.targetDefinition.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue