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

net.snowflake.client.core.SFStatement 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.core;

import static net.snowflake.client.core.SessionUtil.DEFAULT_CLIENT_MEMORY_LIMIT;
import static net.snowflake.client.core.SessionUtil.DEFAULT_CLIENT_PREFETCH_THREADS;
import static net.snowflake.client.core.SessionUtil.MAX_CLIENT_CHUNK_SIZE;
import static net.snowflake.client.core.SessionUtil.MIN_CLIENT_CHUNK_SIZE;
import static net.snowflake.client.jdbc.SnowflakeUtil.systemGetProperty;

import com.fasterxml.jackson.databind.JsonNode;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.concurrent.Callable;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import net.snowflake.client.core.BasicEvent.QueryState;
import net.snowflake.client.core.bind.BindException;
import net.snowflake.client.core.bind.BindUploader;
import net.snowflake.client.jdbc.ErrorCode;
import net.snowflake.client.jdbc.QueryStatusV2;
import net.snowflake.client.jdbc.SnowflakeDriver;
import net.snowflake.client.jdbc.SnowflakeFileTransferAgent;
import net.snowflake.client.jdbc.SnowflakeReauthenticationRequest;
import net.snowflake.client.jdbc.SnowflakeSQLException;
import net.snowflake.client.jdbc.SnowflakeSQLLoggedException;
import net.snowflake.client.jdbc.telemetry.TelemetryData;
import net.snowflake.client.jdbc.telemetry.TelemetryField;
import net.snowflake.client.jdbc.telemetry.TelemetryUtil;
import net.snowflake.client.jdbc.telemetryOOB.TelemetryService;
import net.snowflake.client.log.SFLogger;
import net.snowflake.client.log.SFLoggerFactory;
import net.snowflake.common.core.SqlState;
import org.apache.http.client.methods.HttpRequestBase;

/** Snowflake statement */
public class SFStatement extends SFBaseStatement {

  private static final SFLogger logger = SFLoggerFactory.getLogger(SFStatement.class);

  private SFSession session;

  private SFBaseResultSet resultSet = null;

  private HttpRequestBase httpRequest;

  private Boolean isClosed = false;

  private Integer sequenceId = -1;

  private String requestId = null;

  private String sqlText = null;

  private final AtomicBoolean canceling = new AtomicBoolean(false);

  private boolean isFileTransfer = false;

  private SnowflakeFileTransferAgent transferAgent = null;

  private static final int MAX_BINDING_PARAMS_FOR_LOGGING = 1000;

  /** id used in combine describe and execute */
  private String describeJobUUID;

  // list of child result objects for queries called by the current query, if any
  private List childResults = null;

  // Three parameters adjusted in conservative memory usage mode
  private int conservativePrefetchThreads;
  private int conservativeResultChunkSize;
  private long conservativeMemoryLimit; // in bytes

  public SFStatement(SFSession session) {
    logger.trace("SFStatement(SFSession session)", false);

    this.session = session;
    Integer queryTimeout = session == null ? null : session.getQueryTimeout();
    this.queryTimeout = queryTimeout != null ? queryTimeout : this.queryTimeout;
    verifyArrowSupport();
  }

  private void verifyArrowSupport() {
    if (SnowflakeDriver.isDisableArrowResultFormat()) {
      logger.debug(
          "Disable arrow support: {}", SnowflakeDriver.getDisableArrowResultFormatMessage());
      statementParametersMap.put("JDBC_QUERY_RESULT_FORMAT", "JSON");
    }
  }

  /**
   * Sanity check query text
   *
   * @param sql The SQL statement to check
   * @throws SQLException Will be thrown if sql is null or empty
   */
  private void sanityCheckQuery(String sql) throws SQLException {
    if (sql == null || sql.isEmpty()) {
      throw new SnowflakeSQLException(
          SqlState.SQL_STATEMENT_NOT_YET_COMPLETE, ErrorCode.INVALID_SQL.getMessageCode(), sql);
    }
  }

  /**
   * Execute SQL query with an option for describe only
   *
   * @param sql sql statement
   * @param describeOnly true if describe only
   * @return query result set
   * @throws SQLException if connection is already closed
   * @throws SFException if result set is null
   */
  private SFBaseResultSet executeQuery(
      String sql,
      Map parametersBinding,
      boolean describeOnly,
      boolean asyncExec,
      CallingMethod caller,
      ExecTimeTelemetryData execTimeData)
      throws SQLException, SFException {
    sanityCheckQuery(sql);

    String trimmedSql = sql.trim();

    // snowflake specific client side commands
    if (isFileTransfer(trimmedSql)) {
      // Server side value or Connection string value is false then disable the PUT/GET command
      if ((session != null && !(session.getJdbcEnablePutGet() && session.getEnablePutGet()))) {
        // PUT/GET command disabled either on server side or in the client connection string
        logger.debug("Executing file transfer locally is disabled: {}", sql);
        throw new SnowflakeSQLException("File transfers have been disabled.");
      }

      // PUT/GET command
      logger.debug("Executing file transfer locally: {}", sql);

      return executeFileTransfer(sql);
    }

    // NOTE: It is intentional two describeOnly parameters are specified.
    return executeQueryInternal(
        sql,
        parametersBinding,
        describeOnly,
        describeOnly, // internal query if describeOnly is true
        asyncExec,
        caller,
        execTimeData);
  }

  /**
   * Describe a statement
   *
   * @param sql statement
   * @return metadata of statement including result set metadata and binding information
   * @throws SQLException if connection is already closed
   * @throws SFException if result set is null
   */
  @Override
  public SFPreparedStatementMetaData describe(String sql) throws SFException, SQLException {
    SFBaseResultSet baseResultSet =
        executeQuery(sql, null, true, false, null, new ExecTimeTelemetryData());

    describeJobUUID = baseResultSet.getQueryId();

    return new SFPreparedStatementMetaData(
        baseResultSet.getMetaData(),
        baseResultSet.getStatementType(),
        baseResultSet.getNumberOfBinds(),
        baseResultSet.isArrayBindSupported(),
        baseResultSet.getMetaDataOfBinds(),
        true); // valid metadata
  }

  /**
   * Internal method for executing a query with bindings accepted.
   *
   * 

* * @param sql sql statement * @param parameterBindings binding information * @param describeOnly true if just showing result set metadata * @param internal true if internal command not showing up in the history * @param caller the JDBC method that called this function, null if none * @return snowflake query result set * @throws SQLException if connection is already closed * @throws SFException if result set is null */ SFBaseResultSet executeQueryInternal( String sql, Map parameterBindings, boolean describeOnly, boolean internal, boolean asyncExec, CallingMethod caller, ExecTimeTelemetryData execTimeData) throws SQLException, SFException { resetState(); logger.debug("ExecuteQuery: {}", sql); if (session == null || session.isClosed()) { throw new SQLException("connection is closed"); } Object result = executeHelper( sql, StmtUtil.SF_MEDIA_TYPE, parameterBindings, describeOnly, internal, asyncExec, execTimeData); if (result == null) { throw new SnowflakeSQLLoggedException( session, ErrorCode.INTERNAL_ERROR.getMessageCode(), SqlState.INTERNAL_ERROR, "got null result"); } /* * we sort the result if the connection is in sorting mode */ Object sortProperty = session.getSessionPropertyByKey("sort"); boolean sortResult = sortProperty != null && (Boolean) sortProperty; logger.debug("Creating result set", false); try { JsonNode jsonResult = (JsonNode) result; resultSet = SFResultSetFactory.getResultSet(jsonResult, this, sortResult, execTimeData); childResults = ResultUtil.getChildResults(session, requestId, jsonResult); // if child results are available, skip over this result set and set the // current result to the first child's result. // we still construct the first result set for its side effects. if (!childResults.isEmpty()) { SFStatementType type = childResults.get(0).getType(); // ensure first query type matches the calling JDBC method, if exists if (caller == CallingMethod.EXECUTE_QUERY && !type.isGenerateResultSet()) { throw new SnowflakeSQLLoggedException( session, ErrorCode.QUERY_FIRST_RESULT_NOT_RESULT_SET); } else if (caller == CallingMethod.EXECUTE_UPDATE && type.isGenerateResultSet()) { throw new SnowflakeSQLLoggedException( session, ErrorCode.UPDATE_FIRST_RESULT_NOT_UPDATE_COUNT); } // this will update resultSet to point to the first child result before we return it getMoreResults(); } } catch (SnowflakeSQLException | OutOfMemoryError ex) { // snow-24428: no need to generate incident for exceptions we generate // snow-29403: or client OOM throw ex; } catch (Throwable ex) { // SNOW-22813 log exception logger.error("Exception creating result", ex); throw new SFException( ErrorCode.INTERNAL_ERROR, IncidentUtil.oneLiner("exception creating result", ex)); } logger.debug("Done creating result set", false); if (asyncExec) { session.addQueryToActiveQueryList(resultSet.getQueryId()); } execTimeData.setQueryId(resultSet.getQueryId()); return resultSet; } /** * Set a time bomb to cancel the outstanding query when timeout is reached. * * @param executor object to execute statement cancel request */ private void setTimeBomb(ScheduledExecutorService executor) { class TimeBombTask implements Callable { private final SFStatement statement; private TimeBombTask(SFStatement statement) { this.statement = statement; } @Override public Void call() throws SQLException { try { statement.cancel(CancellationReason.TIMEOUT); } catch (SFException ex) { throw new SnowflakeSQLLoggedException( session, ex.getSqlState(), ex.getVendorCode(), ex, ex.getParams()); } return null; } } executor.schedule(new TimeBombTask(this), this.queryTimeout, TimeUnit.SECONDS); } /** * A helper method to build URL and submit the SQL to snowflake for exec * * @param sql sql statement * @param mediaType media type * @param bindValues map of binding values * @param describeOnly whether only show the result set metadata * @param internal run internal query not showing up in history * @return raw json response * @throws SFException if query is canceled * @throws SnowflakeSQLException if query is already running */ public Object executeHelper( String sql, String mediaType, Map bindValues, boolean describeOnly, boolean internal, boolean asyncExec, ExecTimeTelemetryData execTimeData) throws SnowflakeSQLException, SFException { ScheduledExecutorService executor = null; try { synchronized (this) { if (isClosed) { throw new SFException(ErrorCode.STATEMENT_CLOSED); } // initialize a sequence id if not closed or not for aborting if (canceling.get()) { // nothing to do if canceled throw new SFException(ErrorCode.QUERY_CANCELED); } if (this.requestId != null) { throw new SnowflakeSQLLoggedException( session, ErrorCode.STATEMENT_ALREADY_RUNNING_QUERY.getMessageCode(), SqlState.FEATURE_NOT_SUPPORTED); } this.requestId = UUIDUtils.getUUID().toString(); execTimeData.setRequestId(requestId); this.sequenceId = session.getAndIncrementSequenceId(); this.sqlText = sql; } EventUtil.triggerStateTransition( BasicEvent.QueryState.QUERY_STARTED, String.format(QueryState.QUERY_STARTED.getArgString(), requestId)); // if there are a large number of bind values, we should upload them to stage // instead of passing them in the payload (if enabled) execTimeData.setBindStart(); int numBinds = BindUploader.arrayBindValueCount(bindValues); String bindStagePath = null; if (0 < session.getArrayBindStageThreshold() && session.getArrayBindStageThreshold() <= numBinds && !describeOnly && BindUploader.isArrayBind(bindValues)) { try (BindUploader uploader = BindUploader.newInstance(session, requestId)) { uploader.upload(bindValues); bindStagePath = uploader.getStagePath(); } catch (BindException ex) { logger.debug( "Exception encountered trying to upload binds to stage with input stream. Attaching" + " binds in payload instead. ", ex); TelemetryData errorLog = TelemetryUtil.buildJobData(this.requestId, ex.type.field, 1); this.session.getTelemetryClient().addLogToBatch(errorLog); } catch (SQLException ex) { logger.debug( "Exception encountered trying to upload binds to stage with input stream. Attaching" + " binds in payload instead. ", ex); TelemetryData errorLog = TelemetryUtil.buildJobData(this.requestId, TelemetryField.FAILED_BIND_UPLOAD, 1); this.session.getTelemetryClient().addLogToBatch(errorLog); } } if (session.isConservativeMemoryUsageEnabled()) { logger.debug("JDBC conservative memory usage is enabled.", false); calculateConservativeMemoryUsage(); } StmtUtil.StmtInput stmtInput = new StmtUtil.StmtInput(); stmtInput .setSql(sql) .setMediaType(mediaType) .setInternal(internal) .setDescribeOnly(describeOnly) .setAsync(asyncExec) .setServerUrl(session.getServerUrl()) .setRequestId(requestId) .setSequenceId(sequenceId) .setParametersMap(statementParametersMap) .setSessionToken(session.getSessionToken()) .setNetworkTimeoutInMillis(session.getNetworkTimeoutInMilli()) .setInjectSocketTimeout(session.getInjectSocketTimeout()) .setInjectClientPause(session.getInjectClientPause()) .setCanceling(canceling) .setRetry(false) .setDescribedJobId(describeJobUUID) .setCombineDescribe(session.getEnableCombineDescribe()) .setQuerySubmissionTime(System.currentTimeMillis()) .setServiceName(session.getServiceName()) .setOCSPMode(session.getOCSPMode()) .setHttpClientSettingsKey(session.getHttpClientKey()) .setMaxRetries(session.getMaxHttpRetries()) .setQueryContextDTO(session.isAsyncSession() ? null : session.getQueryContextDTO()); if (bindStagePath != null) { stmtInput.setBindValues(null).setBindStage(bindStagePath); // use the new SQL format for this query so dates/timestamps are parsed correctly setUseNewSqlFormat(true); statementParametersMap.put("TIMESTAMP_INPUT_FORMAT", "AUTO"); } else { stmtInput.setBindValues(bindValues).setBindStage(null); } if (numBinds > 0 && session.getPreparedStatementLogging()) { if (numBinds > MAX_BINDING_PARAMS_FOR_LOGGING) { logger.info( "Number of binds exceeds logging limit. Printing off {} binding parameters.", MAX_BINDING_PARAMS_FOR_LOGGING); } else { logger.info("Printing off {} binding parameters.", numBinds); } int counter = 0; // if it's an array bind, print off the first few rows from each column. if (BindUploader.isArrayBind(bindValues)) { int numRowsPrinted = MAX_BINDING_PARAMS_FOR_LOGGING / bindValues.size(); if (numRowsPrinted <= 0) { numRowsPrinted = 1; } for (Map.Entry entry : bindValues.entrySet()) { List bindRows = (List) entry.getValue().getValue(); if (numRowsPrinted >= bindRows.size()) { numRowsPrinted = bindRows.size(); } String rows = "["; for (int i = 0; i < numRowsPrinted; i++) { rows += bindRows.get(i) + ", "; } rows += "]"; logger.info("Column {}: {}", entry.getKey(), rows); counter += numRowsPrinted; if (counter >= MAX_BINDING_PARAMS_FOR_LOGGING) { break; } } } // not an array, just a bunch of columns else { for (Map.Entry entry : bindValues.entrySet()) { if (counter >= MAX_BINDING_PARAMS_FOR_LOGGING) { break; } counter++; logger.info("Column {}: {}", entry.getKey(), entry.getValue().getValue()); } } } execTimeData.setBindEnd(); if (canceling.get()) { logger.debug("Query cancelled", false); throw new SFException(ErrorCode.QUERY_CANCELED); } // if timeout is set, start a thread to cancel the request after timeout // reached. if (this.queryTimeout > 0) { executor = Executors.newScheduledThreadPool(1); setTimeBomb(executor); } StmtUtil.StmtOutput stmtOutput = null; boolean sessionRenewed; do { sessionRenewed = false; try { stmtOutput = StmtUtil.execute(stmtInput, execTimeData); break; } catch (SnowflakeSQLException ex) { if (ex.getErrorCode() == Constants.SESSION_EXPIRED_GS_CODE) { try { // renew the session session.renewSession(stmtInput.sessionToken); } catch (SnowflakeReauthenticationRequest ex0) { if (session.isExternalbrowserAuthenticator()) { reauthenticate(); } else { throw ex0; } } // SNOW-18822: reset session token for the statement stmtInput.setSessionToken(session.getSessionToken()); stmtInput.setRetry(true); sessionRenewed = true; execTimeData.incrementRetryCount(); execTimeData.addRetryLocation("renewSession"); logger.debug("Session got renewed, will retry", false); } else { throw ex; } } } while (sessionRenewed && !canceling.get()); // Debugging/Testing for incidents if (Boolean.TRUE .toString() .equalsIgnoreCase(systemGetProperty("snowflake.enable_incident_test1"))) { throw new SFException(ErrorCode.STATEMENT_CLOSED); } synchronized (this) { /* * done with the remote execution of the query. set sequenceId to -1 * and request id to null so that we don't try to abort it upon canceling. */ this.sequenceId = -1; this.requestId = null; } if (canceling.get()) { // If we are here, this is the context for the initial query that // is being canceled. Raise an exception anyway here even if // the server fails to abort it. throw new SFException(ErrorCode.QUERY_CANCELED); } logger.debug("Returning from executeHelper", false); if (stmtOutput != null) { return stmtOutput.getResult(); } throw new SFException(ErrorCode.INTERNAL_ERROR); } catch (SFException | SnowflakeSQLException ex) { isClosed = true; throw ex; } finally { if (executor != null) { executor.shutdownNow(); } // if this query enabled the new SQL format, re-disable it now setUseNewSqlFormat(false); } } /** * calculate conservative memory limit and the number of prefetch threads before query execution */ private void calculateConservativeMemoryUsage() { int clientMemoryLimit = session.getClientMemoryLimit(); int clientPrefetchThread = session.getClientPrefetchThreads(); int clientChunkSize = session.getClientResultChunkSize(); long memoryLimitInBytes; if (clientMemoryLimit == DEFAULT_CLIENT_MEMORY_LIMIT) { // this is all default scenario // only allows JDBC use at most 80% of free memory long freeMemoryToUse = Runtime.getRuntime().freeMemory() * 8 / 10; memoryLimitInBytes = Math.min( (long) 2 * clientPrefetchThread * clientChunkSize * 1024 * 1024, freeMemoryToUse); } else { memoryLimitInBytes = (long) clientMemoryLimit * 1024 * 1024; } conservativeMemoryLimit = memoryLimitInBytes; reducePrefetchThreadsAndChunkSizeToFitMemoryLimit( conservativeMemoryLimit, clientPrefetchThread, clientChunkSize); } private void updateConservativeResultChunkSize(int clientChunkSize) { if (clientChunkSize != conservativeResultChunkSize) { logger.debug( "conservativeResultChunkSize changed from {} to {}", conservativeResultChunkSize, clientChunkSize); conservativeResultChunkSize = clientChunkSize; statementParametersMap.put("CLIENT_RESULT_CHUNK_SIZE", conservativeResultChunkSize); } } private void reducePrefetchThreadsAndChunkSizeToFitMemoryLimit( long clientMemoryLimit, int clientPrefetchThread, int clientChunkSize) { if (clientPrefetchThread != DEFAULT_CLIENT_PREFETCH_THREADS) { // prefetch threads are configured so only reduce chunk size conservativePrefetchThreads = clientPrefetchThread; for (; clientChunkSize >= MIN_CLIENT_CHUNK_SIZE; clientChunkSize -= session.getConservativeMemoryAdjustStep()) { if (clientMemoryLimit >= (long) 2 * clientPrefetchThread * clientChunkSize * 1024 * 1024) { updateConservativeResultChunkSize(clientChunkSize); return; } } updateConservativeResultChunkSize(MIN_CLIENT_CHUNK_SIZE); } else { // reduce both prefetch threads and chunk size while (clientPrefetchThread > 1) { for (clientChunkSize = MAX_CLIENT_CHUNK_SIZE; clientChunkSize >= MIN_CLIENT_CHUNK_SIZE; clientChunkSize -= session.getConservativeMemoryAdjustStep()) { if (clientMemoryLimit >= (long) 2 * clientPrefetchThread * clientChunkSize * 1024 * 1024) { conservativePrefetchThreads = clientPrefetchThread; updateConservativeResultChunkSize(clientChunkSize); return; } } clientPrefetchThread--; } conservativePrefetchThreads = clientPrefetchThread; updateConservativeResultChunkSize(MIN_CLIENT_CHUNK_SIZE); } } /** * @return conservative prefetch threads before fetching results */ public int getConservativePrefetchThreads() { return conservativePrefetchThreads; } /** * @return conservative memory limit before fetching results */ public long getConservativeMemoryLimit() { return conservativeMemoryLimit; } /** * Return an array of child query ID for the given query ID. * *

If the given query ID is for a multiple statements query, it returns an array of its child * statements, otherwise, it returns an array to include the given query ID. * * @param queryID The given query ID * @return An array of child query IDs * @throws SQLException If the query is running or the corresponding query is FAILED. */ @Override public String[] getChildQueryIds(String queryID) throws SQLException { QueryStatusV2 qs = session.getQueryStatusV2(queryID); if (qs.isStillRunning()) { throw new SQLException( "Status of query associated with resultSet is " + qs.getDescription() + ". Results not generated."); } try { JsonNode jsonResult = StmtUtil.getQueryResultJSON(queryID, session); List childResults = ResultUtil.getChildResults(session, requestId, jsonResult); List resultList = new ArrayList<>(); for (int i = 0; i < childResults.size(); i++) { resultList.add(childResults.get(i).getId()); } if (resultList.isEmpty()) { resultList.add(queryID); } String[] result = new String[resultList.size()]; resultList.toArray(result); return result; } catch (SFException ex) { throw new SnowflakeSQLException(ex); } } @Override public SFBaseResultSet execute( String sql, Map parametersBinding, CallingMethod caller, ExecTimeTelemetryData execTimeData) throws SQLException, SFException { return execute(sql, false, parametersBinding, caller, execTimeData); } private void reauthenticate() throws SFException, SnowflakeSQLException { SFLoginInput input = new SFLoginInput() .setRole(session.getRole()) .setWarehouse(session.getWarehouse()) .setDatabaseName(session.getDatabase()) .setSchemaName(session.getSchema()) .setOCSPMode(session.getOCSPMode()) .setHttpClientSettingsKey(session.getHttpClientKey()); session.open(); } /** * A helper method to build URL and cancel the SQL for exec * * @param sql sql statement * @param mediaType media type * @param cancellationReason reason for the cancellation * @throws SnowflakeSQLException if failed to cancel the statement * @throws SFException if statement is already closed */ private void cancelHelper(String sql, String mediaType, CancellationReason cancellationReason) throws SnowflakeSQLException, SFException { synchronized (this) { if (isClosed) { throw new SFException(ErrorCode.INTERNAL_ERROR, "statement already closed"); } } StmtUtil.StmtInput stmtInput = new StmtUtil.StmtInput(); stmtInput .setServerUrl(session.getServerUrl()) .setSql(sql) .setMediaType(mediaType) .setRequestId(requestId) .setSessionToken(session.getSessionToken()) .setServiceName(session.getServiceName()) .setOCSPMode(session.getOCSPMode()) .setMaxRetries(session.getMaxHttpRetries()) .setHttpClientSettingsKey(session.getHttpClientKey()); StmtUtil.cancel(stmtInput, cancellationReason); synchronized (this) { /* * done with the remote execution of the query. set sequenceId to -1 * and request id to null so that we don't try to abort it again upon * canceling. */ this.sequenceId = -1; this.requestId = null; } } /** * Execute sql * * @param sql sql statement. * @param parametersBinding parameters to bind * @param caller the JDBC interface method that called this method, if any * @return whether there is result set or not * @throws SQLException if failed to execute sql * @throws SFException exception raised from Snowflake components * @throws SQLException if SQL error occurs */ public SFBaseResultSet execute( String sql, boolean asyncExec, Map parametersBinding, CallingMethod caller, ExecTimeTelemetryData execTimeData) throws SQLException, SFException { TelemetryService.getInstance().updateContext(session.getSnowflakeConnectionString()); sanityCheckQuery(sql); session.injectedDelay(); if (session.getPreparedStatementLogging()) { logger.info("Execute: {}", sql); } else { logger.debug("Execute: {}", sql); } String trimmedSql = sql.trim(); if (trimmedSql.length() >= 20 && trimmedSql.toLowerCase().startsWith("set-sf-property")) { executeSetProperty(sql); return null; } return executeQuery(sql, parametersBinding, false, asyncExec, caller, execTimeData); } private SFBaseResultSet executeFileTransfer(String sql) throws SQLException, SFException { session.injectedDelay(); resetState(); logger.debug("Entering executeFileTransfer", false); isFileTransfer = true; transferAgent = new SnowflakeFileTransferAgent(sql, session, this); try { transferAgent.execute(); logger.debug("Setting result set", false); resultSet = (SFFixedViewResultSet) transferAgent.getResultSet(); childResults = Collections.emptyList(); logger.debug("Number of cols: {}", resultSet.getMetaData().getColumnCount()); logger.debug("Completed transferring data", false); return resultSet; } catch (SQLException ex) { logger.debug("Exception: {}", ex.getMessage()); throw ex; } } @Override public void close() { logger.trace("void close()", false); if (requestId != null) { EventUtil.triggerStateTransition( BasicEvent.QueryState.QUERY_ENDED, String.format(QueryState.QUERY_ENDED.getArgString(), requestId)); } resultSet = null; childResults = null; isClosed = true; if (httpRequest != null) { logger.debug("Releasing connection for the http request", false); httpRequest.releaseConnection(); httpRequest = null; } session.getTelemetryClient().sendBatchAsync(); isFileTransfer = false; transferAgent = null; } @Override public void cancel() throws SFException, SQLException { logger.trace("void cancel()", false); cancel(CancellationReason.UNKNOWN); } @Override public void cancel(CancellationReason cancellationReason) throws SFException, SQLException { logger.trace("void cancel(CancellationReason)", false); if (canceling.get()) { logger.debug("Query is already cancelled", false); return; } canceling.set(true); if (isFileTransfer) { if (transferAgent != null) { logger.debug("Cancel file transferring ... ", false); transferAgent.cancel(); } } else { synchronized (this) { // the query hasn't been sent to GS yet, just mark the stmt closed if (requestId == null) { logger.debug("No remote query outstanding", false); return; } } // cancel the query on the server side if it has been issued cancelHelper(this.sqlText, StmtUtil.SF_MEDIA_TYPE, cancellationReason); } } private void resetState() { resultSet = null; childResults = null; if (httpRequest != null) { httpRequest.releaseConnection(); httpRequest = null; } isClosed = false; sequenceId = -1; requestId = null; sqlText = null; canceling.set(false); isFileTransfer = false; transferAgent = null; } @Override public SFBaseSession getSFBaseSession() { return session; } // *NOTE* this new SQL format is incomplete. It should only be used under certain circumstances. private void setUseNewSqlFormat(boolean useNewSqlFormat) throws SFException { this.addProperty("NEW_SQL_FORMAT", useNewSqlFormat); } public boolean getMoreResults() throws SQLException { return getMoreResults(Statement.CLOSE_CURRENT_RESULT); } @Override public boolean getMoreResults(int current) throws SQLException { // clean up current result, if exists if (resultSet != null && (current == Statement.CLOSE_CURRENT_RESULT || current == Statement.CLOSE_ALL_RESULTS)) { resultSet.close(); } resultSet = null; // verify if more results exist if (childResults == null || childResults.isEmpty()) { return false; } // fetch next result using the query id SFChildResult nextResult = childResults.remove(0); try { JsonNode result = StmtUtil.getQueryResultJSON(nextResult.getId(), session); Object sortProperty = session.getSessionPropertyByKey("sort"); boolean sortResult = sortProperty != null && (Boolean) sortProperty; resultSet = SFResultSetFactory.getResultSet(result, this, sortResult, new ExecTimeTelemetryData()); // override statement type so we can treat the result set like a result of // the original statement called (and not the result scan) resultSet.setStatementType(nextResult.getType()); return nextResult.getType().isGenerateResultSet(); } catch (SFException ex) { throw new SnowflakeSQLException(ex); } } @Override public SFBaseResultSet getResultSet() { return resultSet; } @Override public boolean hasChildren() { return !childResults.isEmpty(); } @Override public SFBaseResultSet asyncExecute( String sql, Map parametersBinding, CallingMethod caller, ExecTimeTelemetryData execTimeData) throws SQLException, SFException { return execute(sql, true, parametersBinding, caller, execTimeData); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy