com.github.davidmoten.rx.jdbc.QueryUpdateOperation Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of rxjava-jdbc Show documentation
Show all versions of rxjava-jdbc Show documentation
rx-java Observables for jdbc
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 super Integer> 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 super Integer> 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 super Integer> 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 super Integer> 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 super Integer> 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 super Integer> 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 super Integer> 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 super Integer> subscriber, State state) {
if (subscriber.isUnsubscribed()) {
state.keepGoing = false;
log.debug("unsubscribing");
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy