Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.drools.persistence.jta.JtaTransactionManager Maven / Gradle / Ivy
/*
* Copyright 2011 Red Hat, Inc. and/or its affiliates.
*
* 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.drools.persistence.jta;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import javax.naming.InitialContext;
import javax.naming.NamingException;
import javax.transaction.Status;
import javax.transaction.SystemException;
import javax.transaction.TransactionSynchronizationRegistry;
import javax.transaction.UserTransaction;
import org.drools.persistence.api.TransactionManager;
import org.drools.persistence.api.TransactionSynchronization;
import org.drools.persistence.api.TransactionSynchronizationRegistryHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class JtaTransactionManager
implements
TransactionManager {
Logger logger = LoggerFactory.getLogger( getClass() );
public static final String DEFAULT_USER_TRANSACTION_NAME = "java:comp/UserTransaction";
public static final String[] FALLBACK_TRANSACTION_MANAGER_NAMES = new String[]{"java:comp/TransactionManager", "java:appserver/TransactionManager", "java:pm/TransactionManager", "java:/TransactionManager", System.getProperty("jbpm.tm.jndi.lookup")};
/**
* Standard Java EE 5 JNDI location for the JTA TransactionSynchronizationRegistry.
* Autodetected when available.
*/
public static final String DEFAULT_TRANSACTION_SYNCHRONIZATION_REGISTRY_NAME = "java:comp/TransactionSynchronizationRegistry";
private static final String TRANSACTION_SYNCHRONIZATION_REGISTRY_CLASS_NAME = "javax.transaction.TransactionSynchronizationRegistry";
private static Class< ? > transactionSynchronizationRegistryClass;
private static final ThreadLocal > transactionResources = new ThreadLocal>(){
@Override
protected Map initialValue() {
return Collections.synchronizedMap(new HashMap());
}
};
static {
ClassLoader cl = JtaTransactionManager.class.getClassLoader();
try {
transactionSynchronizationRegistryClass = cl.loadClass( TRANSACTION_SYNCHRONIZATION_REGISTRY_CLASS_NAME );
} catch ( ClassNotFoundException e ) {
// JTA 1.1 API not available... simply proceed the JTA 1.0 way.
}
}
UserTransaction ut;
Object tsr;
javax.transaction.TransactionManager tm;
public JtaTransactionManager(Object ut,
Object tsr,
Object tm) {
if ( ut instanceof UserTransaction ) {
this.ut = ( UserTransaction ) ut;
} else {
this.ut = ( UserTransaction ) ( (ut != null) ? ut : findUserTransaction() );
}
if ( tm instanceof javax.transaction.TransactionManager ) {
this.tm = ( javax.transaction.TransactionManager ) tm;
} else {
this.tm = ( javax.transaction.TransactionManager ) ( (tm != null) ? tm : findTransactionManager( this.ut ) );
}
this.tsr = (tsr != null) ? tsr : findTransactionSynchronizationRegistry( this.ut,
this.tm );
}
protected javax.transaction.TransactionManager findTransactionManager(UserTransaction ut) {
if ( ut instanceof javax.transaction.TransactionManager ) {
logger.debug( "JTA UserTransaction object [{}] implements TransactionManager",
ut );
return (javax.transaction.TransactionManager) ut;
}
InitialContext context = null;
try {
context = new InitialContext();
} catch ( NamingException ex ) {
logger.debug( "Could not initialise JNDI InitialContext",
ex );
return null;
}
// Check fallback JNDI locations.
for ( String jndiName : FALLBACK_TRANSACTION_MANAGER_NAMES ) {
if (jndiName == null) {
continue;
}
try {
javax.transaction.TransactionManager tm = (javax.transaction.TransactionManager) context.lookup( jndiName );
logger.debug( "JTA TransactionManager found at fallback JNDI location [{}]",
jndiName );
return tm;
} catch ( NamingException ex ) {
logger.debug( "No JTA TransactionManager found at fallback JNDI location [{}]",
jndiName );
}
}
// OK, so no JTA TransactionManager is available...
return null;
}
protected UserTransaction findUserTransaction() {
try {
InitialContext context = new InitialContext();
return (UserTransaction) context.lookup( DEFAULT_USER_TRANSACTION_NAME );
} catch ( NamingException ex ) {
logger.debug( "No UserTransaction found at JNDI location [{}]", DEFAULT_USER_TRANSACTION_NAME, ex );
// no user transaction found in default location try alternative ones
// on some app servers access to default UT is not allowed from non manged thread, e.g. timers
try {
return InitialContext.doLookup(System.getProperty("jbpm.ut.jndi.lookup", "java:jboss/UserTransaction"));
} catch (Exception e1) {
logger.debug("Unable to find transaction: {}. Might be running in CMT environment" + ex.getMessage());
return null;
}
}
}
protected Object findTransactionSynchronizationRegistry(UserTransaction ut,
javax.transaction.TransactionManager tm) {
if ( transactionSynchronizationRegistryClass == null ) {
// JTA 1.1 API not present - skip.
logger.debug( "JTA 1.1 [{}] API not available",
TRANSACTION_SYNCHRONIZATION_REGISTRY_CLASS_NAME );
return null;
}
String jndiName = DEFAULT_TRANSACTION_SYNCHRONIZATION_REGISTRY_NAME;
try {
InitialContext context = new InitialContext();
Object tsrObject = context.lookup( jndiName );
logger.debug( "JTA TransactionSynchronizationRegistry found at default JNDI location [{}]",
jndiName );
return tsrObject;
} catch ( NamingException ex ) {
logger.debug( "No JTA TransactionSynchronizationRegistry found at default JNDI location [{}]",
jndiName,
ex );
String customJndiLocation = System.getProperty("jbpm.tsr.jndi.lookup", "java:jboss/TransactionSynchronizationRegistry");
try {
Object tsrObject = InitialContext.doLookup(customJndiLocation);
logger.debug( "JTA TransactionSynchronizationRegistry found at default JNDI location [{}]",
customJndiLocation );
return tsrObject;
} catch (Exception e1) {
logger.debug( "No JTA TransactionSynchronizationRegistry found at default JNDI location [{}]",
customJndiLocation,
ex );
}
}
// Check whether the UserTransaction or TransactionManager implements it...
if ( transactionSynchronizationRegistryClass.isInstance( ut ) ) {
return ut;
}
if ( transactionSynchronizationRegistryClass.isInstance( tm ) ) {
return tm;
}
// OK, so no JTA 1.1 TransactionSynchronizationRegistry is available,
// despite the API being present...
return null;
}
protected UserTransaction getUt() {
if (this.ut == null) {
this.ut = findUserTransaction();
}
return ut;
}
public boolean begin() {
// RHBPMS-4621 - transaction can be marked as rollback
// and still be associated with current thread
// See WFLY-4327
int status = getStatus();
if ( status == TransactionManager.STATUS_ROLLEDBACK ) {
logger.debug("Cleanup of transaction that has been rolled back previously");
rollback(true);
status = getStatus();
}
if ( status == TransactionManager.STATUS_NO_TRANSACTION ) {
try {
getUt().begin();
return true;
} catch ( Exception e ) {
// special WAS handling for cached UserTrnsactions
if (e.getClass().getName().equals("javax.ejb.EJBException")) {
// reinitialize all fields
this.ut = findUserTransaction();
this.tm = findTransactionManager(this.ut);
this.tsr = findTransactionSynchronizationRegistry(this.ut, this.tm);
try {
this.ut.begin();
return true;
} catch (Exception ex) {
logger.warn( "Unable to begin transaction", e);
throw new RuntimeException( "Unable to begin transaction", e );
}
} else {
logger.warn( "Unable to begin transaction", e);
throw new RuntimeException( "Unable to begin transaction", e );
}
}
}
return false;
}
public void commit(boolean transactionOwner) {
if ( transactionOwner ) {
try {
this.ut.commit();
} catch ( Exception e ) {
logger.warn( "Unable to commit transaction", e);
throw new RuntimeException( "Unable to commit transaction",
e );
}
}
transactionResources.get().clear();
}
public void rollback(boolean transactionOwner) {
try {
if (transactionOwner) {
// transaction can be already rolled back,
// exception thrown during beforeCompletion cycle can cause transaction rollback
if (ut.getStatus() != Status.STATUS_NO_TRANSACTION) {
this.ut.rollback();
}
} else {
getUt().setRollbackOnly();
}
} catch ( Exception e ) {
logger.warn( "Unable to rollback transaction", e);
throw new RuntimeException( "Unable to rollback transaction",
e );
}
transactionResources.get().clear();
}
public int getStatus() {
int s;
try {
// use transaction sync registry if available as it is safe way for both BMT and CMT
if (this.tsr != null) {
s = ((TransactionSynchronizationRegistry) this.tsr).getTransactionStatus();
} else {
// if no transaction sync registry available fallback to user transaction
s = this.ut.getStatus();
}
} catch ( SystemException e ) {
throw new RuntimeException( "Unable to get status for transaction",
e );
}
switch ( s ) {
case Status.STATUS_COMMITTED :
return TransactionManager.STATUS_COMMITTED;
case Status.STATUS_ROLLEDBACK :
case Status.STATUS_MARKED_ROLLBACK :
return TransactionManager.STATUS_ROLLEDBACK;
case Status.STATUS_NO_TRANSACTION :
return TransactionManager.STATUS_NO_TRANSACTION;
default :
return TransactionManager.STATUS_UNKNOWN;
}
}
public void registerTransactionSynchronization(final TransactionSynchronization ts) {
if ( this.tsr != null ) {
TransactionSynchronizationRegistryHelper.registerTransactionSynchronization( this.tsr,
ts );
} else if ( this.tm != null ) {
try {
this.tm.getTransaction().registerSynchronization( new JtaTransactionSynchronizationAdapter( ts ) );
} catch ( Exception e ) {
// No JTA TransactionManager available - log a warning.
logger.warn( "Participating in existing JTA transaction, but no JTA TransactionManager or TransactionSychronizationRegistry available: ",
e );
}
} else {
// No JTA TransactionManager available - log a warning.
logger.warn( "Participating in existing JTA transaction, but no JTA TransactionManager or TransactionSychronizationRegistry available: " );
}
}
@Override
public void putResource(Object key, Object resource) {
if (this.tsr != null) {
TransactionSynchronizationRegistryHelper.putResource(this.tsr, key, resource);
} else {
transactionResources.get().put(key, resource);
}
}
@Override
public Object getResource(Object key) {
if (this.tsr != null) {
return TransactionSynchronizationRegistryHelper.getResource(this.tsr, key);
}
return transactionResources.get().get(key);
}
}