All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.snowflake.client.jdbc.SFAsyncResultSet Maven / Gradle / Ivy

There is a newer version: 3.21.0
Show newest version
/*
 * Copyright (c) 2012-2019 Snowflake Computing Inc. All rights reserved.
 */

package net.snowflake.client.jdbc;

import static net.snowflake.client.core.QueryStatus.NO_DATA;

import com.google.api.client.util.Strings;
import java.math.BigDecimal;
import java.sql.Date;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Time;
import java.sql.Timestamp;
import java.util.List;
import java.util.TimeZone;
import net.snowflake.client.core.QueryStatus;
import net.snowflake.client.core.SFBaseResultSet;
import net.snowflake.client.core.SFBaseSession;
import net.snowflake.client.core.SFSession;
import net.snowflake.client.log.SFLogger;
import net.snowflake.client.log.SFLoggerFactory;
import net.snowflake.common.core.SqlState;

/** SFAsyncResultSet implementation. Note: For Snowflake internal use */
public class SFAsyncResultSet extends SnowflakeBaseResultSet
    implements SnowflakeResultSet, ResultSet {
  private static final SFLogger logger = SFLoggerFactory.getLogger(SFAsyncResultSet.class);

  private ResultSet resultSetForNext = new SnowflakeResultSetV1.EmptyResultSet();
  private boolean resultSetForNextInitialized = false;
  private String queryID;
  private SFBaseSession session;
  private Statement extraStatement;
  private QueryStatus lastQueriedStatus = NO_DATA;
  private QueryStatusV2 lastQueriedStatusV2 = QueryStatusV2.empty();

  /**
   * Constructor takes an inputstream from the API response that we get from executing a SQL
   * statement.
   *
   * 

The constructor will fetch the first row (if any) so that it can initialize the * ResultSetMetaData. * * @param sfBaseResultSet snowflake core base result rest object * @param statement query statement that generates this result set * @throws SQLException if failed to construct snowflake result set metadata */ SFAsyncResultSet(SFBaseResultSet sfBaseResultSet, Statement statement) throws SQLException { super(statement); this.queryID = sfBaseResultSet.getQueryId(); this.session = sfBaseResultSet.getSession(); this.extraStatement = statement; this.resultSetMetaData = new SnowflakeResultSetMetaDataV1(sfBaseResultSet.getMetaData()); this.resultSetMetaData.setQueryIdForAsyncResults(this.queryID); this.resultSetMetaData.setQueryType(SnowflakeResultSetMetaDataV1.QueryType.ASYNC); } /** * Constructor takes a result set serializable object to create a sessionless result set. * * @param sfBaseResultSet snowflake core base result rest object * @param resultSetSerializable The result set serializable object which includes all metadata to * create the result set * @throws SQLException if fails to create the result set object */ public SFAsyncResultSet( SFBaseResultSet sfBaseResultSet, SnowflakeResultSetSerializableV1 resultSetSerializable) throws SQLException { super(resultSetSerializable); this.queryID = sfBaseResultSet.getQueryId(); this.resultSetMetaData = new SnowflakeResultSetMetaDataV1(sfBaseResultSet.getMetaData()); this.resultSetMetaData.setQueryIdForAsyncResults(this.queryID); this.resultSetMetaData.setQueryType(SnowflakeResultSetMetaDataV1.QueryType.ASYNC); } public SFAsyncResultSet(String queryID, Statement statement) throws SQLException { super(statement); queryID.trim(); if (!QueryIdValidator.isValid(queryID)) { throw new SQLException( "The provided query ID " + queryID + " is invalid.", SqlState.INVALID_PARAMETER_VALUE); } this.queryID = queryID; } @Override protected void raiseSQLExceptionIfResultSetIsClosed() throws SQLException { if (isClosed()) { throw new SnowflakeSQLException(ErrorCode.RESULTSET_ALREADY_CLOSED); } } @Override public QueryStatus getStatus() throws SQLException { if (session == null) { throw new SQLException("Session not set"); } if (this.queryID == null) { throw new SQLException("QueryID unknown"); } if (this.lastQueriedStatus == QueryStatus.SUCCESS) { return this.lastQueriedStatus; } // if query has completed successfully, cache its success status to avoid unnecessary future // server calls this.lastQueriedStatus = session.getQueryStatus(this.queryID); return this.lastQueriedStatus; } @Override public String getQueryErrorMessage() throws SQLException { return this.lastQueriedStatus.getErrorMessage(); } @Override public QueryStatusV2 getStatusV2() throws SQLException { if (session == null) { throw new SQLException("Session not set"); } if (this.queryID == null) { throw new SQLException("QueryID unknown"); } if (this.lastQueriedStatusV2.isSuccess()) { return this.lastQueriedStatusV2; } this.lastQueriedStatusV2 = session.getQueryStatusV2(this.queryID); // if query has completed successfully, cache its metadata to avoid unnecessary future server // calls return this.lastQueriedStatusV2; } /** * helper function for next() and getMetaData(). Calls result_scan to get resultSet after * asynchronous query call * * @throws SQLException */ private void getRealResults() throws SQLException { if (!resultSetForNextInitialized) { // If query has already succeeded, go straight to result scan to get results if (this.lastQueriedStatus != QueryStatus.SUCCESS) { QueryStatus qs = this.getStatus(); int noDataRetry = 0; final int noDataMaxRetries = 30; final int[] retryPattern = {1, 1, 2, 3, 4, 8, 10}; final int maxIndex = retryPattern.length - 1; int retry = 0; while (qs != QueryStatus.SUCCESS) { // if query is not running due to a failure (Aborted, failed with error, etc), generate // exception if (!QueryStatus.isStillRunning(qs) && qs.getValue() != QueryStatus.SUCCESS.getValue()) { String errorMessage = qs.getErrorMessage(); if (Strings.isNullOrEmpty(errorMessage)) { errorMessage = "No error message available"; } throw new SQLException( "Status of query associated with resultSet is " + qs.getDescription() + ". " + errorMessage + " Results not generated."); } // if no data about the query is returned after about 2 minutes, give up if (qs == NO_DATA) { noDataRetry++; if (noDataRetry >= noDataMaxRetries) { throw new SQLException( "Cannot retrieve data on the status of this query. No information returned from server for queryID={}.", this.queryID); } } try { // Sleep for an amount before trying again. Exponential backoff up to 5 seconds // implemented. Thread.sleep(500 * retryPattern[retry]); } catch (InterruptedException e) { e.printStackTrace(); } if (retry < maxIndex) { retry++; } qs = this.getStatus(); } } resultSetForNext = extraStatement.executeQuery("select * from table(result_scan('" + this.queryID + "'))"); resultSetForNextInitialized = true; } } /** * Advance to next row * * @return true if next row exists, false otherwise * @throws SQLException if failed to move to the next row */ @Override public boolean next() throws SQLException { getMetaData(); return resultSetForNext.next(); } @Override public void close() throws SQLException { close(true); } public void close(boolean removeClosedResultSetFromStatement) throws SQLException { // no SQLException is raised. resultSetForNext.close(); if (sfBaseResultSet != null) { sfBaseResultSet.close(); } if (removeClosedResultSetFromStatement && statement.isWrapperFor(SnowflakeStatementV1.class)) { statement.unwrap(SnowflakeStatementV1.class).removeClosedResultSet(this); } } public String getQueryID() { return this.queryID; } public boolean wasNull() throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.wasNull(); } public String getString(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getString(columnIndex); } public void setSession(SFSession session) { this.session = session; } public void setStatement(Statement statement) { this.extraStatement = statement; } public boolean getBoolean(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getBoolean(columnIndex); } @Override public byte getByte(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getByte(columnIndex); } public short getShort(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getShort(columnIndex); } public int getInt(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getInt(columnIndex); } public long getLong(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getLong(columnIndex); } public float getFloat(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getFloat(columnIndex); } public double getDouble(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getDouble(columnIndex); } public Date getDate(int columnIndex, TimeZone tz) throws SQLException { // Note: currently we provide this API but it does not use TimeZone tz. // TODO: use the time zone passed from the arguments raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getDate(columnIndex); } public Time getTime(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getTime(columnIndex); } public Timestamp getTimestamp(int columnIndex, TimeZone tz) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.unwrap(SnowflakeResultSetV1.class).getTimestamp(columnIndex, tz); } public ResultSetMetaData getMetaData() throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); getRealResults(); this.resultSetMetaData = (SnowflakeResultSetMetaDataV1) resultSetForNext.unwrap(SnowflakeResultSetV1.class).getMetaData(); this.resultSetMetaData.setQueryIdForAsyncResults(this.queryID); this.resultSetMetaData.setQueryType(SnowflakeResultSetMetaDataV1.QueryType.ASYNC); return resultSetMetaData; } public Object getObject(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getObject(columnIndex); } public BigDecimal getBigDecimal(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getBigDecimal(columnIndex); } @Deprecated public BigDecimal getBigDecimal(int columnIndex, int scale) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getBigDecimal(columnIndex, scale); } public byte[] getBytes(int columnIndex) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getBytes(columnIndex); } public int getRow() throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.getRow(); } public boolean isFirst() throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.isFirst(); } public boolean isClosed() throws SQLException { // no exception is raised. if (sfBaseResultSet != null) { return (resultSetForNext.isClosed() && sfBaseResultSet.isClosed()); } return resultSetForNext.isClosed(); } @Override public boolean isLast() throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.isLast(); } @Override public boolean isAfterLast() throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); return resultSetForNext.isAfterLast(); } @Override public boolean isBeforeFirst() throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); // if ResultSet is not initialized yet, this means neither next() nor getMetaData() has been // called. // If next() hasn't been called, we are at the beginning of the ResultSet so should return true. return !resultSetForNextInitialized || resultSetForNext.isBeforeFirst(); } @Override public boolean isWrapperFor(Class iface) throws SQLException { logger.trace("boolean isWrapperFor(Class iface)", false); return iface.isInstance(this); } @SuppressWarnings("unchecked") @Override public T unwrap(Class iface) throws SQLException { logger.trace(" T unwrap(Class iface)", false); if (!iface.isInstance(this)) { throw new SQLException( this.getClass().getName() + " not unwrappable from " + iface.getName()); } return (T) this; } /** * Get a list of ResultSetSerializables for the ResultSet in order to parallel processing * *

Not currently supported for asynchronous result sets. */ @Override public List getResultSetSerializables(long maxSizeInBytes) throws SQLException { raiseSQLExceptionIfResultSetIsClosed(); getRealResults(); return resultSetForNext .unwrap(SnowflakeResultSet.class) .getResultSetSerializables(maxSizeInBytes); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy