bitronix.tm.BitronixTransaction Maven / Gradle / Ivy
/*
* Copyright (C) 2006-2013 Bitronix Software (http://www.bitronix.be)
*
* 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 bitronix.tm;
import bitronix.tm.internal.*;
import bitronix.tm.journal.Journal;
import bitronix.tm.resource.ResourceRegistrar;
import bitronix.tm.resource.common.XAResourceHolder;
import bitronix.tm.resource.common.XAResourceHolderStateVisitor;
import bitronix.tm.timer.TaskScheduler;
import bitronix.tm.twopc.Committer;
import bitronix.tm.twopc.PhaseException;
import bitronix.tm.twopc.Preparer;
import bitronix.tm.twopc.Rollbacker;
import bitronix.tm.twopc.executor.Executor;
import bitronix.tm.utils.*;
import javax.transaction.*;
import javax.transaction.xa.XAException;
import javax.transaction.xa.XAResource;
import java.io.IOException;
import java.util.*;
import java.util.logging.Level;
/**
* Implementation of {@link Transaction}.
*
* @author Ludovic Orban
*/
public class BitronixTransaction
implements Transaction, BitronixTransactionMBean
{
private static final String NOT_STARTED_TEXT = "transaction hasn't started yet";
private static final String EXTRA_ERROR_TEXT = ", extra error=";
private static final java.util.logging.Logger log = java.util.logging.Logger.getLogger(BitronixTransaction.class.toString());
private final XAResourceManager resourceManager;
private final Scheduler synchronizationScheduler = new Scheduler<>();
private final List transactionStatusListeners = new ArrayList<>();
private final Executor executor = TransactionManagerServices.getExecutor();
private final TaskScheduler taskScheduler = TransactionManagerServices.getTaskScheduler();
private final Preparer preparer = new Preparer(executor);
private final Committer committer = new Committer(executor);
private final Rollbacker rollbacker = new Rollbacker(executor);
private volatile int status = Status.STATUS_NO_TRANSACTION;
private volatile boolean timeout = false;
private volatile Date timeoutDate;
/* management */
private volatile String threadName;
private volatile Date startDate;
private volatile StackTrace activationStackTrace;
/**
* Constructor BitronixTransaction creates a new BitronixTransaction instance.
*/
public BitronixTransaction()
{
Uid gtrid = UidGenerator.generateUid();
if (LogDebugCheck.isDebugEnabled())
{
log.finer("creating new transaction with GTRID [" + gtrid + "]");
}
this.resourceManager = new XAResourceManager(gtrid);
this.threadName = Thread.currentThread()
.getName();
}
/**
* Method buildZeroTransactionDebugMessage ...
*
* @param activationStackTrace
* of type StackTrace
* @param commitStackTrace
* of type StackTrace
*
* @return String
*/
static String buildZeroTransactionDebugMessage(StackTrace activationStackTrace, StackTrace commitStackTrace)
{
String lineSeparator = System.getProperty("line.separator");
StringBuilder sb = new StringBuilder();
sb.append("committed transaction with 0 enlisted resource")
.append(lineSeparator);
sb.append("==================== Began at ====================")
.append(lineSeparator);
sb.append(ExceptionUtils.getStackTrace(activationStackTrace))
.append(lineSeparator);
sb.append("==================== Committed at ====================")
.append(lineSeparator);
sb.append(ExceptionUtils.getStackTrace(commitStackTrace))
.append(lineSeparator);
return sb.toString();
}
/**
* Method getSynchronizationScheduler returns the synchronizationScheduler of this BitronixTransaction object.
*
* @return the synchronizationScheduler (type Scheduler Synchronization ) of this BitronixTransaction object.
*/
public Scheduler getSynchronizationScheduler()
{
return synchronizationScheduler;
}
/**
* Method commit ...
*
* @throws RollbackException
* when
* @throws HeuristicMixedException
* when
* @throws HeuristicRollbackException
* when
* @throws SystemException
* when
*/
@Override
public void commit() throws RollbackException, HeuristicMixedException, HeuristicRollbackException, SystemException
{
if (status == Status.STATUS_NO_TRANSACTION)
{
throw new IllegalStateException(NOT_STARTED_TEXT);
}
if (isDone())
{
throw new IllegalStateException("transaction is done, cannot commit it");
}
taskScheduler.cancelTransactionTimeout(this);
// beforeCompletion must be called before the check to STATUS_MARKED_ROLLBACK as the synchronization
// can still set the status to STATUS_MARKED_ROLLBACK.
try
{
fireBeforeCompletionEvent();
}
catch (BitronixSystemException ex)
{
rollback();
throw new BitronixRollbackException("SystemException thrown during beforeCompletion cycle caused transaction rollback", ex);
}
catch (RuntimeException ex)
{
rollback();
throw new BitronixRollbackException("RuntimeException thrown during beforeCompletion cycle caused transaction rollback", ex);
}
// The following if statements and try/catch block must not be included in the prepare try-catch block as
// they call rollback().
// Doing so would call fireAfterCompletionEvent() twice in case one of those conditions are true.
if (timedOut())
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("transaction timed out");
}
rollback();
throw new BitronixRollbackException("transaction timed out and has been rolled back");
}
try
{
delistUnclosedResources(XAResource.TMSUCCESS);
}
catch (BitronixRollbackException ex)
{
if (LogDebugCheck.isDebugEnabled())
{
log.log(Level.FINER, "delistment error causing transaction rollback", ex);
}
rollback();
// the caught BitronixRollbackException's message is pre-formatted to be appended to this message
throw new BitronixRollbackException("delistment error caused transaction rollback" + ex.getMessage());
}
if (status == Status.STATUS_MARKED_ROLLBACK)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("transaction marked as rollback only");
}
rollback();
throw new BitronixRollbackException("transaction was marked as rollback only and has been rolled back");
}
try
{
List interestedResources;
// prepare phase
try
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("committing, " + resourceManager.size() + " enlisted resource(s)");
}
interestedResources = preparer.prepare(this);
}
catch (RollbackException ex)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("caught rollback exception during prepare, trying to rollback");
}
// rollbackPrepareFailure might throw a SystemException that will 'swallow' the RollbackException which is
// what we want in that case as the transaction has not been rolled back and some resources are now left in-doubt.
rollbackPrepareFailure(ex);
throw new BitronixRollbackException("transaction failed to prepare: " + this, ex);
}
// commit phase
if (LogDebugCheck.isDebugEnabled())
{
log.finer(interestedResources.size() + " interested resource(s)");
}
committer.commit(this, interestedResources);
if (resourceManager.size() == 0 && TransactionManagerServices.getConfiguration()
.isDebugZeroResourceTransaction())
{
log.warning(buildZeroTransactionDebugMessage(activationStackTrace, new StackTrace()));
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("successfully committed " + this);
}
}
finally
{
fireAfterCompletionEvent();
}
}
/**
* Method delistResource ...
*
* @param xaResource
* of type XAResource
* @param flag
* of type int
*
* @return boolean
*
* @throws SystemException
* when
*/
@Override
public boolean delistResource(XAResource xaResource, int flag) throws SystemException
{
if (status == Status.STATUS_NO_TRANSACTION)
{
throw new IllegalStateException(NOT_STARTED_TEXT);
}
if (flag != XAResource.TMSUCCESS && flag != XAResource.TMSUSPEND && flag != XAResource.TMFAIL)
{
throw new BitronixSystemException("can only delist with SUCCESS, SUSPEND, FAIL - was: " + Decoder.decodeXAResourceFlag(flag));
}
if (isWorking())
{
throw new IllegalStateException("transaction is being committed or rolled back, cannot delist any resource now");
}
XAResourceHolder resourceHolder = ResourceRegistrar.findXAResourceHolder(xaResource);
if (resourceHolder == null)
{
throw new BitronixSystemException("unknown XAResource " + xaResource + ", it does not belong to a registered resource");
}
class LocalVisitor
implements XAResourceHolderStateVisitor
{
private final List exceptions = new ArrayList<>();
private final List resourceStates = new ArrayList<>();
private boolean result = true;
/**
* Called when visiting all {@link bitronix.tm.internal.XAResourceHolderState}s.
* @param xaResourceHolderState the currently visited {@link bitronix.tm.internal.XAResourceHolderState}
* @return return true
to continue visitation, false
to stop visitation
*/
@Override
public boolean visit(XAResourceHolderState xaResourceHolderState)
{
try
{
result &= delistResource(xaResourceHolderState, flag);
}
catch (SystemException ex)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("failed to delist resource state " + xaResourceHolderState);
}
exceptions.add(ex);
resourceStates.add(xaResourceHolderState);
}
return true; // continue visitation
}
}
LocalVisitor xaResourceHolderStateVisitor = new LocalVisitor();
resourceHolder.acceptVisitorForXAResourceHolderStates(resourceManager.getGtrid(), xaResourceHolderStateVisitor);
if (!xaResourceHolderStateVisitor.exceptions.isEmpty())
{
BitronixMultiSystemException multiSystemException = new BitronixMultiSystemException("error delisting resource", xaResourceHolderStateVisitor.exceptions,
xaResourceHolderStateVisitor.resourceStates);
if (!multiSystemException.isUnilateralRollback())
{
throw multiSystemException;
}
else
{
if (LogDebugCheck.isDebugEnabled())
{
log.log(Level.FINER, "unilateral rollback of resource " + resourceHolder, multiSystemException);
}
}
}
return xaResourceHolderStateVisitor.result;
}
/**
* Method enlistResource ...
*
* @param xaResource
* of type XAResource
*
* @return boolean
*
* @throws RollbackException
* when
* @throws IllegalStateException
* when
* @throws SystemException
* when
*/
@Override
public boolean enlistResource(XAResource xaResource) throws RollbackException, SystemException
{
if (status == Status.STATUS_NO_TRANSACTION)
{
throw new IllegalStateException(NOT_STARTED_TEXT);
}
if (status == Status.STATUS_MARKED_ROLLBACK)
{
throw new BitronixRollbackException("transaction has been marked as rollback only");
}
if (isDone())
{
throw new IllegalStateException("transaction started or finished 2PC, cannot enlist any more resource");
}
XAResourceHolder resourceHolder = ResourceRegistrar.findXAResourceHolder(xaResource);
if (resourceHolder == null)
{
throw new BitronixSystemException("unknown XAResource " + xaResource + ", it does not belong to a registered resource");
}
XAResourceHolderState resourceHolderState = new XAResourceHolderState(resourceHolder, resourceHolder.getResourceBean());
// resource timeout must be set here so manually enlisted resources can receive it
resourceHolderState.setTransactionTimeoutDate(timeoutDate);
try
{
resourceManager.enlist(resourceHolderState);
}
catch (XAException ex)
{
String extraErrorDetails = TransactionManagerServices.getExceptionAnalyzer()
.extractExtraXAExceptionDetails(ex);
if (BitronixXAException.isUnilateralRollback(ex))
{
// if the resource unilaterally rolled back, the transaction will never be able to commit -> mark it as rollback only
setStatus(Status.STATUS_MARKED_ROLLBACK);
throw new BitronixRollbackException("resource " + resourceHolderState + " unilaterally rolled back, error=" +
Decoder.decodeXAExceptionErrorCode(ex) + (extraErrorDetails == null ? "" : EXTRA_ERROR_TEXT + extraErrorDetails), ex);
}
throw new BitronixSystemException("cannot enlist " + resourceHolderState + ", error=" +
Decoder.decodeXAExceptionErrorCode(ex) + (extraErrorDetails == null ? "" : EXTRA_ERROR_TEXT + extraErrorDetails), ex);
}
resourceHolder.putXAResourceHolderState(resourceHolderState.getXid(), resourceHolderState);
return true;
}
/**
* Method getStatus returns the status of this BitronixTransaction object.
*
* @return the status (type int) of this BitronixTransaction object.
*
* @throws SystemException
* when
*/
@Override
public int getStatus() throws SystemException
{
return status;
}
/**
* Method registerSynchronization ...
*
* @param synchronization
* of type Synchronization
*
* @throws RollbackException
* when
*/
@Override
public void registerSynchronization(Synchronization synchronization) throws RollbackException
{
if (status == Status.STATUS_NO_TRANSACTION)
{
throw new IllegalStateException(NOT_STARTED_TEXT);
}
if (status == Status.STATUS_MARKED_ROLLBACK)
{
throw new BitronixRollbackException("transaction has been marked as rollback only");
}
if (isDone())
{
throw new IllegalStateException("transaction is done, cannot register any more synchronization");
}
if (LogDebugCheck.isDebugEnabled())
{
log.finer("registering synchronization " + synchronization);
}
synchronizationScheduler.add(synchronization, Scheduler.DEFAULT_POSITION);
}
/**
* Method rollback ...
*
* @throws SystemException
* when
*/
@Override
public void rollback() throws SystemException
{
if (status == Status.STATUS_NO_TRANSACTION)
{
throw new IllegalStateException(NOT_STARTED_TEXT);
}
if (isDone())
{
throw new IllegalStateException("transaction is done, cannot roll it back");
}
taskScheduler.cancelTransactionTimeout(this);
try
{
delistUnclosedResources(XAResource.TMSUCCESS);
}
catch (BitronixRollbackException ex)
{
if (LogDebugCheck.isDebugEnabled())
{
log.log(Level.FINER, "some resource(s) failed delistment", ex);
}
}
try
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("rolling back, " + resourceManager.size() + " enlisted resource(s)");
}
List resourcesToRollback = new ArrayList<>();
List allResources = resourceManager.getAllResources();
for (XAResourceHolderState resource : allResources)
{
if (!resource.isFailed())
{
resourcesToRollback.add(resource);
}
}
rollbacker.rollback(this, resourcesToRollback);
if (LogDebugCheck.isDebugEnabled())
{
log.finer("successfully rolled back " + this);
}
}
catch (HeuristicMixedException ex)
{
throw new BitronixSystemException("transaction partly committed and partly rolled back. Resources are now inconsistent !", ex);
}
catch (HeuristicCommitException ex)
{
throw new BitronixSystemException("transaction committed instead of rolled back. Resources are now inconsistent !", ex);
}
finally
{
fireAfterCompletionEvent();
}
}
/**
* Method setRollbackOnly ...
*
* @throws SystemException
* when
*/
@Override
public void setRollbackOnly() throws SystemException
{
if (status == Status.STATUS_NO_TRANSACTION)
{
throw new IllegalStateException(NOT_STARTED_TEXT);
}
if (isDone())
{
throw new IllegalStateException("transaction is done, cannot change its status");
}
setStatus(Status.STATUS_MARKED_ROLLBACK);
}
/**
* Method setStatus sets the status of this BitronixTransaction object.
*
* @param status
* the status of this BitronixTransaction object.
*
* @throws SystemException
* when
*/
public void setStatus(int status) throws SystemException
{
setStatus(status, resourceManager.collectUniqueNames());
}
/**
* Method isDone returns the done of this BitronixTransaction object.
*
* @return the done (type boolean) of this BitronixTransaction object.
*/
private boolean isDone()
{
switch (status)
{
case Status.STATUS_PREPARING:
case Status.STATUS_PREPARED:
case Status.STATUS_COMMITTING:
case Status.STATUS_COMMITTED:
case Status.STATUS_ROLLING_BACK:
case Status.STATUS_ROLLEDBACK:
return true;
default:
return false;
}
}
/**
* Method isWorking returns the working of this BitronixTransaction object.
*
* @return the working (type boolean) of this BitronixTransaction object.
*/
private boolean isWorking()
{
switch (status)
{
case Status.STATUS_PREPARING:
case Status.STATUS_PREPARED:
case Status.STATUS_COMMITTING:
case Status.STATUS_ROLLING_BACK:
return true;
default:
return false;
}
}
/**
* Method delistResource ...
*
* @param resourceHolderState
* of type XAResourceHolderState
* @param flag
* of type int
*
* @return boolean
*
* @throws SystemException
* when
*/
private boolean delistResource(XAResourceHolderState resourceHolderState, int flag) throws SystemException
{
try
{
return resourceManager.delist(resourceHolderState, flag);
}
catch (XAException ex)
{
// if the resource could not be delisted, the transaction must not commit -> mark it as rollback only
if (status != Status.STATUS_MARKED_ROLLBACK)
{
setStatus(Status.STATUS_MARKED_ROLLBACK);
}
String extraErrorDetails = TransactionManagerServices.getExceptionAnalyzer()
.extractExtraXAExceptionDetails(ex);
if (BitronixXAException.isUnilateralRollback(ex))
{
// The resource unilaterally rolled back here. We have to throw an exception to indicate this but
// The signature of this method is inherited from javax.transaction.Transaction. Thereof, we have choice
// between creating a sub-exception of SystemException or using a RuntimeException. Is that the best way
// forward as this 'hidden' exception can be left throw out at unexpected locations where SystemException
// should be rethrown but the exception thrown here should be catched & handled... ?
throw new BitronixRollbackSystemException("resource " + resourceHolderState + " unilaterally rolled back, error=" +
Decoder.decodeXAExceptionErrorCode(ex) + (extraErrorDetails == null ? "" : EXTRA_ERROR_TEXT + extraErrorDetails), ex);
}
throw new BitronixSystemException("cannot delist " + resourceHolderState + ", error=" + Decoder.decodeXAExceptionErrorCode(ex) +
(extraErrorDetails == null ? "" : EXTRA_ERROR_TEXT + extraErrorDetails), ex);
}
}
/**
* Method setStatus ...
*
* @param status
* of type int
* @param uniqueNames
* of type Set String
*
* @throws SystemException
* when
*/
public void setStatus(int status, Set uniqueNames) throws SystemException
{
try
{
boolean force = (resourceManager.size() > 1) && (status == Status.STATUS_COMMITTING);
if (LogDebugCheck.isDebugEnabled())
{
log.finer("changing transaction status to " + Decoder.decodeStatus(status) + (force ? " (forced)" : ""));
}
int oldStatus = this.status;
this.status = status;
Journal journal = TransactionManagerServices.getJournal();
journal.log(status, resourceManager.getGtrid(), uniqueNames);
if (force)
{
journal.force();
}
if (status == Status.STATUS_ACTIVE)
{
ManagementRegistrar.register("bitronix.tm:type=Transaction,Gtrid=" + resourceManager.getGtrid(), this);
}
fireTransactionStatusChangedEvent(oldStatus, status);
}
catch (IOException ex)
{
// if we cannot log, the TM must stop managing TX until the problem is fixed
throw new BitronixSystemException("error logging status", ex);
}
}
/**
* Method fireTransactionStatusChangedEvent ...
*
* @param oldStatus
* of type int
* @param newStatus
* of type int
*/
private void fireTransactionStatusChangedEvent(int oldStatus, int newStatus)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("transaction status is changing from " + Decoder.decodeStatus(oldStatus) + " to " +
Decoder.decodeStatus(newStatus) + " - executing " + transactionStatusListeners.size() + " listener(s)");
}
for (TransactionStatusChangeListener listener : transactionStatusListeners)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("executing TransactionStatusChangeListener " + listener);
}
listener.statusChanged(oldStatus, newStatus);
if (LogDebugCheck.isDebugEnabled())
{
log.finer("executed TransactionStatusChangeListener " + listener);
}
}
}
/**
* Run all registered Synchronizations' beforeCompletion() method. Be aware that this method can change the
* transaction status to mark it as rollback only for instance.
*
* @throws bitronix.tm.internal.BitronixSystemException
* if status changing due to a synchronization throwing an
* exception fails.
*/
private void fireBeforeCompletionEvent() throws SystemException
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("before completion, " + synchronizationScheduler.size() + " synchronization(s) to execute");
}
Iterator it = synchronizationScheduler.reverseIterator();
while (it.hasNext())
{
Synchronization synchronization = it.next();
try
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("executing synchronization " + synchronization);
}
synchronization.beforeCompletion();
}
catch (RuntimeException ex)
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("Synchronization.beforeCompletion() call failed for " + synchronization + ", marking transaction as rollback only - " + ex);
}
setStatus(Status.STATUS_MARKED_ROLLBACK);
throw ex;
}
}
}
/**
* Method getResourceManager returns the resourceManager of this BitronixTransaction object.
*
* @return the resourceManager (type XAResourceManager) of this BitronixTransaction object.
*/
public XAResourceManager getResourceManager()
{
return resourceManager;
}
/**
* Method timedOut ...
*
* @return boolean
*/
public boolean timedOut()
{
return timeout;
}
/**
* Method timeout ...
*
* @throws SystemException
* when
*/
public void timeout() throws SystemException
{
this.timeout = true;
setStatus(Status.STATUS_MARKED_ROLLBACK);
log.warning("transaction timed out: " + this);
}
/**
* Method addTransactionStatusChangeListener ...
*
* @param listener
* of type TransactionStatusChangeListener
*/
public void addTransactionStatusChangeListener(TransactionStatusChangeListener listener)
{
transactionStatusListeners.add(listener);
}
/*
* Internal impl
*/
/**
* Method hashCode ...
*
* @return int
*/
@Override
public int hashCode()
{
return resourceManager.getGtrid()
.hashCode();
}
/**
* Method equals ...
*
* @param obj
* of type Object
*
* @return boolean
*/
@Override
public boolean equals(Object obj)
{
if (obj instanceof BitronixTransaction)
{
BitronixTransaction tx = (BitronixTransaction) obj;
return resourceManager.getGtrid()
.equals(tx.resourceManager.getGtrid());
}
return false;
}
/**
* Method toString ...
*
* @return String
*/
@Override
public String toString()
{
return "a Bitronix Transaction with GTRID [" + resourceManager.getGtrid() + "], status=" + Decoder.decodeStatus(status) + ", " + resourceManager.size() +
" resource(s) enlisted (started " + startDate + ")";
}
/**
* Delist all resources that have not been closed before calling tm.commit(). This basically means calling
* XAResource.end() on all resource that has not been ended yet.
*
* @param flag
* the flag to pass to XAResource.end(). Either TMSUCCESS or TMFAIL.
*
* @throws bitronix.tm.internal.BitronixRollbackException
* if some resources unilaterally rolled back before end() call.
*/
private void delistUnclosedResources(int flag) throws BitronixRollbackException
{
List allResources = resourceManager.getAllResources();
List rolledBackResources = new ArrayList<>();
List failedResources = new ArrayList<>();
for (XAResourceHolderState resource : allResources)
{
if (!resource.isEnded())
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("found unclosed resource to delist: " + resource);
}
try
{
delistResource(resource, flag);
}
catch (BitronixRollbackSystemException ex)
{
rolledBackResources.add(resource);
if (LogDebugCheck.isDebugEnabled())
{
log.log(Level.FINER, "resource unilaterally rolled back: " + resource, ex);
}
}
catch (SystemException ex)
{
failedResources.add(resource);
log.log(Level.WARNING, "error delisting resource, assuming unilateral rollback: " + resource, ex);
}
}
else if (LogDebugCheck.isDebugEnabled())
{
log.finer("no need to delist already closed resource: " + resource);
}
} // for
if (!rolledBackResources.isEmpty() || !failedResources.isEmpty())
{
String lineSeparator = System.getProperty("line.separator");
StringBuilder sb = new StringBuilder();
if (!rolledBackResources.isEmpty())
{
sb.append(lineSeparator);
sb.append(" resource(s) ");
sb.append(Decoder.collectResourcesNames(rolledBackResources));
sb.append(" unilaterally rolled back");
}
if (!failedResources.isEmpty())
{
sb.append(lineSeparator);
sb.append(" resource(s) ");
sb.append(Decoder.collectResourcesNames(failedResources));
sb.append(" could not be delisted");
}
throw new BitronixRollbackException(sb.toString());
}
}
/**
* Rollback resources after a phase 1 prepare failure. All resources must be rolled back as prepared ones
* are in-doubt and non-prepared ones have started/ended work done that must also be cleaned.
*
* @param rbEx
* the thrown rollback exception.
*
* @throws BitronixSystemException
* when a resource could not rollback prepapared state.
*/
private void rollbackPrepareFailure(RollbackException rbEx) throws BitronixSystemException
{
List interestedResources = resourceManager.getAllResources();
try
{
rollbacker.rollback(this, interestedResources);
if (LogDebugCheck.isDebugEnabled())
{
log.finer("rollback after prepare failure succeeded");
}
}
catch (Exception ex)
{
// let's merge both exceptions' PhaseException to report a complete error message
PhaseException preparePhaseEx = (PhaseException) rbEx.getCause();
PhaseException rollbackPhaseEx = (PhaseException) ex.getCause();
List exceptions = new ArrayList<>();
List resources = new ArrayList<>();
exceptions.addAll(preparePhaseEx.getExceptions());
exceptions.addAll(rollbackPhaseEx.getExceptions());
resources.addAll(preparePhaseEx.getResourceStates());
resources.addAll(rollbackPhaseEx.getResourceStates());
throw new BitronixSystemException("transaction partially prepared and only partially rolled back. Some resources might be left in doubt!",
new PhaseException(exceptions, resources));
}
}
/**
* Method setActive sets the active of this BitronixTransaction object.
*
* @param timeout
* the active of this BitronixTransaction object.
*
* @throws SystemException
* when
*/
public void setActive(int timeout) throws SystemException
{
if (status != Status.STATUS_NO_TRANSACTION)
{
throw new IllegalStateException("transaction has already started");
}
setStatus(Status.STATUS_ACTIVE);
this.startDate = new Date(MonotonicClock.currentTimeMillis());
this.timeoutDate = new Date(MonotonicClock.currentTimeMillis() + (timeout * 1000L));
if (TransactionManagerServices.getConfiguration()
.isDebugZeroResourceTransaction())
{
this.activationStackTrace = new StackTrace();
}
taskScheduler.scheduleTransactionTimeout(this, timeoutDate);
}
/**
* Method fireAfterCompletionEvent ...
*/
private void fireAfterCompletionEvent()
{
// this TX is no longer in-flight -> remove this transaction's state from all XAResourceHolders
getResourceManager().clearXAResourceHolderStates();
if (LogDebugCheck.isDebugEnabled())
{
log.finer("after completion, " + synchronizationScheduler.size() + " synchronization(s) to execute");
}
for (Synchronization synchronization : synchronizationScheduler)
{
try
{
if (LogDebugCheck.isDebugEnabled())
{
log.finer("executing synchronization " + synchronization + " with status=" + Decoder.decodeStatus(status));
}
synchronization.afterCompletion(status);
}
catch (Exception ex)
{
log.log(Level.WARNING, "Synchronization.afterCompletion() call failed for " + synchronization, ex);
}
}
ManagementRegistrar.unregister("bitronix.tm:type=Transaction,Gtrid=" + resourceManager.getGtrid());
}
/* management */
/**
* Method getGtrid returns the gtrid of this BitronixTransaction object.
*
* @return the gtrid (type String) of this BitronixTransaction object.
*/
@Override
public String getGtrid()
{
return resourceManager.getGtrid()
.toString();
}
/**
* Method getStatusDescription returns the statusDescription of this BitronixTransaction object.
*
* @return the statusDescription (type String) of this BitronixTransaction object.
*/
@Override
public String getStatusDescription()
{
return Decoder.decodeStatus(status);
}
/**
* Method getThreadName returns the threadName of this BitronixTransaction object.
*
* @return the threadName (type String) of this BitronixTransaction object.
*/
@Override
public String getThreadName()
{
return threadName;
}
/**
* Method getStartDate returns the startDate of this BitronixTransaction object.
*
* @return the startDate (type Date) of this BitronixTransaction object.
*/
@Override
public Date getStartDate()
{
return startDate;
}
/**
* Method getEnlistedResourcesUniqueNames returns the enlistedResourcesUniqueNames of this BitronixTransaction object.
*
* @return the enlistedResourcesUniqueNames (type Collection String ) of this BitronixTransaction object.
*/
@Override
public Collection getEnlistedResourcesUniqueNames()
{
return resourceManager.collectUniqueNames();
}
/**
* Returns the activation {@link StackTrace} if it is available.
*
* @return the call stack of where the transaction began
*/
StackTrace getActivationStackTrace()
{
return activationStackTrace;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy