org.sql2o.Sql2o Maven / Gradle / Ivy
package org.sql2o;
import lombok.extern.slf4j.Slf4j;
import org.sql2o.connectionsources.DataSourceConnectionSource;
import org.sql2o.connectionsources.ConnectionSource;
import org.sql2o.quirks.Quirks;
import org.sql2o.quirks.QuirksDetector;
import javax.sql.DataSource;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.Map;
/**
* Sql2o is the main class for the sql2o library.
*
* An Sql2o
instance represents a way of connecting to one specific database.
* To create a new instance, one need to specify either jdbc-url, username and password for the database or a data source.
*
* Internally the Sql2o instance uses a data source to create jdbc connections to the database. If url, username and password
* was specified in the constructor, a simple data source is created, which works as a simple wrapper around the jdbc
* driver.
*
* Some jdbc implementations have quirks, therefore it may be necessary to use a constructor with the quirks parameter.
* When quirks are specified, Sql2o will use workarounds to avoid these quirks.
* @author Lars Aaberg
*/
@Slf4j
public class Sql2o {
final Quirks quirks;
private Map defaultColumnMappings;
private boolean defaultCaseSensitive;
private ConnectionSource connectionSource;
public Sql2o(String jndiLookup) {
this(JndiDataSource.getJndiDatasource(jndiLookup));
}
/**
* Creates a new instance of the Sql2o class. Internally this constructor will create a {@link GenericDatasource},
* and call the {@link Sql2o#Sql2o(DataSource)} constructor which takes a DataSource as parameter.
* @param url JDBC database url
* @param user database username
* @param pass database password
*/
public Sql2o(String url, String user, String pass){
this(url, user, pass, QuirksDetector.forURL(url));
}
/**
* Created a new instance of the Sql2o class. Internally this constructor will create a {@link GenericDatasource},
* and call the {@link Sql2o#Sql2o(DataSource)} constructor which takes a DataSource as parameter.
* @param url JDBC database url
* @param user database username
* @param pass database password
* @param quirks {@link Quirks} allows sql2o to work around known quirks and issues in different JDBC drivers.
*/
public Sql2o(String url, String user, String pass, Quirks quirks) {
this(new GenericDatasource(url, user, pass), quirks);
}
/**
* Creates a new instance of the Sql2o class, which uses the given DataSource to acquire connections to the database.
* @param dataSource The DataSource Sql2o uses to acquire connections to the database.
*/
public Sql2o(DataSource dataSource) {
this(dataSource, QuirksDetector.forObject(dataSource));
}
/**
* Creates a new instance of the Sql2o class, which uses the given DataSource to acquire connections to the database.
* @param dataSource The DataSource Sql2o uses to acquire connections to the database.
* @param quirks {@link Quirks} allows sql2o to work around known quirks and issues in different JDBC drivers.
*/
public Sql2o(DataSource dataSource, Quirks quirks){
this.connectionSource = new DataSourceConnectionSource(dataSource);
this.quirks=quirks;
this.defaultColumnMappings = new HashMap<>();
}
public Quirks getQuirks() {
return quirks;
}
/**
* Gets the {@link ConnectionSource} that Sql2o uses internally to acquire database connections.
* @return The ConnectionSource instance
*/
public ConnectionSource getConnectionSource() {
return connectionSource;
}
/**
* Sets the {@link ConnectionSource} that Sql2o uses internally to acquire database connections.
* @param connectionSource the ConnectionSource instance to use
*/
public void setConnectionSource(ConnectionSource connectionSource) {
this.connectionSource = connectionSource;
}
/**
* Gets the default column mappings Map. column mappings added to this Map are always available when Sql2o attempts
* to map between result sets and object instances.
* @return The {@link Map} instance, which Sql2o internally uses to map column names with property
* names.
*/
public Map getDefaultColumnMappings() {
return defaultColumnMappings;
}
/**
* Sets the default column mappings Map.
* @param defaultColumnMappings A {@link Map} instance Sql2o uses internally to map between column names and
* property names.
*/
public void setDefaultColumnMappings(Map defaultColumnMappings) {
this.defaultColumnMappings = defaultColumnMappings;
}
/**
* Gets value indicating if this instance of Sql2o is case sensitive when mapping between columns names and property
* names.
* @return
*/
public boolean isDefaultCaseSensitive() {
return defaultCaseSensitive;
}
/**
* Sets a value indicating if this instance of Sql2o is case sensitive when mapping between columns names and property
* names. This should almost always be false, because most relational databases are not case sensitive.
* @param defaultCaseSensitive
*/
public void setDefaultCaseSensitive(boolean defaultCaseSensitive) {
this.defaultCaseSensitive = defaultCaseSensitive;
}
/**
* Opens a connection to the database
* @param connectionSource the {@link ConnectionSource} implementation substitution,
* that will be used instead of one from {@link Sql2o} instance.
* @return instance of the {@link Connection} class.
*/
public Connection open(ConnectionSource connectionSource) {
return new Connection(this, connectionSource, false);
}
/**
* Opens a connection to the database
* @return instance of the {@link Connection} class.
*/
public Connection open() {
return new Connection(this, false);
}
/**
* Invokes the run method on the {@link StatementRunnableWithResult} instance. This method guarantees that
* the connection is closed properly, when either the run method completes or if an exception occurs.
* @param runnable
* @param argument
* @param
* @return
*/
@SuppressWarnings("unchecked")
public V withConnection(StatementRunnableWithResult runnable, Object argument) {
Connection connection = null;
try{
connection = open();
return (V)runnable.run(connection, argument);
} catch (Throwable t) {
throw new Sql2oException("An error occurred while executing StatementRunnable", t);
} finally {
if (connection != null) {
connection.close();
}
}
}
/**
* Invokes the run method on the {@link StatementRunnableWithResult} instance. This method guarantees that
* the connection is closed properly, when either the run method completes or if an exception occurs.
* @param runnable
* @param
* @return
*/
public V withConnection(StatementRunnableWithResult runnable) {
return withConnection(runnable, null);
}
/**
* Invokes the run method on the {@link StatementRunnableWithResult} instance. This method guarantees that
* the connection is closed properly, when either the run method completes or if an exception occurs.
* @param runnable
*/
public void withConnection(StatementRunnable runnable) {
withConnection(runnable, null);
}
/**
* Invokes the run method on the {@link StatementRunnableWithResult} instance. This method guarantees that
* the connection is closed properly, when either the run method completes or if an exception occurs.
* @param runnable
* @param argument
*/
public void withConnection(StatementRunnable runnable, Object argument) {
Connection connection = null;
try{
connection = open();
runnable.run(connection, argument);
} catch (Throwable t) {
throw new Sql2oException("An error occurred while executing StatementRunnable", t);
} finally{
if (connection != null) {
connection.close();
}
}
}
/**
* Begins a transaction with the given isolation level. Every statement executed on the return {@link Connection}
* instance, will be executed in the transaction. It is very important to always call either the {@link Connection#commit()}
* method or the {@link Connection#rollback()} method to close the transaction. Use proper try-catch logic.
* @param isolationLevel the isolation level of the transaction
* @return the {@link Connection} instance to use to run statements in the transaction.
*/
public Connection beginTransaction(int isolationLevel){
return beginTransaction(getConnectionSource(), isolationLevel);
}
/**
* Begins a transaction with the given isolation level. Every statement executed on the return {@link Connection}
* instance, will be executed in the transaction. It is very important to always call either the {@link Connection#commit()}
* method or the {@link Connection#rollback()} method to close the transaction. Use proper try-catch logic.
* @param connectionSource the {@link ConnectionSource} implementation substitution,
* that will be used instead of one from {@link Sql2o} instance.
* @param isolationLevel the isolation level of the transaction
* @return the {@link Connection} instance to use to run statements in the transaction.
*/
public Connection beginTransaction(ConnectionSource connectionSource, int isolationLevel) {
Connection connection = new Connection(this, connectionSource, false);
boolean success = false;
try {
connection.getJdbcConnection().setAutoCommit(false);
connection.getJdbcConnection().setTransactionIsolation(isolationLevel);
success = true;
} catch (SQLException e) {
throw new Sql2oException("Could not start the transaction - " + e.getMessage(), e);
} finally {
if (!success) {
connection.close();
}
}
return connection;
}
/**
* Begins a transaction with isolation level {@link java.sql.Connection#TRANSACTION_READ_COMMITTED}. Every statement executed on the return {@link Connection}
* instance, will be executed in the transaction. It is very important to always call either the {@link Connection#commit()}
* method or the {@link Connection#rollback()} method to close the transaction. Use proper try-catch logic.
* @return the {@link Connection} instance to use to run statements in the transaction.
*/
public Connection beginTransaction(){
return this.beginTransaction(java.sql.Connection.TRANSACTION_READ_COMMITTED);
}
/**
* Begins a transaction with isolation level {@link java.sql.Connection#TRANSACTION_READ_COMMITTED}. Every statement executed on the return {@link Connection}
* instance, will be executed in the transaction. It is very important to always call either the {@link Connection#commit()}
* method or the {@link Connection#rollback()} method to close the transaction. Use proper try-catch logic.
* @param connectionSource the {@link ConnectionSource} implementation substitution,
* that will be used instead of one from {@link Sql2o} instance.
* @return the {@link Connection} instance to use to run statements in the transaction.
*/
public Connection beginTransaction(ConnectionSource connectionSource) {
return this.beginTransaction(connectionSource, java.sql.Connection.TRANSACTION_READ_COMMITTED);
}
/**
* Calls the {@link StatementRunnable#run(Connection, Object)} method on the {@link StatementRunnable} parameter. All statements
* run on the {@link Connection} instance in the {@link StatementRunnable#run(Connection, Object) run} method will be
* executed in a transaction. The transaction will automatically be committed if the {@link StatementRunnable#run(Connection, Object) run}
* method finishes without throwing an exception. If an exception is thrown within the {@link StatementRunnable#run(Connection, Object) run} method,
* the transaction will automatically be rolled back.
*
* The isolation level of the transaction will be set to {@link java.sql.Connection#TRANSACTION_READ_COMMITTED}
* @param runnable The {@link StatementRunnable} instance.
*/
public void runInTransaction(StatementRunnable runnable){
runInTransaction(runnable, null);
}
/**
* Calls the {@link StatementRunnable#run(Connection, Object)} method on the {@link StatementRunnable} parameter. All statements
* run on the {@link Connection} instance in the {@link StatementRunnable#run(Connection, Object) run} method will be
* executed in a transaction. The transaction will automatically be committed if the {@link StatementRunnable#run(Connection, Object) run}
* method finishes without throwing an exception. If an exception is thrown within the {@link StatementRunnable#run(Connection, Object) run} method,
* the transaction will automatically be rolled back.
*
* The isolation level of the transaction will be set to {@link java.sql.Connection#TRANSACTION_READ_COMMITTED}
* @param runnable The {@link StatementRunnable} instance.
* @param argument An argument which will be forwarded to the {@link StatementRunnable#run(Connection, Object) run} method
*/
public void runInTransaction(StatementRunnable runnable, Object argument){
runInTransaction(runnable, argument, java.sql.Connection.TRANSACTION_READ_COMMITTED);
}
/**
* Calls the {@link StatementRunnable#run(Connection, Object)} method on the {@link StatementRunnable} parameter. All statements
* run on the {@link Connection} instance in the {@link StatementRunnable#run(Connection, Object) run} method will be
* executed in a transaction. The transaction will automatically be committed if the {@link StatementRunnable#run(Connection, Object) run}
* method finishes without throwing an exception. If an exception is thrown within the {@link StatementRunnable#run(Connection, Object) run} method,
* the transaction will automatically be rolled back.
* @param runnable The {@link StatementRunnable} instance.
* @param argument An argument which will be forwarded to the {@link StatementRunnable#run(Connection, Object) run} method
* @param isolationLevel The isolation level of the transaction
*/
public void runInTransaction(StatementRunnable runnable, Object argument, int isolationLevel){
Connection connection = this.beginTransaction(isolationLevel);
connection.setRollbackOnException(false);
try {
runnable.run(connection, argument);
} catch (Throwable throwable) {
connection.rollback();
throw new Sql2oException("An error occurred while executing StatementRunnable. Transaction is rolled back.", throwable);
}
connection.commit();
}
public V runInTransaction(StatementRunnableWithResult runnableWithResult){
return runInTransaction(runnableWithResult, null);
}
public V runInTransaction(StatementRunnableWithResult runnableWithResult, Object argument){
return runInTransaction(runnableWithResult, argument, java.sql.Connection.TRANSACTION_READ_COMMITTED);
}
@SuppressWarnings("unchecked")
public V runInTransaction(StatementRunnableWithResult runnableWithResult, Object argument, int isolationLevel){
Connection connection = this.beginTransaction(isolationLevel);
Object result;
try{
result = runnableWithResult.run(connection, argument);
} catch (Throwable throwable) {
connection.rollback();
throw new Sql2oException("An error occurred while executing StatementRunnableWithResult. Transaction rolled back.", throwable);
}
connection.commit();
return (V)result;
}
}