
com.sun.messaging.jmq.jmsclient.JMSXAWrappedXAResourceImpl Maven / Gradle / Ivy
/*
* Copyright (c) 2000, 2020 Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2021, 2022 Contributors to the Eclipse Foundation
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0, which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception, which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
*/
package com.sun.messaging.jmq.jmsclient;
import static java.lang.System.Logger.Level.DEBUG;
import static java.lang.System.Logger.Level.ERROR;
import static java.lang.System.Logger.Level.INFO;
import jakarta.jms.*;
import javax.transaction.xa.*;
import java.util.StringTokenizer;
import java.util.NoSuchElementException;
import com.sun.jms.spi.xa.*;
import com.sun.messaging.AdministeredObject;
import java.lang.System.Logger;
/**
* JMSXAWrappedXAResourceImpl
*
*
* The XAResource interface is a Java mapping of the industry standard XA interface based on the X/Open CAE
* Specification (Distributed Transaction Processing: The XA Specification).
*
*
* The XA interface defines the contract between a Resource Manager and a Transaction Manager in a distributed
* transaction processing (DTP) environment. A JDBC driver or a JMS provider implements this interface to support the
* association between a global transaction and a database or message service connection.
*
*
* The XAResource interface can be supported by any transactional resource that is intended to be used by application
* programs in an environment where transactions are controlled by an external transaction manager. An example of such a
* resource is a database management system. An application may access data through multiple database connections. Each
* database connection is enlisted with the transaction manager as a transactional resource. The transaction manager
* obtains an XAResource for each connection participating in a global transaction. The transaction manager uses the
* start
method to associate the global transaction with the resource, and it uses the end
* method to disassociate the transaction from the resource. The resource manager is responsible for associating the
* global transaction to all work performed on its data between the start and end method invocations.
*
*
* At transaction commit time, the resource managers are informed by the transaction manager to prepare, commit, or
* rollback a transaction according to the two-phase commit protocol.
*
*
* @see javax.transaction.xa.XAResource
*/
public class JMSXAWrappedXAResourceImpl implements XAResource {
private static final boolean debug = JMSXAWrappedConnectionFactoryImpl.debug;
private static Logger logger = System.getLogger(JMSXAWrappedXAResourceImpl.class.getName());
private XAResource xar;
private JMSXAConnectionFactory cf = null;
private boolean isQueue;
private String username = null;
private String password = null;
private boolean closed = false;
private JMSXAWrappedTransactionListener listener = null;
public JMSXAWrappedXAResourceImpl(XAResource xar, boolean isQueue, JMSXAConnectionFactory cf, String username, String password) {
this.xar = xar;
this.cf = cf;
this.isQueue = isQueue;
this.username = username;
this.password = password;
}
protected void setTransactionListener(JMSXAWrappedTransactionListener l) {
listener = l;
}
protected void close() {
closed = true;
}
/**
* Commits the global transaction specified by xid.
*
* @param foreignXid A global transaction identifier
*
* @param onePhase If true, the resource manager should use a one-phase commit protocol to commit the work done on
* behalf of xid.
*
* @exception XAException An error has occurred. Possible XAExceptions are XA_HEURHAZ, XA_HEURCOM, XA_HEURRB,
* XA_HEURMIX, XAER_RMERR, XAER_RMFAIL, XAER_NOTA, XAER_INVAL, or XAER_PROTO.
*
*
* If the resource manager did not commit the transaction and the parameter onePhase is set to true, the resource
* manager may throw one of the XA_RB* exceptions. Upon return, the resource manager has rolled back the branch's work
* and has released all held resources.
*/
@Override
public void commit(Xid foreignXid, boolean onePhase) throws XAException {
if (!closed || xar instanceof com.sun.messaging.jmq.jmsclient.XAResourceImpl) {
dlog("commit(Xid=" + foreignXid + " onePhase=" + onePhase + "), closed=" + closed);
boolean completed = false;
try {
if (listener != null) {
listener.beforeTransactionComplete();
}
xar.commit(foreignXid, onePhase);
completed = true;
} catch (XAException e) {
switch (e.errorCode) {
case XAException.XA_HEURHAZ:
case XAException.XA_HEURCOM:
case XAException.XA_HEURRB:
case XAException.XA_HEURMIX:
completed = false; /* wait forget */
break;
case XAException.XA_RETRY:
completed = false; /* allowed only for 2-phase commit */
break;
case XAException.XA_RBROLLBACK:
case XAException.XA_RBCOMMFAIL:
case XAException.XA_RBDEADLOCK:
case XAException.XA_RBINTEGRITY:
case XAException.XA_RBOTHER:
case XAException.XA_RBPROTO:
case XAException.XA_RBTIMEOUT:
case XAException.XA_RBTRANSIENT:
completed = true; /* allowed only for 1-phase commit */
break;
case XAException.XAER_RMFAIL:
case XAException.XAER_NOTA: /* ?? */
case XAException.XAER_INVAL: /* ?? */
case XAException.XAER_PROTO:
default:
completed = true;
}
throw e;
} finally {
if (listener != null) {
listener.afterTransactionComplete(foreignXid, completed);
}
}
} else {
JMSXAConnection c = null;
try {
dlog("closed commit() ...");
c = getConnection();
XAResource r = getXAResource(c);
dlog("closed commit(Xid=" + foreignXid + " onePhase=" + onePhase + ")");
r.commit(foreignXid, onePhase);
} catch (JMSException e) {
logError(e);
throw new XAException(XAException.XAER_RMFAIL);
} finally {
if (c != null) {
try {
c.close();
} catch (Exception e) {
log(AdministeredObject.cr.getKString(AdministeredObject.cr.W_WARNING), "commit():Connection.close(): " + e.getMessage());
}
}
}
}
}
/**
* Ends the work performed on behalf of a transaction branch. The resource manager disassociates the XA resource from
* the transaction branch specified and lets the transaction complete.
*
*
* If TMSUSPEND is specified in the flags, the transaction branch is temporarily suspended in an incomplete state. The
* transaction context is in a suspended state and must be resumed via the start
method with TMRESUME
* specified.
*
*
*
* If TMFAIL is specified, the portion of work has failed. The resource manager may mark the transaction as
* rollback-only
*
*
*
* If TMSUCCESS is specified, the portion of work has completed successfully.
*
*
* @param foreignXid A global transaction identifier that is the same as the identifier used previously in the
* start
method.
*
* @param flags One of TMSUCCESS, TMFAIL, or TMSUSPEND.
*
* @exception XAException An error has occurred. Possible XAException values are XAER_RMERR, XAER_RMFAILED, XAER_NOTA,
* XAER_INVAL, XAER_PROTO, or XA_RB*.
*/
@Override
public void end(Xid foreignXid, int flags) throws XAException {
boolean completed = false;
try {
if (listener != null) {
listener.beforeTransactionComplete();
}
xar.end(foreignXid, flags);
} catch (XAException e) {
switch (e.errorCode) {
case XAException.XA_RBROLLBACK:
case XAException.XA_RBCOMMFAIL:
case XAException.XA_RBDEADLOCK:
case XAException.XA_RBINTEGRITY:
case XAException.XA_RBOTHER:
case XAException.XA_RBPROTO:
case XAException.XA_RBTIMEOUT:
case XAException.XA_RBTRANSIENT:
completed = false;
break;
default:
completed = true;
}
throw e;
} finally {
if (listener != null) {
listener.afterTransactionComplete(foreignXid, completed);
}
}
}
/**
* Tells the resource manager to forget about a heuristically completed transaction branch.
*
* @param foreignXid A global transaction identifier.
*
* @exception XAException An error has occurred. Possible exception values are XAER_RMERR, XAER_RMFAIL, XAER_NOTA,
* XAER_INVAL, or XAER_PROTO.
*/
@Override
public void forget(Xid foreignXid) throws XAException {
if (!closed || xar instanceof com.sun.messaging.jmq.jmsclient.XAResourceImpl) {
dlog("forget(Xid=" + foreignXid + "), closed=" + closed);
boolean completed = false;
try {
if (listener != null) {
listener.beforeTransactionComplete();
}
xar.forget(foreignXid);
completed = true;
} finally {
if (listener != null) {
listener.afterTransactionComplete(foreignXid, completed);
}
}
} else {
JMSXAConnection c = null;
try {
dlog("closed forget(Xid=" + foreignXid + ")");
c = getConnection();
XAResource r = getXAResource(c);
r.forget(foreignXid);
} catch (JMSException e) {
logError(e);
throw new XAException(XAException.XAER_RMFAIL);
} finally {
if (c != null) {
try {
c.close();
} catch (Exception e) {
log(AdministeredObject.cr.getKString(AdministeredObject.cr.W_WARNING), "forget():Connection.close(): " + e.getMessage());
}
}
}
}
}
/**
* Obtains the current transaction timeout value set for this XAResource instance. If
* XAResource.setTransactionTimeout
was not used prior to invoking this method, the return value is the
* default timeout set for the resource manager; otherwise, the value used in the previous
* setTransactionTimeout
call is returned.
*
* @return the transaction timeout value in seconds.
*
* @exception XAException An error has occurred. Possible exception values are XAER_RMERR and XAER_RMFAIL.
*/
@Override
public int getTransactionTimeout() throws XAException {
return xar.getTransactionTimeout();
}
/**
* This method is called to determine if the resource manager instance represented by the target object is the same as
* the resouce manager instance represented by the parameter xares.
*
* @param foreignXaRes An XAResource object whose resource manager instance is to be compared with the resource manager
* instance of the target object.
*
* @return true if it's the same RM instance; otherwise false.
*
* @exception XAException An error has occurred. Possible exception values are XAER_RMERR and XAER_RMFAIL.
*
*/
@Override
public boolean isSameRM(XAResource foreignXaRes) throws XAException {
dlog(xar.getClass().getName() + ".isSameRM(" + foreignXaRes.getClass().getName() + ")");
return xar.isSameRM(foreignXaRes);
}
/**
* Ask the resource manager to prepare for a transaction commit of the transaction specified in xid.
*
* @param foreignXid A global transaction identifier.
*
* @exception XAException An error has occurred. Possible exception values are: XA_RB*, XAER_RMERR, XAER_RMFAIL,
* XAER_NOTA, XAER_INVAL, or XAER_PROTO.
*
* @return A value indicating the resource manager's vote on the outcome of the transaction. The possible values are:
* XA_RDONLY or XA_OK. If the resource manager wants to roll back the transaction, it should do so by raising an
* appropriate XAException in the prepare method.
*/
@Override
public int prepare(Xid foreignXid) throws XAException {
if (!closed || xar instanceof com.sun.messaging.jmq.jmsclient.XAResourceImpl) {
dlog("prepare(Xid=" + foreignXid + "), closed=" + closed);
boolean completed = false;
int ret = XA_OK;
try {
if (listener != null) {
listener.beforeTransactionComplete();
}
ret = xar.prepare(foreignXid);
if (ret == XA_RDONLY) {
completed = true;
}
return ret;
} catch (XAException e) {
switch (e.errorCode) {
case XAException.XA_RBROLLBACK:
case XAException.XA_RBCOMMFAIL:
case XAException.XA_RBDEADLOCK:
case XAException.XA_RBINTEGRITY:
case XAException.XA_RBOTHER:
case XAException.XA_RBPROTO:
case XAException.XA_RBTIMEOUT:
case XAException.XA_RBTRANSIENT:
completed = true;
break;
default:
completed = true;
}
throw e;
} finally {
if (listener != null) {
listener.afterTransactionComplete(foreignXid, completed);
}
}
} else {
JMSXAConnection c = null;
try {
dlog("closed prepare(Xid=" + foreignXid + ")");
c = getConnection();
XAResource r = getXAResource(c);
return r.prepare(foreignXid);
} catch (JMSException e) {
logError(e);
throw new XAException(XAException.XAER_RMFAIL);
} finally {
if (c != null) {
try {
c.close();
} catch (Exception e) {
log(AdministeredObject.cr.getKString(AdministeredObject.cr.W_WARNING), "prepare():Connection.close():" + e.getMessage());
}
}
}
}
}
/**
* Obtains a list of prepared transaction branches from a resource manager. The transaction manager calls this method
* during recovery to obtain the list of transaction branches that are currently in prepared or heuristically completed
* states.
*
* @param flags One of TMSTARTRSCAN, TMENDRSCAN, TMNOFLAGS. TMNOFLAGS must be used when no other flags are set in the
* parameter.
*
* @exception XAException An error has occurred. Possible values are XAER_RMERR, XAER_RMFAIL, XAER_INVAL, and
* XAER_PROTO.
*
* @return The resource manager returns zero or more XIDs of the transaction branches that are currently in a prepared
* or heuristically completed state. If an error occurs during the operation, the resource manager should throw the
* appropriate XAException.
*
*/
@Override
public Xid[] recover(int flags) throws XAException {
return xar.recover(flags);
}
/**
* Informs the resource manager to roll back work done on behalf of a transaction branch.
*
* @param foreignXid A global transaction identifier.
*
* @exception XAException An error has occurred.
*/
@Override
public void rollback(Xid foreignXid) throws XAException {
if (!closed || xar instanceof com.sun.messaging.jmq.jmsclient.XAResourceImpl) {
dlog("rollback(Xid=" + foreignXid + "), closed=" + closed);
boolean completed = false;
try {
if (listener != null) {
listener.beforeTransactionComplete();
}
xar.rollback(foreignXid);
completed = true;
} catch (XAException e) {
switch (e.errorCode) {
case XAException.XA_HEURHAZ:
case XAException.XA_HEURCOM:
case XAException.XA_HEURRB:
case XAException.XA_HEURMIX:
completed = false; /* wait forget */
break;
case XAException.XA_RBROLLBACK:
case XAException.XA_RBCOMMFAIL:
case XAException.XA_RBDEADLOCK:
case XAException.XA_RBINTEGRITY:
case XAException.XA_RBOTHER:
case XAException.XA_RBPROTO:
case XAException.XA_RBTIMEOUT:
case XAException.XA_RBTRANSIENT:
completed = true; /* typically for tran marked rollback-only */
break;
default:
completed = true;
}
throw e;
} finally {
if (listener != null) {
listener.afterTransactionComplete(foreignXid, completed);
}
}
} else {
JMSXAConnection c = null;
try {
dlog("closed rollback() ...");
c = getConnection();
XAResource r = getXAResource(c);
dlog("closed rollback(Xid=" + foreignXid + ")");
r.rollback(foreignXid);
} catch (JMSException e) {
logError(e);
throw new XAException(XAException.XAER_RMFAIL);
} finally {
if (c != null) {
try {
c.close();
} catch (Exception e) {
log(AdministeredObject.cr.getKString(AdministeredObject.cr.W_WARNING), "rollback():Connection.close():" + e.getMessage());
}
}
}
}
}
/**
*
* Sets the current transaction timeout value for this XAResource
instance. Once set, this timeout value is
* effective until setTransactionTimeout
is invoked again with a different value. To reset the timeout
* value to the default value used by the resource manager, set the value to zero.
*
* If the timeout operation is performed successfully, the method returns true; otherwise false. If a
* resource manager does not support explicitly setting the transaction timeout value, this method returns false.
*
* @param transactionTimeout The transaction timeout value in seconds.
*
* @return true if the transaction timeout value is set successfully; otherwise false.
*
* @exception XAException An error has occurred. Possible exception values are XAER_RMERR, XAER_RMFAIL, or XAER_INVAL.
*/
@Override
public boolean setTransactionTimeout(int transactionTimeout) throws XAException {
return xar.setTransactionTimeout(transactionTimeout);
}
/**
* Starts work on behalf of a transaction branch specified in foreignXid
.
*
* If TMJOIN is specified, the start applies to joining a transaction previously seen by the resource manager. If
* TMRESUME is specified, the start applies to resuming a suspended transaction specified in the parameter
* foreignXid
.
*
* If neither TMJOIN nor TMRESUME is specified and the transaction specified by foreignXid
has previously
* been seen by the resource manager, the resource manager throws the XAException exception with XAER_DUPID error code.
*
* @param foreignXid A global transaction identifier to be associated with the resource.
*
* @param flags One of TMNOFLAGS, TMJOIN, or TMRESUME.
*
* @exception XAException An error has occurred. Possible exceptions are XA_RB*, XAER_RMERR, XAER_RMFAIL, XAER_DUPID,
* XAER_OUTSIDE, XAER_NOTA, XAER_INVAL, or XAER_PROTO.
*
*/
@Override
public void start(Xid foreignXid, int flags) throws XAException {
boolean started = false;
try {
if (listener != null) {
listener.beforeTransactionStart();
}
xar.start(foreignXid, flags);
/*
* assume TMJOIN/TMRESUME has a corresponding TMNOFLAGS start that is, they cause 1 transaction completion
*/
if (flags == TMNOFLAGS) {
started = true;
}
} catch (JMSException e) {
logError(e);
throw new XAException(XAException.XAER_RMFAIL);
} catch (XAException e) {
switch (e.errorCode) {
case XAException.XA_RBROLLBACK:
case XAException.XA_RBCOMMFAIL:
case XAException.XA_RBDEADLOCK:
case XAException.XA_RBINTEGRITY:
case XAException.XA_RBOTHER:
case XAException.XA_RBPROTO:
case XAException.XA_RBTIMEOUT:
case XAException.XA_RBTRANSIENT:
started = false; /* ??? may cause negative 'transactions' count in Session */
break;
default:
started = false;
}
throw e;
} finally {
if (listener != null) {
listener.afterTransactionStart(foreignXid, started);
}
}
}
private JMSXAConnection getConnection() throws JMSException {
JMSXAConnection c = null;
if (isQueue) {
if (username == null) {
c = ((JMSXAQueueConnectionFactory) cf).createXAQueueConnection();
} else {
c = ((JMSXAQueueConnectionFactory) cf).createXAQueueConnection(username, password);
}
} else {
if (username == null) {
c = ((JMSXATopicConnectionFactory) cf).createXATopicConnection();
} else {
c = ((JMSXATopicConnectionFactory) cf).createXATopicConnection(username, password);
}
}
return c;
}
public XAResource getDelegatedXAResource() {
return xar;
}
private XAResource getXAResource(JMSXAConnection c) throws JMSException, XAException {
JMSXASession s;
if (isQueue) {
s = ((JMSXAQueueConnection) c).createXAQueueSession(true, Session.AUTO_ACKNOWLEDGE);
} else {
s = ((JMSXATopicConnection) c).createXATopicSession(true, Session.AUTO_ACKNOWLEDGE);
}
XAResource r = s.getXAResource();
String rc = null;
if (r instanceof com.sun.messaging.jmq.jmsclient.JMSXAWrappedXAResourceImpl) {
rc = ((com.sun.messaging.jmq.jmsclient.JMSXAWrappedXAResourceImpl) r).getDelegatedXAResource().getClass().getName();
} else {
rc = r.getClass().getName();
}
boolean skip = isSystemPropertySetFor("skipIsSameRMCheckForExternalJMSXAResource", rc);
if (r.isSameRM(xar)) {
dlog("isSameRM() true - " + xar.getClass().getName());
try {
int t = xar.getTransactionTimeout();
if (t >= 0) {
r.setTransactionTimeout(t);
}
} catch (Exception e) {
log(AdministeredObject.cr.getKString(AdministeredObject.cr.W_WARNING), "get/setTransactionTimeout(): " + e.getMessage());
}
} else {
if (!skip) {
log("Error:", "isSameRM() false - " + xar.getClass().getName());
throw new XAException(XAException.XAER_RMFAIL);
} else {
log(AdministeredObject.cr.getKString(AdministeredObject.cr.W_WARNING), "isSameRM() false, ignore. - " + xar.getClass().getName());
}
}
return r;
}
public static boolean isSystemPropertySetFor(String propName, String propValue) {
boolean isSet = false;
String values = System.getProperty(propName);
if (values == null) {
return false;
}
StringTokenizer token = new StringTokenizer(values, ",");
try {
while (token.hasMoreTokens()) {
if (propValue.equals((token.nextToken().trim()))) {
isSet = true;
break;
}
}
} catch (NoSuchElementException e) {
}
return isSet;
}
private static void dlog(String msg) {
if (debug) {
logger.log(DEBUG, msg);
}
}
private static void logError(Exception e) {
logger.log(ERROR, e.getMessage(), e);
}
private static void log(String level, String msg) {
logger.log(INFO, () -> level + ": " + msg);
}
}
/****************
* SeeBeyond's Issue: Session close causing RollbackException on commit in MDB CMT
*
*
* According to SeeBeyond, in its XASession implementation, session close will cause not only local transactions to
* rollback but also global (XA) transactions to rollback.
*
* SeeBeyond claimed that to change their implementation is too risky and is not something they want to consider. To
* support Sun customer Syntegra, we are planning to provide a workaround in our wrapper classes for AS7 foreign JMS
* provider to better work with SeeBeyond's JMS implementation.
*
* The workaround is to delay session close until a started XA transaction completes.
*
* There are possibly 3 approaches to address the problem:
*
*
* 1. Code changes in MQ source in JMS wrapper classes for foreign JMS providers cons: no access to
* Transaction/TransactionManager objects therefore can't get transaction completion notification using
* jakarta.transaction.Synchronization.
*
* pros: only need to patch MQ jar file affect only foreign JMS provider support in AS7
*
* 2. Code changes in AS7 source in JMS wrapper classes cons: affects MQ integration source in AS7 would need patch AS7
*
* 3. Code changes in AS7 source and MQ source pros: scope the change in AS7 source as small as possible
*
* cons: would need patch both AS7 and MQ jar file
*
*
* Since we are planning to provide the workaround for only one customer, we are planning to use #1 approach with some
* limitations.
*
* Notes on Modifications on MQ's JMS Wrapper Classes for Foreign JMS Providers
* --------------------------------------------------------------------------- 1. Transaction completion detection . Use
* XAResource.commit/rollback (including forget call) calls to detect transaction completion
*
* . XAException error codes will also be checked for XAResource.commit/rollback calls to distinguish those XAExceptions
* that indicates transaction completion and those indicates transaction not completed*
*
*
* limitation: Since we can't use jakarta.transaction.Synchronization, our transaction completion detection through
* XAResource object would not be bullet-proof. There can be edge cases where a transaction completion is not detected
* which would cause Session leak or where a Session can be closed too early due to vagueness in the XA/JTA specs for
* some edge cases. But it's our understanding that these edge cases should not be common. Understanding SeeBeyond's XA
* resource manager implementation detail maybe able to help narrow down some of these edge cases.
*
* This transaction completion detection also has the assumption that a transaction started on a XAResource object will
* always be completed on the same XAResource object. We expect SeeBeyond's XAResource implementation meets this
* requirement since its XAResource.isSameRM always returns false for two different XAResource objects.
*
* Since Connection will not be hard-closed until all its Sessions have been hard-closed (see #3), session leak would
* also lead to connection leak. Therefore applications should provide some connection monitoring/ cleanup mechanism to
* periodically monitor/cleanup leaked connections if Connection leak does become an issue.
*
* : see more details at XAResource XAException Checks below
*
*
* 2. JMS Session close semantic . Session close is deferred until transaction completes
*
* . Session.close is NOT deferred if there is a Session MessageListener
*
* . Operations on 'soft-closed' Session will NOT be prevented
*
*
* limitations: . no JMS Session close semantic of throwing IllegalStateException when using the closed Session or its
* producers/consumers . no JMS Session close semantic of unblocking MessageConsumer.receive . no JMS Session close
* semantic of waiting MessageConsumer.receive orderly returning the message . no JMS Session close semantic of waiting
* for all MessageConsumer's MessageListeners return
*
*
* 3. JMS Connection close semantic . Connection close is deferred until all sessions has been hard-closed
*
* . Operations (except create XASession) on 'soft-closed' Connection will NOT be prevented
*
* . See 'JMS Session close semantic'
*
* limitation: . no JMS Connection close semantic of throwing IllegalStateException when using a closed Connection
* (except create XASession) or its Sessions . no JMS Connection close semantic of only return until message processing
* has been orderly shut down
*
*
* 4. The Session.close()/Connection.close deferral is turned on by setting following jvm system property
*
* delaySessionCloseForExternalJMSXAResource
*
* which is a ',' separated list of fully-qualified class name strings that are XAResource implementation classes for an
* external JMS provider.
*
*
* XAResource XAException Checks ----------------------------- A XASession will only do soft-close if there is any
* uncompleted XA transaction that has been started on its XAResource (and there is no Session MessageListener set for
* the XASession). A soft-closed XASession will be hard-closed only when all XA transactions that have started on its
* XAResource have been completed or if its Connection is closed.
*
* 1. XAResource.commit
*
* The transaction will not mark as completed for following XAExceptions: (expecting XAResource.forget to be called
* later XA spec: A resource manager that heuristically completes work done on behalf of a transaction branch must keep
* track of that branch along with the decision (that is, heuristically committed, rolled back, or mixed) XA_HEURHAZ
* XA_HEURCOM XA_HEURRB XA_HEURMIX
*
* XA_RETRY (XA spec: All resources held on behalf of *xid remain in a prepared state until commitment is possible.)
*
*
* The transaction will be marked as completed for following XAExceptions: (XA spec: Upon return the resource manager
* has rolled back the branch's work and has released all held resources. Only applicable to 1-phase commit)
* XA_RBROLLBACK XA_RBCOMMFAIL XA_RBDEADLOCK XA_RBINTEGRITY XA_RBOTHER XA_RBPROTO XA_RBTIMEOUT XA_RBTRANSIENT
*
* All other XAExceptions will mark the transaction completion (XA spec: not clear)
*
*
* 2. XAResource.rollback
*
* The transaction will not mark as completed for following XAExceptions: (expecting XAResource.forget to be called
* later XA spec: A resource manager that heuristically completes work done on behalf of a transaction branch must keep
* track of that branch along with the decision (that is, heuristically committed, rolled back, or mixed) XA_HEURHAZ
* XA_HEURCOM XA_HEURRB XA_HEURMIX
*
*
* The transaction will be marked as completed for following XAExceptions: (XA spec: The resource manager has rolled
* back the transaction branch's work and has released all held resources. typically occurs when the branch was already
* marked rollback-only)
*
* XA_RBROLLBACK XA_RBCOMMFAIL XA_RBDEADLOCK XA_RBINTEGRITY XA_RBOTHER XA_RBPROTO XA_RBTIMEOUT XA_RBTRANSIENT
*
* All other XAExceptions will mark the transaction completion (CA spec: not clear)
*
*
* 3. XAResource.forget
*
* The transaction is marked as completed for any XAExceptions (XA spec: not clear)
*
*
* 4. XAResource.start A new transaction is not marked as started for XAExceptions: (XA spec: The resource manager has
* not associated the transaction branch with the thread of control and has marked *xid rollback-only.)
*
* XA_RBROLLBACK XA_RBCOMMFAIL XA_RBDEADLOCK XA_RBINTEGRITY XA_RBOTHER XA_RBPROTO XA_RBTIMEOUT XA_RBTRANSIENT
*
* A transaction is not marked as started for any other XAExceptions
*
* If TMJOIN or TMRESUME flag set, it will not mark a new transaction as started regardless it throws XAException or not
*
*
* 5. XAResource.end
*
* The transaction will not be marked as completed for following XAExceptions: (expecting rollback to be called XA spec:
* The resource manager has dissociated the transaction branch from the thread of control and has marked rollback-only
* the work performed on behalf of *xid.)
*
* XA_RBROLLBACK XA_RBCOMMFAIL XA_RBDEADLOCK XA_RBINTEGRITY XA_RBOTHER XA_RBPROTO XA_RBTIMEOUT XA_RBTRANSIENT
*
* The transaction will be marked as completed for any other XAExceptions (XA spec: not clear)
*
* The transaction is not marked as completed when no XAException
*
* 5. XAResource.prepare
*
* The transaction will be marked as completed if returns XA_RDONLY (XA spec: XA_RDONLY - The resource manager may
* release all resources and forget about the branch. committed)
*
* The transaction will not be marked as completed if returns XA_OK (XA spec: A resource manager cannot erase its
* knowledge of a branch until the transaction manager calls either commit or rollback)
*
* The transaction will be marked as completed for following XAException: (XA spec: - Upon return, the resource manager
* has rolled back the branch's work and has released all held resources.)
*
* XA_RBROLLBACK XA_RBCOMMFAIL XA_RBDEADLOCK XA_RBINTEGRITY XA_RBOTHER XA_RBPROTO XA_RBTIMEOUT XA_RBTRANSIENT
*
* The transaction will be marked as completed for any other XAException
*****************/