diff --git a/lib/connection/connection.js b/lib/connection/connection.js index 7e1b31a02..0e9f515c3 100644 --- a/lib/connection/connection.js +++ b/lib/connection/connection.js @@ -5,6 +5,7 @@ const { v4: uuidv4 } = require('uuid'); const Url = require('url'); const QueryString = require('querystring'); const GSErrors = require('../constants/gs_errors') +const QueryStatus = require('../constants/query_status'); var Util = require('../util'); var Errors = require('../errors'); @@ -37,6 +38,15 @@ function Connection(context) // generate an id for the connection var id = uuidv4(); + // async max retry and retry pattern from python connector + const asyncNoDataMaxRetry = 24; + const asyncRetryPattern = [1, 1, 2, 3, 4, 8, 10]; + const asyncRetryInMilliseconds = 500; + + // Custom regex based on uuid validate + // Unable to directly use uuid validate because the queryId returned from the server doesn't match the regex + const queryIdRegex = new RegExp(/^(?:[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}|00000000-0000-0000-0000-000000000000)$/i); + //Make session tokens available for testing this.getTokens = function () { @@ -411,6 +421,164 @@ function Connection(context) return this; }; + /** + * Gets the response containing the status of the query based on queryId. + * + * @param {String} queryId + * + * @returns {Object} the query response + */ + async function getQueryResponse(queryId) { + // Check if queryId exists and is valid uuid + Errors.checkArgumentExists(Util.exists(queryId), + ErrorCodes.ERR_CONN_FETCH_RESULT_MISSING_QUERY_ID); + Errors.checkArgumentValid(queryIdRegex.test(queryId), + ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID, queryId); + + // Form the request options + const options = + { + method: 'GET', + url: Url.format( + { + pathname: `/monitoring/queries/${queryId}` + }), + }; + + // Get the response containing the query status + const response = await services.sf.requestAsync(options); + + return response['data']; + } + + /** + * Extracts the status of the query from the query response. + * + * @param {Object} queryResponse + * + * @returns {String} the query status. + */ + function extractQueryStatus(queryResponse) { + const queries = queryResponse['data']['queries']; + let status = QueryStatus.code.NO_DATA; // default status + if (queries.length > 0) { + status = queries[0]['status']; + } + + return status; + } + + /** + * Gets the status of the query based on queryId. + * + * @param {String} queryId + * + * @returns {String} the query status. + */ + this.getQueryStatus = async function (queryId) { + return extractQueryStatus(await getQueryResponse(queryId)); + }; + + /** + * Gets the status of the query based on queryId and throws if there's an error. + * + * @param {String} queryId + * + * @returns {String} the query status. + */ + this.getQueryStatusThrowIfError = async function (queryId) { + const status = this.getQueryStatus(queryId); + + let message, code, sqlState = null; + + if (this.isAnError(status)) { + const response = await getQueryResponse(queryId); + message = response['message'] || ''; + code = response['code'] || -1; + + if (response['data']) { + message += response['data']['queries'].length > 0 ? response['data']['queries'][0]['errorMessage'] : ''; + sqlState = response['data']['sqlState']; + } + + throw Errors.createOperationFailedError( + code, response, message, sqlState); + } + + return status; + }; + + /** + * Gets the results from a previously ran query based on queryId + * + * @param {Object} options + * + * @returns {Object} + */ + this.getResultsFromQueryId = async function (options) { + const queryId = options.queryId; + let status, noDataCounter = 0, retryPatternPos = 0; + + // Wait until query has finished executing + let queryStillExecuting = true; + while (queryStillExecuting) { + // Check if query is still running and trigger exception if it failed + status = await this.getQueryStatusThrowIfError(queryId); + queryStillExecuting = this.isStillRunning(status); + if (!queryStillExecuting) { + break; + } + + // Timeout based on query status retry rules + await new Promise((resolve) => { + setTimeout(() => resolve(), asyncRetryInMilliseconds * asyncRetryPattern[retryPatternPos]); + }); + + // If no data, increment the no data counter + if (QueryStatus.code[status] == QueryStatus.code.NO_DATA) { + noDataCounter++; + // Check if retry for no data is exceeded + if (noDataCounter > asyncNoDataMaxRetry) { + throw Errors.createClientError( + ErrorCodes.ERR_GET_RESULTS_QUERY_ID_NO_DATA, true, queryId); + } + } + + if (retryPatternPos < asyncRetryPattern.length - 1) { + retryPatternPos++; + } + } + + if (QueryStatus.code[status] != QueryStatus.code.SUCCESS) { + throw Errors.createClientError( + ErrorCodes.ERR_GET_RESULTS_QUERY_ID_NOT_SUCCESS_STATUS, true, queryId, status); + } + + return this.fetchResult(options); + }; + + /** + * Checks whether the given status is currently running. + * + * @param {String} status + * + * @returns {Boolean} + */ + this.isStillRunning = function (status) { + return QueryStatus.runningStatuses.includes(QueryStatus.code[status]); + }; + + /** + * Checks whether the given status means that there has been an error. + * + * @param {String} status + * + * @returns {Boolean} + */ + this.isAnError = function (status) { + return QueryStatus.errorStatuses.includes(QueryStatus.code[status]); + }; + /** * Returns a serialized version of this connection. * diff --git a/lib/connection/statement.js b/lib/connection/statement.js index 130d6bc30..0e531aeee 100644 --- a/lib/connection/statement.js +++ b/lib/connection/statement.js @@ -33,6 +33,11 @@ var statementTypes = FILE_POST_EXEC: 'FILE_POST_EXEC' }; +const queryCodes = { + QUERY_IN_PROGRESS: '333333', // GS code: the query is in progress + QUERY_IN_PROGRESS_ASYNC: '333334' // GS code: the query is detached +}; + exports.createContext = function ( options, services, connectionConfig) { @@ -364,6 +369,12 @@ function createContextPreExec( RowMode.checkRowModeValid(rowMode); } + // if an asyncExec flag is specified, make sure it's boolean + if (Util.exists(statementOptions.asyncExec)) { + Errors.checkArgumentValid(Util.isBoolean(statementOptions.asyncExec), + ErrorCodes.ERR_CONN_EXEC_STMT_INVALID_ASYNC_EXEC); + } + // create a statement context var statementContext = createStatementContext(); @@ -374,6 +385,7 @@ function createContextPreExec( statementContext.multiResultIds = statementOptions.multiResultIds; statementContext.multiCurId = statementOptions.multiCurId; statementContext.rowMode = statementOptions.rowMode; + statementContext.asyncExec = statementOptions.asyncExec; // if a binds array is specified, add it to the statement context if (Util.exists(statementOptions.binds)) @@ -682,10 +694,13 @@ function invokeStatementComplete(statement, context) streamResult = context.connectionConfig.getStreamResult(); } - // if the result will be streamed later, + // if the result will be streamed later or in asyncExec mode, // invoke the complete callback right away if (streamResult) { context.complete(Errors.externalize(context.resultError), statement); + } else if (context.asyncExec) { + // return the result object with the query ID inside. + context.complete(null, statement, context.result); } else { process.nextTick(function () { // aggregate all the rows into an array and pass this @@ -779,6 +794,12 @@ function createOnStatementRequestSuccRow(statement, context) // if we don't already have a result if (!context.result) { + if (body.code === queryCodes.QUERY_IN_PROGRESS_ASYNC) { + context.result = { + queryId: body.data.queryId + }; + return; + } if (body.data.resultIds != undefined && body.data.resultIds.length > 0) { //multi statements @@ -1330,6 +1351,11 @@ function sendRequestPreExec(statementContext, onResultAvailable) json.queryContextDTO = statementContext.services.sf.getQueryContextDTO(); } + // include the asyncExec flag if a value was specified + if (Util.exists(statementContext.asyncExec)) { + json.asyncExec = statementContext.asyncExec; + } + // use the snowflake service to issue the request sendSfRequest(statementContext, { @@ -1645,9 +1671,15 @@ function buildResultRequestCallback( statementContext.queryId = body.data.queryId; // if the result is not ready yet, extract the result url from the response - // and issue a GET request to try to fetch the result again - if (body && (body.code === '333333' || body.code === '333334')) - { + // and issue a GET request to try to fetch the result again unless asyncExec is enabled. + if (body && (body.code === queryCodes.QUERY_IN_PROGRESS + || body.code === queryCodes.QUERY_IN_PROGRESS_ASYNC)) { + + if (statementContext.asyncExec) { + await onResultAvailable.call(null, err, body); + return; + } + // extract the result url from the response and try to get the result // again sendSfRequest(statementContext, diff --git a/lib/constants/error_messages.js b/lib/constants/error_messages.js index fe7157917..9da8d13bd 100644 --- a/lib/constants/error_messages.js +++ b/lib/constants/error_messages.js @@ -110,12 +110,13 @@ exports[409010] = 'Invalid streamResult flag. The specified value must be a bool exports[409011] = 'Invalid fetchAsString value. The specified value must be an Array.'; exports[409012] = 'Invalid fetchAsString type: %s. The supported types are: String, Boolean, Number, Date, Buffer, and JSON.'; exports[409013] = 'Invalid requestId. The specified value must be a string.'; +exports[409014] = 'Invalid asyncExec. The specified value must be a boolean.' // 410001 exports[410001] = 'Fetch-result options must be specified.'; exports[410002] = 'Invalid options. The specified value must be an object.'; -exports[410003] = 'A statement id must be specified.'; -exports[410004] = 'Invalid statement id. The specified value must be a string.'; +exports[410003] = 'A query id/statement id must be specified.'; +exports[410004] = 'Invalid query id/statement id. The specified value must be a string.'; exports[410005] = 'Invalid complete callback. The specified value must be a function.'; exports[410006] = 'Invalid streamResult flag. The specified value must be a boolean.'; exports[410007] = 'Invalid fetchAsString value. The specified value must be an Array.'; @@ -152,3 +153,8 @@ exports[450004] = 'Invalid each() callback. The specified value must be a functi exports[450005] = 'An end() callback must be specified.'; exports[450006] = 'Invalid end() callback. The specified value must be a function.'; exports[450007] = 'Operation failed because the statement is still in progress.'; + +// 460001 +exports[460001] = 'Invalid queryId: %s'; +exports[460002] = 'Cannot retrieve data. No information returned from server for query %s'; +exports[460003] = 'Status of query %s is %s, results are unavailable'; diff --git a/lib/constants/query_status.js b/lib/constants/query_status.js new file mode 100644 index 000000000..6bc4cf2a4 --- /dev/null +++ b/lib/constants/query_status.js @@ -0,0 +1,45 @@ +/* + * Copyright (c) 2023 Snowflake Computing Inc. All rights reserved. + */ + +const code = {}; + +code.RUNNING = 'RUNNING'; +code.ABORTING = 'ABORTING'; +code.SUCCESS = 'SUCCESS'; +code.FAILED_WITH_ERROR = 'FAILED_WITH_ERROR'; +code.ABORTED = 'ABORTED'; +code.QUEUED = 'QUEUED'; +code.FAILED_WITH_INCIDENT = 'FAILED_WITH_INCIDENT'; +code.DISCONNECTED = 'DISCONNECTED'; +code.RESUMING_WAREHOUSE = 'RESUMING_WAREHOUSE'; +// purposeful typo.Is present in QueryDTO.java +code.QUEUED_REPARING_WAREHOUSE = 'QUEUED_REPARING_WAREHOUSE'; +code.RESTARTED = 'RESTARTED'; +code.BLOCKED = 'BLOCKED'; +code.NO_DATA = 'NO_DATA'; + +// All running query statuses +const runningStatuses = + [ + code.RUNNING, + code.RESUMING_WAREHOUSE, + code.QUEUED, + code.QUEUED_REPARING_WAREHOUSE, + code.NO_DATA, + ]; + +// All error query statuses +const errorStatuses = + [ + code.ABORTING, + code.FAILED_WITH_ERROR, + code.ABORTED, + code.FAILED_WITH_INCIDENT, + code.DISCONNECTED, + code.BLOCKED, + ]; + +exports.code = code; +exports.runningStatuses = runningStatuses; +exports.errorStatuses = errorStatuses; diff --git a/lib/errors.js b/lib/errors.js index 3b08a888b..cdbb61177 100644 --- a/lib/errors.js +++ b/lib/errors.js @@ -115,6 +115,7 @@ codes.ERR_CONN_EXEC_STMT_INVALID_STREAM_RESULT = 409010; codes.ERR_CONN_EXEC_STMT_INVALID_FETCH_AS_STRING = 409011; codes.ERR_CONN_EXEC_STMT_INVALID_FETCH_AS_STRING_VALUES = 409012; codes.ERR_CONN_EXEC_STMT_INVALID_REQUEST_ID = 409013; +codes.ERR_CONN_EXEC_STMT_INVALID_ASYNC_EXEC = 409014; // 410001 codes.ERR_CONN_FETCH_RESULT_MISSING_OPTIONS = 410001; @@ -159,6 +160,11 @@ codes.ERR_STMT_FETCH_ROWS_MISSING_END = 450005; codes.ERR_STMT_FETCH_ROWS_INVALID_END = 450006; codes.ERR_STMT_FETCH_ROWS_FETCHING_RESULT = 450007; +// 460001 +codes.ERR_GET_RESPONSE_QUERY_INVALID_UUID = 460001; +codes.ERR_GET_RESULTS_QUERY_ID_NO_DATA = 460002; +codes.ERR_GET_RESULTS_QUERY_ID_NOT_SUCCESS_STATUS = 460003; + exports.codes = codes; /** diff --git a/lib/http/base.js b/lib/http/base.js index 2d92f9650..190e1ca5a 100644 --- a/lib/http/base.js +++ b/lib/http/base.js @@ -85,7 +85,14 @@ HttpClient.prototype.requestAsync = async function (options) { const requestOptions = prepareRequestOptions.call(this, options); - return axios.request(requestOptions); + let response = await axios.request(requestOptions); + + if (Util.isString(response['data']) && + response['headers']['content-type'] === 'application/json') { + response['data'] = JSON.parse(response['data']); + } + + return response; }; /** diff --git a/lib/services/sf.js b/lib/services/sf.js index c805b393d..26cd86463 100644 --- a/lib/services/sf.js +++ b/lib/services/sf.js @@ -258,6 +258,15 @@ function SnowflakeService(connectionConfig, httpClient, config) new OperationRequest(options).validate().execute(); }; + /** + * Issues a generic async request to Snowflake. + * + * @param {Object} options + */ + this.requestAsync = async function (options) { + return await new OperationRequest(options).validate().executeAsync(); + }; + /** * Terminates the current connection to Snowflake. * @@ -407,9 +416,8 @@ function SnowflakeService(connectionConfig, httpClient, config) /** * @inheritDoc */ - OperationRequest.prototype.executeAsync = function () - { - return currentState.requestAsync(this.options); + OperationRequest.prototype.executeAsync = async function () { + return await currentState.requestAsync(this.options); }; /** @@ -737,24 +745,24 @@ function StateAbstract(options) } /** - * Sends out the post request. + * Sends out the request. * * @returns {Object} the request that was issued. */ - Request.prototype.sendAsync = function () - { + Request.prototype.sendAsync = async function () { // pre-process the request options this.preprocessOptions(this.requestOptions); - var url = this.requestOptions.absoluteUrl; - var header = this.requestOptions.headers; - var body = this.requestOptions.json; + const options = + { + method: this.requestOptions.method, + headers: this.requestOptions.headers, + url: this.requestOptions.absoluteUrl, + json: this.requestOptions.json + }; - // issue the http request - return axios - .post(url, body, { - headers: header - }); + // issue the async http request + return await httpClient.requestAsync(options); }; /** @@ -1355,10 +1363,9 @@ StateConnected.prototype.connect = function (options) }); }; -StateConnected.prototype.requestAsync = function (options) -{ +StateConnected.prototype.requestAsync = async function (options) { // create a session token request from the options and send out the request - return this.createSessionTokenRequest(options).sendAsync(); + return await this.createSessionTokenRequest(options).sendAsync(); }; /** diff --git a/test/integration/testExecuteAsync.js b/test/integration/testExecuteAsync.js new file mode 100644 index 000000000..a81363259 --- /dev/null +++ b/test/integration/testExecuteAsync.js @@ -0,0 +1,246 @@ +/* + * Copyright (c) 2015-2019 Snowflake Computing Inc. All rights reserved. + */ + +const assert = require('assert'); +const async = require('async'); +const testUtil = require('./testUtil'); +const ErrorCodes = require('../../lib/errors').codes; +const QueryStatus = require('../../lib/constants/query_status').code; + +describe('ExecuteAsync test', function () { + let connection; + + before(async () => { + connection = testUtil.createConnection(); + await testUtil.connectAsync(connection); + }); + + after(async () => { + await testUtil.destroyConnectionAsync(connection); + }); + + it('testAsyncQueryWithPromise', function (done) { + const expectedSeconds = 3; + const sqlText = `CALL SYSTEM$WAIT(${expectedSeconds}, 'SECONDS')`; + let queryId; + + async.series( + [ + // Execute query in async mode + function (callback) { + connection.execute({ + sqlText: sqlText, + asyncExec: true, + complete: async function (err, stmt) { + assert.ok(!err); + queryId = stmt.getQueryId(); + const status = await connection.getQueryStatus(queryId); + assert.ok(connection.isStillRunning(status)); + callback(); + } + }); + }, + // Get results using query id + async function () { + const statement = await connection.getResultsFromQueryId({ queryId: queryId }); + + await new Promise((resolve, reject) => { + statement.streamRows() + .on('error', (err) => reject(err)) + .on('data', (row) => assert.strictEqual(row['SYSTEM$WAIT'], `waited ${expectedSeconds} seconds`)) + .on('end', async () => { + const status = await connection.getQueryStatus(queryId); + assert.strictEqual(QueryStatus[status], QueryStatus.SUCCESS); + resolve(); + }); + }); + } + ], + done + ); + }); + + it('testAsyncQueryWithCallback', function (done) { + const expectedSeconds = 3; + const sqlText = `CALL SYSTEM$WAIT(${expectedSeconds}, 'SECONDS')`; + let queryId; + + async.series( + [ + // Execute query in async mode + function (callback) { + connection.execute({ + sqlText: sqlText, + asyncExec: true, + complete: async function (err, stmt) { + assert.ok(!err); + queryId = stmt.getQueryId(); + const status = await connection.getQueryStatus(queryId); + assert.ok(connection.isStillRunning(status)); + callback(); + } + }); + }, + // Get results using query id + function (callback) { + connection.getResultsFromQueryId({ + queryId: queryId, + complete: async function (err, _stmt, rows) { + assert.ok(!err); + const status = await connection.getQueryStatus(queryId); + assert.strictEqual(QueryStatus[status], QueryStatus.SUCCESS); + assert.strictEqual(rows[0]['SYSTEM$WAIT'], `waited ${expectedSeconds} seconds`); + callback(); + } + }); + } + ], + done + ); + }); + + it('testFailedQueryThrowsError', function (done) { + const sqlText = 'select * from fakeTable'; + const timeoutInMs = 1000; // 1 second + let queryId; + + async.series( + [ + // Execute query in async mode + function (callback) { + connection.execute({ + sqlText: sqlText, + asyncExec: true, + complete: async function (err, stmt) { + assert.ok(!err); + queryId = stmt.getQueryId(); + callback(); + } + }); + }, + async function () { + // Wait for query to finish executing + while (connection.isStillRunning(await connection.getQueryStatus(queryId))) { + await new Promise((resolve) => { + setTimeout(() => resolve(), timeoutInMs); + }); + } + + // Check query status failed + const status = await connection.getQueryStatus(queryId); + assert.strictEqual(QueryStatus[status], QueryStatus.FAILED_WITH_ERROR); + assert.ok(connection.isAnError(status)); + + // Check getting the query status throws an error + try { + await connection.getQueryStatusThrowIfError(queryId); + assert.fail(); + } catch (err) { + assert.ok(err); + } + + // Check getting the results throws an error + try { + await connection.getResultsFromQueryId({ queryId: queryId }); + assert.fail(); + } catch (err) { + assert.ok(err); + } + } + ], + done + ); + }); + + it('testMixedSyncAndAsyncQueries', function (done) { + const expectedSeconds = '3'; + const sqlTextForAsync = `CALL SYSTEM$WAIT(${expectedSeconds}, 'SECONDS')`; + const sqlTextForSync = 'select 1'; + let queryId; + + async.series( + [ + // Execute query in async mode + function (callback) { + connection.execute({ + sqlText: sqlTextForAsync, + asyncExec: true, + complete: async function (err, stmt) { + assert.ok(!err); + queryId = stmt.getQueryId(); + const status = await connection.getQueryStatus(queryId); + assert.ok(connection.isStillRunning(status)); + callback(); + } + }); + }, + // Execute a different query in non-async mode + function (callback) { + testUtil.executeCmd(connection, sqlTextForSync, callback); + }, + // Get results using query id + function (callback) { + connection.getResultsFromQueryId({ + queryId: queryId, + complete: async function (_err, _stmt, rows) { + const status = await connection.getQueryStatus(queryId); + assert.strictEqual(QueryStatus[status], QueryStatus.SUCCESS); + assert.strictEqual(rows[0]['SYSTEM$WAIT'], `waited ${expectedSeconds} seconds`); + callback(); + } + }); + } + ], + done + ); + }); + + it('testGetStatusOfInvalidQueryId', async function () { + const fakeQueryId = 'fakeQueryId'; + + // Get the query status using an invalid query id + try { + // Should fail from invalid uuid + await connection.getQueryStatus(fakeQueryId); + assert.fail(); + } catch (err) { + assert.strictEqual(err.code, ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID); + } + }); + + it('testGetResultsOfInvalidQueryId', async function () { + const fakeQueryId = 'fakeQueryId'; + + // Get the queryresults using an invalid query id + try { + // Should fail from invalid uuid + await connection.getResultsFromQueryId({ queryId: fakeQueryId }); + assert.fail(); + } catch (err) { + assert.strictEqual(err.code, ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID); + } + }); + + it('testGetStatusOfUnknownQueryId', async function () { + const unknownQueryId = '12345678-1234-4123-A123-123456789012'; + + // Get the query status using an unknown query id + const status = await connection.getQueryStatus(unknownQueryId); + assert.strictEqual(QueryStatus[status], QueryStatus.NO_DATA); + }); + + // The test retries until it reaches the max retry count and sometimes it fails due to timeout + it.skip('testGetResultsOfUnknownQueryId', async function () { + const unknownQueryId = '12345678-1234-4123-A123-123456789012'; + + // Get the query results using an unknown query id + try { + // Should fail from exceeding NO_DATA retry count + await connection.getResultsFromQueryId({ queryId: unknownQueryId }); + assert.fail(); + } catch (err) { + assert.strictEqual(err.code, ErrorCodes.ERR_GET_RESULTS_QUERY_ID_NO_DATA); + } + }); +}); diff --git a/test/unit/mock/mock_http_client.js b/test/unit/mock/mock_http_client.js index 551733cf7..b76137e92 100644 --- a/test/unit/mock/mock_http_client.js +++ b/test/unit/mock/mock_http_client.js @@ -77,6 +77,36 @@ MockHttpClient.prototype.request = function (request) }, delay); }; +/** + * Issues an async request. + * + * @param {Object} request the request options. + */ +MockHttpClient.prototype.requestAsync = function (request) { + // build the request-to-output map if this is the first request + if (!this._mapRequestToOutput) { + this._mapRequestToOutput = + buildRequestToOutputMap(buildRequestOutputMappings(this._clientInfo)); + } + + // Closing a connection includes a requestID as a query parameter in the url + // Example: http://fake504.snowflakecomputing.com/session?delete=true&requestId=a40454c6-c3bb-4824-b0f3-bae041d9d6a2 + if (request.url.includes('session?delete=true')) { + // Remove the requestID query parameter for the mock HTTP client + request.url = request.url.substring(0, request.url.indexOf('&requestId=')); + } + + // get the output of the specified request from the map + const requestOutput = this._mapRequestToOutput[serializeRequest(request)]; + + Errors.assertInternal(Util.isObject(requestOutput), + 'no response available for: ' + serializeRequest(request)); + + const response = JSON.parse(JSON.stringify(requestOutput.response)); + + return response; +}; + /** * Builds a map in which the keys are requests (or rather, serialized versions * of the requests) and the values are the outputs of the corresponding request @@ -995,6 +1025,37 @@ function buildRequestOutputMappings(clientInfo) } } }, + { + request: + { + method: 'GET', + url: 'http://fakeaccount.snowflakecomputing.com/monitoring/queries/00000000-0000-0000-0000-000000000000', + headers: + { + 'Accept': 'application/json', + 'Authorization': 'Snowflake Token="SESSION_TOKEN"', + 'Content-Type': 'application/json' + } + }, + output: + { + err: null, + response: + { + statusCode: 200, + statusMessage: "OK", + data: + { + code: null, + data: { + queries: [{ status: 'RESTARTED' }] + }, + message: null, + success: true + } + } + } + }, { request: { diff --git a/test/unit/snowflake_test.js b/test/unit/snowflake_test.js index f95e7d071..f83cd6eed 100644 --- a/test/unit/snowflake_test.js +++ b/test/unit/snowflake_test.js @@ -5,6 +5,7 @@ var Util = require('./../../lib/util'); var ErrorCodes = require('./../../lib/errors').codes; var MockTestUtil = require('./mock/mock_test_util'); +const QueryStatus = require('./../../lib/constants/query_status').code; var assert = require('assert'); var async = require('async'); @@ -1481,6 +1482,316 @@ describe('statement.cancel()', function () }); }); +describe('connection.getResultsFromQueryId() asynchronous errors', function () { + const queryId = '00000000-0000-0000-0000-000000000000'; + + it('not success status', function (done) { + const connection = snowflake.createConnection(connectionOptions); + async.series( + [ + function (callback) { + connection.connect(function (err) { + assert.ok(!err, JSON.stringify(err)); + callback(); + }); + }, + async function () { + try { + await connection.getResultsFromQueryId({ queryId: queryId }); + assert.fail(); + } catch (err) { + assert.strictEqual(err.code, ErrorCodes.ERR_GET_RESULTS_QUERY_ID_NOT_SUCCESS_STATUS); + } + }, + function (callback) { + connection.destroy(function (err) { + assert.ok(!err, JSON.stringify(err)); + callback(); + }); + } + ], + done + ); + }); +}); + +describe('connection.getResultsFromQueryId() synchronous errors', function () { + const connection = snowflake.createConnection(connectionOptions); + + const testCases = + [ + { + name: 'missing queryId', + options: {}, + errorCode: ErrorCodes.ERR_CONN_FETCH_RESULT_MISSING_QUERY_ID + }, + { + name: 'undefined queryId', + options: { queryId: undefined }, + errorCode: ErrorCodes.ERR_CONN_FETCH_RESULT_MISSING_QUERY_ID + }, + { + name: 'null queryId', + options: { queryId: null }, + errorCode: ErrorCodes.ERR_CONN_FETCH_RESULT_MISSING_QUERY_ID + }, + { + name: 'non-string queryId', + options: { queryId: 123 }, + errorCode: ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID + }, + { + name: 'invalid queryId', + options: { queryId: 'invalidQueryId' }, + errorCode: ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID + } + ]; + + testCases.forEach(testCase => { + it(testCase.name, async function () { + try { + await connection.getResultsFromQueryId(testCase.options); + } catch (err) { + assert.strictEqual(err.code, testCase.errorCode); + } + }); + }); +}); + +describe('connection.getQueryStatus() synchronous errors', function () { + const connection = snowflake.createConnection(connectionOptions); + + const testCases = + [ + { + name: 'undefined queryId', + queryId: undefined, + errorCode: ErrorCodes.ERR_CONN_FETCH_RESULT_MISSING_QUERY_ID + }, + { + name: 'null queryId', + queryId: null, + errorCode: ErrorCodes.ERR_CONN_FETCH_RESULT_MISSING_QUERY_ID + }, + { + name: 'non-string queryId', + queryId: 123, + errorCode: ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID + }, + { + name: 'invalid queryId', + queryId: 'invalidQueryId', + errorCode: ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID + } + ]; + + testCases.forEach(testCase => { + it(testCase.name, async function () { + try { + await connection.getQueryStatus(testCase.queryId); + } catch (err) { + assert.strictEqual(err.code, testCase.errorCode); + } + }); + }); +}); + +describe('connection.getQueryStatusThrowIfError() synchronous errors', function () { + const connection = snowflake.createConnection(connectionOptions); + + const testCases = + [ + { + name: 'undefined queryId', + queryId: undefined, + errorCode: ErrorCodes.ERR_CONN_FETCH_RESULT_MISSING_QUERY_ID + }, + { + name: 'null queryId', + queryId: null, + errorCode: ErrorCodes.ERR_CONN_FETCH_RESULT_MISSING_QUERY_ID + }, + { + name: 'non-string queryId', + queryId: 123, + errorCode: ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID + }, + { + name: 'invalid queryId', + queryId: 'invalidQueryId', + errorCode: ErrorCodes.ERR_GET_RESPONSE_QUERY_INVALID_UUID + } + ]; + + testCases.forEach(testCase => { + it(testCase.name, async function () { + try { + await connection.getQueryStatusThrowIfError(testCase.queryId); + } catch (err) { + assert.strictEqual(err.code, testCase.errorCode); + } + }); + }); +}); + +describe('snowflake.isStillRunning()', function () { + const connection = snowflake.createConnection(connectionOptions); + + const testCases = + [ + { + name: 'Running', + status: QueryStatus.RUNNING, + expectedValue: true + }, + { + name: 'Aborting', + status: QueryStatus.ABORTING, + expectedValue: false + }, + { + name: 'Success', + status: QueryStatus.SUCCESS, + expectedValue: false + }, + { + name: 'Failed with error', + status: QueryStatus.FAILED_WITH_ERROR, + expectedValue: false + }, + { + name: 'Aborted', + status: QueryStatus.ABORTED, + expectedValue: false + }, + { + name: 'Queued', + status: QueryStatus.QUEUED, + expectedValue: true + }, + { + name: 'Failed with incident', + status: QueryStatus.FAILED_WITH_INCIDENT, + expectedValue: false + }, + { + name: 'Disconnected', + status: QueryStatus.DISCONNECTED, + expectedValue: false + }, + { + name: 'Resuming warehouse', + status: QueryStatus.RESUMING_WAREHOUSE, + expectedValue: true + }, + { + name: 'Queued repairing warehouse', + status: QueryStatus.QUEUED_REPARING_WAREHOUSE, + expectedValue: true + }, + { + name: 'Restarted', + status: QueryStatus.RESTARTED, + expectedValue: false + }, + { + name: 'Blocked', + status: QueryStatus.BLOCKED, + expectedValue: false + }, + { + name: 'No data', + status: QueryStatus.NO_DATA, + expectedValue: true + }, + ]; + + testCases.forEach(testCase => { + it(testCase.name, function () { + assert.strictEqual(testCase.expectedValue, connection.isStillRunning(testCase.status)); + }); + }); +}); + +describe('snowflake.isAnError()', function () { + const connection = snowflake.createConnection(connectionOptions); + + const testCases = + [ + { + name: 'Running', + status: QueryStatus.RUNNING, + expectedValue: false + }, + { + name: 'Aborting', + status: QueryStatus.ABORTING, + expectedValue: true + }, + { + name: 'Success', + status: QueryStatus.SUCCESS, + expectedValue: false + }, + { + name: 'Failed with error', + status: QueryStatus.FAILED_WITH_ERROR, + expectedValue: true + }, + { + name: 'Aborted', + status: QueryStatus.ABORTED, + expectedValue: true + }, + { + name: 'Queued', + status: QueryStatus.QUEUED, + expectedValue: false + }, + { + name: 'Failed with incident', + status: QueryStatus.FAILED_WITH_INCIDENT, + expectedValue: true + }, + { + name: 'Disconnected', + status: QueryStatus.DISCONNECTED, + expectedValue: true + }, + { + name: 'Resuming warehouse', + status: QueryStatus.RESUMING_WAREHOUSE, + expectedValue: false + }, + { + name: 'Queued repairing warehouse', + status: QueryStatus.QUEUED_REPARING_WAREHOUSE, + expectedValue: false + }, + { + name: 'Restarted', + status: QueryStatus.RESTARTED, + expectedValue: false + }, + { + name: 'Blocked', + status: QueryStatus.BLOCKED, + expectedValue: true + }, + { + name: 'No data', + status: QueryStatus.NO_DATA, + expectedValue: false + }, + ]; + + testCases.forEach(testCase => { + it(testCase.name, function () { + assert.strictEqual(testCase.expectedValue, connection.isAnError(testCase.status)); + }); + }); +}); + describe('connection.destroy()', function () { it('destroy without connecting', function (done)