org.firebirdsql.gds.ng.wire.version11.V11Statement 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.gds.ng.wire.version11;
import org.firebirdsql.gds.ISCConstants;
import org.firebirdsql.gds.ng.wire.AsyncFetchStatus;
import org.firebirdsql.gds.ng.CursorFlag;
import org.firebirdsql.gds.ng.DeferredResponse;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.gds.ng.FetchDirection;
import org.firebirdsql.gds.ng.LockCloseable;
import org.firebirdsql.gds.ng.OperationCloseHandle;
import org.firebirdsql.gds.ng.StatementState;
import org.firebirdsql.gds.ng.WarningMessageCallback;
import org.firebirdsql.gds.ng.wire.DeferredAction;
import org.firebirdsql.gds.ng.wire.FbWireDatabase;
import org.firebirdsql.gds.ng.wire.Response;
import org.firebirdsql.gds.ng.wire.version10.V10Statement;
import java.io.IOException;
import java.sql.SQLException;
import java.util.Optional;
/**
* {@link org.firebirdsql.gds.ng.wire.FbWireStatement} implementation for the version 11 wire protocol.
*
* @author Mark Rotteveel
* @since 3.0
*/
public class V11Statement extends V10Statement {
protected AsyncFetchStatus asyncFetchStatus = AsyncFetchStatus.nonePending();
/**
* Creates a new instance of V11Statement for the specified database.
*
* @param database
* FbWireDatabase implementation
*/
public V11Statement(FbWireDatabase database) {
super(database);
}
@Override
public void prepare(final String statementText) throws SQLException {
try (var ignored = withLock()) {
final StatementState initialState = checkPrepareAllowed();
resetAll();
int expectedResponseCount = 0;
try {
if (initialState == StatementState.NEW) {
sendAllocate();
// We're assuming allocation is successful, as changing it when processing the response breaks
// the state transition in sendPrepare
switchState(StatementState.ALLOCATED);
expectedResponseCount++;
} else {
checkStatementValid();
}
sendPrepare(statementText);
expectedResponseCount++;
getXdrOut().flush();
} catch (IOException e) {
switchState(StatementState.ERROR);
throw FbExceptionBuilder.ioWriteError(e);
}
try {
final FbWireDatabase db = getDatabase();
try {
if (initialState == StatementState.NEW) {
try {
expectedResponseCount--;
processAllocateResponse(db.readGenericResponse(getStatementWarningCallback()));
} catch (SQLException e) {
forceState(StatementState.NEW);
throw e;
}
}
try {
expectedResponseCount--;
processPrepareResponse(db.readGenericResponse(getStatementWarningCallback()));
} catch (SQLException e) {
switchState(StatementState.ALLOCATED);
throw e;
}
} finally {
db.consumePackets(expectedResponseCount, getStatementWarningCallback());
}
} catch (IOException e) {
switchState(StatementState.ERROR);
throw FbExceptionBuilder.ioReadError(e);
}
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
@Override
public void fetchRows(int fetchSize) throws SQLException {
try (LockCloseable ignored = withLock()) {
if (!completeAsyncFetch()) {
super.fetchRows(fetchSize);
}
}
}
@Override
public final void asyncFetchRows(int fetchSize) throws SQLException {
try (LockCloseable ignored = withLock()) {
checkStatementHasOpenCursor();
checkFetchSize(fetchSize);
if (isSkipAsyncFetch(fetchSize)) return;
sendAsyncFetch(fetchSize);
asyncFetchStatus = AsyncFetchStatus.pending();
getDatabase().enqueueDeferredAction(wrapDeferredResponse(
new DeferredResponse<>() {
@Override
public void onResponse(AsyncFetchStatus responseStatus) {
asyncFetchStatus = responseStatus;
}
@Override
public void onException(Exception exception) {
SQLException sqlException;
if (exception instanceof SQLException sqle) {
sqlException = sqle;
} else if (exception instanceof IOException ioe) {
sqlException = FbExceptionBuilder.ioReadError(ioe);
} else {
sqlException = new SQLException("Unexpected exception occurred during async fetch",
exception);
}
asyncFetchStatus = AsyncFetchStatus.completedWithException(sqlException);
}
},
this::processAsyncFetchResponse, false));
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
/**
* Conditions for not performing an async fetch.
*
* @param fetchSize
* requested fetch size
* @return {@code true} if async fetch should be skipped, {@code false} if async fetch should be performed
*/
private boolean isSkipAsyncFetch(int fetchSize) {
return isAfterLast()
|| asyncFetchStatus.isPending()
|| fetchSize == 1
|| getCursorName() != null
|| isCursorFlagSet(CursorFlag.CURSOR_TYPE_SCROLLABLE)
|| isAsyncFetchDisabled();
}
private boolean isAsyncFetchDisabled() {
return !getDatabase().getConnectionProperties().isAsyncFetch();
}
private void sendAsyncFetch(int fetchSize) throws SQLException {
try (OperationCloseHandle operationCloseHandle = signalAsyncFetchStart()){
if (operationCloseHandle.isCancelled()) {
// operation was synchronously cancelled from an OperationAware implementation
throw FbExceptionBuilder.toException(ISCConstants.isc_cancelled);
}
sendFetch(fetchSize);
getXdrOut().flush();
} catch (IOException e) {
switchState(StatementState.ERROR);
throw FbExceptionBuilder.ioWriteError(e);
}
}
private AsyncFetchStatus processAsyncFetchResponse(Response initialResponse) {
try (OperationCloseHandle ignored = signalAsyncFetchComplete()){
processFetchResponse(FetchDirection.FORWARD, initialResponse);
return AsyncFetchStatus.completed();
} catch (IOException e) {
try {
switchState(StatementState.ERROR);
} catch (SQLException ignored) {
// ERROR is always a valid transition, so no exception should get thrown
}
return AsyncFetchStatus.completedWithException(FbExceptionBuilder.ioReadError(e));
} catch (SQLException e) {
return AsyncFetchStatus.completedWithException(e);
}
}
/**
* Completes a pending async fetch.
*
* @return {@code true} if a pending async fetch was processed, {@code false} if there was no pending async fetch.
* @throws SQLException
* from unsuccessfully completed async fetch, or for errors try to complete deferred actions
* @since 6
*/
protected boolean completeAsyncFetch() throws SQLException {
try (LockCloseable ignored = withLock()) {
checkStatementValid();
if (!asyncFetchStatus.isPending()) return false;
try {
// Throw exception from previously unsuccessfully completed async fetch
Optional fetchException = asyncFetchStatus.exception();
if (fetchException.isPresent()) {
throw fetchException.get();
}
getDatabase().completeDeferredActions();
// Throw exception from unsuccessfully completed async fetch
fetchException = asyncFetchStatus.exception();
if (fetchException.isPresent()) {
throw fetchException.get();
}
return true;
} finally {
asyncFetchStatus = AsyncFetchStatus.completed();
}
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
@Override
protected void free(final int option) throws SQLException {
try (LockCloseable ignored = withLock()) {
try {
doFreePacket(option);
// Don't flush close of cursor, only flush drop or unprepare of statement. This balances network
// efficiencies with preventing statements retaining locks on metadata objects too long
if (option != ISCConstants.DSQL_close) {
getXdrOut().flush();
}
// process response later
getDatabase().enqueueDeferredAction(new DeferredAction() {
@Override
public void processResponse(Response response) {
processFreeResponse(response);
}
@Override
public WarningMessageCallback getWarningMessageCallback() {
return getStatementWarningCallback();
}
@Override
public boolean requiresSync() {
return option == ISCConstants.DSQL_close;
}
});
} catch (IOException e) {
switchState(StatementState.ERROR);
throw FbExceptionBuilder.ioWriteError(e);
}
}
}
@Override
protected void reset(boolean resetAll) {
try (LockCloseable ignored = withLock()) {
try {
super.reset(resetAll);
} finally {
asyncFetchStatus.exception().ifPresent(e -> System.getLogger(getClass().getName())
.log(System.Logger.Level.TRACE, "Ignored pending async fetch exception during reset", e));
asyncFetchStatus = AsyncFetchStatus.nonePending();
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy