org.firebirdsql.jdbc.FBUpdatableFetcher 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.ng.fields.RowValue;
import org.jspecify.annotations.NullMarked;
import org.jspecify.annotations.Nullable;
import java.sql.SQLException;
import java.sql.SQLNonTransientException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static java.lang.String.format;
import static java.util.Collections.emptyList;
import static java.util.Collections.emptyMap;
/**
* Decorator that handles tracking updates, deletes and inserts of an updatable result set.
*
* This fetcher handles the updatable result set behaviour defined in jdp-2021-04
* for server-side scrollable cursors and jdp-2024-05
* for emulated scrollable cursors.
*
*
* This behaviour can be summarized as: updates are visible, deletes are visible (with a deletion marker row),
* and inserts occur at the end of the cursor.
*
*
* @since 5
*/
@NullMarked
final class FBUpdatableFetcher implements FBFetcher {
private final FBFetcher fetcher;
private final RowValue deletedRowMarker;
private FBObjectListener.FetcherListener fetcherListener;
private Map modifiedRows = new HashMap<>();
private int position;
private List insertedRows = new ArrayList<>();
private int firstInsertPosition;
private int fetcherSize = -1;
private final InternalFetcherListener rowListener = new InternalFetcherListener();
/**
* Creates an updatable fetcher wrapping a real fetcher.
*
* @param fetcher
* The fetcher decorated by this fetcher
* @param fetcherListener
* Fetcher listener
* @param deletedRowMarker
* Deleted row marker
*/
FBUpdatableFetcher(FBFetcher fetcher, FBObjectListener.FetcherListener fetcherListener, RowValue deletedRowMarker) {
if (!deletedRowMarker.isDeletedRowMarker()) {
throw new IllegalArgumentException("deletedRowMarker should return true for isDeletedRowMarker()");
}
this.fetcher = fetcher;
fetcher.setFetcherListener(rowListener);
this.fetcherListener = fetcherListener;
this.deletedRowMarker = deletedRowMarker;
}
@Override
public FetchConfig getFetchConfig() {
return fetcher.getFetchConfig();
}
@Override
public void setReadOnly() throws SQLException {
throw new SQLNonTransientException("This fetcher implementation cannot be marked read-only",
SQLStateConstants.SQL_STATE_INVALID_ATTR_VALUE);
}
private boolean notifyFetcherRow(int position) throws SQLException {
return notifyRow(position, rowListener.lastReceivedRow);
}
private boolean notifyInsertedRow(int position) throws SQLException {
int insertPosition = insertPosition(position);
RowValue rowValue = insertPosition < insertedRows.size() ? insertedRows.get(insertPosition) : null;
return notifyRow(position, rowValue);
}
private boolean notifyRow(int position, @Nullable RowValue originalRowValue) throws SQLException {
//noinspection DataFlowIssue : Disagree with IntelliJs nullability inference for this method
RowValue rowValue = modifiedRows.getOrDefault(position, originalRowValue);
fetcherListener.rowChanged(fetcher, rowValue);
//noinspection ConstantValue : Disagree with IntelliJs nullability inference
return rowValue != null;
}
private int insertPosition(int position) throws SQLException {
int firstInsertPosition = firstInsertPosition();
if (position < firstInsertPosition) {
throw new SQLException(format("Implementation error: %d is not a valid insert-position (minimum: %d)",
position, firstInsertPosition));
}
return position - firstInsertPosition();
}
private int firstInsertPosition() throws SQLException {
int firstInsertPosition = this.firstInsertPosition;
if (firstInsertPosition == 0) {
return this.firstInsertPosition = fetcherSize() + 1;
}
return firstInsertPosition;
}
private int fetcherSize() throws SQLException {
int fetcherSize = this.fetcherSize;
if (fetcherSize == -1) {
return this.fetcherSize = fetcher.size();
}
return fetcherSize;
}
private int cappedPosition(int position) throws SQLException {
return Math.max(0, Math.min(size() + 1, position));
}
@Override
public boolean first() throws SQLException {
this.position = 1;
return fetcher.first() ? notifyFetcherRow(1) : notifyInsertedRow(1);
}
@Override
public boolean last() throws SQLException {
if (insertedRows.isEmpty()) {
fetcher.last();
int position = this.position = fetcher.currentPosition();
return notifyFetcherRow(position);
} else {
int position = this.position = size();
fetcher.afterLast();
return notifyInsertedRow(position);
}
}
@Override
public boolean previous() throws SQLException {
int position = this.position = cappedPosition(this.position - 1);
if (position <= fetcherSize()) {
fetcher.previous();
assert position == fetcher.currentPosition() : "Position discrepancy: " + position + " <> " + fetcher.currentPosition();
return notifyFetcherRow(position);
} else {
fetcher.afterLast();
return notifyInsertedRow(position);
}
}
@Override
public boolean next() throws SQLException {
int position = this.position = cappedPosition(this.position + 1);
if (position <= fetcherSize()) {
fetcher.next();
assert position == fetcher.currentPosition() : "Position discrepancy: " + position + " <> " + fetcher.currentPosition();
return notifyFetcherRow(position);
} else {
fetcher.afterLast();
return notifyInsertedRow(position);
}
}
@Override
public boolean absolute(int row) throws SQLException {
int position = row >= 0 ? cappedPosition(row) : cappedPosition(size() + 1 + row);
return internalAbsolute(position);
}
private boolean internalAbsolute(int position) throws SQLException {
this.position = position;
if (position <= fetcherSize()) {
fetcher.absolute(position);
return notifyFetcherRow(position);
} else {
fetcher.afterLast();
return notifyInsertedRow(position);
}
}
@Override
public boolean relative(int row) throws SQLException {
int position = cappedPosition(this.position + row);
return internalAbsolute(position);
}
@Override
public void beforeFirst() throws SQLException {
position = 0;
fetcher.beforeFirst();
notifyFetcherRow(0);
}
@Override
public void afterLast() throws SQLException {
int position = this.position = size() + 1;
fetcher.afterLast();
notifyFetcherRow(position);
}
@Override
public void close() throws SQLException {
close(CompletionReason.OTHER);
}
@Override
public void close(CompletionReason completionReason) throws SQLException {
try {
fetcher.close(completionReason);
} finally {
modifiedRows.clear();
modifiedRows = emptyMap();
insertedRows.clear();
insertedRows = emptyList();
}
}
@Override
public boolean isClosed() {
return fetcher.isClosed();
}
@Override
public int getRowNum() throws SQLException {
int position = this.position;
return position <= size() ? position : 0;
}
@Override
public boolean isEmpty() throws SQLException {
return size() == 0;
}
@Override
public boolean isBeforeFirst() throws SQLException {
return position == 0;
}
@Override
public boolean isFirst() throws SQLException {
return position == 1 && !isEmpty();
}
@Override
public boolean isLast() throws SQLException {
int size = size();
return position == size && size > 0;
}
@Override
public boolean isAfterLast() throws SQLException {
return position > size();
}
@Override
public void beforeExecuteInsert() throws SQLException {
fetcher.beforeExecuteInsert();
}
@Override
public void insertRow(RowValue data) throws SQLException {
insertedRows.add(data);
fetcherListener.rowChanged(this, data);
}
@Override
public void deleteRow() throws SQLException {
modifiedRows.put(position, deletedRowMarker);
fetcherListener.rowChanged(this, deletedRowMarker);
}
@Override
public void updateRow(RowValue data) throws SQLException {
modifiedRows.put(position, data);
fetcherListener.rowChanged(this, data);
}
@Override
public void renotifyCurrentRow() throws SQLException {
int position = this.position;
// We can reuse the lastReceivedRow of the fetcher listener if the current row is not stored in this fetcher
if (position <= fetcherSize()) {
notifyFetcherRow(position);
} else {
notifyInsertedRow(position);
}
}
@Override
public int getFetchSize() throws SQLException {
return fetcher.getFetchSize();
}
@Override
public void setFetchSize(int fetchSize) throws SQLException {
fetcher.setFetchSize(fetchSize);
}
@Override
public int getFetchDirection() throws SQLException {
return fetcher.getFetchDirection();
}
@Override
public void setFetchDirection(int direction) throws SQLException {
fetcher.setFetchDirection(direction);
}
@Override
public int currentPosition() {
return position;
}
@Override
public int size() throws SQLException {
return fetcherSize() + insertedRows.size();
}
@Override
public void setFetcherListener(FBObjectListener.FetcherListener fetcherListener) {
this.fetcherListener = fetcherListener;
}
@Override
public boolean rowInserted() throws SQLException {
return !(isBeforeFirst() || isAfterLast()) && firstInsertPosition() <= position;
}
@Override
public boolean rowUpdated() throws SQLException {
if (isBeforeFirst() || isAfterLast()) {
return false;
}
RowValue rowValue = modifiedRows.get(position);
return !(rowValue == null || rowValue.isDeletedRowMarker());
}
@Override
public boolean rowDeleted() throws SQLException {
if (isBeforeFirst() || isAfterLast()) {
return false;
}
RowValue rowValue = modifiedRows.get(position);
return rowValue != null && rowValue.isDeletedRowMarker();
}
private static final class InternalFetcherListener implements FBObjectListener.FetcherListener {
@Nullable RowValue lastReceivedRow;
@Override
public void rowChanged(FBFetcher fetcher, @Nullable RowValue newRow) {
lastReceivedRow = newRow;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy