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

com.landawn.abacus.jdbc.SQLTransaction Maven / Gradle / Ivy

/*
 * Copyright (C) 2015 HaiYang Li
 *
 * 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.landawn.abacus.jdbc;

import java.sql.Connection;
import java.sql.SQLException;
import java.util.Map;
import java.util.Stack;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;

import com.landawn.abacus.exception.UncheckedSQLException;
import com.landawn.abacus.logging.Logger;
import com.landawn.abacus.logging.LoggerFactory;
import com.landawn.abacus.util.Fn;
import com.landawn.abacus.util.N;
import com.landawn.abacus.util.Strings;
import com.landawn.abacus.util.Throwables;

// TODO: Auto-generated Javadoc
/**
 * DO NOT CLOSE the connection manually. It will be automatically closed after the transaction is committed or rolled back.
 *
 */
@SuppressWarnings("resource")
public final class SQLTransaction implements Transaction, AutoCloseable {

    private static final Logger logger = LoggerFactory.getLogger(SQLTransaction.class);

    private static final Map threadTransactionMap = new ConcurrentHashMap<>();
    // private static final Map attachedThreadTransactionMap = new ConcurrentHashMap<>();

    private final String _id; //NOSONAR

    private final String _timedId; //NOSONAR

    private final javax.sql.DataSource _ds; //NOSONAR

    private final Connection _conn; //NOSONAR

    private final boolean _closeConnection; //NOSONAR

    private final boolean _originalAutoCommit; //NOSONAR

    private final int _originalIsolationLevel; //NOSONAR

    private Transaction.Status _status = Status.ACTIVE; //NOSONAR

    private final AtomicInteger _refCount = new AtomicInteger(); //NOSONAR

    private final Stack _isolationLevelStack = new Stack<>(); //NOSONAR

    private final Stack _isForUpdateOnlyStack = new Stack<>(); //NOSONAR

    private IsolationLevel _isolationLevel; //NOSONAR

    private boolean _isForUpdateOnly; //NOSONAR

    private boolean _isMarkedByCommitPreviously = false; //NOSONAR

    SQLTransaction(final javax.sql.DataSource ds, final Connection conn, final IsolationLevel isolationLevel, final CreatedBy creator,
            final boolean closeConnection) throws SQLException {
        N.checkArgNotNull(conn);
        N.checkArgNotNull(isolationLevel);

        _id = getTransactionId(ds, creator);
        _timedId = _id + "_" + System.currentTimeMillis();
        _ds = ds;
        _conn = conn;
        _isolationLevel = isolationLevel;
        _closeConnection = closeConnection;

        _originalAutoCommit = conn.getAutoCommit();
        _originalIsolationLevel = conn.getTransactionIsolation();

        conn.setAutoCommit(false);

        if (isolationLevel != IsolationLevel.DEFAULT) {
            conn.setTransactionIsolation(isolationLevel.intValue());
        }
    }

    /**
     * Returns the unique identifier of the transaction.
     *
     * @return the unique identifier of the transaction.
     */
    @Override
    public String id() {
        return _timedId;
    }

    /**
     * Returns the connection associated with this transaction.
     *
     * @return the connection associated with this transaction.
     */
    public Connection connection() {
        return _conn;
    }

    /**
     * Returns the isolation level of the transaction.
     *
     * @return the isolation level of the transaction.
     */
    @Override
    public IsolationLevel isolationLevel() {
        return _isolationLevel;
    }

    /**
     * Returns the current status of the transaction.
     *
     * @return the current status of the transaction.
     */
    @Override
    public Transaction.Status status() {
        return _status;
    }

    /**
     * Checks if the transaction is active.
     *
     * @return {@code true} if the transaction is active, {@code false} otherwise.
     */
    @Override
    public boolean isActive() {
        return _status == Status.ACTIVE;
    }

    //    /**
    //     * Attaches this transaction to current thread.
    //     *
    //     */
    //    public void attach() {
    //        final String currentThreadName = Thread.currentThread().getName();
    //        final String resourceId = ttid.substring(ttid.lastIndexOf('_') + 1);
    //        final String targetTTID = currentThreadName + "_" + resourceId;
    //
    //        if (attachedThreadTransactionMap.containsKey(targetTTID)) {
    //            throw new IllegalStateException("Transaction(id=" + attachedThreadTransactionMap.get(targetTTID).id()
    //                    + ") has already been attached to current thread: " + currentThreadName);
    //        } else if (threadTransactionMap.containsKey(targetTTID)) {
    //            throw new IllegalStateException(
    //                    "Transaction(id=" + threadTransactionMap.get(targetTTID).id() + ") has already been created in current thread: " + currentThreadName);
    //        }
    //
    //        attachedThreadTransactionMap.put(targetTTID, this);
    //        threadTransactionMap.put(targetTTID, this);
    //    }
    //
    //    public void detach() {
    //        final String currentThreadName = Thread.currentThread().getName();
    //        final String resourceId = ttid.substring(ttid.lastIndexOf('_') + 1);
    //        final String targetTTID = currentThreadName + "_" + resourceId;
    //
    //        if (!attachedThreadTransactionMap.containsKey(targetTTID)) {
    //            throw new IllegalStateException(
    //                    "Transaction(id=" + attachedThreadTransactionMap.get(targetTTID).id() + ") is not attached to current thread: " + currentThreadName);
    //        }
    //
    //        threadTransactionMap.remove(targetTTID);
    //        attachedThreadTransactionMap.remove(targetTTID);
    //    }

    /**
     * Commits the current transaction.
     *
     * @throws UncheckedSQLException if an SQL error occurs during the commit.
     */
    @Override
    public void commit() throws UncheckedSQLException {
        commit(Fn.emptyAction());
    }

    /**
     * Commits the current transaction and executes the specified action after the commit.
     *
     * @param actionAfterCommit the action to be executed after the current transaction is committed successfully.
     * @throws UncheckedSQLException if an SQL error occurs during the commit.
     */
    void commit(final Runnable actionAfterCommit) throws UncheckedSQLException {
        final int refCount = decrementAndGetRef();
        _isMarkedByCommitPreviously = true;

        if (refCount > 0) {
            return;
        } else if (refCount < 0) {
            logger.warn("Transaction(id={}) is already: {}. This committing is ignored", _timedId, _status);
            return;
        }

        if (_status == Status.MARKED_ROLLBACK) {
            logger.warn("Transaction(id={}) will be rolled back because it's marked for roll back only", _timedId);
            executeRollback();
            return;
        }

        if (_status != Status.ACTIVE) {
            throw new IllegalArgumentException("Transaction(id=" + _timedId + ") is already: " + _status + ". It can not be committed"); //NOSONAR
        }

        logger.info("Committing transaction(id={})", _timedId);

        _status = Status.FAILED_COMMIT;

        try {
            if (_originalAutoCommit) {
                _conn.commit();
            }

            _status = Status.COMMITTED;
        } catch (final SQLException e) {
            throw new UncheckedSQLException("Failed to commit transaction(id=" + _id + ")", e);
        } finally {
            if (_status == Status.COMMITTED) {
                logger.info("Transaction(id={}) has been committed successfully", _timedId);

                resetAndCloseConnection();

                actionAfterCommit.run();
            } else {
                logger.warn("Failed to commit transaction(id={}). It will automatically be rolled back ", _timedId);
                executeRollback();
            }
        }
    }

    /**
     * Rolls back the current transaction.
     *
     * 
     * 
     *   final SQLTransaction tran = JdbcUtil.beginTransaction(IsolationLevel.READ_COMMITTED);
     *   try {
     *       // sqlExecutor.insert(...);
     *       // sqlExecutor.update(...);
     *       // sqlExecutor.query(...);
     *
     *       tran.commit();
     *   } finally {
     *       // The connection will be automatically closed after the transaction is committed or rolled back.
     *       tran.rollbackIfNotCommitted();
     *   }
     * 
     * 
* * @throws UncheckedSQLException the unchecked SQL exception * @deprecated replaced by {@code #rollbackIfNotCommitted()} */ @Deprecated @Override public void rollback() throws UncheckedSQLException { rollback(Fn.emptyAction()); } /** * Rolls back the current transaction and executes the specified action after the rollback. * * @param actionAfterRollback the action to be executed after the current transaction is rolled back, not successfully or not. * @throws UncheckedSQLException if an SQL error occurs during the rollback. */ void rollback(final Runnable actionAfterRollback) throws UncheckedSQLException { final int refCount = decrementAndGetRef(); _isMarkedByCommitPreviously = true; if (refCount > 0) { _status = Status.MARKED_ROLLBACK; return; } else if (refCount < 0) { logger.warn("Transaction(id={}) is already: {}. This rollback is ignored", _timedId, _status); return; } if (!(_status == Status.ACTIVE || _status == Status.MARKED_ROLLBACK || _status == Status.FAILED_COMMIT)) { throw new IllegalStateException("Transaction(id=" + _timedId + ") is already: " + _status); } executeRollback(actionAfterRollback); } /** * Rolls back the transaction if it has not been committed successfully. * * @throws UncheckedSQLException if an SQL error occurs during the rollback. */ @Override public void rollbackIfNotCommitted() throws UncheckedSQLException { if (_isMarkedByCommitPreviously) { // Do nothing. It happened in finally block. _isMarkedByCommitPreviously = false; return; } final int refCount = decrementAndGetRef(); if (refCount > 0) { _status = Status.MARKED_ROLLBACK; return; } else if (refCount < 0) { if (refCount == -1 && (_status == Status.COMMITTED || _status == Status.FAILED_COMMIT || _status == Status.ROLLED_BACK || _status == Status.FAILED_ROLLBACK)) { // Do nothing. It happened in finally block. } else { logger.warn("Transaction(id={}) is already: {}. This rollback is ignored", _timedId, _status); } return; } if (!(_status == Status.ACTIVE || _status == Status.MARKED_ROLLBACK || _status == Status.FAILED_COMMIT || _status == Status.FAILED_ROLLBACK)) { throw new IllegalArgumentException("Transaction(id=" + _timedId + ") is already: " + _status + ". It can not be rolled back"); } executeRollback(); } private void executeRollback() throws UncheckedSQLException { executeRollback(Fn.emptyAction()); } /** * * @throws UncheckedSQLException the unchecked SQL exception */ private void executeRollback(final Runnable actionAfterRollback) throws UncheckedSQLException { logger.warn("Rolling back transaction(id={})", _timedId); _status = Status.FAILED_ROLLBACK; try { if (_originalAutoCommit) { _conn.rollback(); } _status = Status.ROLLED_BACK; } catch (final SQLException e) { throw new UncheckedSQLException(e); } finally { if (_status == Status.ROLLED_BACK) { logger.warn("Transaction(id={}) has been rolled back successfully", _timedId); } else { logger.warn("Failed to roll back transaction(id={})", _timedId); } resetAndCloseConnection(); actionAfterRollback.run(); } } /** * Reset and close connection. */ private void resetAndCloseConnection() { try { _conn.setAutoCommit(_originalAutoCommit); _conn.setTransactionIsolation(_originalIsolationLevel); } catch (final SQLException e) { logger.warn("Failed to reset connection", e); } finally { if (_closeConnection) { JdbcUtil.releaseConnection(_conn, _ds); } } } /** * Increment and get ref. * * @param isolationLevel * @param forUpdateOnly * @return */ synchronized int incrementAndGetRef(final IsolationLevel isolationLevel, final boolean forUpdateOnly) { if (_status != Status.ACTIVE) { throw new IllegalStateException("Transaction(id=" + _timedId + ") is already: " + _status); } _isMarkedByCommitPreviously = false; if (_conn != null) { try { if (isolationLevel == IsolationLevel.DEFAULT) { // conn.setTransactionIsolation(originalIsolationLevel); } else { _conn.setTransactionIsolation(isolationLevel.intValue()); } } catch (final SQLException e) { throw new UncheckedSQLException(e); } } if (_refCount.get() > 0) { _isolationLevelStack.push(_isolationLevel); _isForUpdateOnlyStack.push(_isForUpdateOnly); } _isolationLevel = isolationLevel; _isForUpdateOnly = forUpdateOnly; return _refCount.incrementAndGet(); } /** * Decrement and get ref. * * @return * @throws UncheckedSQLException the unchecked SQL exception */ synchronized int decrementAndGetRef() throws UncheckedSQLException { final int res = _refCount.decrementAndGet(); if (res == 0) { threadTransactionMap.remove(_id); logger.info("Finishing transaction(id={})", _timedId); logger.debug("Remaining active transactions: {}", threadTransactionMap.values()); } else if (res > 0) { _isolationLevel = _isolationLevelStack.pop(); _isForUpdateOnly = _isForUpdateOnlyStack.pop(); if (_conn != null) { try { if (_isolationLevel == IsolationLevel.DEFAULT) { _conn.setTransactionIsolation(_originalIsolationLevel); } else { _conn.setTransactionIsolation(_isolationLevel.intValue()); } } catch (final SQLException e) { throw new UncheckedSQLException(e); } } } return res; } /** * Checks if is for update only. * * @return {@code true}, if is for update only */ boolean isForUpdateOnly() { return _isForUpdateOnly; } /** * * @param ds * @param creator * @return */ static String getTransactionId(final javax.sql.DataSource ds, final CreatedBy creator) { return Strings.concat(System.identityHashCode(ds), "_", Thread.currentThread().getId(), "_", Thread.currentThread().getName(), "_", creator.ordinal()); } /** * * @param ds * @param creator * @return */ static SQLTransaction getTransaction(final javax.sql.DataSource ds, final CreatedBy creator) { return threadTransactionMap.get(getTransactionId(ds, creator)); } /** * * @param tran * @return */ static SQLTransaction putTransaction(final SQLTransaction tran) { return threadTransactionMap.put(tran._id, tran); } /** * Executes the specified {@code Runnable} outside of this transaction. * * @param the type of exception that the {@code Runnable} might throw. * @param cmd the {@code Runnable} to be executed outside of this transaction. * @throws E if the {@code Runnable} throws an exception. */ public void runNotInMe(final Throwables.Runnable cmd) throws E { synchronized (_id) { //NOSONAR threadTransactionMap.remove(_id); try { cmd.run(); } finally { if (threadTransactionMap.put(_id, this) != null) { throw new IllegalStateException("Another transaction is opened but not closed in 'Transaction.runNotInMe'."); //NOSONAR } } } } /** * Executes the specified {@code Callable} outside of this transaction. * * @param the type of the result returned by the {@code Callable}. * @param the type of exception that the {@code Callable} might throw. * @param cmd the {@code Callable} to be executed outside of this transaction. * @return the result returned by the {@code Callable}. * @throws E if the {@code Callable} throws an exception. */ public R callNotInMe(final Throwables.Callable cmd) throws E { synchronized (_id) { //NOSONAR threadTransactionMap.remove(_id); try { return cmd.call(); } finally { if (threadTransactionMap.put(_id, this) != null) { throw new IllegalStateException("Another transaction is opened but not closed in 'Transaction.callNotInMe'."); //NOSONAR } } } } /** * {@code rollbackIfNotCommitted} is called. * * @see #rollbackIfNotCommitted() */ @Override public void close() { rollbackIfNotCommitted(); } /** * * * @return */ @Override public int hashCode() { return _timedId.hashCode(); } /** * * @param obj * @return {@code true}, if successful */ @Override public boolean equals(final Object obj) { return obj instanceof SQLTransaction && _timedId.equals(((SQLTransaction) obj)._timedId); } /** * * * @return */ @Override public String toString() { return "SQLTransaction={id=" + _timedId + "}"; } /** * The Enum CreatedBy. */ enum CreatedBy { /** * Global for all. */ JDBC_UTIL, /** * SQLExecutor. * @deprecated not used */ SQL_EXECUTOR } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy