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

com.jcabi.jdbc.JdbcSession Maven / Gradle / Ivy

There is a newer version: 0.19.0
Show newest version
/**
 * Copyright (c) 2012-2014, JCabi.com
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met: 1) Redistributions of source code must retain the above
 * copyright notice, this list of conditions and the following
 * disclaimer. 2) Redistributions in binary form must reproduce the above
 * copyright notice, this list of conditions and the following
 * disclaimer in the documentation and/or other materials provided
 * with the distribution. 3) Neither the name of the jcabi.com nor
 * the names of its contributors may be used to endorse or promote
 * products derived from this software without specific prior written
 * permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT
 * NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
 * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
 * THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
 * OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package com.jcabi.jdbc;

import com.jcabi.aspects.Loggable;
import java.sql.Connection;
import java.sql.Date;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Collection;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.atomic.AtomicReference;
import javax.sql.DataSource;
import javax.validation.constraints.NotNull;
import lombok.EqualsAndHashCode;
import lombok.ToString;

/**
 * Universal JDBC wrapper.
 *
 * 

Execute a simple SQL query over a JDBC data source: * *

 String name = new JdbcSession(source)
 *   .sql("SELECT name FROM foo WHERE id = ?")
 *   .set(123)
 *   .select(
 *     new Outcome<String>() {
 *       @Override
 *       public String handle(final ResultSet rset) throws SQLException {
 *         rset.next();
 *         return rset.getString(1);
 *       }
 *     }
 *   );
* *

There are a number of convenient pre-defined outcomes, like * {@link Outcome#VOID}, {@link Outcome#NOT_EMPTY}, {@link Outcome#UPDATE_COUNT} * {@link SingleOutcome}, etc. * *

Methods {@link #insert(Outcome)}, * {@link #update(Outcome)}, * {@link #execute()}, and * {@link #select(Outcome)} clean the list of arguments pre-set by * {@link #set(Object)}. The class can be used for a complex transaction, when * it's necessary to perform a number of SQL statements in a group. For * example, the following construct will execute two SQL queries, in a single * transaction and will "commit" at the end (or rollback the entire transaction * in case of any error in between): * *

 new JdbcSession(source)
 *   .autocommit(false)
 *   .sql("START TRANSACTION")
 *   .execute()
 *   .sql("DELETE FROM foo WHERE id = ?")
 *   .set(444)
 *   .execute()
 *   .set(555)
 *   .execute()
 *   .commit();
* *

The following SQL queries will be sent to the database: * *

 START TRANSACTION;
 * DELETE FROM foo WHERE id = 444;
 * DELETE FROM foo WHERE id = 555;
 * COMMIT;
* *

{@link #autocommit(boolean)} (with {@code false} as an argument) * can be used when it's necessary to execute * a statement and leave the connection open. For example when shutting down * the database through SQL: * *

 new JdbcSession(/* H2 Database data source */)
 *   .autocommit(false)
 *   .sql("SHUTDOWN COMPACT")
 *   .execute();
* *

This class is thread-safe. * * @author Yegor Bugayenko ([email protected]) * @version $Id$ * @since 0.1.8 */ @ToString @EqualsAndHashCode(of = { "source", "connection", "args", "auto", "query" }) @Loggable(Loggable.DEBUG) @SuppressWarnings({ "PMD.TooManyMethods", "PMD.CloseResource" }) public final class JdbcSession { /** * JDBC DataSource to get connections from. */ private final transient DataSource source; /** * Arguments. */ private final transient Collection args = new CopyOnWriteArrayList(); /** * Connection currently open. */ private final transient AtomicReference connection = new AtomicReference(); /** * Shall we close/autocommit automatically? */ private transient boolean auto = true; /** * The query to use. */ private transient String query; /** * Public ctor. * @param src Data source */ public JdbcSession( @NotNull(message = "data source can't be NULL") final DataSource src) { this.source = src; } /** * Public ctor. * @param cnx Connection to use * @since 0.10 */ public JdbcSession( @NotNull(message = "connection can't be NULL") final Connection cnx) { this.source = new StaticSource(cnx); } /** * Use this SQL query (with optional parameters inside). * *

The query will be used in {@link PreparedStatement}, that's why * you can use the same formatting as there. Arguments shall be marked * as {@code "?"} (question marks). For example: * *

 String name = new JdbcSession(source)
     *   .sql("INSERT INTO foo (id, name) VALUES (?, ?)")
     *   .set(556677)
     *   .set("Jeffrey Lebowski")
     *   .insert(Outcome.VOID);
* * @param sql The SQL query to use * @return This object */ public JdbcSession sql( @NotNull(message = "SQL query can't be NULL") final String sql) { synchronized (this.args) { this.query = sql; } return this; } /** * Shall we auto-commit? * *

By default this flag is set to TRUE, which means that methods * {@link #insert(Outcome)}, {@link #execute()}, and * {@link #select(Outcome)} will * call {@link Connection#commit()} after * their successful execution. * * @param autocommit Shall we? * @return This object */ public JdbcSession autocommit(final boolean autocommit) { synchronized (this.args) { this.auto = autocommit; } return this; } /** * Set new parameter for the query. * *

The following types are supported: {@link Boolean}, {@link Date}, * {@link Utc}, {@link Long}, {@link Integer}. All other types will be * converted to {@link String} using their {@code toString()} methods. * * @param value The value to add * @return This object */ public JdbcSession set(final Object value) { this.args.add(value); return this; } /** * Commit the transaction (calls {@link Connection#commit()} and then * {@link Connection#close()}). * @throws SQLException If fails to do the SQL operation */ public void commit() throws SQLException { final Connection conn = this.connection.get(); if (conn == null) { throw new IllegalStateException( "connection is not open, can't commit" ); } conn.commit(); this.disconnect(); } /** * Make SQL {@code INSERT} request. * *

{@link Outcome} will receive * a {@link ResultSet} of generated keys. * *

JDBC connection is opened and, optionally, closed by this method. * * @param outcome The outcome of the operation * @return The result * @param Type of response * @throws SQLException If fails */ public T insert( @NotNull(message = "outcome can't be NULL") final Outcome outcome) throws SQLException { return this.run( outcome, new JdbcSession.Fetcher() { @Override public ResultSet fetch(final PreparedStatement stmt) throws SQLException { stmt.execute(); return stmt.getGeneratedKeys(); } @Override public PreparedStatement statement(final Connection conn) throws SQLException { return conn.prepareStatement( JdbcSession.this.query, Statement.RETURN_GENERATED_KEYS ); } } ); } /** * Make SQL {@code UPDATE} request. * *

JDBC connection is opened and, optionally, closed by this method. * * @param Type of result expected * @param outcome Outcome of the operation * @return This object * @throws SQLException If fails */ public T update( @NotNull(message = "outcome can't be NULL") final Outcome outcome) throws SQLException { return this.run( outcome, new JdbcSession.Fetcher() { @Override public ResultSet fetch(final PreparedStatement stmt) throws SQLException { stmt.executeUpdate(); return null; } @Override public PreparedStatement statement(final Connection conn) throws SQLException { return conn.prepareStatement( JdbcSession.this.query, Statement.RETURN_GENERATED_KEYS ); } } ); } /** * Make SQL request expecting no response from the server. * *

This method should be used for schema manipulation statements, * like CREATE TABLE, CREATE INDEX, DROP COLUMN, etc. and server-side * instructions that return no data back. Main difference between this * one and {@code #execute()} is that the later requests JDBC to return * generated keys. When SQL server doesn't return any keys this may * cause runtime exceptions in JDBC. * *

JDBC connection is opened and, optionally, closed by this method. * * @return This object * @throws SQLException If fails * @since 0.9 */ public JdbcSession execute() throws SQLException { this.run( Outcome.VOID, new JdbcSession.Fetcher() { @Override public ResultSet fetch(final PreparedStatement stmt) throws SQLException { stmt.execute(); return null; } @Override public PreparedStatement statement(final Connection conn) throws SQLException { return conn.prepareStatement(JdbcSession.this.query); } } ); return this; } /** * Make SQL {@code SELECT} request. * *

JDBC connection is opened and, optionally, closed by this method. * * @param outcome The outcome of the operaton * @return The result * @param Type of response * @throws SQLException If fails */ public T select( @NotNull(message = "outcome can't be NULL") final Outcome outcome) throws SQLException { return this.run( outcome, new JdbcSession.Fetcher() { @Override public ResultSet fetch(final PreparedStatement stmt) throws SQLException { return stmt.executeQuery(); } @Override public PreparedStatement statement(final Connection conn) throws SQLException { return conn.prepareStatement(JdbcSession.this.query); } } ); } /** * Run with this outcome, and this fetcher. * @param outcome The outcome of the operation * @param fetcher Fetcher of result set * @return The result * @param Type of response * @throws SQLException If fails * @checkstyle ExecutableStatementCount (100 lines) */ private T run(final Outcome outcome, final JdbcSession.Fetcher fetcher) throws SQLException { if (this.query == null) { throw new IllegalStateException("call #sql() first"); } final Connection conn = this.connect(); T result; try { conn.setAutoCommit(this.auto); final PreparedStatement stmt = fetcher.statement(conn); try { this.parametrize(stmt); final ResultSet rset = fetcher.fetch(stmt); // @checkstyle NestedTryDepth (5 lines) try { result = outcome.handle(rset, stmt); } finally { if (rset != null) { rset.close(); } } } finally { stmt.close(); } } catch (final SQLException ex) { if (!this.auto) { conn.rollback(); this.disconnect(); } throw new SQLException(ex); } finally { if (this.auto) { this.disconnect(); } this.args.clear(); } return result; } /** * Open connection and cache it locally in the class. * @return Connection to use * @throws SQLException If fails */ private Connection connect() throws SQLException { synchronized (this.args) { if (this.connection.get() == null) { this.connection.set(this.source.getConnection()); } return this.connection.get(); } } /** * Close connection if it's open (runtime exception otherwise). * @throws SQLException If fails to do the SQL operation */ private void disconnect() throws SQLException { final Connection conn = this.connection.getAndSet(null); if (conn == null) { throw new IllegalStateException( "connection is not open, can't close" ); } conn.close(); } /** * Add params to the statement. * @param stmt The statement to parametrize * @throws SQLException If some problem */ private void parametrize(final PreparedStatement stmt) throws SQLException { int pos = 1; for (final Object arg : this.args) { if (arg == null) { stmt.setString(pos, null); } else if (arg instanceof Long) { stmt.setLong(pos, Long.class.cast(arg)); } else if (arg instanceof Boolean) { stmt.setBoolean(pos, Boolean.class.cast(arg)); } else if (arg instanceof Date) { stmt.setDate(pos, Date.class.cast(arg)); } else if (arg instanceof Integer) { stmt.setInt(pos, Integer.class.cast(arg)); } else if (arg instanceof Utc) { Utc.class.cast(arg).setTimestamp(stmt, pos); } else { stmt.setString(pos, arg.toString()); } ++pos; } } /** * The fetcher. */ public interface Fetcher { /** * Create prepare statement. * @param conn Open connection * @return The statement * @throws SQLException If some problem */ PreparedStatement statement(Connection conn) throws SQLException; /** * Fetch result set from statement. * @param stmt The statement * @return The result set * @throws SQLException If some problem */ ResultSet fetch(PreparedStatement stmt) throws SQLException; } }