com.impossibl.postgres.jdbc.PGStatement Maven / Gradle / Ivy
Go to download
A new JDBC driver for PostgreSQL aimed at supporting the advanced features of JDBC and Postgres.
/**
* Copyright (c) 2013, impossibl.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:
*
* * Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* * 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.
* * Neither the name of impossibl.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 OWNER 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.impossibl.postgres.jdbc;
import com.impossibl.postgres.jdbc.Housekeeper.CleanupRunnable;
import com.impossibl.postgres.protocol.BindExecCommand;
import com.impossibl.postgres.protocol.CloseCommand;
import com.impossibl.postgres.protocol.Command;
import com.impossibl.postgres.protocol.DataRow;
import com.impossibl.postgres.protocol.QueryCommand;
import com.impossibl.postgres.protocol.ResultField;
import com.impossibl.postgres.protocol.ServerObjectType;
import com.impossibl.postgres.system.Settings;
import com.impossibl.postgres.types.Type;
import static com.impossibl.postgres.jdbc.Exceptions.CLOSED_STATEMENT;
import static com.impossibl.postgres.jdbc.Exceptions.ILLEGAL_ARGUMENT;
import static com.impossibl.postgres.jdbc.Exceptions.NOT_IMPLEMENTED;
import static com.impossibl.postgres.jdbc.Exceptions.UNWRAP_ERROR;
import static com.impossibl.postgres.protocol.ServerObjectType.Statement;
import java.lang.ref.WeakReference;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.SQLWarning;
import java.sql.Statement;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import static java.util.concurrent.TimeUnit.SECONDS;
abstract class PGStatement implements Statement {
protected static final String CACHED_STATEMENT_PREFIX = "cached-";
protected static final String NO_CACHE_STATEMENT_PREFIX = "nocache-";
/**
* Cleans up server resources in the event of leaking statements
*
* @author kdubb
*
*/
static class Cleanup implements CleanupRunnable {
PGConnectionImpl connection;
String name;
List> resultSets;
StackTraceElement[] allocationStackTrace;
public Cleanup(PGConnectionImpl connection, String name, List> resultSets) {
this.connection = connection;
this.name = name;
this.resultSets = resultSets;
this.allocationStackTrace = new Exception().getStackTrace();
}
@Override
public String getKind() {
return "statement";
}
@Override
public StackTraceElement[] getAllocationStackTrace() {
return allocationStackTrace;
}
@Override
public void run() {
closeResultSets(resultSets);
try {
dispose(connection, ServerObjectType.Statement, name);
}
catch (SQLException e) {
// Ignore...
}
connection.handleStatementClosure(null);
connection = null;
}
}
PGConnectionImpl connection;
String cursorName;
int resultSetType;
int resultSetConcurrency;
int resultSetHoldability;
int fetchDirection;
String name;
boolean processEscapes;
List resultFields;
Integer maxRows;
Integer fetchSize;
Integer maxFieldSize;
QueryCommand command;
List resultBatches;
boolean autoClose;
List> activeResultSets;
PGResultSet generatedKeysResultSet;
SQLWarning warningChain;
int queryTimeout;
final Housekeeper.Ref housekeeper;
final Object cleanupKey;
PGStatement(PGConnectionImpl connection, int resultSetType, int resultSetConcurrency, int resultSetHoldability, String name, List resultFields) {
super();
this.connection = connection;
this.resultSetType = resultSetType;
this.resultSetConcurrency = resultSetConcurrency;
this.resultSetHoldability = resultSetHoldability;
this.fetchDirection = ResultSet.FETCH_FORWARD;
this.name = name;
this.processEscapes = true;
this.resultFields = resultFields;
this.activeResultSets = new ArrayList<>();
this.generatedKeysResultSet = null;
this.fetchSize = connection.getDefaultFetchSize() != Settings.DEFAULT_FETCH_SIZE_DEFAULT ? Integer.valueOf(connection.getDefaultFetchSize()) : null;
this.housekeeper = connection.housekeeper;
if (this.housekeeper != null)
this.cleanupKey = this.housekeeper.add(this, new Cleanup(connection, name, activeResultSets));
else
this.cleanupKey = null;
}
/**
* Ensure the connection is not closed
*
* @throws SQLException
* If the connection is closed
*/
void checkClosed() throws SQLException {
if (isClosed())
throw CLOSED_STATEMENT;
}
/**
* Disposes of the named server object
*
* @param objectType
* Type of object to dispose of
* @param objectName
* Name of the object to dispose of
* @throws SQLException
* If an error occurs during disposal
*/
public static void dispose(PGConnectionImpl connection, ServerObjectType objectType, String objectName) throws SQLException {
if (objectName == null)
return;
CloseCommand close = connection.getProtocol().createClose(objectType, objectName);
connection.execute(close, false);
}
void dispose(Command command) throws SQLException {
if (command instanceof BindExecCommand) {
dispose(connection, ServerObjectType.Portal, ((BindExecCommand)command).getPortalName());
}
}
void releaseResultBatches() {
if (resultBatches != null) {
for (QueryCommand.ResultBatch resultBatch : resultBatches) {
resultBatch.release();
}
resultBatches = null;
}
}
/**
* Closes the given list of result-sets
*
* @throws SQLException
* If an error occurs closing a result set
*/
static void closeResultSets(List> resultSets) {
for (WeakReference resultSetRef : resultSets) {
PGResultSet resultSet = resultSetRef.get();
if (resultSet != null) {
try {
resultSet.internalClose();
}
catch (SQLException e) {
//Ignore...
}
}
}
resultSets.clear();
}
/**
* Closes all active result sets for this statement
*
* @throws SQLException
* If an error occurs closing a result set
*/
void closeResultSets() throws SQLException {
closeResultSets(activeResultSets);
generatedKeysResultSet = null;
}
/**
* Called by result sets to notify the statement of their closure. Removes
* the result set from the active set of result sets. If auto-close is
* enabled this closes the statement when the last result set is closed.
*
* @param resultSet
* The result set that is closing
* @throws SQLException
* If an error occurs closing a result set
*/
void handleResultSetClosure(PGResultSet resultSet) throws SQLException {
//Remove given & abandoned result sets
Iterator> resultSetRefIter = activeResultSets.iterator();
while (resultSetRefIter.hasNext()) {
WeakReference resultSetRef = resultSetRefIter.next();
PGResultSet rs = resultSetRef.get();
if (rs == null) {
resultSetRefIter.remove();
}
else if (rs == resultSet) {
resultSetRefIter.remove();
break;
}
}
//Handle auto closing
if (autoClose && activeResultSets.isEmpty()) {
close();
}
}
/**
* Determines whether or not the current statement state requires a named
* portal or could use the unnamed portal instead
*
* @return true when a named portal is required and false when it is not
*/
boolean needsNamedPortal() {
return fetchSize != null;
}
/**
* Cleans up all resources, including active result sets
*
* @throws SQLException
* If an error occurs closing result sets or
*/
void internalClose() throws SQLException {
closeResultSets();
releaseResultBatches();
if (name != null && !name.startsWith(CACHED_STATEMENT_PREFIX)) {
dispose(connection, Statement, name);
}
if (housekeeper != null)
housekeeper.remove(cleanupKey);
connection = null;
command = null;
resultFields = null;
generatedKeysResultSet = null;
}
boolean hasResults() {
return !resultBatches.isEmpty() &&
resultBatches.get(0).results != null;
}
boolean hasUpdateCount() {
return !resultBatches.isEmpty() &&
resultBatches.get(0).rowsAffected != null;
}
/**
* Execute the given sql
*
* @param sql
* SQL text to execute
* @return True if command returned results or false if not
* @throws SQLException
* If an error occurred during statement execution
*/
public boolean executeSimple(String sql) throws SQLException {
closeResultSets();
releaseResultBatches();
command = connection.getProtocol().createQuery(sql);
if (maxFieldSize != null)
command.setMaxFieldLength(maxFieldSize);
warningChain = connection.execute(command, true);
resultBatches = new ArrayList<>(command.getResultBatches());
return hasResults();
}
/**
* Execute the named statement. It must have previously been parsed and
* ready to be bound and executed.
*
* @param statementName Name of backend statement to execute or null
* @param parameterTypes List of parameter types
* @param parameterValues List of parmaeter values
* @return true if command returned results or false if not
* @throws SQLException
* If an error occurred durring statement execution
*/
public boolean executeStatement(String statementName, List parameterTypes, List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy