Skip to content

Commit

Permalink
SNOW-734920: Add possibility to configure keep alive value
Browse files Browse the repository at this point in the history
  • Loading branch information
sfc-gh-pmotacki committed Sep 29, 2023
1 parent f93a2e4 commit 2a12f30
Show file tree
Hide file tree
Showing 6 changed files with 118 additions and 57 deletions.
1 change: 1 addition & 0 deletions lib/constants/error_messages.js
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ exports[403002] = 'Invalid insecureConnect option. The specified value must be a
exports[403003] = 'Invalid OCSP mode. The specified value must be FAIL_CLOSED, FAIL_OPEN, or INSECURE_MODE.';
exports[403004] = 'Invalid custom JSON parser. The specified value must be a function.';
exports[403005] = 'Invalid custom XML parser. The specified value must be a function.';
exports[403006] = 'Invalid keep alive value. The specified value must be a boolean.';

// 404001
exports[404001] = 'Connection options must be specified.';
Expand Down
9 changes: 9 additions & 0 deletions lib/core.js
Original file line number Diff line number Diff line change
Expand Up @@ -217,6 +217,15 @@ function Core(options)
} else if (Util.exists(xmlParserConfig)) {
GlobalConfig.createXmlColumnVariantParserWithParameters(xmlParserConfig);
}

let keepAlive = options.keepAlive;
if (Util.exists(keepAlive))
{
Errors.checkArgumentValid(Util.isBoolean(keepAlive),
ErrorCodes.ERR_GLOBAL_CONFIGURE_INVALID_KEEP_ALIVE);

GlobalConfig.setKeepAlive(keepAlive);
}
}
};

Expand Down
1 change: 1 addition & 0 deletions lib/errors.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ codes.ERR_GLOBAL_CONFIGURE_INVALID_INSECURE_CONNECT = 403002;
codes.ERR_GLOBAL_CONFIGURE_INVALID_OCSP_MODE = 403003;
codes.ERR_GLOBAL_CONFIGURE_INVALID_JSON_PARSER = 403004;
codes.ERR_GLOBAL_CONFIGURE_INVALID_XML_PARSER = 403005;
codes.ERR_GLOBAL_CONFIGURE_INVALID_KEEP_ALIVE = 403006;

// 404001
codes.ERR_CONN_CREATE_MISSING_OPTIONS = 404001;
Expand Down
26 changes: 25 additions & 1 deletion lib/global_config.js
Original file line number Diff line number Diff line change
Expand Up @@ -255,5 +255,29 @@ function createXmlColumnVariantParser(config) {
throw new Error(validateResult.err.msg);
}
};

}

let keepAlive = true;

/**
* Updates the value of the 'keepAlive' parameter.
*
* @param {boolean} value
*/
exports.setKeepAlive = function (value)
{
// validate input
Errors.assertInternal(Util.isBoolean(value));
keepAlive = value;
};

/**
* Returns the overriden value of 'keepAlive' or default if not set. Default value is true
*
* @param {boolean} value
*/
exports.keepAlive = function (config) {
return keepAlive;
};


13 changes: 7 additions & 6 deletions lib/http/node.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ const Base = require('./base');
const HttpsAgent = require('../agent/https_ocsp_agent');
const HttpsProxyAgent = require('../agent/https_proxy_ocsp_agent');
const HttpAgent = require('http').Agent;
const GlobalConfig = require('../../lib/global_config');
const Logger = require('../logger');

/**
Expand Down Expand Up @@ -47,7 +48,7 @@ function getFromCacheOrCreate(agentClass, options, url) {
const parsed = Url.parse(url);
const protocol = parsed.protocol || 'http:';
const port = parsed.port || (protocol === 'http:' ? '80' : '443');
const agentId = `${protocol}//${parsed.hostname}:${port}`
const agentId = `${protocol}//${parsed.hostname}:${port}/${options.keepAlive}`
if (httpsAgentCache.has(agentId)) {
Logger.getInstance().trace(`Get agent with id: ${agentId} from cache`);
return httpsAgentCache.get(agentId);
Expand All @@ -72,11 +73,11 @@ function prepareProxyAgentOptions(agentOptions, proxy) {

function isBypassProxy(proxy, url, bypassProxy) {
if (proxy && proxy.noProxy) {
var bypassList = proxy.noProxy.split("|");
let bypassList = proxy.noProxy.split("|");
for (let i = 0; i < bypassList.length; i++) {
host = bypassList[i].trim();
host = host.replace("*", ".*?");
var matches = url.match(host);
let matches = url.match(host);
if (matches) {
Logger.getInstance().debug("bypassing proxy for %s", url);
return true;
Expand All @@ -89,9 +90,9 @@ function isBypassProxy(proxy, url, bypassProxy) {
/**
* @inheritDoc
*/
NodeHttpClient.prototype.getAgent = function (url, proxy, mock) {
NodeHttpClient.prototype.getAgent = function (url, proxy, mock, defaultAgentOptions) {
const isHttps = Url.parse(url).protocol === 'https:';
let agentOptions = {keepAlive: false};
let agentOptions = {keepAlive: GlobalConfig.keepAlive()};
var bypassProxy = isBypassProxy(proxy, url);

if (mock) {
Expand All @@ -106,7 +107,7 @@ NodeHttpClient.prototype.getAgent = function (url, proxy, mock) {
return getFromCacheOrCreate(HttpsAgent, agentOptions, url);
}
} else if (proxy && !bypassProxy) {
let proxyAgentOptions = prepareProxyAgentOptions();
const proxyAgentOptions = prepareProxyAgentOptions(agentOptions, proxy);
return getFromCacheOrCreate(HttpAgent, proxyAgentOptions, url);
} else {
return getFromCacheOrCreate(HttpAgent, agentOptions, url);
Expand Down
125 changes: 75 additions & 50 deletions test/integration/testKeepAlive.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,81 +7,106 @@ const connOption = require('./connectionOptions');
const async = require('async');
const testUtil = require('./testUtil');
const { performance } = require('perf_hooks');
const assert = require("assert");

describe.skip('keepAlive perf test', function ()
describe('keepAlive test', function ()
{
this.timeout(1000000);
let connection;
const loopCount = 5;
const rowCount = 10;
const tableName = 'test_keepalive000';

var connection;
const LOOP_COUNT = 100;
const createTableWithRandomStrings = `CREATE OR REPLACE TABLE ${tableName} (value string)
AS select randstr(200, random()) from table (generator(rowcount =>${rowCount}))`;

before(function (done)
{
console.time('execution');
done();
});
after(function (done)
{
testUtil.destroyConnection(connection, done);
after(async () => {
await testUtil.dropTablesIgnoringErrorsAsync(connection, [tableName]);
await testUtil.destroyConnectionAsync(connection);
});

it('Run query in loop', function (done)
{
let sumWithKeepAlive = 0;
let sumWithoutKeepAlive = 0;
async.series(
[
// Create the connection
function (callback)
// Run the query loopCount times
async function ()
{
console.time('connecting')
connection = snowflake.createConnection(connOption.valid);
connection.connect(function (err, conn)
await testUtil.connectAsync(connection);
await testUtil.executeCmdAsync(connection, createTableWithRandomStrings);
for (let count = 1; count <= loopCount; count++)
{
if (err)
{
console.error('Unable to connect: ' + err.message);
} else
await new Promise((resolve, reject) =>
{
console.timeEnd('connecting');
callback();
}
});
const start = performance.now();
connection.execute({
sqlText: `SELECT VALUE from ${tableName} limit ${rowCount};`,
streamResult: true,
complete: function (err, stmt) {
if (err) {
done(err);
} else {
stmt.streamRows()
.on('error', function (err) {
done(err);
})
.on('data', function (row) {
return;
})
.on('end', function (row) {
const end = performance.now();
const time = end - start;
sumWithKeepAlive += time;
resolve();
});
}
}
});
});
}
},
// Run the query LOOP_COUNT times
async function ()
{
var sum = 0;
for (var count = 1; count <= LOOP_COUNT; count++)
snowflake.configure({keepAlive: false});
connection = snowflake.createConnection(connOption.valid);
await testUtil.connectAsync(connection);
await testUtil.executeCmdAsync(connection, createTableWithRandomStrings);
for (let count = 1; count <= loopCount; count++)
{
await new Promise((resolve, reject) =>
{
var start = performance.now();
var statement = connection.execute({
sqlText: "SELECT VALUE from VARIANT_TABLE2 limit 10000;",
complete: function (err, stmt, rows)
{
var stream = statement.streamRows();
stream.on('error', function (err)
{
const start = performance.now();
connection.execute({
sqlText: `SELECT VALUE from ${tableName} limit ${rowCount};`,
streamResult: true,
complete: function (err, stmt) {
if (err) {
done(err);
});
stream.on('data', function (row)
{
return;
});
stream.on('end', function (row)
{
var end = performance.now();
var time = end - start;
console.log("query: " + time);
sum += time;
resolve();
});
} else {
stmt.streamRows()
.on('error', function (err) {
done(err);
})
.on('data', function (row) {
return;
})
.on('end', function (row) {
const end = performance.now();
const time = end - start;
sumWithoutKeepAlive += time;
resolve();
});
}
}
});
});
}
console.log("query average: " + (sum / LOOP_COUNT));
console.timeEnd('execution');
},
async function ()
{
assert.ok(sumWithoutKeepAlive/2 > sumWithKeepAlive, 'With keep alive queries should work more thdn two times faster');
}
],
done
Expand Down

0 comments on commit 2a12f30

Please sign in to comment.