net.snowflake.client.jdbc.SFAsyncResultSet Maven / Gradle / Ivy
/*
* 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);
}
}