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

com.github.davidmoten.rx.jdbc.QueryUpdateOperation Maven / Gradle / Ivy

There is a newer version: 0.7.19
Show newest version
package com.github.davidmoten.rx.jdbc;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.util.List;
import java.util.concurrent.atomic.AtomicBoolean;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import rx.Observable;
import rx.Observable.OnSubscribe;
import rx.Subscriber;

/**
 * Executes the update query.
 */
class QueryUpdateOperation {

    private static final Logger log = LoggerFactory.getLogger(QueryUpdateOperation.class);

    static final String BEGIN_TRANSACTION = "begin";

    /**
     * Special sql command that brings about a rollback.
     */
    static final String ROLLBACK = "rollback";

    /**
     * Special sql command that brings about a commit.
     */
    static final String COMMIT = "commit";

    /**
     * Returns an Observable of the results of pushing one set of parameters
     * through a select query.
     * 
     * @param params
     *            one set of parameters to be run with the query
     * @return
     */
    static Observable execute(QueryUpdate query, List parameters) {
        return Observable.create(new QueryUpdateOnSubscribe(query, parameters));
    }

    private static class QueryUpdateOnSubscribe implements OnSubscribe {

        /**
         * The query to be executed.
         */
        private final QueryUpdate query;

        /**
         * The parameters to run the query against (may be a subset of the query
         * parameters specified in the query because the query may be run
         * multiple times with multiple sets of parameters).
         */
        private final List parameters;

        private static class State {
            volatile boolean keepGoing = true;
            volatile Connection con;
            volatile PreparedStatement ps;
            final AtomicBoolean closed = new AtomicBoolean(false);
        }

        /**
         * Constructor.
         * 
         * @param query
         * @param parameters
         */
        private QueryUpdateOnSubscribe(QueryUpdate query, List parameters) {
            this.query = query;
            this.parameters = parameters;
        }

        @Override
        public void call(Subscriber subscriber) {
            final State state = new State();
            try {

                if (isBeginTransaction())
                    performBeginTransaction(subscriber);
                else {
                    getConnection(state);
                    if (isCommit())
                        performCommit(subscriber, state);
                    else if (isRollback())
                        performRollback(subscriber, state);
                    else
                        performUpdate(subscriber, state);
                    close(state);
                }

                complete(subscriber);

            } catch (Exception e) {
                try {
                    close(state);
                } finally {
                    handleException(e, subscriber);
                }
            }
        }

        private boolean isBeginTransaction() {
            return query.sql().equals(BEGIN_TRANSACTION);
        }

        private void performBeginTransaction(Subscriber subscriber) {
            query.context().beginTransactionObserve();
            log.debug("beginTransaction emitting 1");
            subscriber.onNext(Integer.valueOf(1));
            log.debug("emitted 1");
        }

        /**
         * Gets the current connection.
         */
        private void getConnection(State state) {
            log.debug("getting connection");
            state.con = query.context().connectionProvider().get();
            log.debug("cp={}", query.context().connectionProvider());
        }

        /**
         * Returns true if and only if the sql statement is a commit command.
         * 
         * @return if is commit
         */
        private boolean isCommit() {
            return query.sql().equals(COMMIT);
        }

        /**
         * Returns true if and only if the sql statement is a rollback command.
         * 
         * @return if is rollback
         */
        private boolean isRollback() {
            return query.sql().equals(ROLLBACK);
        }

        /**
         * Commits the current transaction. Throws {@link RuntimeException} if
         * connection is in autoCommit mode.
         * 
         * @param subscriber
         * @param state
         */
        private void performCommit(Subscriber subscriber, State state) {
            query.context().endTransactionObserve();
            checkSubscription(subscriber, state);
            if (!state.keepGoing)
                return;

            log.debug("committing");
            Conditions.checkTrue(!Util.isAutoCommit(state.con));
            Util.commit(state.con);
            // must close before onNext so that connection is released and is
            // available to a query that might process the onNext
            close(state);

            checkSubscription(subscriber, state);
            if (!state.keepGoing)
                return;

            subscriber.onNext(Integer.valueOf(1));
            log.debug("committed");
        }

        /**
         * Rolls back the current transaction. Throws {@link RuntimeException}
         * if connection is in autoCommit mode.
         * 
         * @param subscriber
         * @param state
         */
        private void performRollback(Subscriber subscriber, State state) {
            log.debug("rolling back");
            query.context().endTransactionObserve();
            Conditions.checkTrue(!Util.isAutoCommit(state.con));
            Util.rollback(state.con);
            // must close before onNext so that connection is released and is
            // available to a query that might process the onNext
            close(state);
            subscriber.onNext(Integer.valueOf(0));
            log.debug("rolled back");
        }

        /**
         * Executes the prepared statement.
         * 
         * @param subscriber
         * 
         * @throws SQLException
         */
        private void performUpdate(Subscriber subscriber, State state)
                throws SQLException {
            checkSubscription(subscriber, state);
            if (!state.keepGoing)
                return;

            state.ps = state.con.prepareStatement(query.sql());
            Util.setParameters(state.ps, parameters);

            checkSubscription(subscriber, state);
            if (!state.keepGoing)
                return;

            int count;
            try {
                log.debug("executing sql={}, parameters {}", query.sql(), parameters);
                count = state.ps.executeUpdate();
                log.debug("executed ps={}", state.ps);
            } catch (SQLException e) {
                throw new SQLException("failed to execute sql=" + query.sql(), e);
            }
            // must close before onNext so that connection is released and is
            // available to a query that might process the onNext
            close(state);
            checkSubscription(subscriber, state);
            if (!state.keepGoing)
                return;
            log.debug("onNext");
            subscriber.onNext(count);
        }

        /**
         * Notify observer that sequence is complete.
         * 
         * @param subscriber
         * @param state
         */
        private void complete(Subscriber subscriber) {
            if (!subscriber.isUnsubscribed()) {
                log.debug("onCompleted");
                subscriber.onCompleted();
            } else
                log.debug("unsubscribed");
        }

        /**
         * Notify observer of an error.
         * 
         * @param e
         * @param subscriber
         */
        private void handleException(Exception e, Subscriber subscriber) {
            log.debug("onError: ", e.getMessage());
            if (subscriber.isUnsubscribed())
                log.debug("unsubscribed");
            else {
                subscriber.onError(e);
            }
        }

        /**
         * Cancels a running PreparedStatement, closing it and the current
         * Connection but only if auto commit mode.
         */
        private void close(State state) {
            // ensure close happens once only to avoid race conditions
            if (state.closed.compareAndSet(false, true)) {
                Util.closeQuietly(state.ps);
                if (isCommit() || isRollback())
                    Util.closeQuietly(state.con);
                else
                    Util.closeQuietlyIfAutoCommit(state.con);
            }
        }

        private void checkSubscription(Subscriber subscriber, State state) {
            if (subscriber.isUnsubscribed()) {
                state.keepGoing = false;
                log.debug("unsubscribing");
            }
        }

    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy