org.firebirdsql.gds.ng.AbstractFbStatement Maven / Gradle / Ivy
Show all versions of jaybird Show documentation
/*
* 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;
import org.firebirdsql.gds.ISCConstants;
import org.firebirdsql.gds.JaybirdErrorCodes;
import org.firebirdsql.gds.ng.fields.RowDescriptor;
import org.firebirdsql.gds.ng.fields.RowValue;
import org.firebirdsql.gds.ng.listeners.*;
import org.firebirdsql.jdbc.FBDriverNotCapableException;
import java.sql.SQLException;
import java.sql.SQLNonTransientException;
import java.util.Collections;
import java.util.EnumSet;
import java.util.Set;
import static java.lang.String.format;
import static java.lang.System.Logger.Level.DEBUG;
import static org.firebirdsql.gds.ng.TransactionHelper.checkTransactionActive;
/**
* @author Mark Rotteveel
* @since 3.0
*/
public abstract class AbstractFbStatement implements FbStatement {
private static final long MAX_STATEMENT_TIMEOUT = 0xFF_FF_FF_FFL;
/**
* Set of states that will be reset to {@link StatementState#PREPARED} on transaction change
*/
private static final Set RESET_TO_PREPARED = Collections.unmodifiableSet(
EnumSet.of(StatementState.EXECUTING, StatementState.CURSOR_OPEN));
private static final System.Logger log = System.getLogger(AbstractFbStatement.class.getName());
// NOTE: BEFORE_FIRST is also used for statements that don't produce rows
private static final int BEFORE_FIRST = -1;
private static final int IN_CURSOR = 0;
private static final int AFTER_LAST = 1;
protected final StatementListenerDispatcher statementListenerDispatcher = new StatementListenerDispatcher();
private final WarningMessageCallback warningCallback = warning ->
statementListenerDispatcher.warningReceived(AbstractFbStatement.this, warning);
protected final ExceptionListenerDispatcher exceptionListenerDispatcher = new ExceptionListenerDispatcher(this);
private volatile int cursorPosition = BEFORE_FIRST;
// Indicates whether at least one fetch was done for the current cursor
private boolean fetched;
private volatile StatementState state = StatementState.NEW;
private volatile StatementType type = StatementType.NONE;
@SuppressWarnings("java:S3077")
private volatile RowDescriptor parameterDescriptor;
@SuppressWarnings("java:S3077")
private volatile RowDescriptor fieldDescriptor;
@SuppressWarnings("java:S3077")
private volatile FbTransaction transaction;
private String cursorName;
private long timeout;
private final TransactionListener transactionListener = new TransactionListener() {
@Override
@SuppressWarnings({ "ThrowFromFinallyBlock", "java:S1143", "java:S1163" })
public void transactionStateChanged(FbTransaction transaction, TransactionState newState,
TransactionState previousState) {
if (getTransaction() != transaction) {
transaction.removeTransactionListener(this);
return;
}
if (newState == TransactionState.COMMITTED || newState == TransactionState.ROLLED_BACK) {
try (LockCloseable ignored = withLock()) {
try {
if (RESET_TO_PREPARED.contains(getState())) {
// Cursor has been closed due to commit, rollback, etc., back to prepared state
try {
switchState(StatementState.PREPARED);
} catch (SQLException e) {
throw new IllegalStateException("Received an SQLException when none was expected", e);
}
reset(false);
}
} finally {
transaction.removeTransactionListener(this);
try {
setTransaction(null);
} catch (SQLException e) {
throw new IllegalStateException("Received an SQLException when none was expected", e);
}
}
}
}
}
};
protected AbstractFbStatement() {
exceptionListenerDispatcher.addListener(new StatementCancelledListener());
statementListenerDispatcher.addListener(new SelfListener());
}
/**
* Gets the {@link TransactionListener} instance for this statement.
*
* This method should only be called by this object itself. Subclasses may provide their own transaction listener,
* but the instance returned by this method should be the same for the lifetime of this {@link FbStatement}.
*
*
* @return The transaction listener instance for this statement.
*/
protected final TransactionListener getTransactionListener() {
return transactionListener;
}
protected final WarningMessageCallback getStatementWarningCallback() {
return warningCallback;
}
@Override
public void asyncFetchRows(int fetchSize) throws SQLException {
try (LockCloseable ignored = withLock()) {
checkStatementValid();
// Async fetch not supported, doing nothing (see javadoc in FbStatement)
}
}
@Override
public final boolean hasFetched() {
return fetched;
}
/**
* Validates if {@code fetchSize} is positive, otherwise throws an exception.
*
* @param fetchSize fetch size (must be positive)
* @throws SQLException if {@code fetchSize} is smaller than {@code 1}
* @since 6
*/
protected final void checkFetchSize(int fetchSize) throws SQLException {
if (fetchSize > 0) return;
throw FbExceptionBuilder.forException(JaybirdErrorCodes.jb_invalidFetchSize)
.messageParameter(fetchSize)
.toSQLException();
}
@Override
public void close() throws SQLException {
if (getState() == StatementState.CLOSED) return;
try (LockCloseable ignored = withLock()) {
try {
final StatementState currentState = getState();
forceState(StatementState.CLOSING);
if (currentState != StatementState.NEW) {
free(ISCConstants.DSQL_drop);
}
} finally {
forceState(StatementState.CLOSED);
setType(StatementType.NONE);
statementListenerDispatcher.shutdown();
setTransaction(null);
}
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
} finally {
exceptionListenerDispatcher.shutdown();
}
}
@Override
public final void closeCursor() throws SQLException {
closeCursor(false);
}
@Override
@SuppressWarnings("java:S1141")
public final void closeCursor(boolean transactionEnd) throws SQLException {
try (LockCloseable ignored = withLock()) {
if (!getState().isCursorOpen()) return;
try {
if (!transactionEnd && getType().isTypeWithCursor()) {
free(ISCConstants.DSQL_close);
}
// TODO Any statement types that cannot be prepared and would need to go to ALLOCATED?
switchState(StatementState.PREPARED);
} catch (SQLException e) {
// TODO Close in case of exception?
switchState(StatementState.ERROR);
throw e;
}
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
@Override
public final void ensureClosedCursor(boolean transactionEnd) throws SQLException {
if (getState().isCursorOpen()) {
closeCursor(transactionEnd);
}
}
@Override
public void unprepare() throws SQLException {
if (getDatabase().getServerVersion().isEqualOrAbove(2, 5)) {
try (LockCloseable ignored = withLock()) {
StatementState currentState = getState();
// Cannot unprepare if NEW, and unpreparing if ALLOCATED makes no sense
if (!(currentState == StatementState.NEW || currentState == StatementState.ALLOCATED)) {
// TODO This throws an exception if a transition to ALLOCATED is not allowed, consider if this
// is desired, or if checking validity of transition and only logging a warning is better
switchState(StatementState.ALLOCATED);
free(ISCConstants.DSQL_unprepare);
}
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
} else {
closeCursor();
}
}
@Override
public final StatementState getState() {
return state;
}
/**
* Sets the StatementState.
*
* @param newState
* New state
* @throws SQLException
* When the state is changed to an illegal next state
*/
protected final void switchState(final StatementState newState) throws SQLException {
try (LockCloseable ignored = withLock()) {
final StatementState currentState = state;
if (currentState == newState || currentState == StatementState.CLOSED) return;
if (currentState.isValidTransition(newState)) {
state = newState;
statementListenerDispatcher.statementStateChanged(this, newState, currentState);
} else {
throw new SQLNonTransientException(format("Statement state %s only allows next states %s, received %s",
currentState, currentState.validTransitionSet(), newState));
}
}
}
/**
* Forces the statement to the specified state without throwing an exception if this is not a valid transition.
*
* Does nothing if current state is CLOSED.
*
*
* @param newState
* New state
* @see #switchState(StatementState)
*/
protected void forceState(final StatementState newState) {
try (LockCloseable ignored = withLock()) {
final StatementState currentState = state;
if (currentState == newState || currentState == StatementState.CLOSED) return;
if (!currentState.isValidTransition(newState) && log.isLoggable(DEBUG)) {
log.log(DEBUG, "Forced statement transition is invalid; state %s only allows next states %s, forced to set %s"
.formatted(currentState, currentState.validTransitionSet(), newState),
new IllegalStateException("debugging stacktrace"));
}
state = newState;
statementListenerDispatcher.statementStateChanged(this, newState, currentState);
}
}
@Override
public final StatementType getType() {
return type;
}
/**
* Sets the StatementType
*
* @param type
* New type
*/
protected void setType(StatementType type) {
this.type = type;
}
/**
* Queues row data for consumption
*
* @param rowData
* Row data
*/
protected final void queueRowData(RowValue rowData) {
cursorPosition = IN_CURSOR;
statementListenerDispatcher.receivedRow(this, rowData);
}
/**
* Marks the cursor position as before-first.
*
* All registered {@link org.firebirdsql.gds.ng.listeners.StatementListener} instances are notified for
* the {@link org.firebirdsql.gds.ng.listeners.StatementListener#beforeFirst(FbStatement)} event.
*
*/
protected final void setBeforeFirst() {
setBeforeFirst(true);
}
private void setBeforeFirst(boolean notify) {
cursorPosition = BEFORE_FIRST;
if (notify) {
statementListenerDispatcher.beforeFirst(this);
}
}
protected final boolean isBeforeFirst() {
return cursorPosition == BEFORE_FIRST;
}
/**
* Marks the cursor position as after-last.
*
* All registered {@link org.firebirdsql.gds.ng.listeners.StatementListener} instances are notified for
* the {@link org.firebirdsql.gds.ng.listeners.StatementListener#afterLast(FbStatement)} event.
*
*/
protected final void setAfterLast() {
cursorPosition = AFTER_LAST;
statementListenerDispatcher.afterLast(this);
}
protected final boolean isAfterLast() {
return cursorPosition == AFTER_LAST;
}
/**
* Reset statement state, equivalent to calling {@link #reset(boolean)} with {@code false}.
*/
protected final void reset() {
reset(false);
}
/**
* Reset statement state and clear parameter description, equivalent to calling {@link #reset(boolean)} with
* {@code true}.
*/
protected final void resetAll() {
reset(true);
}
/**
* Resets the statement for next execution. Implementation in derived class must lock on
* {@link #withLock()} and call {@code super.reset(resetAll)}
*
* @param resetAll
* Also reset field and parameter info
*/
protected void reset(boolean resetAll) {
try (LockCloseable ignored = withLock()) {
StatementType statementType = getType();
// Don't notify for non result-set types, but ensure the cursor position value is before-first
setBeforeFirst(statementType.isTypeWithCursor() || statementType.isTypeWithSingletonResult());
if (resetAll) {
setParameterDescriptor(null);
setRowDescriptor(null);
setType(StatementType.NONE);
}
}
}
private static final Set PREPARE_ALLOWED_STATES = Collections.unmodifiableSet(
EnumSet.of(StatementState.NEW, StatementState.ALLOCATED, StatementState.PREPARED));
/**
* Is a call to {@link #prepare(String)} allowed for the supplied {@link StatementState}.
*
* @param state
* The statement state
* @return {@code true} call to {@code prepare} is allowed
*/
protected boolean isPrepareAllowed(final StatementState state) {
return PREPARE_ALLOWED_STATES.contains(state);
}
/**
* Checks if prepare is allowed, returning the current statement state.
*
* This method checks if the current transaction is active, and if the current statement state allows preparing
* a new statement.
*
*
* @return statement state
* @throws SQLException
* if there is no active transaction, or the current state does not allow statement prepare
* @since 6
*/
protected final StatementState checkPrepareAllowed() throws SQLException {
checkTransactionActive(getTransaction());
final StatementState initialState = getState();
if (!isPrepareAllowed(initialState)) {
throw FbExceptionBuilder.forNonTransientException(JaybirdErrorCodes.jb_prepareNotAllowedByState)
.messageParameter(initialState)
.toSQLException();
}
return initialState;
}
@Override
public final RowDescriptor getParameterDescriptor() {
return parameterDescriptor;
}
/**
* Sets the parameter descriptor.
*
* @param parameterDescriptor
* Parameter descriptor
*/
protected void setParameterDescriptor(RowDescriptor parameterDescriptor) {
this.parameterDescriptor = parameterDescriptor;
}
@Override
public final RowDescriptor getRowDescriptor() {
return fieldDescriptor;
}
/**
* Sets the (result set) row descriptor.
*
* @param rowDescriptor
* Row descriptor
*/
protected void setRowDescriptor(RowDescriptor rowDescriptor) {
this.fieldDescriptor = rowDescriptor;
}
/**
* @return The (full) statement info request items.
* @see #getParameterDescriptionInfoRequestItems()
*/
public byte[] getStatementInfoRequestItems() {
return ((AbstractFbDatabase>) getDatabase()).getStatementInfoRequestItems();
}
/**
* @return The {@code isc_info_sql_describe_vars} info request items.
* @see #getStatementInfoRequestItems()
*/
public byte[] getParameterDescriptionInfoRequestItems() {
return ((AbstractFbDatabase>) getDatabase()).getParameterDescriptionInfoRequestItems();
}
@Override
public final void fetchScroll(FetchType fetchType, int fetchSize, int position) throws SQLException {
if (fetchType == FetchType.NEXT) {
fetchRows(fetchSize);
return;
}
try {
fetchScrollImpl(fetchType, fetchSize, position);
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
/**
* Implementation of {@link #fetchScroll(FetchType, int, int)}.
*
* An implementation should not notify {@code exceptionListenerDispatcher}, as that is already handled in
* {@link #fetchScroll(FetchType, int, int)}.
*
*
* The implementation of {@link #fetchScroll(FetchType, int, int)} redirects {@link FetchType#NEXT} to
* {@link #fetchRows(int)}. The implementation does need to handle {@code NEXT}, but only when actually implementing
* the other scroll direction.
*
*
* @throws java.sql.SQLFeatureNotSupportedException
* If the protocol version or the implementation does not support scroll fetch (even for {@code NEXT})
* @throws SQLException
* For database access errors, when called on a closed statement, when no cursor is open, or for serverside
* error conditions
* @see #fetchScroll(FetchType, int, int)
* @see #supportsFetchScroll()
*/
protected void fetchScrollImpl(FetchType fetchType, int fetchSize, int position) throws SQLException {
throw new FBDriverNotCapableException("implementation does not support fetchScroll");
}
/**
* Request statement info.
*
* @param requestItems
* Array of info items to request
* @param bufferLength
* Response buffer length to use
* @param infoProcessor
* Implementation of {@link InfoProcessor} to transform
* the info response
* @return Transformed info response of type T
* @throws SQLException
* For errors retrieving or transforming the response.
*/
@Override
public final T getSqlInfo(final byte[] requestItems, final int bufferLength,
final InfoProcessor infoProcessor) throws SQLException {
final byte[] sqlInfo = getSqlInfo(requestItems, bufferLength);
try {
return infoProcessor.process(sqlInfo);
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
@Override
public final T getCursorInfo(byte[] requestItems, int bufferLength, InfoProcessor infoProcessor)
throws SQLException {
final byte[] sqlInfo = getCursorInfo(requestItems, bufferLength);
try {
return infoProcessor.process(sqlInfo);
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
@Override
public final byte[] getCursorInfo(byte[] requestItems, int bufferLength) throws SQLException {
try {
checkStatementValid();
return getCursorInfoImpl(requestItems, bufferLength);
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
/**
* Implementation of {@link #getCursorInfo(byte[], int)}.
*
* An implementation should not notify {@code exceptionListenerDispatcher}, as that is already handled in
* {@link #getCursorInfo(byte[], int)}.
*
*
* @throws SQLException
* For errors retrieving or transforming the response
* @throws java.sql.SQLFeatureNotSupportedException
* If requesting cursor info is not supported (Firebird 4.0 or earlier, or native implementation)
* @see #getCursorInfo(byte[], int)
* @see #supportsCursorInfo()
*/
protected byte[] getCursorInfoImpl(byte[] requestItems, int bufferLength) throws SQLException {
throw new FBDriverNotCapableException("implementation does not support getCursorInfo: " + getClass());
}
@Override
public final String getExecutionPlan() throws SQLException {
final ExecutionPlanProcessor processor = createExecutionPlanProcessor();
return getSqlInfo(processor.getDescribePlanInfoItems(), getDefaultSqlInfoSize(), processor);
}
@Override
public final String getExplainedExecutionPlan() throws SQLException {
try (LockCloseable ignored = withLock()) {
checkExplainedExecutionPlanSupport();
final ExecutionPlanProcessor processor = createExecutionPlanProcessor();
return getSqlInfo(processor.getDescribeExplainedPlanInfoItems(), getDefaultSqlInfoSize(), processor);
}
}
private void checkExplainedExecutionPlanSupport() throws SQLException {
try {
checkStatementValid();
if (!getDatabase().getServerVersion().isEqualOrAbove(3, 0)) {
throw FbExceptionBuilder.toException(JaybirdErrorCodes.jb_explainedExecutionPlanNotSupported);
}
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
/**
* @return New instance of {@link ExecutionPlanProcessor} (or subclass) for this statement.
*/
protected ExecutionPlanProcessor createExecutionPlanProcessor() {
return new ExecutionPlanProcessor(this);
}
@Override
public SqlCountHolder getSqlCounts() throws SQLException {
try {
checkStatementValid();
if (getState() == StatementState.CURSOR_OPEN && !isAfterLast()) {
// We disallow fetching count when we haven't fetched all rows yet.
throw FbExceptionBuilder.toNonTransientException(JaybirdErrorCodes.jb_closeCursorBeforeCount);
}
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
final SqlCountProcessor countProcessor = createSqlCountProcessor();
// NOTE: implementation of SqlCountProcessor assumes the specified size is sufficient (actual requirement is 49 bytes max) and does not handle truncation
final SqlCountHolder sqlCounts = getSqlInfo(countProcessor.getRecordCountInfoItems(), 64, countProcessor);
statementListenerDispatcher.sqlCounts(this, sqlCounts);
return sqlCounts;
}
/**
* @return New instance of {@link SqlCountProcessor} (or subclass) for this statement.
*/
protected SqlCountProcessor createSqlCountProcessor() {
return new SqlCountProcessor();
}
/**
* Frees the currently allocated statement. Either close the cursor with {@link ISCConstants#DSQL_close} or drop
* the statement handle using {@link ISCConstants#DSQL_drop}.
*
* @param option
* Free option
*/
protected abstract void free(int option) throws SQLException;
@Override
public final void validateParameters(final RowValue parameters) throws SQLException {
final RowDescriptor parameterDescriptor = getParameterDescriptor();
final int expectedSize = parameterDescriptor != null ? parameterDescriptor.getCount() : 0;
final int actualSize = parameters.getCount();
if (actualSize != expectedSize) {
throw FbExceptionBuilder.forNonTransientException(JaybirdErrorCodes.jb_invalidParameterCount)
.messageParameter(expectedSize, actualSize)
.toSQLException();
}
for (int fieldIndex = 0; fieldIndex < actualSize; fieldIndex++) {
if (!parameters.isInitialized(fieldIndex)) {
// Communicating 1-based index, so it doesn't cause confusion when JDBC user sees this.
throw FbExceptionBuilder.forTransientException(JaybirdErrorCodes.jb_parameterNotSet)
.messageParameter(fieldIndex + 1)
.toSQLException();
}
}
}
@Override
public final void addStatementListener(StatementListener statementListener) {
if (getState() == StatementState.CLOSED) return;
statementListenerDispatcher.addListener(statementListener);
}
@Override
public void addWeakStatementListener(StatementListener statementListener) {
if (getState() == StatementState.CLOSED) return;
statementListenerDispatcher.addWeakListener(statementListener);
}
@Override
public final void removeStatementListener(StatementListener statementListener) {
statementListenerDispatcher.removeListener(statementListener);
}
@Override
public final void addExceptionListener(ExceptionListener listener) {
exceptionListenerDispatcher.addListener(listener);
}
@Override
public final void removeExceptionListener(ExceptionListener listener) {
exceptionListenerDispatcher.removeListener(listener);
}
/**
* Checks if this statement is not in {@link StatementState#CLOSED}, {@link StatementState#CLOSING},
* {@link StatementState#NEW} or {@link StatementState#ERROR}, and throws an {@code SQLException} if it is.
*
* @throws SQLException
* when this statement is closed or in error state
*/
protected final void checkStatementValid() throws SQLException {
int errorCode;
switch (getState()) {
case NEW -> errorCode = JaybirdErrorCodes.jb_stmtNotAllocated;
case CLOSING, CLOSED -> errorCode = JaybirdErrorCodes.jb_stmtClosed;
case ERROR -> errorCode = JaybirdErrorCodes.jb_stmtInErrorRequireClose;
default -> {
// Valid state, continue
return;
}
}
throw FbExceptionBuilder.toNonTransientException(errorCode);
}
/**
* Performs the same check as {@link #checkStatementValid()}, but considers {@code ignoreState} as valid.
*
* @param ignoreState
* the invalid state (see {@link #checkStatementValid()} to ignore
* @throws SQLException
* when this statement is closed or in error state
*/
protected final void checkStatementValid(StatementState ignoreState) throws SQLException {
if (ignoreState != getState()) {
checkStatementValid();
}
}
/**
* Checks if statement is valid with {@link #checkStatementValid()} and then checks if the current statement state
* has an open cursor.
*
* @throws SQLException
* when this statement is closed or in error state, or has no open cursor
* @since 6
*/
protected final void checkStatementHasOpenCursor() throws SQLException {
checkStatementValid();
if (!getState().isCursorOpen()) {
throw FbExceptionBuilder.toException(ISCConstants.isc_cursor_not_open);
}
}
@Override
public FbTransaction getTransaction() {
return transaction;
}
/**
* Method to decide if a transaction implementation class is valid for the statement implementation.
*
* Eg a {@link org.firebirdsql.gds.ng.wire.version10.V10Statement} will only work with an
* {@link org.firebirdsql.gds.ng.wire.FbWireTransaction} implementation.
*
*
* @param transactionClass
* Class of the transaction
* @return {@code true} when the transaction class is valid for the statement implementation.
*/
protected abstract boolean isValidTransactionClass(Class extends FbTransaction> transactionClass);
@Override
public final void setTransaction(final FbTransaction newTransaction) throws SQLException {
// TODO Should we really notify the exception listener for errors here?
try {
if (newTransaction == null || isValidTransactionClass(newTransaction.getClass())) {
// TODO Is there a statement or transaction state where we should not be switching transactions?
// Probably an error to switch when newTransaction is not null and current state is ERROR, CURSOR_OPEN, EXECUTING, CLOSING or CLOSED
try (LockCloseable ignored = withLock()) {
if (newTransaction == transaction) return;
if (transaction != null) {
transaction.removeTransactionListener(getTransactionListener());
}
transaction = newTransaction;
if (newTransaction != null) {
newTransaction.addTransactionListener(getTransactionListener());
}
}
} else {
throw FbExceptionBuilder.forNonTransientException(JaybirdErrorCodes.jb_invalidTransactionHandleType)
.messageParameter(newTransaction.getClass())
.toSQLException();
}
} catch (SQLNonTransientException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
@Override
public void setTimeout(long statementTimeout) throws SQLException {
try {
if (statementTimeout < 0) {
throw FbExceptionBuilder.toNonTransientException(JaybirdErrorCodes.jb_invalidTimeout);
}
try (LockCloseable ignored = withLock()) {
checkStatementValid(StatementState.NEW);
this.timeout = statementTimeout;
}
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
@Override
public long getTimeout() throws SQLException {
try (LockCloseable ignored = withLock()) {
checkStatementValid(StatementState.NEW);
return timeout;
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
/**
* {@inheritDoc}
*
* This method takes out a lock, and checks statement validity, then calls {@link #setCursorNameImpl(String)},
* and stores the cursor name on successful completion. Any exceptions will be notified on
* {@code exceptionListenerDispatcher}. To override the behaviour of this method, implement/override
* {@link #setCursorNameImpl(String)}.
*
*/
@Override
public final void setCursorName(String cursorName) throws SQLException {
try (LockCloseable ignored = withLock()) {
checkStatementValid();
setCursorNameImpl(cursorName);
this.cursorName = cursorName;
} catch (SQLException e) {
exceptionListenerDispatcher.errorOccurred(e);
throw e;
}
}
/**
* Implementation of {@link #setCursorName(String)}.
*
* The caller of this method will take out the lock, check statement validity and call
* {@code exceptionListenerDispatcher} for exceptions, so implementations of this method do not need to do so.
*
*
* @param cursorName
* Name of the cursor
* @throws SQLException
* If this statement is closed, or if the cursor name is set and {@code cursorName} is different from the
* current cursor name
*/
protected abstract void setCursorNameImpl(String cursorName) throws SQLException;
/**
* Gets the cursor name.
*
* The cursor name is cleared by a new statement prepare.
*
*
* @return the current cursor name
* @since 6
*/
protected final String getCursorName() {
return cursorName;
}
/**
* @return The timeout value, or {@code 0} if the timeout is larger than supported
* @throws SQLException
* If the statement is invalid
*/
protected long getAllowedTimeout() throws SQLException {
long timeout = getTimeout();
if (timeout > MAX_STATEMENT_TIMEOUT) {
return 0;
}
return timeout;
}
/**
* Parse the statement info response in {@code statementInfoResponse}. If the response is truncated, a new
* request is done using {@link #getStatementInfoRequestItems()}
*
* @param statementInfoResponse
* Statement info response
*/
protected void parseStatementInfo(final byte[] statementInfoResponse) throws SQLException {
final StatementInfoProcessor infoProcessor = new StatementInfoProcessor(this, this.getDatabase());
InfoProcessor.StatementInfo statementInfo = infoProcessor.process(statementInfoResponse);
setType(statementInfo.getStatementType());
setRowDescriptor(statementInfo.getFields());
setParameterDescriptor(statementInfo.getParameters());
}
/**
* @return {@code true} if this is a stored procedure (or other singleton result producing statement) with at least 1 output field
*/
protected final boolean hasSingletonResult() {
return getType().isTypeWithSingletonResult() && hasFields();
}
/**
* @return {@code true} if this statement has at least one output field (either singleton or result set)
*/
protected final boolean hasFields() {
RowDescriptor fieldDescriptor = getRowDescriptor();
return fieldDescriptor != null && fieldDescriptor.getCount() > 0;
}
/**
* Signals the start of an execute for this statement.
*
* @return {@code OperationCloseHandle} handle for the operation
*/
protected final OperationCloseHandle signalExecute() {
return FbDatabaseOperation.signalExecute(getDatabase());
}
/**
* Signals the start of a fetch for this statement.
*
* @return {@code OperationCloseHandle} handle for the operation
*/
protected final OperationCloseHandle signalFetch() {
return FbDatabaseOperation.signalFetch(getDatabase(), this::fetchExecuted);
}
protected final OperationCloseHandle signalAsyncFetchStart() {
return FbDatabaseOperation.signalAsyncFetchStart(getDatabase(), this::fetchExecuted);
}
protected final OperationCloseHandle signalAsyncFetchComplete() {
return FbDatabaseOperation.signalAsyncFetchComplete(getDatabase());
}
private void fetchExecuted() {
fetched = true;
}
/**
* Listener to reset the statement state when it has been cancelled due to statement timeout.
*/
private final class StatementCancelledListener implements ExceptionListener {
@SuppressWarnings("java:S1301")
@Override
public void errorOccurred(Object source, SQLException ex) {
if (source != AbstractFbStatement.this) {
return;
}
switch (ex.getErrorCode()) {
// Close cursor so statement can be reused
case ISCConstants.isc_cfg_stmt_timeout, ISCConstants.isc_att_stmt_timeout,
ISCConstants.isc_req_stmt_timeout -> {
try {
closeCursor();
} catch (SQLException e) {
log.log(System.Logger.Level.ERROR, "Unable to close cursor after statement timeout", e);
}
}
default -> {
// do nothing
}
}
}
}
/**
* Listener that allows a statement to listen to itself, so it can react to its own actions or state transitions.
*/
private final class SelfListener implements StatementListener {
@Override
public void statementStateChanged(FbStatement sender, StatementState newState, StatementState previousState) {
// Any statement state change indicates existing 'fetched' information is no longer valid
fetched = false;
if (newState == StatementState.PREPARING) {
cursorName = null;
}
}
}
}