All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.infinispan.transaction.impl.LocalTransaction Maven / Gradle / Ivy

There is a newer version: 9.1.7.Final
Show newest version
package org.infinispan.transaction.impl;

import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.Map;
import java.util.Set;

import javax.transaction.Transaction;

import org.infinispan.commands.write.ClearCommand;
import org.infinispan.commands.write.WriteCommand;
import org.infinispan.commons.CacheException;
import org.infinispan.commons.util.CollectionFactory;
import org.infinispan.container.entries.CacheEntry;
import org.infinispan.context.Flag;
import org.infinispan.context.impl.FlagBitSets;
import org.infinispan.remoting.transport.Address;
import org.infinispan.transaction.xa.GlobalTransaction;
import org.infinispan.util.logging.Log;
import org.infinispan.util.logging.LogFactory;

/**
 * Object that holds transaction's state on the node where it originated; as opposed to {@link RemoteTransaction}.
 *
 * @author [email protected]
 * @author Pedro Ruivo
 * @since 5.0
 */
public abstract class LocalTransaction extends AbstractCacheTransaction {

   private static final Log log = LogFactory.getLog(LocalTransaction.class);
   private static final boolean trace = log.isTraceEnabled();

   private Set
remoteLockedNodes; private Set readKeys = null; private final Transaction transaction; private final boolean implicitTransaction; private volatile boolean isFromRemoteSite; private boolean prepareSent; private boolean commitOrRollbackSent; public LocalTransaction(Transaction transaction, GlobalTransaction tx, boolean implicitTransaction, int topologyId, long txCreationTime) { super(tx, topologyId, txCreationTime); this.transaction = transaction; this.implicitTransaction = implicitTransaction; } public final void addModification(WriteCommand mod) { if (trace) log.tracef("Adding modification %s. Mod list is %s", mod, modifications); if (modifications == null) { // we need to synchronize this collection to be able to get a valid snapshot from another thread during state transfer modifications = Collections.synchronizedList(new LinkedList()); } if (mod.hasAnyFlag(FlagBitSets.CACHE_MODE_LOCAL)) { hasLocalOnlyModifications = true; } modifications.add(mod); } public void locksAcquired(Collection
nodes) { if (trace) log.tracef("Adding remote locks on %s. Remote locks are %s", nodes, remoteLockedNodes); if (remoteLockedNodes == null) remoteLockedNodes = new HashSet<>(nodes); else remoteLockedNodes.addAll(nodes); } public Collection
getRemoteLocksAcquired(){ if (remoteLockedNodes == null) return Collections.emptySet(); return remoteLockedNodes; } public void clearRemoteLocksAcquired() { if (remoteLockedNodes != null) remoteLockedNodes.clear(); } public Transaction getTransaction() { return transaction; } @Override public Map getLookedUpEntries() { return lookedUpEntries == null ? Collections.emptyMap() : lookedUpEntries; } public boolean isImplicitTransaction() { return implicitTransaction; } @Override public void putLookedUpEntry(Object key, CacheEntry e) { if (isMarkedForRollback()) { throw new CacheException("This transaction is marked for rollback and cannot acquire locks!"); } if (lookedUpEntries == null) lookedUpEntries = CollectionFactory.makeMap(4); lookedUpEntries.put(key, e); } @Override public void putLookedUpEntries(Map entries) { if (isMarkedForRollback()) { throw new CacheException("This transaction is marked for rollback and cannot acquire locks!"); } if (lookedUpEntries == null) { lookedUpEntries = CollectionFactory.makeMap(entries); } else { lookedUpEntries.putAll(entries); } } public boolean isReadOnly() { return modifications == null || modifications.isEmpty(); } public abstract boolean isEnlisted(); @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; LocalTransaction that = (LocalTransaction) o; return tx.getId() == that.tx.getId(); } @Override public int hashCode() { long id = tx.getId(); return (int)(id ^ (id >>> 32)); } @Override public String toString() { return "LocalTransaction{" + "remoteLockedNodes=" + remoteLockedNodes + ", isMarkedForRollback=" + isMarkedForRollback() + ", lockedKeys=" + getLockedKeys() + ", backupKeyLocks=" + getBackupLockedKeys() + ", topologyId=" + topologyId + ", stateTransferFlag=" + getStateTransferFlag() + "} " + super.toString(); } @Override public void addReadKey(Object key) { if (readKeys == null) readKeys = new HashSet<>(2); readKeys.add(key); } @Override public boolean keyRead(Object key) { return readKeys != null && readKeys.contains(key); } public void setStateTransferFlag(Flag stateTransferFlag) { if (this.getStateTransferFlag() == null && (stateTransferFlag == Flag.PUT_FOR_STATE_TRANSFER || stateTransferFlag == Flag.PUT_FOR_X_SITE_STATE_TRANSFER)) { internalSetStateTransferFlag(stateTransferFlag); } } /** * When x-site replication is used, this returns when this operation * happens as a result of backing up data from a remote site. */ public boolean isFromRemoteSite() { return isFromRemoteSite; } /** * @see #isFromRemoteSite() */ public void setFromRemoteSite(boolean fromRemoteSite) { isFromRemoteSite = fromRemoteSite; } /** * Calculates the list of nodes to which a commit/rollback needs to be sent based on the nodes to which prepare * was sent. If the commit/rollback is to be sent in the same topologyId, then the 'recipients' param is returned back. * If the current topologyId is different than the topologyId of this transaction ({@link #getTopologyId()} then * this method returns the reunion between 'recipients' and {@link #getRemoteLocksAcquired()} from which it discards * the members that left. */ public Collection
getCommitNodes(Collection
recipients, int currentTopologyId, Collection
members) { if (trace) log.tracef("getCommitNodes recipients=%s, currentTopologyId=%s, members=%s, txTopologyId=%s", recipients, currentTopologyId, members, getTopologyId()); if (hasModification(ClearCommand.class)) { return members; } if (recipients == null) { return null; } if (getTopologyId() == currentTopologyId) { return recipients; } Set
allRecipients = new HashSet<>(getRemoteLocksAcquired()); allRecipients.addAll(recipients); allRecipients.retainAll(members); if (trace) log.tracef("The merged list of nodes to send commit/rollback is %s", allRecipients); return allRecipients; } /** * Sets the prepare sent for this transaction */ public final void markPrepareSent() { prepareSent = true; } /** * @return true if the prepare was sent to the other nodes */ public final boolean isPrepareSent() { return prepareSent; } /** * Sets the commit or rollback sent for this transaction */ public final void markCommitOrRollbackSent() { commitOrRollbackSent = true; } /** * @return true if the commit or rollback was sent to the other nodes */ public final boolean isCommitOrRollbackSent() { return commitOrRollbackSent; } }