org.firebirdsql.jdbc.FBRowUpdater Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jaybird Show documentation
Show all versions of jaybird Show documentation
JDBC Driver for the Firebird RDBMS
/*
* Firebird Open Source 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 org.firebirdsql.gds.impl.GDSHelper;
import org.firebirdsql.gds.ng.FbStatement;
import org.firebirdsql.gds.ng.LockCloseable;
import org.firebirdsql.gds.ng.fields.FieldDescriptor;
import org.firebirdsql.gds.ng.fields.RowDescriptor;
import org.firebirdsql.gds.ng.fields.RowValue;
import org.firebirdsql.gds.ng.listeners.StatementListener;
import org.firebirdsql.jaybird.util.SQLExceptionChainBuilder;
import org.firebirdsql.jaybird.util.UncheckedSQLException;
import org.firebirdsql.jdbc.field.FBField;
import org.firebirdsql.jdbc.field.FBFlushableField;
import org.firebirdsql.jdbc.field.FieldDataProvider;
import java.sql.DatabaseMetaData;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.stream.StreamSupport;
import static org.firebirdsql.jaybird.util.StringUtils.isNullOrEmpty;
import static org.firebirdsql.jdbc.SQLStateConstants.SQL_STATE_INVALID_CURSOR_STATE;
/**
* 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
* @author Mark Rotteveel
*/
final class FBRowUpdater implements FirebirdRowUpdater {
// Estimated average column length of 10 + 2 quote characters + comma (for pre-sizing string builders).
// We could be precise by summing over the field descriptors, but we don't want to waste cycles on it.
private static final int EST_COLUMN_SIZE = 13;
// Estimated size for update/delete/select statement (for pre-sizing string builders).
// This is probably still too small for some cases, but will prevent a number of resizes from the default size
private static final int EST_STATEMENT_SIZE = 64;
private static final String ROW_INSERT = "insert";
private static final String ROW_CURRENT = "current";
private static final String ROW_UPDATE = "update";
private static final String ROW_OLD = "old";
private static final byte[][] EMPTY_2D_BYTES = new byte[0][];
private final String tableName;
private final FBObjectListener.ResultSetListener rsListener;
private final GDSHelper gdsHelper;
private final RowDescriptor rowDescriptor;
private final List fields;
private final QuoteStrategy quoteStrategy;
private final FbStatement[] statements = new FbStatement[4];
private final List keyColumns;
private final RowValue newRow;
private RowValue oldRow;
private boolean inInsertRow;
private boolean closed;
private boolean processing;
FBRowUpdater(FBConnection connection, RowDescriptor rowDescriptor, boolean cached,
FBObjectListener.ResultSetListener rsListener) throws SQLException {
tableName = requireSingleTableName(rowDescriptor);
keyColumns = deriveKeyColumns(tableName, rowDescriptor, connection.getMetaData());
this.rsListener = rsListener;
gdsHelper = connection.getGDSHelper();
quoteStrategy = connection.getQuoteStrategy();
fields = createFields(rowDescriptor, cached);
newRow = rowDescriptor.createDefaultFieldValues();
this.rowDescriptor = rowDescriptor;
}
private List createFields(RowDescriptor rowDescriptor, boolean cached) throws SQLException {
try {
return StreamSupport.stream(rowDescriptor.spliterator(), false)
.map(fieldDescriptor -> createFieldUnchecked(fieldDescriptor, cached))
.toList();
} catch (UncheckedSQLException e) {
throw e.getCause();
}
}
private FBField createFieldUnchecked(FieldDescriptor fieldDescriptor, boolean cached) {
try {
return FBField.createField(
fieldDescriptor, new FieldDataProviderImpl(fieldDescriptor.getPosition()), gdsHelper, cached);
} catch (SQLException e) {
throw new UncheckedSQLException(e);
}
}
/**
* Returns the single table name referenced by {@code rowDescriptor}, or throws an exception if there are no or
* multiple table names.
*
* @param rowDescriptor
* row descriptor
* @return non-null table name
* @throws SQLException
* if {@code rowDescriptor} references multiple table names or no table names at all
*/
private static String requireSingleTableName(RowDescriptor rowDescriptor) throws SQLException {
// find the table name (there can be only one table per updatable result set)
String tableName = null;
for (FieldDescriptor fieldDescriptor : rowDescriptor) {
// TODO This will not detect derived columns in the prefix of the select list
if (tableName == null) {
tableName = fieldDescriptor.getOriginalTableName();
} else if (!Objects.equals(tableName, fieldDescriptor.getOriginalTableName())) {
throw new FBResultSetNotUpdatableException(
"Underlying result set references at least two relations: %s and %s."
.formatted(tableName, fieldDescriptor.getOriginalTableName()));
}
}
if (isNullOrEmpty(tableName)) {
throw new FBResultSetNotUpdatableException("Underlying result set references no relations");
}
return tableName;
}
private void notifyExecutionStarted() throws SQLException {
if (closed) throw new SQLException("Corresponding result set is closed.", SQL_STATE_INVALID_CURSOR_STATE);
if (processing) return;
rsListener.executionStarted(this);
processing = true;
}
private void notifyExecutionCompleted(boolean success) throws SQLException {
if (!processing) return;
rsListener.executionCompleted(this, success);
processing = false;
}
private void deallocateStatement(FbStatement handle, SQLExceptionChainBuilder chain) {
if (handle == null) return;
try {
handle.close();
} catch (SQLException ex) {
chain.append(ex);
}
}
@Override
public void close() throws SQLException {
closed = true;
var chain = new SQLExceptionChainBuilder();
for (FbStatement statement : statements) {
deallocateStatement(statement, chain);
}
try {
notifyExecutionCompleted(true);
} catch (SQLException e) {
chain.append(e);
}
chain.throwIfPresent();
}
@Override
public void setRow(RowValue row) {
oldRow = row;
newRow.reset();
inInsertRow = false;
}
@Override
public void cancelRowUpdates() {
newRow.reset();
inInsertRow = false;
}
@Override
public FBField getField(int fieldPosition) {
return fields.get(fieldPosition);
}
/**
* This method derives the key columns that uniquely identify the row in the result set.
*
* The key column(s) are the best row identifier, or {@code RDB$DB_KEY} if available. The columns of that 'best row
* identifier' (or the DB key) must be a subset of the selected columns. If no suitable columns are found, an
* exception is thrown.
*
*
* @return immutable list of columns that uniquely identify the row, for use in the WHERE clause of the UPDATE,
* DELETE, or SELECT statements created in this class.
* @throws FBResultSetNotUpdatableException
* if there are no suitable columns to identify a row uniquely
* @throws SQLException
* for errors looking up the best row identifier
*/
private static List deriveKeyColumns(String tableName, RowDescriptor rowDescriptor,
DatabaseMetaData dbmd) throws SQLException {
// first try best row identifier
List keyColumns = keyColumnsOfBestRowIdentifier( tableName, rowDescriptor, dbmd);
if (keyColumns.isEmpty()) {
// best row identifier not available or not fully matched, fallback to RDB$DB_KEY
// NOTE: fallback is updatable, but may not be insertable (e.g. if missing PK column(s) are not generated)!
keyColumns = keyColumnsOfDbKey(rowDescriptor);
if (keyColumns.isEmpty()) {
// we did not find the columns of the best row identifier or RDB$DB_KEY in our result set,
// throw an exception, since we cannot reliably identify the row.
throw new FBResultSetNotUpdatableException("Underlying result set does not contain all columns that "
+ "form 'best row identifier' and no RDB$DB_KEY was available as fallback");
}
}
return List.copyOf(keyColumns);
}
/**
* Derives the key columns based on {@code DatabaseMetaData.getBestRowIdentifier}.
*
* The 'best row identifier' are the primary key columns or {@code RDB$DB_KEY} if there is no primary key.
*
*
* @return a list of columns in the best row identifier, or empty if there is no best row identifier, or not all
* columns of the best row identifier exist in {@code rowDescriptor}
* @throws SQLException
* for errors looking up the best row identifier
*/
private static List keyColumnsOfBestRowIdentifier(String tableName, RowDescriptor rowDescriptor,
DatabaseMetaData dbmd) throws SQLException {
try (ResultSet bestRowIdentifier = dbmd
.getBestRowIdentifier("", "", tableName, DatabaseMetaData.bestRowTransaction, true)) {
int bestRowIdentifierColumnCount = 0;
List keyColumns = new ArrayList<>();
while (bestRowIdentifier.next()) {
bestRowIdentifierColumnCount++;
String columnName = bestRowIdentifier.getString(2);
if (columnName == null) continue;
for (FieldDescriptor fieldDescriptor : rowDescriptor) {
// NOTE: We only use the first occurrence of a column
// TODO repeated columns in an updatable result set might be problematic in and of itself, maybe we
// need to explicitly disallow it.
if ("RDB$DB_KEY".equals(columnName) && fieldDescriptor.isDbKey()) {
// special handling for the RDB$DB_KEY columns that must be referenced as RDB$DB_KEY in select
// and where, but in metadata are represented as DB_KEY
return List.of(fieldDescriptor);
} else if (columnName.equals(fieldDescriptor.getOriginalName())) {
keyColumns.add(fieldDescriptor);
}
}
// column of best row identifier not found, stop matching process
if (keyColumns.size() != bestRowIdentifierColumnCount) {
return List.of();
}
}
return keyColumns;
}
}
/**
* Derives the key column based on {@code RDB$DB_KEY}, if present in the result set.
*
* This is intended as a fallback when {@code keyColumnsOfBestRowIdentifier} returns an empty list.
*
*
* @return list with the (single) {@link FieldDescriptor} of the {@code RDB$DB_KEY} column, or an empty list if
* {@code rowDescriptor} contains no {@code RDB$DB_KEY} column
*/
private static List keyColumnsOfDbKey(RowDescriptor rowDescriptor) {
for (FieldDescriptor fieldDescriptor : rowDescriptor) {
if (fieldDescriptor.isDbKey()) {
return List.of(fieldDescriptor);
}
}
return List.of();
}
private void appendWhereClause(StringBuilder sb) {
sb.append("where ");
// handle the RDB$DB_KEY case first
if (keyColumns.get(0).isDbKey()) {
sb.append("RDB$DB_KEY=?");
return;
}
// no RDB$DB_KEY update was used, so loop through the key columns and build the WHERE clause
boolean first = true;
for (FieldDescriptor fieldDescriptor : keyColumns) {
if (first) {
first = false;
} else {
sb.append("\nand ");
}
quoteStrategy.appendQuoted(fieldDescriptor.getOriginalName(), sb).append("=?");
}
}
private String buildUpdateStatement() {
// TODO raise exception if there are no updated columns, or do nothing?
var sb = new StringBuilder(EST_STATEMENT_SIZE + newRow.initializedCount() * EST_COLUMN_SIZE)
.append("update ");
quoteStrategy.appendQuoted(tableName, sb).append(" set ");
boolean first = true;
for (FieldDescriptor fieldDescriptor : rowDescriptor) {
if (!newRow.isInitialized(fieldDescriptor.getPosition()) || fieldDescriptor.isDbKey()) continue;
if (first) {
first = false;
} else {
sb.append(",\n\t");
}
quoteStrategy.appendQuoted(fieldDescriptor.getOriginalName(), sb).append("=?");
}
sb.append('\n');
appendWhereClause(sb);
return sb.toString();
}
private String buildDeleteStatement() {
var sb = new StringBuilder(EST_STATEMENT_SIZE).append("delete from ");
quoteStrategy.appendQuoted(tableName, sb).append('\n');
appendWhereClause(sb);
return sb.toString();
}
private String buildInsertStatement() {
// TODO raise exception if there are no initialized columns, or use INSERT .. DEFAULT VALUES?
final int initializedColumnCount = newRow.initializedCount();
var columns = new StringBuilder(initializedColumnCount * EST_COLUMN_SIZE);
var params = new StringBuilder(initializedColumnCount * 2);
boolean first = true;
for (FieldDescriptor fieldDescriptor : rowDescriptor) {
if (!newRow.isInitialized(fieldDescriptor.getPosition()) || fieldDescriptor.isDbKey()) continue;
if (first) {
first = false;
} else {
columns.append(',');
params.append(',');
}
quoteStrategy.appendQuoted(fieldDescriptor.getOriginalName(), columns);
params.append('?');
}
// 27 = length of appended literals + 2 quote characters
var sb = new StringBuilder(27 + tableName.length() + columns.length() + params.length()).append("insert into ");
quoteStrategy.appendQuoted(tableName, sb)
.append(" (").append(columns).append(") values (").append(params).append(')');
return sb.toString();
}
private String buildSelectStatement() {
var columns = new StringBuilder(rowDescriptor.getCount() * EST_COLUMN_SIZE);
boolean first = true;
for (FieldDescriptor fieldDescriptor : rowDescriptor) {
if (first) {
first = false;
} else {
columns.append(',');
}
// special handling of RDB$DB_KEY, since Firebird returns DB_KEY column name instead of the correct one
if (fieldDescriptor.isDbKey()) {
columns.append("RDB$DB_KEY");
} else {
quoteStrategy.appendQuoted(fieldDescriptor.getOriginalName(), columns);
}
}
var sb = new StringBuilder(EST_STATEMENT_SIZE + columns.length())
.append("select ").append(columns).append("\nfrom ");
quoteStrategy.appendQuoted(tableName, sb).append('\n');
appendWhereClause(sb);
return sb.toString();
}
private static final int UPDATE_STATEMENT_TYPE = 0;
private static final int DELETE_STATEMENT_TYPE = 1;
private static final int INSERT_STATEMENT_TYPE = 2;
private static final int SELECT_STATEMENT_TYPE = 3;
private void modifyRow(int statementType) throws SQLException {
try (LockCloseable ignored = gdsHelper.withLock()) {
boolean success = false;
try {
notifyExecutionStarted();
executeStatement(statementType, getStatementWithTransaction(statementType));
success = true;
} finally {
notifyExecutionCompleted(success);
}
}
}
@SuppressWarnings("resource")
private FbStatement getStatementWithTransaction(int statementType) throws SQLException {
FbStatement stmt = statements[statementType];
if (stmt == null) {
return statements[statementType] = gdsHelper.allocateStatement();
} else {
stmt.setTransaction(gdsHelper.getCurrentTransaction());
return stmt;
}
}
@Override
public void updateRow() throws SQLException {
modifyRow(UPDATE_STATEMENT_TYPE);
}
@Override
public void deleteRow() throws SQLException {
modifyRow(DELETE_STATEMENT_TYPE);
}
@Override
public void insertRow() throws SQLException {
modifyRow(INSERT_STATEMENT_TYPE);
}
@Override
public void refreshRow() throws SQLException {
try (LockCloseable ignored = gdsHelper.withLock()) {
boolean success = false;
try {
notifyExecutionStarted();
FbStatement selectStatement = getStatementWithTransaction(SELECT_STATEMENT_TYPE);
final RowListener rowListener = new RowListener();
selectStatement.addStatementListener(rowListener);
try {
executeStatement(SELECT_STATEMENT_TYPE, selectStatement);
// should fetch one row anyway
selectStatement.fetchRows(10);
List rows = rowListener.getRows();
if (rows.isEmpty()) {
throw new SQLException("No rows could be fetched.");
}
if (rows.size() > 1) {
throw new SQLException("More then one row fetched.");
}
setRow(rows.get(0));
} finally {
selectStatement.removeStatementListener(rowListener);
selectStatement.closeCursor();
}
success = true;
} finally {
notifyExecutionCompleted(success);
}
}
}
private void executeStatement(int statementType, FbStatement stmt) throws SQLException {
if (statementType != INSERT_STATEMENT_TYPE) {
if (inInsertRow) {
throw new SQLException("Only insertRow() is allowed when result set is positioned on insert row.");
} else if (oldRow == null) {
throw new SQLException("Result set is not positioned on a row.");
}
}
// Flushable field can update the value, which in turn can change the parameter distribution
flushFields();
stmt.prepare(generateStatementText(statementType));
List params = new ArrayList<>(newRow.initializedCount() + keyColumns.size());
// Set parameters of new values
if (statementType == UPDATE_STATEMENT_TYPE || statementType == INSERT_STATEMENT_TYPE) {
for (FieldDescriptor fieldDescriptor : rowDescriptor) {
if (newRow.isInitialized(fieldDescriptor.getPosition()) && !fieldDescriptor.isDbKey()) {
params.add(newRow.getFieldData(fieldDescriptor.getPosition()));
}
}
}
// Set parameters of where clause
if (statementType != INSERT_STATEMENT_TYPE) {
for (FieldDescriptor keyColumn : keyColumns) {
params.add(oldRow.getFieldData(keyColumn.getPosition()));
}
}
stmt.execute(RowValue.of(params.toArray(EMPTY_2D_BYTES)));
}
private void flushFields() throws SQLException {
for (FBField field : fields) {
if (field instanceof FBFlushableField flushableField) {
flushableField.flushCachedData();
}
}
}
private String generateStatementText(int statementType) {
return switch (statementType) {
case UPDATE_STATEMENT_TYPE -> buildUpdateStatement();
case DELETE_STATEMENT_TYPE -> buildDeleteStatement();
case INSERT_STATEMENT_TYPE -> buildInsertStatement();
case SELECT_STATEMENT_TYPE -> buildSelectStatement();
default -> throw new IllegalArgumentException("Incorrect statement type specified.");
};
}
@Override
public RowValue getNewRow() throws SQLException {
if (inInsertRow) {
throw wrongRow(ROW_UPDATE, ROW_INSERT);
}
RowValue newRowCopy = rowDescriptor.createDefaultFieldValues();
for (int i = 0; i < rowDescriptor.getCount(); i++) {
byte[] fieldData = getFieldData(i);
newRowCopy.setFieldData(i, fieldData != null ? fieldData.clone() : null);
}
return newRowCopy;
}
private byte[] getFieldData(int field) {
RowValue source = newRow.isInitialized(field) || inInsertRow ? newRow : oldRow;
return source.getFieldData(field);
}
@Override
public RowValue getInsertRow() throws SQLException {
if (inInsertRow) {
RowValue newRowCopy = newRow.deepCopy();
newRowCopy.initializeFields();
return newRowCopy;
}
throw wrongRow(ROW_INSERT, ROW_CURRENT);
}
@Override
public RowValue getOldRow() throws SQLException {
if (inInsertRow) {
throw wrongRow(ROW_OLD, ROW_INSERT);
}
return oldRow;
}
private static SQLException wrongRow(String expectedRow, String actualRow) {
return new SQLException(
"Cannot return %s row, currently positioned on %s row".formatted(expectedRow, actualRow));
}
@Override
public void moveToInsertRow() {
inInsertRow = true;
newRow.reset();
}
@Override
public void moveToCurrentRow() {
inInsertRow = false;
newRow.reset();
}
private static final class RowListener implements StatementListener {
// expect 0 or 1 rows (2 or more would mean the key columns didn't identify a row uniquely)
private final List rows = new ArrayList<>(1);
@Override
public void receivedRow(FbStatement sender, RowValue rowValue) {
rows.add(rowValue);
}
public List getRows() {
return rows;
}
}
private final class FieldDataProviderImpl implements FieldDataProvider {
private final int field;
FieldDataProviderImpl(int field) {
this.field = field;
}
@Override
public byte[] getFieldData() {
return FBRowUpdater.this.getFieldData(field);
}
@Override
public void setFieldData(byte[] data) {
newRow.setFieldData(field, data);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy