org.firebirdsql.jdbc.FBRowUpdater Maven / Gradle / Ivy
Show all versions of jaybird-jdk17 Show documentation
/*
* $Id: FBRowUpdater.java 60362 2014-12-13 14:24:32Z mrotteveel $
*
* Firebird Open Source JavaEE Connector - JDBC Driver
*
* Distributable under LGPL license.
* You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* LGPL License for more details.
*
* This file was created by members of the firebird development team.
* All individual contributions remain the Copyright (C) of those
* individuals. Contributors to this file are either listed here or
* can be obtained from a source control history command.
*
* All rights reserved.
*/
package org.firebirdsql.jdbc;
import java.sql.DatabaseMetaData;
import java.sql.ResultSet;
import java.sql.SQLException;
import org.firebirdsql.gds.*;
import org.firebirdsql.gds.impl.AbstractIscStmtHandle;
import org.firebirdsql.gds.impl.GDSHelper;
import org.firebirdsql.jdbc.field.FBField;
import org.firebirdsql.jdbc.field.FBFlushableField;
import org.firebirdsql.jdbc.field.FieldDataProvider;
import org.firebirdsql.util.SQLExceptionChainBuilder;
/**
* Class responsible for modifying updatable result sets.
*
* A result set is updatable if and only if:
*
* - It is a subset of a single table and includes all columns from the
* table's primary key (in other words, includes all best row identifiers) or
* RDB$DB_KEY column (in this case tables without primary key can be updated
* too).
*
*
- If base table columns not included in the result set allow NULL values,
* result set allows inserting rows into it.
*
*
- The result set's SELECT statement does not contain subqueries, a
* DISTINCT predicate, a HAVING clause, aggregate functions, joined tables,
* user-defined functions, or stored procedures.
*
*
* If the result set definition does not meet these conditions, it is considered
* read-only.
*
* @author Roman Rokytskyy
*/
public class FBRowUpdater implements FirebirdRowUpdater {
private static final int PARAMETER_UNUSED = 0;
private static final int PARAMETER_USED = 1;
private static final int PARAMETER_DBKEY = 2;
private final GDSHelper gdsHelper;
private final Synchronizable syncProvider;
private final XSQLVAR[] xsqlvars;
private final FBField[] fields;
private final QuoteStrategy quoteStrategy;
private boolean inInsertRow;
private byte[][] newRow;
private byte[][] oldRow;
private byte[][] insertRow;
private boolean[] updatedFlags;
private String tableName;
private AbstractIscStmtHandle updateStatement;
private AbstractIscStmtHandle deleteStatement;
private AbstractIscStmtHandle insertStatement;
private AbstractIscStmtHandle selectStatement;
private final FBObjectListener.ResultSetListener rsListener;
private boolean closed;
private boolean processing;
public FBRowUpdater(GDSHelper connection, XSQLVAR[] xsqlvars,
Synchronizable syncProvider, boolean cached,
FBObjectListener.ResultSetListener rsListener) throws SQLException {
this.rsListener = rsListener;
this.gdsHelper = connection;
this.syncProvider = syncProvider;
this.xsqlvars = new XSQLVAR[xsqlvars.length];
this.fields = new FBField[xsqlvars.length];
quoteStrategy = QuoteStrategy.forDialect(connection.getDialect());
for (int i = 0; i < xsqlvars.length; i++) {
XSQLVAR xsqlvar = xsqlvars[i].deepCopy();
this.xsqlvars[i] = xsqlvar;
}
newRow = new byte[xsqlvars.length][];
updatedFlags = new boolean[xsqlvars.length];
for (int i = 0; i < this.xsqlvars.length; i++) {
final int fieldPos = i;
// implementation of the FieldDataProvider interface
FieldDataProvider dataProvider = new FieldDataProvider() {
public byte[] getFieldData() {
if (updatedFlags[fieldPos]) {
if (inInsertRow)
return insertRow[fieldPos];
else
return newRow[fieldPos];
} else
return oldRow[fieldPos];
}
public void setFieldData(byte[] data) {
if (inInsertRow)
insertRow[fieldPos] = data;
else
newRow[fieldPos] = data;
updatedFlags[fieldPos] = true;
}
};
fields[i] = FBField.createField(this.xsqlvars[i], dataProvider, connection, cached);
}
// find the table name (there can be only one table per result set)
for (int i = 0; i < xsqlvars.length; i++) {
if (tableName == null)
tableName = xsqlvars[i].relname;
else
if (!tableName.equals(xsqlvars[i].relname))
throw new FBResultSetNotUpdatableException(
"Underlying result set references at least two relations: " +
tableName + " and " + xsqlvars[i].relname + ".");
}
}
private void notifyExecutionStarted() throws SQLException {
if (closed)
throw new FBSQLException("Corresponding result set is closed.");
if (processing)
return;
rsListener.executionStarted(this);
this.processing = true;
}
private void notifyExecutionCompleted(boolean success) throws SQLException {
if (!processing)
return;
rsListener.executionCompleted(this, success);
this.processing = false;
}
private void deallocateStatement(AbstractIscStmtHandle handle, SQLExceptionChainBuilder chain) {
try {
if (handle != null)
gdsHelper.closeStatement(handle, true);
} catch(GDSException ex) {
chain.append(new FBSQLException(ex));
}
}
public void close() throws SQLException {
SQLExceptionChainBuilder chain = new SQLExceptionChainBuilder();
deallocateStatement(selectStatement, chain);
deallocateStatement(insertStatement, chain);
deallocateStatement(updateStatement, chain);
deallocateStatement(deleteStatement, chain);
// TODO: Close not completed by throw at this point?
if (chain.hasException())
throw chain.getException();
this.closed = true;
if (processing)
notifyExecutionCompleted(true);
}
public void setRow(byte[][] row) {
this.oldRow = row;
this.updatedFlags = new boolean[xsqlvars.length];
this.inInsertRow = false;
}
public void cancelRowUpdates() {
this.newRow = new byte[xsqlvars.length][];
this.updatedFlags = new boolean[xsqlvars.length];
this.inInsertRow = false;
}
public FBField getField(int fieldPosition) {
return fields[fieldPosition];
}
/**
* This method gets the parameter mask for the UPDATE or DELETE statement.
* Parameter mask is an array of booleans, where array item is set to true,
* if the appropriate field should be included in WHERE clause of the
* UPDATE or DELETE statement.
*
* This method obtains the parameter mask from the best row identifiers, in
* other words set of columns that form "best row identifiers" must be a
* subset of the selected columns (no distinction is made whether columns
* are real or are pseudo-columns). If no
*
* @return array of booleans that represent parameter mask.
*/
private int[] getParameterMask() throws SQLException {
// loop through the "best row identifiers" and set appropriate flags.
FBDatabaseMetaData metaData = new FBDatabaseMetaData(gdsHelper);
ResultSet bestRowIdentifier = metaData.getBestRowIdentifier(
"", "", tableName, DatabaseMetaData.bestRowSession, true);
try {
int[] result = new int[xsqlvars.length];
boolean hasParams = false;
while(bestRowIdentifier.next()) {
String columnName = bestRowIdentifier.getString(2);
if (columnName == null)
continue;
boolean found = false;
for (int i = 0; i < xsqlvars.length; i++) {
// special handling for the RDB$DB_KEY columns that must be
// selected as RDB$DB_KEY, but in XSQLVAR are represented
// as DB_KEY
if ("RDB$DB_KEY".equals(columnName) && "DB_KEY".equals(xsqlvars[i].sqlname)) {
result[i] = PARAMETER_DBKEY;
found = true;
} else if (columnName.equals(xsqlvars[i].sqlname)) {
result[i] = PARAMETER_USED;
found = true;
}
}
// if we did not found a column from the best row identifier
// in our result set, throw an exception, since we cannot
// reliably identify the row.
if (!found)
throw new FBResultSetNotUpdatableException(
"Underlying result set does not contain all columns " +
"that form 'best row identifier'.");
else
hasParams = true;
}
if (!hasParams)
throw new FBResultSetNotUpdatableException(
"No columns that can be used in WHERE clause could be " +
"found.");
return result;
} finally {
bestRowIdentifier.close();
}
}
private void appendWhereClause(StringBuffer sb, int[] parameterMask) {
sb.append("WHERE ");
// handle the RDB$DB_KEY case first
boolean hasDbKey = false;
for (int i = 0; i < parameterMask.length; i++) {
if (parameterMask[i] == PARAMETER_DBKEY) {
hasDbKey = true;
break;
}
}
if (hasDbKey) {
sb.append("RDB$DB_KEY = ?");
return;
}
// if we are here, then no RDB$DB_KEY update was used
// therefore loop through the parameters and build the
// WHERE clause
boolean first = true;
for (int i = 0; i < xsqlvars.length; i++) {
if (parameterMask[i] == PARAMETER_UNUSED) continue;
if (!first) sb.append(" AND");
sb.append("\n\t");
quoteStrategy.appendQuoted(xsqlvars[i].sqlname, sb).append(" = ?");
first = false;
}
}
private String buildUpdateStatement(int[] parameterMask) {
StringBuffer sb = new StringBuffer();
sb.append("UPDATE ");
quoteStrategy.appendQuoted(tableName, sb);
sb.append("\nSET\n");
boolean first = true;
for (int i = 0; i < xsqlvars.length; i++) {
if (!updatedFlags[i])
continue;
if (!first)
sb.append(",");
sb.append("\n\t");
quoteStrategy.appendQuoted(xsqlvars[i].sqlname, sb).append(" = ?");
first = false;
}
sb.append('\n');
appendWhereClause(sb, parameterMask);
return sb.toString();
}
private String buildDeleteStatement(int[] parameterMask) {
StringBuffer sb = new StringBuffer();
sb.append("DELETE FROM ");
quoteStrategy.appendQuoted(tableName, sb).append('\n');
appendWhereClause(sb, parameterMask);
return sb.toString();
}
private String buildInsertStatement() {
StringBuffer sb = new StringBuffer();
StringBuffer columns = new StringBuffer();
StringBuffer params = new StringBuffer();
sb.append("INSERT INTO ");
quoteStrategy.appendQuoted(tableName, sb);
boolean first = true;
for (int i = 0; i < xsqlvars.length; i++) {
if (!updatedFlags[i])
continue;
if (!first) {
columns.append(", ");
params.append(", ");
}
quoteStrategy.appendQuoted(xsqlvars[i].sqlname, columns);
params.append('?');
first = false;
}
sb.append('(').append(columns).append(") VALUES (").append(params).append(')');
return sb.toString();
}
private String buildSelectStatement(int[] parameterMask) {
StringBuffer sb = new StringBuffer();
StringBuffer columns = new StringBuffer();
sb.append("SELECT");
boolean first = true;
for (int i = 0; i < xsqlvars.length; i++) {
if (!first)
columns.append(", ");
// do special handling of RDB$DB_KEY, since Firebird returns
// DB_KEY column name instead of the correct one
if ("DB_KEY".equals(xsqlvars[i].sqlname)
&& ((xsqlvars[i].sqltype & ~1) == ISCConstants.SQL_TEXT)
&& xsqlvars[i].sqllen == 8)
columns.append("RDB$DB_KEY");
else
quoteStrategy.appendQuoted(xsqlvars[i].sqlname, columns);
first = false;
}
sb.append("\n\t").append(columns).append('\n')
.append("FROM\n\t");
quoteStrategy.appendQuoted(tableName, sb).append('\n');
appendWhereClause(sb, parameterMask);
return sb.toString();
}
private static final int UPDATE_STATEMENT_TYPE = 1;
private static final int DELETE_STATEMENT_TYPE = 2;
private static final int INSERT_STATEMENT_TYPE = 3;
private static final int SELECT_STATEMENT_TYPE = 4;
public void updateRow() throws SQLException {
boolean success = false;
synchronized(syncProvider.getSynchronizationObject()) {
try {
notifyExecutionStarted();
if (updateStatement == null)
updateStatement = gdsHelper.allocateStatement();
executeStatement(UPDATE_STATEMENT_TYPE, updateStatement);
success = true;
} catch(GDSException ex) {
throw new FBSQLException(ex);
} finally {
notifyExecutionCompleted(success);
}
}
}
public void deleteRow() throws SQLException {
boolean success = false;
synchronized(syncProvider.getSynchronizationObject()) {
try {
notifyExecutionStarted();
if (deleteStatement == null)
deleteStatement = gdsHelper.allocateStatement();
executeStatement(DELETE_STATEMENT_TYPE, deleteStatement);
success = true;
} catch(GDSException ex) {
throw new FBSQLException(ex);
} finally {
notifyExecutionCompleted(success);
}
}
}
public void insertRow() throws SQLException {
boolean success = false;
synchronized(syncProvider.getSynchronizationObject()) {
try {
notifyExecutionStarted();
if (insertStatement == null)
insertStatement = gdsHelper.allocateStatement();
executeStatement(INSERT_STATEMENT_TYPE, insertStatement);
success = true;
} catch(GDSException ex) {
throw new FBSQLException(ex);
} finally {
notifyExecutionCompleted(success);
}
}
}
public void refreshRow() throws SQLException {
boolean success = false;
synchronized(syncProvider.getSynchronizationObject()) {
try {
notifyExecutionStarted();
if (selectStatement == null)
selectStatement = gdsHelper.allocateStatement();
try {
executeStatement(SELECT_STATEMENT_TYPE, selectStatement);
// should fetch one row anyway
gdsHelper.fetch(selectStatement, 10);
Object[] rows = selectStatement.getRows();
if (selectStatement.size() == 0)
throw new FBSQLException("No rows could be fetched.");
if (selectStatement.size() > 1)
throw new FBSQLException("More then one row fetched.");
setRow((byte[][])rows[0]);
} finally {
gdsHelper.closeStatement(selectStatement, false);
selectStatement = null;
}
success = true;
} catch(GDSException ex) {
throw new FBSQLException(ex);
} finally {
notifyExecutionCompleted(success);
}
}
}
private void executeStatement(int statementType, AbstractIscStmtHandle stmt) throws SQLException {
try {
if (!stmt.isValid())
throw new FBSQLException("Corresponding connection is not valid.",
FBSQLException.SQL_STATE_CONNECTION_FAILURE_IN_TX);
if (inInsertRow && statementType != INSERT_STATEMENT_TYPE)
throw new FBSQLException("Only insertRow() is allowed when " +
"result set is positioned on insert row.");
if (statementType != INSERT_STATEMENT_TYPE && oldRow == null)
throw new FBSQLException("Result set is not positioned on a row.");
// we have to flush before constructing the parameters
// since flushable field can update the value, which
// in turn can change the parameter distribution
for (int i = 0; i < xsqlvars.length; i++) {
if (fields[i] instanceof FBFlushableField)
((FBFlushableField)fields[i]).flushCachedData();
}
int[] parameterMask = getParameterMask();
String sql;
switch(statementType) {
case UPDATE_STATEMENT_TYPE :
sql = buildUpdateStatement(parameterMask);
break;
case DELETE_STATEMENT_TYPE :
sql = buildDeleteStatement(parameterMask);
break;
case INSERT_STATEMENT_TYPE :
sql = buildInsertStatement();
break;
case SELECT_STATEMENT_TYPE :
sql = buildSelectStatement(parameterMask);
break;
default :
throw new IllegalArgumentException(
"Incorrect statement type specified.");
}
gdsHelper.prepareStatement(stmt, sql, true);
XSQLVAR[] params = stmt.getInSqlda().sqlvar;
int paramIterator = 0;
if (statementType == UPDATE_STATEMENT_TYPE) {
for (int i = 0; i < xsqlvars.length; i++) {
if (!updatedFlags[i])
continue;
params[paramIterator].copyFrom(xsqlvars[i]);
params[paramIterator].sqldata = newRow[i];
paramIterator++;
}
}
for (int i = 0; i < xsqlvars.length; i++) {
if (parameterMask[i] == PARAMETER_UNUSED && statementType != INSERT_STATEMENT_TYPE)
continue;
else
if (!updatedFlags[i] && statementType == INSERT_STATEMENT_TYPE)
continue;
params[paramIterator].copyFrom(xsqlvars[i]);
if (statementType == INSERT_STATEMENT_TYPE)
params[paramIterator].sqldata = insertRow[i];
else
params[paramIterator].sqldata = oldRow[i];
paramIterator++;
}
gdsHelper.executeStatement(stmt, false);
// TODO think about adding COMMIT RETAIN in the auto-commit mode
} catch(GDSException ex) {
throw new FBSQLException(ex);
}
}
public boolean rowInserted() throws SQLException {
return false;
}
public boolean rowDeleted() throws SQLException {
return false;
}
public boolean rowUpdated() throws SQLException {
return false;
}
public byte[][] getNewRow() {
byte[][] result = new byte[oldRow.length][];
for (int i = 0; i < result.length; i++) {
if (updatedFlags[i]) {
if (newRow[i] == null)
result[i] = null;
else {
result[i] = new byte[newRow[i].length];
System.arraycopy(newRow[i], 0, result[i], 0, newRow[i].length);
}
} else {
if (oldRow[i] == null) {
result[i] = null;
} else {
result[i] = new byte[oldRow[i].length];
System.arraycopy(oldRow[i], 0, result[i], 0, oldRow[i].length);
}
}
}
return result;
}
public byte[][] getInsertRow() {
return insertRow;
}
public byte[][] getOldRow() {
return oldRow;
}
public void moveToInsertRow() throws SQLException {
inInsertRow = true;
insertRow = new byte[xsqlvars.length][];
this.updatedFlags = new boolean[xsqlvars.length];
}
public void moveToCurrentRow() throws SQLException {
inInsertRow = false;
insertRow = new byte[xsqlvars.length][];
this.updatedFlags = new boolean[xsqlvars.length];
}
}