se.laz.casual.jca.CasualXAResource Maven / Gradle / Ivy
/*
* Copyright (c) 2017 - 2024, The casual project. All rights reserved.
*
* This software is licensed under the MIT license, https://opensource.org/licenses/MIT
*/
package se.laz.casual.jca;
import se.laz.casual.api.flags.Flag;
import se.laz.casual.api.flags.XAFlags;
import se.laz.casual.api.network.protocol.messages.CasualNWMessage;
import se.laz.casual.api.util.PrettyPrinter;
import se.laz.casual.api.xa.XAReturnCode;
import se.laz.casual.api.xa.XID;
import se.laz.casual.network.protocol.messages.CasualNWMessageImpl;
import se.laz.casual.network.protocol.messages.transaction.CasualTransactionResourceCommitReplyMessage;
import se.laz.casual.network.protocol.messages.transaction.CasualTransactionResourceCommitRequestMessage;
import se.laz.casual.network.protocol.messages.transaction.CasualTransactionResourcePrepareReplyMessage;
import se.laz.casual.network.protocol.messages.transaction.CasualTransactionResourcePrepareRequestMessage;
import se.laz.casual.network.protocol.messages.transaction.CasualTransactionResourceRollbackReplyMessage;
import se.laz.casual.network.protocol.messages.transaction.CasualTransactionResourceRollbackRequestMessage;
import javax.transaction.xa.XAException;
import javax.transaction.xa.XAResource;
import javax.transaction.xa.Xid;
import java.util.UUID;
import java.util.concurrent.CompletableFuture;
import java.util.logging.Logger;
/**
* @author jone
*/
public class CasualXAResource implements XAResource
{
private static final Logger LOG = Logger.getLogger(CasualXAResource.class.getName());
private static final Xid[] NO_XIDS = {};
private final CasualManagedConnection casualManagedConnection;
private final int resourceManagerId;
private Xid currentXid = XID.NULL_XID;
private boolean readOnly = false;
public CasualXAResource(final CasualManagedConnection connection, int resourceManagerId)
{
casualManagedConnection = connection;
this.resourceManagerId = resourceManagerId;
}
public Xid getCurrentXid()
{
return currentXid;
}
@Override
public void commit(Xid xid, boolean onePhaseCommit) throws XAException
{
Flag flags = Flag.of(XAFlags.TMNOFLAGS);
if (onePhaseCommit)
{
flags = Flag.of(XAFlags.TMONEPHASE);
}
LOG.finest(() -> String.format("trying to commit, xid: %s ( %s ) onePhase?%b", PrettyPrinter.casualStringify(xid), xid, onePhaseCommit));
CasualTransactionResourceCommitRequestMessage commitRequest =
CasualTransactionResourceCommitRequestMessage.of(UUID.randomUUID(), xid, resourceManagerId, flags);
CasualNWMessage requestEnvelope = CasualNWMessageImpl.of(UUID.randomUUID(), commitRequest);
CompletableFuture> replyEnvelopeFuture = casualManagedConnection.getNetworkConnection().request(requestEnvelope);
CasualNWMessage replyEnvelope = replyEnvelopeFuture.join();
CasualTransactionResourceCommitReplyMessage replyMsg = replyEnvelope.getMessage();
throwWhenTransactionErrorCode(replyMsg.getTransactionReturnCode());
LOG.finest(() -> String.format("commited, xid: %s ( %s )", PrettyPrinter.casualStringify(xid), xid));
}
/**
* Removes the
*
* @param xid transaction id
* @param flag - TMSUCCESS, TMFAIL, or TMSUSPEND.
* @throws XAException if end fails.
*/
@Override
// java:S1301 - opinionated and, in this case - wrong
@SuppressWarnings("java:S1301")
public void end(Xid xid, int flag) throws XAException
{
LOG.finest(()-> String.format("end, xid: %s (%s) flag: %d, %s ", PrettyPrinter.casualStringify(xid), xid, flag, XAFlags.unmarshall(flag)));
CasualResourceManager.getInstance().remove(domainId(), xid);
disassociate();
XAFlags f = XAFlags.unmarshall(flag);
switch(f)
{
case TMSUCCESS, TMFAIL, TMSUSPEND:
break;
default:
LOG.finest(()->"throwing XAException.XAER_RMFAIL");
throw new XAException(XAException.XAER_RMFAIL);
}
}
/**
* Removes the records for a heuristically completed
* transaction
*
* @param xid - ID of heuristically complete transaction
* @throws XAException - Possible exception values are XAER_RMERR, XAER_RMFAIL, XAER_NOTA, XAER_INVAL, or XAER_PROTO.
*/
@Override
public void forget(Xid xid) throws XAException
{
throw new UnsupportedOperationException("Not implemented");
}
@Override
public int getTransactionTimeout() throws XAException
{
return casualManagedConnection.getTransactionTimeout();
}
@Override
public boolean isSameRM(XAResource xaResource) throws XAException
{
if(xaResource instanceof CasualXAResource casualXAResource)
{
return casualXAResource.casualManagedConnection.getDomainId().equals(casualManagedConnection.getDomainId());
}
return false;
}
@Override
public int prepare(Xid xid) throws XAException
{
if (isReadOnly())
{
return XAResource.XA_RDONLY;
}
LOG.finest(() -> String.format("trying to prepare, xid: %s ( %s )", PrettyPrinter.casualStringify(xid), xid));
Flag flags = Flag.of(XAFlags.TMNOFLAGS);
CasualTransactionResourcePrepareRequestMessage prepareRequest = CasualTransactionResourcePrepareRequestMessage.of(UUID.randomUUID(), xid, resourceManagerId, flags);
CasualNWMessage requestEnvelope = CasualNWMessageImpl.of(UUID.randomUUID(), prepareRequest);
CompletableFuture> replyEnvelopeFuture = casualManagedConnection.getNetworkConnection().request(requestEnvelope);
CasualNWMessage replyEnvelope = replyEnvelopeFuture.join();
CasualTransactionResourcePrepareReplyMessage replyMsg = replyEnvelope.getMessage();
throwWhenTransactionErrorCode(replyMsg.getTransactionReturnCode());
LOG.finest(() -> String.format("prepared, xid: %s ( %s ), XA_RDONLY: %b", PrettyPrinter.casualStringify(xid), xid, XAReturnCode.XA_RDONLY == replyMsg.getTransactionReturnCode()));
return replyMsg.getTransactionReturnCode().getId();
}
@Override
public Xid[] recover(int i) throws XAException
{
return NO_XIDS;
}
@Override
public void rollback(Xid xid) throws XAException
{
LOG.finest(() -> String.format("trying to rollback, xid: %s ( %s )", PrettyPrinter.casualStringify(xid), xid));
Flag flags = Flag.of(XAFlags.TMNOFLAGS);
CasualTransactionResourceRollbackRequestMessage request =
CasualTransactionResourceRollbackRequestMessage.of(UUID.randomUUID(), xid, resourceManagerId, flags);
CasualNWMessage requestEnvelope = CasualNWMessageImpl.of(UUID.randomUUID(), request);
CompletableFuture> replyEnvelopeFuture = casualManagedConnection.getNetworkConnection().request(requestEnvelope);
CasualNWMessage replyEnvelope = replyEnvelopeFuture.join();
CasualTransactionResourceRollbackReplyMessage replyMsg = replyEnvelope.getMessage();
throwWhenTransactionErrorCode(replyMsg.getTransactionReturnCode());
LOG.finest(() -> String.format("rolled, xid: %s ( %s )", PrettyPrinter.casualStringify(xid), xid));
}
@Override
public boolean setTransactionTimeout(int i) throws XAException
{
casualManagedConnection.setTransactionTimeout(i);
return true;
}
@Override
public void start(Xid xid, int i) throws XAException
{
LOG.finest(()-> String.format("start, xid: %s (%s) flag: %d, %s ", PrettyPrinter.casualStringify(xid), xid, i, XAFlags.unmarshall(i)));
readOnly = false;
if(!(XAFlags.TMJOIN.getValue() == i || XAFlags.TMRESUME.getValue() == i) &&
CasualResourceManager.getInstance().isPending(domainId(), xid))
{
LOG.finest(()->"throwing XAException.XAER_DUPID");
throw new XAException(XAException.XAER_DUPID);
}
associate(xid);
if(!CasualResourceManager.getInstance().isPending(domainId(), currentXid))
{
CasualResourceManager.getInstance().put(domainId(), currentXid);
}
}
@Override
public String toString()
{
return "CasualXAResource{" +
"currentXid=" + currentXid +
'}';
}
private DomainId domainId()
{
return casualManagedConnection.getDomainId();
}
private void associate(Xid xid)
{
currentXid = xid;
}
public void disassociate()
{
currentXid = XID.NULL_XID;
}
public void setReadOnly()
{
readOnly = true;
}
public boolean isReadOnly()
{
return readOnly;
}
// java:S1301 - opinionated and, in this case - wrong
@SuppressWarnings("java:S1301")
private void throwWhenTransactionErrorCode(final XAReturnCode transactionReturnCode) throws XAException
{
switch( transactionReturnCode )
{
case XA_OK, XA_RDONLY:
break;
default:
LOG.finest(()->"throwing XAException for XAReturnCode: " + transactionReturnCode);
throw new XAException( transactionReturnCode.getId());
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy