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

org.infinispan.configuration.cache.TransactionConfigurationBuilder Maven / Gradle / Ivy

There is a newer version: 9.1.7.Final
Show newest version
package org.infinispan.configuration.cache;

import static org.infinispan.configuration.cache.TransactionConfiguration.*;

import java.lang.invoke.MethodHandles;
import java.util.concurrent.TimeUnit;

import javax.transaction.Synchronization;
import javax.transaction.TransactionManager;
import javax.transaction.xa.XAResource;

import org.infinispan.commons.configuration.Builder;
import org.infinispan.commons.configuration.attributes.Attribute;
import org.infinispan.commons.configuration.attributes.AttributeSet;
import org.infinispan.configuration.global.GlobalConfiguration;
import org.infinispan.transaction.LockingMode;
import org.infinispan.transaction.TransactionMode;
import org.infinispan.transaction.TransactionProtocol;
import org.infinispan.transaction.lookup.TransactionManagerLookup;
import org.infinispan.transaction.lookup.TransactionSynchronizationRegistryLookup;
import org.infinispan.util.logging.Log;
import org.infinispan.util.logging.LogFactory;

/**
 * Defines transactional (JTA) characteristics of the cache.
 *
 * @author pmuir
 * @author Pedro Ruivo
 */
public class TransactionConfigurationBuilder extends AbstractConfigurationChildBuilder implements Builder {
   private static final Log log = LogFactory.getLog(MethodHandles.lookup().lookupClass());
   private final AttributeSet attributes;
   private final RecoveryConfigurationBuilder recovery;

   TransactionConfigurationBuilder(ConfigurationBuilder builder) {
      super(builder);
      attributes = TransactionConfiguration.attributeDefinitionSet();
      this.recovery = new RecoveryConfigurationBuilder(this);
   }

   /**
    * If the cache is transactional (i.e. {@link #transactionMode(org.infinispan.transaction.TransactionMode)} == TransactionMode.TRANSACTIONAL)
    * and transactionAutoCommit is enabled then for single operation transactions
    * the user doesn't need to manually start a transaction, but a transactions
    * is injected by the system. Defaults to true.
    */
   public TransactionConfigurationBuilder autoCommit(boolean b) {
      attributes.attribute(AUTO_COMMIT).set(b);
      return this;
   }

   /**
    * If there are any ongoing transactions when a cache is stopped, Infinispan waits for ongoing
    * remote and local transactions to finish. The amount of time to wait for is defined by the
    * cache stop timeout. It is recommended that this value does not exceed the transaction timeout
    * because even if a new transaction was started just before the cache was stopped, this could
    * only last as long as the transaction timeout allows it.
    * 

* This configuration property may be adjusted at runtime */ public TransactionConfigurationBuilder cacheStopTimeout(long l) { attributes.attribute(CACHE_STOP_TIMEOUT).set(l); return this; } /** * If there are any ongoing transactions when a cache is stopped, Infinispan waits for ongoing * remote and local transactions to finish. The amount of time to wait for is defined by the * cache stop timeout. It is recommended that this value does not exceed the transaction timeout * because even if a new transaction was started just before the cache was stopped, this could * only last as long as the transaction timeout allows it. *

* This configuration property may be adjusted at runtime */ public TransactionConfigurationBuilder cacheStopTimeout(long l, TimeUnit unit) { return cacheStopTimeout(unit.toMillis(l)); } /** * Only has effect for DIST mode and when useEagerLocking is set to true. When this is enabled, * then only one node is locked in the cluster, disregarding numOwners config. On the opposite, * if this is false, then on all cache.lock() calls numOwners RPCs are being performed. The node * that gets locked is the main data owner, i.e. the node where data would reside if * numOwners==1. If the node where the lock resides crashes, then the transaction is marked for * rollback - data is in a consistent state, no fault tolerance. * * @deprecated starting with Infinispan 5.1 single node locking is used by default */ @Deprecated public TransactionConfigurationBuilder eagerLockingSingleNode(boolean b) { attributes.attribute(EAGER_LOCKING_SINGLE_NODE).set(b); return this; } /** * Configures whether the cache uses optimistic or pessimistic locking. If the cache is not * transactional then the locking mode is ignored. * * @see org.infinispan.configuration.cache.TransactionConfiguration#transactionMode() */ public TransactionConfigurationBuilder lockingMode(LockingMode lockingMode) { attributes.attribute(LOCKING_MODE).set(lockingMode); return this; } LockingMode lockingMode() { return attributes.attribute(LOCKING_MODE).get(); } /** * If true, the cluster-wide commit phase in two-phase commit (2PC) transactions will be * synchronous, so Infinispan will wait for responses from all nodes to which the commit was * sent. Otherwise, the commit phase will be asynchronous. Keeping it as false improves * performance of 2PC transactions, but it can lead to inconsistencies when a backup owner * only commits the transaction after the primary owner released the lock. */ public TransactionConfigurationBuilder syncCommitPhase(boolean b) { attributes.attribute(SYNC_COMMIT_PHASE).set(b); return this; } /** * See {@link #syncCommitPhase(boolean)} * * @return {@code true} if sync commit phase is enabled */ boolean syncCommitPhase() { return attributes.attribute(SYNC_COMMIT_PHASE).get(); } /** * If true, the cluster-wide rollback phase in two-phase commit (2PC) transactions will be * synchronous, so Infinispan will wait for responses from all nodes to which the rollback was * sent. Otherwise, the rollback phase will be asynchronous. * * Keeping it as false can lead to inconsistencies when a transaction is rolled back because of * a commit timeout, as a backup owner could commit the transaction after the primary released the lock. */ public TransactionConfigurationBuilder syncRollbackPhase(boolean b) { attributes.attribute(SYNC_ROLLBACK_PHASE).set(b); return this; } /** * Configure Transaction manager lookup directly using an instance of TransactionManagerLookup. * Calling this method marks the cache as transactional. */ public TransactionConfigurationBuilder transactionManagerLookup(TransactionManagerLookup tml) { attributes.attribute(TRANSACTION_MANAGER_LOOKUP).set(tml); if (tml != null) { this.transactionMode(TransactionMode.TRANSACTIONAL); } return this; } /** * Configure Transaction Synchronization Registry lookup directly using an instance of * TransactionManagerLookup. Calling this method marks the cache as transactional. */ public TransactionConfigurationBuilder transactionSynchronizationRegistryLookup(TransactionSynchronizationRegistryLookup lookup) { attributes.attribute(TRANSACTION_SYNCHRONIZATION_REGISTRY_LOOKUP).set(lookup); return this; } public TransactionConfigurationBuilder transactionMode(TransactionMode transactionMode) { attributes.attribute(TRANSACTION_MODE).set(transactionMode); return this; } TransactionMode transactionMode() { if (attributes.attribute(TRANSACTION_MODE).isModified()) { return attributes.attribute(TRANSACTION_MODE).get(); } else { return null; } } /** * Prevents more than one transaction being written to a key by enforcing cluster-wide locks * on each write operation. Infinispan attempts to obtain locks on specified cache keys across * all nodes in a cluster. All locks are released during the commit or rollback phase. * This configuration might be used when a high contention on keys is occurring, resulting in * inefficiencies and unexpected roll back operations. * * @deprecated Starting with Infinispan 5.1 eager locking is replaced with pessimistic locking and can * be enforced by setting transaction's locking mode to PESSIMISTIC. */ @Deprecated public TransactionConfigurationBuilder useEagerLocking(boolean b) { this.attributes.attribute(USE_EAGER_LOCKING).set(b); return this; } /** * Configures whether the cache registers a synchronization with the transaction manager, or registers itself as an * XA resource. It is often unnecessary to register as a full XA resource unless you intend to make use of recovery * as well, and registering a synchronization is significantly more efficient. * @param b if true, {@link Synchronization}s are used rather than {@link XAResource}s when communicating with a {@link TransactionManager}. */ public TransactionConfigurationBuilder useSynchronization(boolean b) { attributes.attribute(USE_SYNCHRONIZATION).set(b); return this; } /** * See {@link #useSynchronization(boolean)} * * @return {@code true} if synchronization enlistment is enabled */ boolean useSynchronization() { return attributes.attribute(USE_SYNCHRONIZATION).get(); } /** * This method allows configuration of the transaction recovery cache. When this method is * called, it automatically enables recovery. So, if you want it to be disabled, make sure you * call {@link org.infinispan.config.FluentConfiguration.RecoveryConfig#disable()} */ public RecoveryConfigurationBuilder recovery() { return recovery; } /** * Before Infinispan 5.1 you could access the cache both transactionally and * non-transactionally. Naturally the non-transactional access is faster and * offers less consistency guarantees. From Infinispan 5.1 onwards, mixed * access is no longer supported, so if you wanna speed up transactional * caches and you're ready to trade some consistency guarantees, you can * enable use1PcForAutoCommitTransactions.

* * What this configuration option does is force an induced transaction, * that has been started by Infinispan as a result of enabling autoCommit, * to commit in a single phase. So only 1 RPC instead of 2RPCs as in the * case of a full 2 Phase Commit (2PC). */ public TransactionConfigurationBuilder use1PcForAutoCommitTransactions(boolean b) { attributes.attribute(USE_1_PC_FOR_AUTO_COMMIT_TRANSACTIONS).set(b); return this; } /** *The time interval (millis) at which the thread that cleans up transaction completion information kicks in. Defaults to 30000. */ public TransactionConfigurationBuilder reaperWakeUpInterval(long interval) { attributes.attribute(REAPER_WAKE_UP_INTERVAL).set(interval); return this; } /** * The duration (millis) in which to keep information about the completion of a transaction. Defaults to 60000. */ public TransactionConfigurationBuilder completedTxTimeout(long timeout) { attributes.attribute(COMPLETED_TX_TIMEOUT).set(timeout); return this; } public TransactionConfigurationBuilder transactionProtocol(TransactionProtocol transactionProtocol) { attributes.attribute(TRANSACTION_PROTOCOL).set(transactionProtocol); return this; } /** * See {@link #transactionProtocol(TransactionProtocol)}. * * @return the current configured {@link TransactionProtocol}. */ public TransactionProtocol transactionProtocol() { return attributes.attribute(TRANSACTION_PROTOCOL).get(); } /** * @return are transactional notifications ( * {@link org.infinispan.notifications.cachelistener.annotation.TransactionRegistered} and * {@link org.infinispan.notifications.cachelistener.annotation.TransactionCompleted}) triggered? */ public TransactionConfigurationBuilder notifications(boolean enabled) { attributes.attribute(NOTIFICATIONS).set(enabled); return this; } @Override public void validate() { Attribute reaperWakeUpInterval = attributes.attribute(REAPER_WAKE_UP_INTERVAL); Attribute completedTxTimeout = attributes.attribute(COMPLETED_TX_TIMEOUT); if (reaperWakeUpInterval.get()< 0) throw log.invalidReaperWakeUpInterval(reaperWakeUpInterval.get()); if (completedTxTimeout.get() < 0) throw log.invalidCompletedTxTimeout(completedTxTimeout.get()); if(attributes.attribute(TRANSACTION_PROTOCOL).get() == TransactionProtocol.TOTAL_ORDER) { //total order only supports transactional caches if(transactionMode() != TransactionMode.TRANSACTIONAL) { throw log.invalidTxModeForTotalOrder(transactionMode()); } //total order only supports replicated and distributed mode if(!clustering().cacheMode().isReplicated() && !clustering().cacheMode().isDistributed()) { throw log.invalidCacheModeForTotalOrder(clustering().cacheMode().friendlyCacheModeString()); } if (lockingMode() != LockingMode.OPTIMISTIC) { throw log.invalidLockingModeForTotalOrder(lockingMode()); } } if (!attributes.attribute(NOTIFICATIONS).get()) { log.transactionNotificationsDisabled(); } recovery.validate(); } @Override public void validate(GlobalConfiguration globalConfig) { recovery.validate(globalConfig); } @Override public TransactionConfiguration create() { if (attributes.attribute(USE_EAGER_LOCKING).get()) { lockingMode(LockingMode.PESSIMISTIC); } if (transactionMode() == null && getBuilder().invocationBatching().isEnabled()) transactionMode(TransactionMode.TRANSACTIONAL); else if (transactionMode() == null) transactionMode(TransactionMode.NON_TRANSACTIONAL); return new TransactionConfiguration(attributes.protect(), recovery.create()); } @Override public TransactionConfigurationBuilder read(TransactionConfiguration template) { this.attributes.read(template.attributes()); this.recovery.read(template.recovery()); return this; } @Override public String toString() { return "TransactionConfigurationBuilder [attributes=" + attributes + ", recovery=" + recovery + "]"; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy