org.infinispan.statetransfer.StateTransferInterceptor Maven / Gradle / Ivy
package org.infinispan.statetransfer;
import java.util.concurrent.CompletableFuture;
import org.infinispan.commands.TopologyAffectedCommand;
import org.infinispan.commands.VisitableCommand;
import org.infinispan.commands.control.LockControlCommand;
import org.infinispan.commands.functional.ReadWriteKeyCommand;
import org.infinispan.commands.functional.ReadWriteKeyValueCommand;
import org.infinispan.commands.functional.ReadWriteManyCommand;
import org.infinispan.commands.functional.ReadWriteManyEntriesCommand;
import org.infinispan.commands.functional.WriteOnlyKeyCommand;
import org.infinispan.commands.functional.WriteOnlyKeyValueCommand;
import org.infinispan.commands.functional.WriteOnlyManyCommand;
import org.infinispan.commands.functional.WriteOnlyManyEntriesCommand;
import org.infinispan.commands.tx.CommitCommand;
import org.infinispan.commands.tx.PrepareCommand;
import org.infinispan.commands.tx.RollbackCommand;
import org.infinispan.commands.tx.TransactionBoundaryCommand;
import org.infinispan.commands.write.ClearCommand;
import org.infinispan.commands.write.ComputeCommand;
import org.infinispan.commands.write.ComputeIfAbsentCommand;
import org.infinispan.commands.write.EvictCommand;
import org.infinispan.commands.write.InvalidateCommand;
import org.infinispan.commands.write.InvalidateL1Command;
import org.infinispan.commands.write.PutKeyValueCommand;
import org.infinispan.commands.write.PutMapCommand;
import org.infinispan.commands.write.RemoveCommand;
import org.infinispan.commands.write.ReplaceCommand;
import org.infinispan.commands.write.WriteCommand;
import org.infinispan.context.InvocationContext;
import org.infinispan.context.impl.FlagBitSets;
import org.infinispan.context.impl.TxInvocationContext;
import org.infinispan.interceptors.InvocationFinallyFunction;
import org.infinispan.interceptors.impl.BaseStateTransferInterceptor;
import org.infinispan.remoting.RemoteException;
import org.infinispan.remoting.responses.UnsureResponse;
import org.infinispan.remoting.transport.Address;
import org.infinispan.remoting.transport.jgroups.SuspectException;
import org.infinispan.util.logging.Log;
import org.infinispan.util.logging.LogFactory;
/**
* This interceptor has two tasks:
*
* - If the command's topology id is higher than the current topology id,
* wait for the node to receive transaction data for the new topology id.
* - If the topology id changed during a command's execution, retry the command, but only on the
* originator (which replicates it to the new owners).
*
*
* If the cache is configured with asynchronous replication, owners cannot signal to the originator that they
* saw a new topology, so instead each owner forwards the command to all the other owners in the new topology.
*
* @author [email protected]
*/
public class StateTransferInterceptor extends BaseStateTransferInterceptor {
private static final Log log = LogFactory.getLog(StateTransferInterceptor.class);
private static final boolean trace = log.isTraceEnabled();
private final InvocationFinallyFunction handleTxReturn = this::handleTxReturn;
private final InvocationFinallyFunction handleTxWriteReturn = this::handleTxWriteReturn;
private final InvocationFinallyFunction handleNonTxWriteReturn = this::handleNonTxWriteReturn;
@Override
public Object visitPrepareCommand(TxInvocationContext ctx, PrepareCommand command)
throws Throwable {
return handleTxCommand(ctx, command);
}
@Override
public Object visitCommitCommand(TxInvocationContext ctx, CommitCommand command)
throws Throwable {
return handleTxCommand(ctx, command);
}
@Override
public Object visitRollbackCommand(TxInvocationContext ctx, RollbackCommand command)
throws Throwable {
return handleTxCommand(ctx, command);
}
@Override
public Object visitLockControlCommand(TxInvocationContext ctx, LockControlCommand command)
throws Throwable {
if (trace) log.tracef("handleTxCommand for command %s, origin %s", command, getOrigin(ctx));
updateTopologyId(command);
return invokeNextAndHandle(ctx, command, handleTxReturn);
}
@Override
public Object visitPutKeyValueCommand(InvocationContext ctx, PutKeyValueCommand command)
throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitPutMapCommand(InvocationContext ctx, PutMapCommand command)
throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitRemoveCommand(InvocationContext ctx, RemoveCommand command)
throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitReplaceCommand(InvocationContext ctx, ReplaceCommand command)
throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitComputeCommand(InvocationContext ctx, ComputeCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitComputeIfAbsentCommand(InvocationContext ctx, ComputeIfAbsentCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitClearCommand(InvocationContext ctx, ClearCommand command)
throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitInvalidateCommand(InvocationContext ctx, InvalidateCommand command)
throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitInvalidateL1Command(InvocationContext ctx, InvalidateL1Command command)
throws Throwable {
// no need to forward this command
return invokeNext(ctx, command);
}
@Override
public Object visitEvictCommand(InvocationContext ctx, EvictCommand command)
throws Throwable {
// it's not necessary to propagate eviction to the new owners in case of state transfer
return invokeNext(ctx, command);
}
@Override
public Object visitReadWriteKeyValueCommand(InvocationContext ctx,
ReadWriteKeyValueCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitReadWriteKeyCommand(InvocationContext ctx, ReadWriteKeyCommand command)
throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitWriteOnlyKeyCommand(InvocationContext ctx, WriteOnlyKeyCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitWriteOnlyManyEntriesCommand(InvocationContext ctx, WriteOnlyManyEntriesCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitWriteOnlyKeyValueCommand(InvocationContext ctx, WriteOnlyKeyValueCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitWriteOnlyManyCommand(InvocationContext ctx, WriteOnlyManyCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitReadWriteManyCommand(InvocationContext ctx, ReadWriteManyCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
@Override
public Object visitReadWriteManyEntriesCommand(InvocationContext ctx, ReadWriteManyEntriesCommand command) throws Throwable {
return handleWriteCommand(ctx, command);
}
/**
* Special processing required for transaction commands.
*
*/
private Object handleTxCommand(TxInvocationContext ctx, TransactionBoundaryCommand command) {
if (trace) log.tracef("handleTxCommand for command %s, origin %s", command, getOrigin(ctx));
updateTopologyId(command);
return invokeNextAndHandle(ctx, command, handleTxReturn);
}
private Address getOrigin(TxInvocationContext ctx) {
// For local commands we may not have a GlobalTransaction yet
return ctx.isOriginLocal() ? ctx.getOrigin() : ctx.getGlobalTransaction().getAddress();
}
private Object handleTxReturn(InvocationContext ctx,
VisitableCommand command, Object rv, Throwable t) throws Throwable {
TransactionBoundaryCommand txCommand = (TransactionBoundaryCommand) command;
int retryTopologyId = -1;
int currentTopology = currentTopologyId();
if (t instanceof OutdatedTopologyException || t instanceof AllOwnersLostException) {
// This can only happen on the originator
retryTopologyId = Math.max(currentTopology, txCommand.getTopologyId() + 1);
} else if (t != null) {
throw t;
}
if (ctx.isOriginLocal()) {
// On the originator, we only retry if we got an OutdatedTopologyException
// Which could be caused either by an owner leaving or by an owner having a newer topology
// No need to retry just because we have a new topology on the originator, all entries were
// wrapped anyway
if (retryTopologyId > 0) {
// Only the originator can retry the command
txCommand.setTopologyId(retryTopologyId);
if (txCommand instanceof PrepareCommand) {
((PrepareCommand) txCommand).setRetriedCommand(true);
}
CompletableFuture transactionDataFuture = stateTransferLock.transactionDataFuture(retryTopologyId);
return retryWhenDone(transactionDataFuture, retryTopologyId, ctx, txCommand, handleTxReturn);
}
} else {
if (currentTopology > txCommand.getTopologyId()) {
// Signal the originator to retry
return UnsureResponse.INSTANCE;
}
}
return rv;
}
private Object handleWriteCommand(InvocationContext ctx, WriteCommand command) {
if (ctx.isInTxScope()) {
return handleTxWriteCommand(ctx, command);
} else {
return handleNonTxWriteCommand(ctx, command);
}
}
private Object handleTxWriteCommand(InvocationContext ctx, WriteCommand command) {
if (trace) log.tracef("handleTxWriteCommand for command %s, origin %s", command, ctx.getOrigin());
updateTopologyId(command);
return invokeNextAndHandle(ctx, command, handleTxWriteReturn);
}
private Object handleTxWriteReturn(InvocationContext rCtx, VisitableCommand rCommand, Object rv, Throwable t)
throws Throwable {
int retryTopologyId = -1;
WriteCommand writeCommand = (WriteCommand) rCommand;
if (t instanceof OutdatedTopologyException || t instanceof AllOwnersLostException) {
// This can only happen on the originator
retryTopologyId = Math.max(currentTopologyId(), writeCommand.getTopologyId() + 1);
} else if (t != null) {
throw t;
}
if (rCtx.isOriginLocal()) {
// On the originator, we only retry if we got an OutdatedTopologyException
// Which could be caused either by an owner leaving or by an owner having a newer topology
// No need to retry just because we have a new topology on the originator, all entries were
// wrapped anyway
if (retryTopologyId > 0) {
// Only the originator can retry the command
writeCommand.setTopologyId(retryTopologyId);
CompletableFuture transactionDataFuture = stateTransferLock.transactionDataFuture(retryTopologyId);
return retryWhenDone(transactionDataFuture, retryTopologyId, rCtx, writeCommand, handleTxWriteReturn);
}
} else {
if (currentTopologyId() > writeCommand.getTopologyId()) {
// Signal the originator to retry
return UnsureResponse.INSTANCE;
}
}
return rv;
}
/**
* For non-tx write commands, we retry the command locally if the topology changed.
* But we only retry on the originator, and only if the command doesn't have
* the {@code CACHE_MODE_LOCAL} flag.
*/
private Object handleNonTxWriteCommand(InvocationContext ctx, WriteCommand command) {
if (trace) log.tracef("handleNonTxWriteCommand for command %s, topology id %d", command, command.getTopologyId());
updateTopologyId(command);
// Only catch OutdatedTopologyExceptions on the originator
if (!ctx.isOriginLocal()) {
return invokeNext(ctx, command);
}
return invokeNextAndHandle(ctx, command, handleNonTxWriteReturn);
}
private Object handleNonTxWriteReturn(InvocationContext rCtx,
VisitableCommand rCommand, Object rv, Throwable t) throws Throwable {
if (t == null)
return rv;
Throwable ce = t;
while (ce instanceof RemoteException) {
ce = ce.getCause();
}
if (!(ce instanceof OutdatedTopologyException) && !(ce instanceof SuspectException) && !(ce instanceof AllOwnersLostException))
throw t;
// We increment the topology id so that updateTopologyIdAndWaitForTransactionData waits for the
// next topology.
// Without this, we could retry the command too fast and we could get the
// OutdatedTopologyException again.
int currentTopologyId = currentTopologyId();
WriteCommand writeCommand = (WriteCommand) rCommand;
int newTopologyId = getNewTopologyId(ce, currentTopologyId, writeCommand);
if (trace)
log.tracef("Retrying command because of topology change, current topology is %d (requested: %d): %s",
currentTopologyId, newTopologyId, writeCommand);
writeCommand.setTopologyId(newTopologyId);
writeCommand.addFlags(FlagBitSets.COMMAND_RETRY);
// In non-tx context, waiting for transaction data is equal to waiting for topology
CompletableFuture transactionDataFuture = stateTransferLock.transactionDataFuture(newTopologyId);
return retryWhenDone(transactionDataFuture, newTopologyId, rCtx, writeCommand, handleNonTxWriteReturn);
}
@Override
public Object handleDefault(InvocationContext ctx, VisitableCommand command)
throws Throwable {
if (command instanceof TopologyAffectedCommand) {
return handleTopologyAffectedCommand(ctx, command, ctx.getOrigin());
} else {
return invokeNext(ctx, command);
}
}
private Object handleTopologyAffectedCommand(InvocationContext ctx,
VisitableCommand command, Address origin) {
if (trace) log.tracef("handleTopologyAffectedCommand for command %s, origin %s", command, origin);
updateTopologyId((TopologyAffectedCommand) command);
return invokeNext(ctx, command);
}
@Override
protected Log getLog() {
return log;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy