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

com.hazelcast.transaction.impl.TransactionImpl Maven / Gradle / Ivy

There is a newer version: 5.5.0
Show newest version
/*
 * Copyright (c) 2008-2021, Hazelcast, Inc. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.hazelcast.transaction.impl;

import com.hazelcast.cluster.Address;
import com.hazelcast.core.HazelcastInstanceNotActiveException;
import com.hazelcast.core.MemberLeftException;
import com.hazelcast.internal.cluster.ClusterService;
import com.hazelcast.internal.util.FutureUtil.ExceptionHandler;
import com.hazelcast.logging.ILogger;
import com.hazelcast.spi.exception.TargetNotMemberException;
import com.hazelcast.spi.impl.NodeEngine;
import com.hazelcast.spi.impl.operationservice.Operation;
import com.hazelcast.spi.impl.operationservice.OperationService;
import com.hazelcast.transaction.TransactionException;
import com.hazelcast.transaction.TransactionNotActiveException;
import com.hazelcast.transaction.TransactionOptions;
import com.hazelcast.transaction.TransactionTimedOutException;
import com.hazelcast.transaction.impl.operations.CreateTxBackupLogOperation;
import com.hazelcast.transaction.impl.operations.PurgeTxBackupLogOperation;
import com.hazelcast.transaction.impl.operations.ReplicateTxBackupLogOperation;
import com.hazelcast.transaction.impl.operations.RollbackTxBackupLogOperation;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.TimeoutException;
import java.util.logging.Level;

import static com.hazelcast.internal.util.Clock.currentTimeMillis;
import static com.hazelcast.internal.util.ExceptionUtil.rethrow;
import static com.hazelcast.internal.util.FutureUtil.RETHROW_TRANSACTION_EXCEPTION;
import static com.hazelcast.internal.util.FutureUtil.logAllExceptions;
import static com.hazelcast.internal.util.FutureUtil.waitUntilAllRespondedWithDeadline;
import static com.hazelcast.internal.util.FutureUtil.waitWithDeadline;
import static com.hazelcast.internal.util.UuidUtil.newUnsecureUUID;
import static com.hazelcast.transaction.TransactionOptions.TransactionType;
import static com.hazelcast.transaction.TransactionOptions.TransactionType.ONE_PHASE;
import static com.hazelcast.transaction.TransactionOptions.TransactionType.TWO_PHASE;
import static com.hazelcast.transaction.impl.Transaction.State.ACTIVE;
import static com.hazelcast.transaction.impl.Transaction.State.COMMITTED;
import static com.hazelcast.transaction.impl.Transaction.State.COMMITTING;
import static com.hazelcast.transaction.impl.Transaction.State.COMMIT_FAILED;
import static com.hazelcast.transaction.impl.Transaction.State.NO_TXN;
import static com.hazelcast.transaction.impl.Transaction.State.PREPARED;
import static com.hazelcast.transaction.impl.Transaction.State.PREPARING;
import static com.hazelcast.transaction.impl.Transaction.State.ROLLED_BACK;
import static com.hazelcast.transaction.impl.Transaction.State.ROLLING_BACK;
import static com.hazelcast.transaction.impl.TransactionManagerServiceImpl.SERVICE_NAME;
import static java.lang.Boolean.TRUE;
import static java.util.concurrent.TimeUnit.MILLISECONDS;

@SuppressWarnings("checkstyle:methodcount")
public class TransactionImpl implements Transaction {

    private static final Address[] EMPTY_ADDRESSES = new Address[0];
    private static final ThreadLocal TRANSACTION_EXISTS = new ThreadLocal();

    private final ExceptionHandler rollbackExceptionHandler;
    private final ExceptionHandler rollbackTxExceptionHandler;
    private final ExceptionHandler replicationTxExceptionHandler;
    private final TransactionManagerServiceImpl transactionManagerService;
    private final NodeEngine nodeEngine;
    private final UUID txnId;
    private final int durability;
    private final TransactionType transactionType;
    private final boolean checkThreadAccess;
    private final ILogger logger;
    private final UUID txOwnerUuid;
    private final TransactionLog transactionLog;
    private Long threadId;
    private long timeoutMillis;
    private State state = NO_TXN;
    private long startTime;
    private Address[] backupAddresses = EMPTY_ADDRESSES;
    private boolean backupLogsCreated;
    private boolean originatedFromClient;

    public TransactionImpl(@Nonnull TransactionManagerServiceImpl transactionManagerService,
                           @Nonnull NodeEngine nodeEngine,
                           @Nonnull TransactionOptions options,
                           @Nullable UUID txOwnerUuid) {
        this(transactionManagerService, nodeEngine, options, txOwnerUuid, false);
    }

    public TransactionImpl(@Nonnull TransactionManagerServiceImpl transactionManagerService,
                           @Nonnull NodeEngine nodeEngine,
                           @Nonnull TransactionOptions options,
                           @Nullable UUID txOwnerUuid,
                           boolean originatedFromClient) {
        this.transactionLog = new TransactionLog();
        this.transactionManagerService = transactionManagerService;
        this.nodeEngine = nodeEngine;
        this.txnId = newUnsecureUUID();
        this.timeoutMillis = options.getTimeoutMillis();
        this.transactionType = options.getTransactionType();
        this.durability = transactionType == ONE_PHASE ? 0 : options.getDurability();
        this.txOwnerUuid = txOwnerUuid == null ? nodeEngine.getLocalMember().getUuid() : txOwnerUuid;
        this.checkThreadAccess = txOwnerUuid == null;

        this.logger = nodeEngine.getLogger(getClass());
        this.rollbackExceptionHandler = logAllExceptions(logger, "Error during rollback!", Level.FINEST);
        this.rollbackTxExceptionHandler = logAllExceptions(logger, "Error during tx rollback backup!", Level.FINEST);
        this.replicationTxExceptionHandler = createReplicationTxExceptionHandler(logger);
        this.originatedFromClient = originatedFromClient;
    }

    // used by tx backups
    TransactionImpl(TransactionManagerServiceImpl transactionManagerService, NodeEngine nodeEngine,
                    UUID txnId, List transactionLog, long timeoutMillis,
                    long startTime, UUID txOwnerUuid) {
        this.transactionLog = new TransactionLog(transactionLog);
        this.transactionManagerService = transactionManagerService;
        this.nodeEngine = nodeEngine;
        this.txnId = txnId;
        this.timeoutMillis = timeoutMillis;
        this.startTime = startTime;
        this.durability = 0;
        this.transactionType = TWO_PHASE;
        this.state = PREPARED;
        this.txOwnerUuid = txOwnerUuid;
        this.checkThreadAccess = false;

        this.logger = nodeEngine.getLogger(getClass());
        this.rollbackExceptionHandler = logAllExceptions(logger, "Error during rollback!", Level.FINEST);
        this.rollbackTxExceptionHandler = logAllExceptions(logger, "Error during tx rollback backup!", Level.FINEST);
        this.replicationTxExceptionHandler = createReplicationTxExceptionHandler(logger);
    }

    @Override
    public UUID getTxnId() {
        return txnId;
    }

    public long getStartTime() {
        return startTime;
    }

    @Override
    public UUID getOwnerUuid() {
        return txOwnerUuid;
    }

    public boolean isOriginatedFromClient() {
        return originatedFromClient;
    }

    @Override
    public State getState() {
        return state;
    }

    @Override
    public long getTimeoutMillis() {
        return timeoutMillis;
    }

    protected TransactionLog getTransactionLog() {
        return transactionLog;
    }

    @Override
    public void add(TransactionLogRecord record) {
        if (state != Transaction.State.ACTIVE) {
            throw new TransactionNotActiveException("Transaction is not active!");
        }
        checkThread();
        transactionLog.add(record);
    }

    @Override
    public TransactionLogRecord get(Object key) {
        return transactionLog.get(key);
    }

    @Override
    public void remove(Object key) {
        transactionLog.remove(key);
    }

    private void checkThread() {
        if (checkThreadAccess && threadId != null && threadId.longValue() != Thread.currentThread().getId()) {
            throw new IllegalStateException("Transaction cannot span multiple threads!");
        }
    }

    @Override
    public void begin() throws IllegalStateException {
        if (state == ACTIVE) {
            throw new IllegalStateException("Transaction is already active");
        }
        if (TRANSACTION_EXISTS.get() != null) {
            throw new IllegalStateException("Nested transactions are not allowed!");
        }
        startTime = currentTimeMillis();
        backupAddresses = transactionManagerService.pickBackupLogAddresses(durability);

        //init caller thread
        if (threadId == null) {
            threadId = Thread.currentThread().getId();
            setThreadFlag(TRUE);
        }
        state = ACTIVE;
        transactionManagerService.startCount.inc();
    }

    private void setThreadFlag(Boolean flag) {
        if (checkThreadAccess) {
            TRANSACTION_EXISTS.set(flag);
        }
    }

    @Override
    public void prepare() throws TransactionException {
        if (state != ACTIVE) {
            throw new TransactionNotActiveException("Transaction is not active");
        }

        checkThread();
        checkTimeout();
        try {
            createBackupLogs();
            state = PREPARING;
            List futures = transactionLog.prepare(nodeEngine);
            waitUntilAllRespondedWithDeadline(futures, timeoutMillis, MILLISECONDS, RETHROW_TRANSACTION_EXCEPTION);
            state = PREPARED;
            replicateTxnLog();
        } catch (Throwable e) {
            throw rethrow(e, TransactionException.class);
        }
    }

    /**
     * Checks if this Transaction needs to be prepared.
     * 

* Preparing a transaction costs time since the backup log potentially needs to be copied and * each logrecord needs to prepare its content (e.g. by acquiring locks). This takes time. *

* If a transaction is local or if there is 1 or 0 items in the transaction log, instead of * preparing, we are just going to try to commit. If the lock is still acquired, the write * succeeds, and if the lock isn't acquired, the write fails; this is the same effect as a * prepare would have. * * @return true if {@link #prepare()} is required. */ public boolean requiresPrepare() { if (transactionType == ONE_PHASE) { return false; } return transactionLog.size() > 1; } @Override public void commit() throws TransactionException, IllegalStateException { try { if (transactionType == TWO_PHASE) { if (transactionLog.size() > 1) { if (state != PREPARED) { throw new IllegalStateException("Transaction is not prepared"); } } else { // when a transaction log contains less than 2 items, it can be committed without preparing if (state != PREPARED && state != ACTIVE) { throw new IllegalStateException("Transaction is not prepared or active"); } } } else if (transactionType == ONE_PHASE && state != ACTIVE) { throw new IllegalStateException("Transaction is not active"); } checkThread(); checkTimeout(); try { state = COMMITTING; List futures = transactionLog.commit(nodeEngine); waitWithDeadline(futures, Long.MAX_VALUE, MILLISECONDS, RETHROW_TRANSACTION_EXCEPTION); state = COMMITTED; transactionManagerService.commitCount.inc(); transactionLog.onCommitSuccess(); } catch (Throwable e) { state = COMMIT_FAILED; transactionLog.onCommitFailure(); throw rethrow(e, TransactionException.class); } finally { purgeBackupLogs(); } } finally { setThreadFlag(null); } } private void checkTimeout() throws TransactionTimedOutException { if (startTime + timeoutMillis < currentTimeMillis()) { throw new TransactionTimedOutException("Transaction is timed-out!"); } } @Override public void rollback() throws IllegalStateException { try { if (state == NO_TXN || state == ROLLED_BACK) { throw new IllegalStateException("Transaction is not active"); } checkThread(); state = ROLLING_BACK; try { //TODO: Do we need both a purge and rollback? rollbackBackupLogs(); List futures = transactionLog.rollback(nodeEngine); waitWithDeadline(futures, Long.MAX_VALUE, MILLISECONDS, rollbackExceptionHandler); purgeBackupLogs(); } catch (Throwable e) { throw rethrow(e); } finally { state = ROLLED_BACK; transactionManagerService.rollbackCount.inc(); } } finally { setThreadFlag(null); } } private void replicateTxnLog() { if (skipBackupLogReplication()) { return; } OperationService operationService = nodeEngine.getOperationService(); ClusterService clusterService = nodeEngine.getClusterService(); List futures = new ArrayList(backupAddresses.length); for (Address backupAddress : backupAddresses) { if (clusterService.getMember(backupAddress) != null) { Operation op = createReplicateTxBackupLogOperation(); Future f = operationService.invokeOnTarget(SERVICE_NAME, op, backupAddress); futures.add(f); } } waitWithDeadline(futures, timeoutMillis, MILLISECONDS, replicationTxExceptionHandler); } /** * Some data-structures like the Transaction List rely on (empty) backup logs to be created before any change on the * data-structure is made. That is why when such a data-structure is loaded, it should the creation. *

* Not every data-structure, e.g. the Transactional Map, relies on it and in some cases can even skip it. */ public void ensureBackupLogsExist() { // we can't take the TransactionLog size in consideration because this call can be made before an item // is added to the transactionLog. if (backupLogsCreated || backupAddresses.length == 0) { return; } forceCreateBackupLogs(); } private void createBackupLogs() { if (backupLogsCreated || skipBackupLogReplication()) { return; } forceCreateBackupLogs(); } private void forceCreateBackupLogs() { backupLogsCreated = true; OperationService operationService = nodeEngine.getOperationService(); List futures = new ArrayList(backupAddresses.length); for (Address backupAddress : backupAddresses) { if (nodeEngine.getClusterService().getMember(backupAddress) != null) { final CreateTxBackupLogOperation op = createCreateTxBackupLogOperation(); Future f = operationService.invokeOnTarget(SERVICE_NAME, op, backupAddress); futures.add(f); } } waitWithDeadline(futures, timeoutMillis, MILLISECONDS, replicationTxExceptionHandler); } private void rollbackBackupLogs() { if (!backupLogsCreated) { return; } OperationService operationService = nodeEngine.getOperationService(); ClusterService clusterService = nodeEngine.getClusterService(); List futures = new ArrayList(backupAddresses.length); for (Address backupAddress : backupAddresses) { if (clusterService.getMember(backupAddress) != null) { Future f = operationService.invokeOnTarget(SERVICE_NAME, createRollbackTxBackupLogOperation(), backupAddress); futures.add(f); } } waitWithDeadline(futures, timeoutMillis, MILLISECONDS, rollbackTxExceptionHandler); } private void purgeBackupLogs() { if (!backupLogsCreated) { return; } OperationService operationService = nodeEngine.getOperationService(); ClusterService clusterService = nodeEngine.getClusterService(); for (Address backupAddress : backupAddresses) { if (clusterService.getMember(backupAddress) != null) { try { operationService.invokeOnTarget(SERVICE_NAME, createPurgeTxBackupLogOperation(), backupAddress); } catch (Throwable e) { logger.warning("Error during purging backups!", e); } } } } private boolean skipBackupLogReplication() { //todo: what if backupLogsCreated are created? return durability == 0 || transactionLog.size() <= 1 || backupAddresses.length == 0; } protected CreateTxBackupLogOperation createCreateTxBackupLogOperation() { return new CreateTxBackupLogOperation(txOwnerUuid, txnId); } protected ReplicateTxBackupLogOperation createReplicateTxBackupLogOperation() { return new ReplicateTxBackupLogOperation( transactionLog.getRecords(), txOwnerUuid, txnId, timeoutMillis, startTime); } protected RollbackTxBackupLogOperation createRollbackTxBackupLogOperation() { return new RollbackTxBackupLogOperation(txnId); } protected PurgeTxBackupLogOperation createPurgeTxBackupLogOperation() { return new PurgeTxBackupLogOperation(txnId); } @Override public TransactionType getTransactionType() { return transactionType; } @Override public String toString() { return "Transaction{" + "txnId='" + txnId + '\'' + ", state=" + state + ", txType=" + transactionType + ", timeoutMillis=" + timeoutMillis + '}'; } static ExceptionHandler createReplicationTxExceptionHandler(final ILogger logger) { return new ExceptionHandler() { @Override public void handleException(Throwable throwable) { if (throwable instanceof TimeoutException) { throw new TransactionTimedOutException(throwable); } if (throwable instanceof MemberLeftException) { logger.warning("Member left while replicating tx begin: " + throwable); return; } if (throwable instanceof ExecutionException) { Throwable cause = throwable.getCause(); if (cause instanceof TargetNotMemberException || cause instanceof HazelcastInstanceNotActiveException) { logger.warning("Member left while replicating tx begin: " + cause); return; } } throw rethrow(throwable); } }; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy