diff --git a/lib/configuration/client_configuration.js b/lib/configuration/client_configuration.js new file mode 100644 index 000000000..2286cafef --- /dev/null +++ b/lib/configuration/client_configuration.js @@ -0,0 +1,152 @@ +/* + * Copyright (c) 2023 Snowflake Computing Inc. All rights reserved. + */ +const os = require('os'); +const path = require('path'); +const fs = require('fs'); +const clientConfigFileName = 'sf_client_config.json'; + +const Levels = Object.freeze({ + Off: 'OFF', + Error: 'ERROR', + Warn: 'WARN', + Info: 'INFO', + Debug: 'DEBUG', + Trace: 'TRACE' +}); + +const allLevels = Object.values(Levels); + +class ClientConfig { + constructor (loggingConfig) { + this.loggingConfig = loggingConfig; + } +} + +class ClientLoggingConfig { + constructor (logLevel, logPath) { + this.logLevel = logLevel; + this.logPath = logPath; + } +} + +class ConfigurationError extends Error { + name = 'ConfigurationError'; + + constructor(message, cause) { + super(message); + this.cause = cause; + Error.captureStackTrace(this, this.constructor); + } + + toString() { + return this.message + ': ' + this.cause.toString(); + } +} + +/** + * @param value {String} Log level. + * @return {String} normalized log level value. + * @throws {Error} Error for unknown value. + */ +function levelFromString (value) { + const level = value.toUpperCase(); + if (!allLevels.includes(level)) { + throw new Error('Unknown log level: ' + value); + } + return level; +} + +/** + * @param fsPromisesModule {module} filestream module + * @param processModule {processModule} process module + */ +function ConfigurationUtil(fsPromisesModule, processModule) { + + const fsPromises = typeof fsPromisesModule !== 'undefined' ? fsPromisesModule : require('fs/promises'); + const process = typeof processModule !== 'undefined' ? processModule : require('process'); + + /** + * @param configFilePath {String} A path to a client config file. + * @return {Promise} Client configuration. + */ + this.getClientConfig = async function (configFilePath) { + const path = await findConfig(configFilePath); + if (path == null) { + return null; + } + const configFileContents = await readFileConfig(path); + return configFileContents == null ? null : parseConfigFile(configFileContents); + }; + + function readFileConfig (filePath) { + if (!filePath) { + return Promise.resolve(null); + } + return fsPromises.readFile(filePath, { encoding: 'utf8' }) + .catch(err => { + throw new ConfigurationError('Finding client configuration failed', err); + }); + } + + function parseConfigFile (configurationJson) { + try { + const parsedConfiguration = JSON.parse(configurationJson); + validate(parsedConfiguration); + return new ClientConfig( + new ClientLoggingConfig( + logLevel(parsedConfiguration), + logPath(parsedConfiguration) + ) + ); + } catch (err) { + throw new ConfigurationError('Parsing client configuration failed', err); + } + } + + function validate (configuration) { + const level = logLevel(configuration); + if (level != null) { + levelFromString(level); + } + } + + function logLevel (configuration) { + return configuration.common.log_level; + } + + function logPath (configuration) { + return configuration.common.log_path; + } + + function findConfig (filePathFromConnectionString) { + return verifyNotEmpty(filePathFromConnectionString) + .then((filePath) => filePath ?? getFilePathFromEnvironmentVariable()) + .then((filePath) => filePath ?? searchForConfigInDictionary('.')) + .then((filePath) => filePath ?? searchForConfigInDictionary(os.homedir())) + .then((filePath) => filePath ?? searchForConfigInDictionary(os.tmpdir())); + } + + async function verifyNotEmpty (filePath) { + return filePath ? filePath : null; + } + + function getFilePathFromEnvironmentVariable () { + return verifyNotEmpty(process.env.SF_CLIENT_CONFIG_FILE); + } + + async function searchForConfigInDictionary (dictionary) { + const filePath = path.join(dictionary, clientConfigFileName); + return onlyIfFileExists(filePath); + } + + async function onlyIfFileExists (filePath) { + return await fsPromises.access(filePath, fs.constants.F_OK) + .then(() => filePath) + .catch(() => null); + } +} + +exports.Levels = Levels; +exports.levelFromString = levelFromString; +exports.ConfigurationUtil = ConfigurationUtil; diff --git a/lib/connection/connection.js b/lib/connection/connection.js index 5822b4137..fd15e2308 100644 --- a/lib/connection/connection.js +++ b/lib/connection/connection.js @@ -14,6 +14,7 @@ var Statement = require('./statement'); var Parameters = require('../parameters'); var Authenticator = require('../authentication/authentication'); var Logger = require('../logger'); +const {init: initEasyLogging} = require('../logger/easy_logging_starter') const PRIVATELINK_URL_SUFFIX = ".privatelink.snowflakecomputing.com"; @@ -261,13 +262,19 @@ function Connection(context) throw authErr; } - // Request connection - services.sf.connect({ - callback: connectCallback(self, callback), - json: body - }); - - // return the connection to facilitate chaining + initEasyLogging(connectionConfig.clientConfigFile) + .then(() => { + try { + services.sf.connect({ + callback: connectCallback(self, callback), + json: body + }); + } catch (e) { + // we don't expect an error here since callback method should be called + Logger.getInstance().error('Unexpected error from calling callback function', e); + } + }) + .catch(() => callback(Errors.createClientError(ErrorCodes.ERR_CONN_CONNECT_INVALID_CLIENT_CONFIG))); return this; }; @@ -301,6 +308,11 @@ function Connection(context) try { + try { + await initEasyLogging(connectionConfig.clientConfigFile); + } catch (err) { + throw Errors.createClientError(ErrorCodes.ERR_CONN_CONNECT_INVALID_CLIENT_CONFIG); + } await auth.authenticate(connectionConfig.getAuthenticator(), connectionConfig.getServiceName(), connectionConfig.account, diff --git a/lib/connection/connection_config.js b/lib/connection/connection_config.js index 5e355cdfc..7fc76fa5d 100644 --- a/lib/connection/connection_config.js +++ b/lib/connection/connection_config.js @@ -405,6 +405,11 @@ function ConnectionConfig(options, validateCredentials, qaMode, clientInfo) ErrorCodes.ERR_CONN_CREATE_INVALID_GCS_USE_DOWNSCOPED_CREDENTIAL); } + var clientConfigFile = options.clientConfigFile; + if (Util.exists(clientConfigFile)) { + Errors.checkArgumentValid(Util.isString(clientConfigFile), ErrorCodes.ERR_CONN_CREATE_INVALID_CLIENT_CONFIG_FILE); + } + // remember if we're in qa mode this._qaMode = qaMode; @@ -474,7 +479,7 @@ function ConnectionConfig(options, validateCredentials, qaMode, clientInfo) if (Util.exists(options.disableQueryContextCache)) { Errors.checkArgumentValid(Util.isBoolean(options.disableQueryContextCache), ErrorCodes.ERR_CONN_CREATE_INVALID_DISABLED_QUERY_CONTEXT_CACHE); - + disableQueryContextCache = options.disableQueryContextCache; } @@ -494,7 +499,7 @@ function ConnectionConfig(options, validateCredentials, qaMode, clientInfo) if (Util.exists(options.includeRetryReason)) { Errors.checkArgumentValid(Util.isBoolean(options.includeRetryReason), ErrorCodes.ERR_CONN_CREATE_INVALID_INCLUDE_RETRY_REASON); - + includeRetryReason = options.includeRetryReason; } @@ -763,14 +768,14 @@ function ConnectionConfig(options, validateCredentials, qaMode, clientInfo) /** * Returns whether the Retry reason is included or not in the retry url - * + * * @returns {Boolean} */ this.getIncludeRetryReason = function () { return includeRetryReason; } - /** + /** * Returns whether the Query Context Cache is enabled or not by the configuration * * @returns {Boolean} @@ -779,6 +784,15 @@ function ConnectionConfig(options, validateCredentials, qaMode, clientInfo) return disableQueryContextCache; } + /** + * Returns the client config file + * + * @returns {String} + */ + this.getClientConfigFile = function () { + return clientConfigFile; + }; + // save config options this.username = options.username; this.password = options.password; @@ -790,6 +804,7 @@ function ConnectionConfig(options, validateCredentials, qaMode, clientInfo) this.masterToken = options.masterToken; this.masterTokenExpirationTime = options.masterTokenExpirationTime; this.sessionTokenExpirationTime = options.sessionTokenExpirationTime; + this.clientConfigFile = options.clientConfigFile; // create the parameters array var parameters = createParameters(); diff --git a/lib/constants/error_messages.js b/lib/constants/error_messages.js index 6dca66dab..ad6b407c2 100644 --- a/lib/constants/error_messages.js +++ b/lib/constants/error_messages.js @@ -67,6 +67,7 @@ exports[404039] = 'Invalid forceStageBindError. The specified value must be a nu exports[404040] = 'Invalid browser timeout value. The specified value must be a positive number.'; exports[404041] = 'Invalid disablQueryContextCache. The specified value must be a boolean.'; exports[404042] = 'Invalid includeRetryReason. The specified value must be a boolean.' +exports[404043] = 'Invalid clientConfigFile value. The specified value must be a string.'; // 405001 exports[405001] = 'Invalid callback. The specified value must be a function.'; @@ -76,6 +77,7 @@ exports[405501] = 'Connection already in progress.'; exports[405502] = 'Already connected.'; exports[405503] = 'Connection already terminated. Cannot connect again.'; exports[405504] = 'connect() does not work with external browser or okta authenticators, call connectAsync() instead'; +exports[405505] = 'Configuration from client config file failed'; // 406001 exports[406001] = 'Invalid callback. The specified value must be a function.'; diff --git a/lib/core.js b/lib/core.js index e9b2a7be0..abc55b7d4 100644 --- a/lib/core.js +++ b/lib/core.js @@ -166,16 +166,13 @@ function Core(options) */ configure: function (options) { - var logTag = options.logLevel; - if (Util.exists(logTag)) - { - // check that the specified value is a valid tag - Errors.checkArgumentValid(LoggerCore.isValidLogTag(logTag), - ErrorCodes.ERR_GLOBAL_CONFIGURE_INVALID_LOG_LEVEL); - + const logLevel = extractLogLevel(options); + const logFilePath = options.logFilePath; + if (logLevel != null || logFilePath) { Logger.getInstance().configure( { - level: LoggerCore.logTagToLevel(logTag) + level: logLevel, + filePath: logFilePath }); } @@ -228,6 +225,17 @@ function Core(options) } }; + function extractLogLevel(options) { + const logTag = options.logLevel; + if (Util.exists(logTag)) { + Errors.checkArgumentValid(LoggerCore.isValidLogTag(logTag), + ErrorCodes.ERR_GLOBAL_CONFIGURE_INVALID_LOG_LEVEL); + + return LoggerCore.logTagToLevel(logTag) + } + return undefined; + } + // add some read-only constants var nativeTypeValues = DataTypes.NativeTypes.values; Object.defineProperties(instance, diff --git a/lib/errors.js b/lib/errors.js index d543725bb..c04fe7b5a 100644 --- a/lib/errors.js +++ b/lib/errors.js @@ -72,6 +72,7 @@ codes.ERR_CONN_CREATE_INVALID_FORCE_STAGE_BIND_ERROR = 404039; codes.ERR_CONN_CREATE_INVALID_BROWSER_TIMEOUT = 404040; codes.ERR_CONN_CREATE_INVALID_DISABLED_QUERY_CONTEXT_CACHE = 404041 codes.ERR_CONN_CREATE_INVALID_INCLUDE_RETRY_REASON =404042 +codes.ERR_CONN_CREATE_INVALID_CLIENT_CONFIG_FILE = 404043; // 405001 codes.ERR_CONN_CONNECT_INVALID_CALLBACK = 405001; @@ -81,6 +82,7 @@ codes.ERR_CONN_CONNECT_STATUS_CONNECTING = 405501; // sql state: 08002 codes.ERR_CONN_CONNECT_STATUS_CONNECTED = 405502; // sql state: 08002 codes.ERR_CONN_CONNECT_STATUS_DISCONNECTED = 405503; // sql state: 08002 codes.ERR_CONN_CREATE_INVALID_AUTH_CONNECT = 405504; +codes.ERR_CONN_CONNECT_INVALID_CLIENT_CONFIG = 405505; // 406001 codes.ERR_CONN_DESTROY_INVALID_CALLBACK = 406001; diff --git a/lib/logger/browser.js b/lib/logger/browser.js index 5be73508b..0b40ff4ee 100644 --- a/lib/logger/browser.js +++ b/lib/logger/browser.js @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015-2019 Snowflake Computing Inc. All rights reserved. + * Copyright (c) 2015-2023 Snowflake Computing Inc. All rights reserved. */ var Util = require('../util'); @@ -45,8 +45,11 @@ function Logger(options) buffer.push(message); }; + const doNothingToReconfigure = function (filePath) { + }; + // create an inner implementation to which all our methods will be forwarded - var common = Core.createLogger(options, logMessage); + var common = Core.createLogger(options, logMessage, doNothingToReconfigure); /** * Configures this logger. diff --git a/lib/logger/core.js b/lib/logger/core.js index fb128e32c..5ca6abacd 100644 --- a/lib/logger/core.js +++ b/lib/logger/core.js @@ -1,44 +1,51 @@ /* - * Copyright (c) 2015-2019 Snowflake Computing Inc. All rights reserved. + * Copyright (c) 2015-2023 Snowflake Computing Inc. All rights reserved. */ -var moment = require('moment'); -var Util = require('../util'); -var Errors = require('../errors'); -var SecretDetector = new (require('../secret_detector.js'))(); +const moment = require('moment'); +const Util = require('../util'); +const Errors = require('../errors'); +const SecretDetector = new (require('../secret_detector.js'))(); -var LOG_LEVEL_ERROR = +const LOG_LEVEL_OFF = + { + tag: 'OFF', + level: -1 + }; + +const LOG_LEVEL_ERROR = { tag: 'ERROR', level: 0 }; -var LOG_LEVEL_WARN = +const LOG_LEVEL_WARN = { tag: 'WARN', level: 1 }; -var LOG_LEVEL_INFO = +const LOG_LEVEL_INFO = { tag: 'INFO', level: 2 }; -var LOG_LEVEL_DEBUG = +const LOG_LEVEL_DEBUG = { tag: 'DEBUG', level: 3 }; -var LOG_LEVEL_TRACE = +const LOG_LEVEL_TRACE = { tag: 'TRACE', level: 4 }; -var LOG_LEVELS = +const LOG_LEVELS = [ + LOG_LEVEL_OFF, LOG_LEVEL_ERROR, LOG_LEVEL_WARN, LOG_LEVEL_INFO, @@ -47,6 +54,7 @@ var LOG_LEVELS = ]; const LOG_LEVEL_TAGS = { + OFF: LOG_LEVEL_OFF.tag, ERROR: LOG_LEVEL_ERROR.tag, WARN: LOG_LEVEL_WARN.tag, INFO: LOG_LEVEL_INFO.tag, @@ -58,36 +66,35 @@ exports.LOG_LEVEL_TAGS = LOG_LEVEL_TAGS; // create two maps, one in which the key is the log level and the value is the // corresponding log level object, and another in which the key is the log tag // and the value is the corresponding log level -var MAP_LOG_LEVEL_TO_OBJECT = {}; -var MAP_LOG_TAG_TO_LEVEL = {}; -for (var index = 0, length = LOG_LEVELS.length; index < length; index++) -{ - var logLevelObject = LOG_LEVELS[index]; +const MAP_LOG_LEVEL_TO_OBJECT = {}; +const MAP_LOG_TAG_TO_LEVEL = {}; +for (let index = 0, length = LOG_LEVELS.length; index < length; index++) { + const logLevelObject = LOG_LEVELS[index]; MAP_LOG_LEVEL_TO_OBJECT[logLevelObject.level] = logLevelObject; MAP_LOG_TAG_TO_LEVEL[logLevelObject.tag] = logLevelObject.level; } -var DEFAULT_BUFFER_MAX_LENGTH = 500; -var DEFAULT_MESSAGE_MAX_LENGTH = 500; -var DEFAULT_LEVEL = LOG_LEVEL_INFO; +const DEFAULT_BUFFER_MAX_LENGTH = 500; +const DEFAULT_MESSAGE_MAX_LENGTH = 500; +const DEFAULT_LEVEL = LOG_LEVEL_INFO; /** * Creates a new Logger instance. * - * @param {Object} options - * @param {Function} logMessage + * @param options {Object} + * @param logMessage {Function} + * @param reconfigureOperation {Function} Action to perform to change log destination file * * @returns {Object} */ -exports.createLogger = function (options, logMessage) -{ +exports.createLogger = function (options, logMessage, reconfigureOperation) { // a log function must be specified Errors.assertInternal(Util.isFunction(logMessage)); /** * Whether to include the current timestamp in the log message. */ - var includeTimestamp; + let includeTimestamp; /** * The maximum size (in terms of number of messages) to which the log buffer @@ -95,14 +102,14 @@ exports.createLogger = function (options, logMessage) * * @type {Number} */ - var bufferMaxLength; + let bufferMaxLength; /** * The maximum message length. Longer messages will be truncated. * * @type {Number} */ - var messageMaxLength; + let messageMaxLength; /** * The current log level. Any message logged at a lower level won't be added @@ -110,180 +117,159 @@ exports.createLogger = function (options, logMessage) * * @type {Object} */ - var currlevelObject; + let currlevelObject; // create a new logger instance - var logger = - { - /** - * Configures this logger. - * - * @param {Object} options - */ - configure: function (options) - { - var localIncludeTimestamp; - var localBufferMaxLength; - var localMessageMaxLength; - var localLevel; - - // if an options argument is specified - if (Util.exists(options)) - { - // make sure it's an object - Errors.assertInternal(Util.isObject(options)); - - localIncludeTimestamp = options.includeTimestamp; - localBufferMaxLength = options.bufferMaxLength; - localMessageMaxLength = options.messageMaxLength; - localLevel = options.level; - } - - // if an includeTimestamp options is specified, convert it to a boolean - if (Util.exists(localIncludeTimestamp)) - { - includeTimestamp = !!localIncludeTimestamp; - } - else if (!Util.exists(includeTimestamp)) - { - // default to true - includeTimestamp = true; - } - - // if a bufferMaxLength option is specified, make sure - // it's a positive integer before updating the logger option - if (Util.exists(localBufferMaxLength)) - { - Errors.assertInternal( - Util.number.isPositiveInteger(localBufferMaxLength)); - bufferMaxLength = localBufferMaxLength; - } - // initialize logger option if configure() hasn't been called before - else if (!Util.exists(bufferMaxLength)) - { - bufferMaxLength = DEFAULT_BUFFER_MAX_LENGTH; - } - - // if a messageMaxLength option is specified, make sure - // it's a positive integer before updating the logger option - if (Util.exists(localMessageMaxLength)) - { - Errors.assertInternal( - Util.number.isPositiveInteger(localMessageMaxLength)); - messageMaxLength = localMessageMaxLength; - } - // initialize logger option if configure() hasn't been called before - else if (!Util.exists(messageMaxLength)) - { - messageMaxLength = DEFAULT_MESSAGE_MAX_LENGTH; - } - - // if a level option is specified, make sure - // it's valid before updating the logger option - if (Util.exists(localLevel)) - { - Errors.assertInternal( - MAP_LOG_LEVEL_TO_OBJECT.hasOwnProperty(localLevel)); - currlevelObject = MAP_LOG_LEVEL_TO_OBJECT[localLevel]; - } - // initialize logger option if configure() hasn't been called before - else if (!Util.exists(currlevelObject)) - { - currlevelObject = DEFAULT_LEVEL; - } - }, - - /** - * Logs a given message at the error level. - * - * @param {String} message - */ - error: function (message) - { - log(LOG_LEVEL_ERROR, message, getMessageArgs(arguments)); - }, - - /** - * Logs a given message at the warn level. - * - * @param {String} message - */ - warn: function (message) - { - log(LOG_LEVEL_WARN, message, getMessageArgs(arguments)); - }, - - /** - * Logs a given message at the info level. - * - * @param {String} message - */ - info: function (message) - { - log(LOG_LEVEL_INFO, message, getMessageArgs(arguments)); - }, - - /** - * Logs a given message at the debug level. - * - * @param {String} message - */ - debug: function (message) - { - log(LOG_LEVEL_DEBUG, message, getMessageArgs(arguments)); - }, - - /** - * Logs a given message at the trace level. - * - * @param {String} message - */ - trace: function (message) - { - log(LOG_LEVEL_TRACE, message, getMessageArgs(arguments)); - }, - - /** - * Returns the log buffer. - * - * @returns {String[]} - */ - getLogBuffer: function () - { - return []; - }, - - /** - * Returns the level number associated with the current log level. - * - * @returns {Number} - */ - getLevelNumber: function () - { - return currlevelObject.level; - }, - - /** - * Returns the tag associated with the current log level. - * - * @returns {String} - */ - getLevelTag: function () - { - return currlevelObject.tag; - }, - - /** - * Returns a map in which the keys are the level tags and the values are the - * corresponding log levels. - * - * @returns {Object} - */ - getLevelTagsMap: function () - { - return MAP_LOG_TAG_TO_LEVEL; + const logger = { + /** + * Configures this logger. + * + * @param {Object} options + */ + configure: function (options) { + let localIncludeTimestamp; + let localBufferMaxLength; + let localMessageMaxLength; + let localLevel; + let localFilePath; + + // if an options argument is specified + if (Util.exists(options)) { + // make sure it's an object + Errors.assertInternal(Util.isObject(options)); + + localIncludeTimestamp = options.includeTimestamp; + localBufferMaxLength = options.bufferMaxLength; + localMessageMaxLength = options.messageMaxLength; + localLevel = options.level; + localFilePath = options.filePath; + } + + // if an includeTimestamp options is specified, convert it to a boolean + if (Util.exists(localIncludeTimestamp)) { + includeTimestamp = !!localIncludeTimestamp; + } else if (!Util.exists(includeTimestamp)) { + // default to true + includeTimestamp = true; } - }; + + // if a bufferMaxLength option is specified, make sure + // it's a positive integer before updating the logger option + if (Util.exists(localBufferMaxLength)) { + Errors.assertInternal( + Util.number.isPositiveInteger(localBufferMaxLength)); + bufferMaxLength = localBufferMaxLength; + } else if (!Util.exists(bufferMaxLength)) { // initialize logger option if configure() hasn't been called before + bufferMaxLength = DEFAULT_BUFFER_MAX_LENGTH; + } + + // if a messageMaxLength option is specified, make sure + // it's a positive integer before updating the logger option + if (Util.exists(localMessageMaxLength)) { + Errors.assertInternal( + Util.number.isPositiveInteger(localMessageMaxLength)); + messageMaxLength = localMessageMaxLength; + } else if (!Util.exists(messageMaxLength)) { // initialize logger option if configure() hasn't been called before + messageMaxLength = DEFAULT_MESSAGE_MAX_LENGTH; + } + + // if a level option is specified, make sure + // it's valid before updating the logger option + if (Util.exists(localLevel)) { + Errors.assertInternal( + MAP_LOG_LEVEL_TO_OBJECT.hasOwnProperty(localLevel)); + currlevelObject = MAP_LOG_LEVEL_TO_OBJECT[localLevel]; + } else if (!Util.exists(currlevelObject)) { // initialize logger option if configure() hasn't been called before + currlevelObject = DEFAULT_LEVEL; + } + + if (Util.exists(localFilePath)) { + reconfigureOperation(localFilePath); + } + }, + + /** + * Logs a given message at the error level. + * + * @param {String} message + */ + error: function (message) { + log(LOG_LEVEL_ERROR, message, getMessageArgs(arguments)); + }, + + /** + * Logs a given message at the warn level. + * + * @param {String} message + */ + warn: function (message) { + log(LOG_LEVEL_WARN, message, getMessageArgs(arguments)); + }, + + /** + * Logs a given message at the info level. + * + * @param {String} message + */ + info: function (message) { + log(LOG_LEVEL_INFO, message, getMessageArgs(arguments)); + }, + + /** + * Logs a given message at the debug level. + * + * @param {String} message + */ + debug: function (message) { + log(LOG_LEVEL_DEBUG, message, getMessageArgs(arguments)); + }, + + /** + * Logs a given message at the trace level. + * + * @param {String} message + */ + trace: function (message) { + log(LOG_LEVEL_TRACE, message, getMessageArgs(arguments)); + }, + + /** + * Returns the log buffer. + * + * @returns {String[]} + */ + getLogBuffer: function () { + return []; + }, + + /** + * Returns the level number associated with the current log level. + * + * @returns {Number} + */ + getLevelNumber: function () { + return currlevelObject.level; + }, + + /** + * Returns the tag associated with the current log level. + * + * @returns {String} + */ + getLevelTag: function () { + return currlevelObject.tag; + }, + + /** + * Returns a map in which the keys are the level tags and the values are the + * corresponding log levels. + * + * @returns {Object} + */ + getLevelTagsMap: function () { + return MAP_LOG_TAG_TO_LEVEL; + } + }; // configure the logger logger.configure(options); @@ -295,25 +281,21 @@ exports.createLogger = function (options, logMessage) * @param {String} message the message template. * @param {String[]} messageArgs any arguments to substitute into the message. */ - var log = function (targetLevelObject, message, messageArgs) - { + const log = function (targetLevelObject, message, messageArgs) { // the message should not be logged if the target // level is more verbose than the current level - if (targetLevelObject.level <= currlevelObject.level) - { + if (targetLevelObject.level <= currlevelObject.level) { // substitute the messageArgs into the message template messageArgs.unshift(message); message = Util.format.apply(Util, messageArgs); // truncate the message if it's too long - if (message.length > messageMaxLength) - { + if (message.length > messageMaxLength) { message = message.slice(0, messageMaxLength); } // if needed, add the current time to the front of the message - if (includeTimestamp) - { + if (includeTimestamp) { message = Util.format( '[%s]: %s', moment().format('h:mm:ss.SSS A'), message); } @@ -329,10 +311,8 @@ exports.createLogger = function (options, logMessage) return logger; }; -exports.isValidLogTag = function (logTag) -{ - if (!Util.isString(logTag)) - { +exports.isValidLogTag = function (logTag) { + if (!Util.isString(logTag)) { return false; } @@ -341,11 +321,10 @@ exports.isValidLogTag = function (logTag) }; -exports.logTagToLevel = function (logTag) -{ +exports.logTagToLevel = function (logTag) { Errors.assertInternal(Util.isString(logTag)); return MAP_LOG_TAG_TO_LEVEL[logTag.toUpperCase()]; -} +}; /** * Helper method to extract the messageArgs from the arguments passed to @@ -355,7 +334,6 @@ exports.logTagToLevel = function (logTag) * * @returns {*} */ -function getMessageArgs(args) -{ - return Array.prototype.slice.call(args, 1) +function getMessageArgs(args) { + return Array.prototype.slice.call(args, 1); } diff --git a/lib/logger/easy_logging_starter.js b/lib/logger/easy_logging_starter.js new file mode 100644 index 000000000..d6b060780 --- /dev/null +++ b/lib/logger/easy_logging_starter.js @@ -0,0 +1,115 @@ +/* + * Copyright (c) 2023 Snowflake Computing Inc. All rights reserved. + */ +const os = require('os'); +const path = require('path'); +const fsPromises = require('fs/promises'); +const fs = require('fs'); +const {logTagToLevel} = require('./core'); +const {ConfigurationUtil, Levels} = require('../configuration/client_configuration'); +const Logger = require('../logger'); +const clientConfiguration = new ConfigurationUtil(); +const getClientConfig = clientConfiguration.getClientConfig; + +let initTrialParameters; + +/** + * @param {string} configFilePathFromConnectionString + * @returns {Promise} + */ +exports.init = async function (configFilePathFromConnectionString) { + try { + if (!allowedToInitialize(configFilePathFromConnectionString)) { + return; + } + const config = await getClientConfig(configFilePathFromConnectionString); + if (!config) { + initTrialParameters = { + configFilePathFromConnectionString: configFilePathFromConnectionString + }; + return; + } + const logLevel = mapLogLevel(config); + const logPath = await getLogPath(config); + const logger = Logger.getInstance(); + logger.configure({ + level: logLevel, + filePath: path.join(logPath, 'snowflake.log') + }); + logger.easyLoggingConfigureCounter = (logger.easyLoggingConfigureCounter ?? 0) + 1; + initTrialParameters = { + configFilePathFromConnectionString: configFilePathFromConnectionString + }; + } catch (err) { + const error = new EasyLoggingError('Failed to initialize easy logging', err); + Logger.getInstance().error(error.toString(), error); + throw error; + } +}; + +exports.reset = function () { + initTrialParameters = undefined; + Logger.getInstance().easyLoggingConfigureCounter = undefined; +}; + +function allowedToInitialize(configFilePathFromConnectionString) { + const everTriedToInitialize = !!initTrialParameters; + const triedToInitializeWithoutConfigFile = everTriedToInitialize && initTrialParameters.configFilePathFromConnectionString == null; + const isGivenConfigFilePath = !!configFilePathFromConnectionString; + const isAllowedToInitialize = !everTriedToInitialize || (triedToInitializeWithoutConfigFile && isGivenConfigFilePath); + if (!isAllowedToInitialize && initTrialParameters.configFilePathFromConnectionString !== configFilePathFromConnectionString) { + Logger.getInstance().warn(`Easy logging will not be configured for CLIENT_CONFIG_FILE=${configFilePathFromConnectionString} because it was previously configured for a different client config`); + } + return isAllowedToInitialize; +} + +function mapLogLevel(config) { + const configLogLevel = getLogLevel(config); + const logLevelNumber = logTagToLevel(configLogLevel); + if (logLevelNumber === null || logLevelNumber === undefined) { + throw new Error('Failed to convert configuration log level into internal one'); + } + return logLevelNumber; +} + +function getLogLevel(config) { + const logLevel = config.loggingConfig.logLevel; + if (!logLevel) { + Logger.getInstance().warn('LogLevel in client config not found. Using default value: OFF'); + return Levels.Off; + } + return logLevel; +} + +async function getLogPath(config) { + let logPath = config.loggingConfig.logPath; + if (!logPath) { + Logger.getInstance().warn('LogPath in client config not found. Using temporary directory as a default value'); + logPath = os.tmpdir(); + } + const pathWithNodeJsSubdirectory = path.join(logPath, 'nodejs'); + await fsPromises.access(pathWithNodeJsSubdirectory, fs.constants.F_OK) + .then(() => true) + .catch(() => { + try { + return fsPromises.mkdir(pathWithNodeJsSubdirectory, { recursive: true }); + } catch (err) { + throw new EasyLoggingError('Failed to create the directory for logs'); + } + }); + return pathWithNodeJsSubdirectory; +} + +class EasyLoggingError extends Error { + name = 'EasyLoggingError'; + + constructor(message, cause) { + super(message); + this.cause = cause; + Error.captureStackTrace(this, this.constructor); + } + + toString() { + return this.message + ': ' + this.cause.toString(); + } +} diff --git a/lib/logger/node.js b/lib/logger/node.js index c19d6d77c..fc61ae49b 100644 --- a/lib/logger/node.js +++ b/lib/logger/node.js @@ -1,9 +1,11 @@ /* - * Copyright (c) 2015-2019 Snowflake Computing Inc. All rights reserved. + * Copyright (c) 2015-2023 Snowflake Computing Inc. All rights reserved. */ var winston = require('winston'); var Core = require('./core'); +const Util = require('../util'); +const Errors = require('../errors'); /** * Creates a new Logger instance for when we're running in node. @@ -16,12 +18,76 @@ function Logger(options) { var common; var winstonLogger; + const defaultFilePath = 'snowflake.log'; + let filePath = getFilePath(options); this.setLogger = function (logger) { winstonLogger = logger; }; + /** + * This operation is for purpose of tests only. The idea is to force flushing logs to files. + * Winston logger emits 'finish' event before flushes all the transports so waiting for this event on logger is not good enough. + * We would like to close each transport and wait for the 'finish' event on each of them. + * However, it seems that 'finish' event on transports are not emitted + * so effectively we just close all transports and wait a given timeout for each of them. + * + * @param {long} The maximum number of milliseconds to wait after closing each of transports. + * @returns {Promise} + */ + this.closeAndFlushTransports = async function (timeoutMillis) { + return new Promise(async (resolve, reject) => { + if (transportsCreated()) { + for (const transport of winstonLogger.transports) { + await closeAndFlushTransport(transport, timeoutMillis); + } + } + resolve(); + }); + }; + + function transportsCreated() { + try { + winstonLogger.transports; + return true; + } catch (err) { + return false; + } + } + + async function closeAndFlushTransport (transport, timeoutMillis) { + if (!transport.close) { + return Promise.resolve(); + } + return new Promise(async (resolve, reject) => { + transport._markedAsDone = false; + function done() { + if (transport._markedAsDone) { + return; + } + transport._markedAsDone = true; + resolve(); + } + setTimeout(() => { + done(); + }, timeoutMillis); + transport.once('finish', () => { // actually I never found it happened + done(); + }); + transport.close(); + }); + } + + function reconfigureWinstonLogger(filePathInput) { + const currentWinstonLogger = winstonLogger; + filePath = filePathInput ?? filePath; + winstonLogger = null; // it will be created for the first log operation + if (currentWinstonLogger) { + currentWinstonLogger.close(); + } + } + /** * Logs a message at a given level. * @@ -36,13 +102,15 @@ function Logger(options) // initialize the winston logger if needed if (!winstonLogger) { + const transports = 'STDOUT' == filePath.toUpperCase() + ? [ new (winston.transports.Console)() ] + : [ + new (winston.transports.Console)(), + new (winston.transports.File)({filename: filePath}) + ]; winstonLogger = new winston.createLogger( { - transports: - [ - new (winston.transports.Console)(), - new (winston.transports.File)({filename: 'snowflake.log'}) - ], + transports: transports, level: common.getLevelTag(), levels: common.getLevelTagsMap() }); @@ -54,7 +122,15 @@ function Logger(options) }; // create an inner implementation to which all our methods will be forwarded - common = Core.createLogger(options, logMessage); + common = Core.createLogger(options, logMessage, reconfigureWinstonLogger); + + function getFilePath (options) { + if (Util.exists(options)) { + Errors.assertInternal(Util.isObject(options)); + return options.filePath ?? defaultFilePath; + } + return defaultFilePath; + } this.getLevelTag = function () { diff --git a/test/configureLogger.js b/test/configureLogger.js index 2d95eb2ad..dc323d890 100644 --- a/test/configureLogger.js +++ b/test/configureLogger.js @@ -3,7 +3,7 @@ const NodeLogger = require("../lib/logger/node"); const snowflake = require("../lib/snowflake"); /** - * @param logLevel one of ERROR | WARN | INFO | DEBUG | TRACE + * @param logLevel one of OFF | ERROR | WARN | INFO | DEBUG | TRACE */ exports.configureLogger = (logLevel = 'ERROR') => { Logger.setInstance(new NodeLogger()); diff --git a/test/integration/testEasyLoggingOnConnecting.js b/test/integration/testEasyLoggingOnConnecting.js new file mode 100644 index 000000000..449ac465e --- /dev/null +++ b/test/integration/testEasyLoggingOnConnecting.js @@ -0,0 +1,126 @@ +/* + * Copyright (c) 2023 Snowflake Computing Inc. All rights reserved. + */ +const snowflake = require('./../../lib/snowflake'); +const connOption = require('./connectionOptions'); +const Logger = require('./../../lib/logger'); +const {reset: resetEasyLoggingModule} = require('../../lib/logger/easy_logging_starter'); +const path = require('path'); +const os = require('os'); +const fsPromises = require('fs/promises'); +const assert = require('assert'); +const logLevelBefore = Logger.getInstance().getLevel(); +const {codes} = require('./../../lib/errors'); +const errorMessages = require('./../../lib/constants/error_messages'); +let tempDir = null; + +describe('Easy logging tests', function () { + + before(async function () { + tempDir = await fsPromises.mkdtemp(path.join(os.tmpdir(), 'easy_logging_connect_tests_')); + }); + + after(async function () { + await fsPromises.rm(tempDir, { recursive: true, force: true }); + Logger.getInstance().configure({ + level: logLevelBefore, + filePath: 'snowflake.log' + }); + }); + + afterEach(function () { + resetEasyLoggingModule(); + }); + + it('Should apply easy logging config when connection is being opened with callback', function (done) { + const logLevel = 'ERROR'; + createConfigFile(logLevel).then((configFilePath) => { + const configParameters = createConfigParameters(configFilePath); + const connection = snowflake.createConnection(configParameters); + connection.connect((err) => { + if (err) { + done(err); + } else { + assert.strictEqual(Logger.getInstance().getLevelTag(), logLevel); + done(); + } + }); + }); + }); + + it('Should fail for connecting with wrong easy logging config', function (done) { + const logLevel = 'something weird'; + createConfigFile(logLevel).then((configFilePath) => { + const configParameters = createConfigParameters(configFilePath); + const connection = snowflake.createConnection(configParameters); + connection.connect((err) => { + if (err) { + try { + assert.strictEqual(err.message, errorMessages[codes.ERR_CONN_CONNECT_INVALID_CLIENT_CONFIG]); + assert.strictEqual(err.code, codes.ERR_CONN_CONNECT_INVALID_CLIENT_CONFIG); + done(); + } catch (e) { + done(e); + } + } else { + done(new Error('Error should be thrown')); + } + }); + }); + }); + + it('Should apply easy logging config when connection is being opened asynchronously', async function (){ + // given + const logLevel = 'ERROR'; + const configFilePath = await createConfigFile(logLevel); + const configParameters = createConfigParameters(configFilePath); + const connection = snowflake.createConnection(configParameters); + + // when + await connection.connectAsync(); + + // then + assert.strictEqual(Logger.getInstance().getLevelTag(), logLevel); + }); + + it('Should fail to connect asynchronously with wrong easy logging config', async function (){ + // given + const logLevel = 'something weird'; + const configFilePath = await createConfigFile(logLevel); + const configParameters = createConfigParameters(configFilePath); + const connection = snowflake.createConnection(configParameters); + + // expect + await connection.connectAsync(err => { + if (err) { + assert.strictEqual(err.message, errorMessages[codes.ERR_CONN_CONNECT_INVALID_CLIENT_CONFIG]); + assert.strictEqual(err.code, codes.ERR_CONN_CONNECT_INVALID_CLIENT_CONFIG); + } else { + assert.fail('Error should be thrown'); + } + }); + }); + + function createConfigParameters(clientConfigFile) { + const config = {}; + Object.assign(config, connOption.valid); + config.clientConfigFile = clientConfigFile; + return config; + } + + async function createConfigFile(logLevel) { + const configFilePath = path.join(tempDir, 'config.json'); + const configContent = `{ + "common": { + "log_level": "${logLevel}", + "log_path": "${tempDir}" + } + }`; + await writeFile(configFilePath, configContent); + return configFilePath; + } + + async function writeFile (filePath, fileContent) { + await fsPromises.writeFile(filePath, fileContent, { encoding: 'utf8' }); + } +}); diff --git a/test/unit/configuration/configuration_finding_test.js b/test/unit/configuration/configuration_finding_test.js new file mode 100644 index 000000000..8f1d13e2d --- /dev/null +++ b/test/unit/configuration/configuration_finding_test.js @@ -0,0 +1,232 @@ +/* + * Copyright (c) 2023 Snowflake Computing Inc. All rights reserved. + */ +const os = require('os'); +const path = require('path'); +const assert = require('assert'); +const mock = require('mock-require'); +const {Levels, ClientConfig, ClientLoggingConfig, ConfigurationUtil} = require('./../../../lib/configuration/client_configuration'); +const defaultConfigName = 'sf_client_config.json'; +const configInDriverDirectory = path.join('.', defaultConfigName); +const configInHomeDirectory = path.join(os.homedir(), defaultConfigName); +const configInTempDirectory = path.join(os.tmpdir(), defaultConfigName); +const configFromEnvVariable = 'env_config.json'; +const configFromConnectionString = 'conn_config.json'; +const logLevel = Levels.Info; +const logPath = '/some-path/some-directory'; +const fileContent = `{ + "common": { + "log_level": "${logLevel}", + "log_path": "${logPath}" + } +}`; +const clientConfig = { + loggingConfig: { + logLevel: logLevel, + logPath: logPath + } +}; + +describe('Configuration finding tests', function () { + + it('should take config from connection string', async function() { + // given + const fsMock = new FsMock() + .mockFile(configFromConnectionString, fileContent) + .mockFile(configFromEnvVariable, 'random content') + .mockFile(configInDriverDirectory, 'random content') + .mockFile(configInHomeDirectory, 'random content') + .mockFile(configInTempDirectory, 'random content'); + mock('fs/promises', { + access: async function (path) { + return fsMock.access(path); + }, + readFile: async function (path){ + return fsMock.readFile(path); + } + }); + mock('process', { + env: { + SF_CLIENT_CONFIG_FILE: configFromEnvVariable + } + }); + const fsPromises = require('fs/promises'); + const process = require('process'); + const configUtil = new ConfigurationUtil(fsPromises, process); + + // when + const configFound = await configUtil.getClientConfig(configFromConnectionString); + + // then + assert.deepEqual(configFound, clientConfig); + }); + + it('should take config from environmental variable if no input present', async function () { + // given + const fsMock = new FsMock() + .mockFile(configFromEnvVariable, fileContent) + .mockFile(configInDriverDirectory, 'random content') + .mockFile(configInHomeDirectory, 'random content') + .mockFile(configInTempDirectory, 'random content'); + mock('fs/promises', { + access: async function (path) { + return fsMock.access(path); + }, + readFile: async function (path){ + return fsMock.readFile(path); + } + }); + mock('process', { + env: { + SF_CLIENT_CONFIG_FILE: configFromEnvVariable + } + }); + const fsPromises = require('fs/promises'); + const process = require('process'); + const configUtil = new ConfigurationUtil(fsPromises, process); + + // when + const configFound = await configUtil.getClientConfig(null); + + // then + assert.deepEqual(configFound, clientConfig); + }); + + it('should take config from driver directory if no input nor environmental variable present', async function () { + // given + const fsMock = new FsMock() + .mockFile(configInDriverDirectory, fileContent) + .mockFile(configInHomeDirectory, 'random content') + .mockFile(configInTempDirectory, 'random content'); + mock('fs/promises', { + access: async function (path) { + return fsMock.access(path); + }, + readFile: async function (path){ + return fsMock.readFile(path); + } + }); + mock('process', { + env: { + SF_CLIENT_CONFIG_FILE: undefined + } + }); + const fsPromises = require('fs/promises'); + const process = require('process'); + const configUtil = new ConfigurationUtil(fsPromises, process); + + // when + const configFound = await configUtil.getClientConfig(null); + + // then + assert.deepEqual(configFound, clientConfig); + }); + + it('should take config from home directory if no input nor environmental variable nor in driver directory present', async function () { + // given + const fsMock = new FsMock() + .mockFile(configInHomeDirectory, fileContent) + .mockFile(configInTempDirectory, 'random content'); + mock('fs/promises', { + access: async function (path) { + return fsMock.access(path); + }, + readFile: async function (path){ + return fsMock.readFile(path); + } + }); + mock('process', { + env: { + SF_CLIENT_CONFIG_FILE: undefined + } + }); + const fsPromises = require('fs/promises'); + const process = require('process'); + const configUtil = new ConfigurationUtil(fsPromises, process); + + // when + const configFound = await configUtil.getClientConfig(null); + + // then + assert.deepEqual(configFound, clientConfig); + }); + + it('should take config from temp directory if no other location possible', async function () { + // given + const fsMock = new FsMock() + .mockFile(configInTempDirectory, fileContent); + mock('fs/promises', { + access: async function (path) { + return fsMock.access(path); + }, + readFile: async function (path){ + return fsMock.readFile(path); + } + }); + mock('process', { + env: { + SF_CLIENT_CONFIG_FILE: undefined + } + }); + const fsPromises = require('fs/promises'); + const process = require('process'); + const configUtil = new ConfigurationUtil(fsPromises, process); + + // when + const configFound = await configUtil.getClientConfig(null); + + // then + assert.deepEqual(configFound, clientConfig); + }); + + it('should return null if config could not be found', async function () { + // given + const fsMock = new FsMock(); + mock('fs/promises', { + access: async function (path) { + return fsMock.access(path); + }, + readFile: async function (path){ + return fsMock.readFile(path); + } + }); + mock('process', { + env: { + SF_CLIENT_CONFIG_FILE: undefined + } + }); + const fsPromises = require('fs/promises'); + const process = require('process'); + const configUtil = new ConfigurationUtil(fsPromises, process); + + // when + const configFound = await configUtil.getClientConfig(null); + + // then + assert.strictEqual(configFound, null); + }); +}); + +class FsMock { + existingFiles = new Map(); + + constructor () {} + + mockFile(filePath, fileContents) { + this.existingFiles.set(filePath, fileContents); + return this; + } + + async access(filePath) { + if (!this.existingFiles.has(filePath)) { + throw new Error('File does not exist'); + } + } + + async readFile(filePath) { + if (!this.existingFiles.has(filePath)) { + throw new Error('File does not exist'); + } + return this.existingFiles.get(filePath); + } +} diff --git a/test/unit/configuration/configuration_parsing_test.js b/test/unit/configuration/configuration_parsing_test.js new file mode 100644 index 000000000..0f86c86bf --- /dev/null +++ b/test/unit/configuration/configuration_parsing_test.js @@ -0,0 +1,161 @@ +/* + * Copyright (c) 2023 Snowflake Computing Inc. All rights reserved. + */ +const assert = require('assert'); +const {Levels, ConfigurationUtil} = require('./../../../lib/configuration/client_configuration'); +const getClientConfig = new ConfigurationUtil().getClientConfig; +const fsPromises = require('fs/promises'); +const os = require('os'); +const path = require('path'); +let tempDir = null; + +describe('Configuration parsing tests', function () { + + before(async function () { + tempDir = await fsPromises.mkdtemp(path.join(os.tmpdir(), 'conf_parse_tests_')); + }); + + after(async function () { + await fsPromises.rm(tempDir, { recursive: true, force: true }); + }); + + [ + { + testCaseName: 'INFO', + logLevel: Levels.Info.toUpperCase() + }, + { + testCaseName: 'info', + logLevel: Levels.Info.toLowerCase() + }, + ].forEach(({testCaseName, logLevel}) => { + it('should parse json with log level: ' + testCaseName, async function () { + // given + const fileName = 'config.json'; + const filePath = path.join(tempDir, fileName); + const logPath = '/some-path/some-directory'; + const fileContent = `{ + "common": { + "log_level": "${logLevel}", + "log_path": "${logPath}" + } + }`; + await writeFile(filePath, fileContent); + + // when + const configuration = await getClientConfig(filePath); + + // then + assert.equal(configuration.loggingConfig.logLevel, logLevel); + assert.equal(configuration.loggingConfig.logPath, logPath); + }); + }); + + [ + { + testCaseName: 'config with nulls', + fileContent: `{ + "common": { + "log_level": null, + "log_path": null + } + }` + }, + { + testCaseName: 'config with empty common', + fileContent: `{ + "common": {} + }` + }, + ].forEach(({testCaseName, fileContent}) => { + it('should parse config without values: ' + testCaseName, async function () { + // given + const fileName = 'config_nulls_' + replaceSpaces(testCaseName) + '.json'; + const filePath = path.join(tempDir, fileName); + await fsPromises.writeFile(filePath, fileContent, { encoding: 'utf8' }); + + // when + const configuration = await getClientConfig(filePath); + + // then + assert.equal(configuration.logLevel, null); + assert.equal(configuration.logPath, null); + }); + }); + + [ + { + testCaseName: 'null', + filePath: null + }, + { + testCaseName: 'empty string', + filePath: '' + }, + { + testCaseName: 'undefined', + filePath: undefined + } + ].forEach(({testCaseName, filePath}) => { + it('should return null when config file not given: ' + testCaseName, async function () { + // when + const configuration = await getClientConfig(filePath); + + // then + assert.strictEqual(configuration, null); + }); + }); + + it('should fail when config file does not exist', async function () { + // expect + await assert.rejects( + async () => await getClientConfig('./not-existing-config.json'), + (err) => { + assert.strictEqual(err.name, 'ConfigurationError'); + assert.strictEqual(err.message, 'Finding client configuration failed'); + assert.match(err.cause.message, /ENOENT: no such file or directory./); + return true; + }); + }); + + [ + { + testCaseName: 'unknown log level', + fileContent: `{ + "common": { + "log_level": "unknown", + "log_path": "/some-path/some-directory" + } + }` + }, + { + testCaseName: 'no common in config', + fileContent: '{}' + }, + ].forEach(({testCaseName, fileContent}) => { + it('should fail for wrong config content ' + testCaseName, async function () { + // given + const fileName = 'config_wrong_' + replaceSpaces(testCaseName) + '.json'; + const filePath = path.join(tempDir, fileName); + await fsPromises.writeFile(filePath, fileContent, { encoding: 'utf8' }); + + // expect + await assert.rejects( + async () => await getClientConfig(filePath), + (err) => { + assert.strictEqual(err.name, 'ConfigurationError'); + assert.strictEqual(err.message, 'Parsing client configuration failed'); + assert.ok(err.cause); + return true; + }); + }); + }); + + function replaceSpaces(stringValue) { + return stringValue.replace(' ', '_'); + } + + async function writeFile (filePath, fileContent) { + await fsPromises.writeFile(filePath, fileContent, { encoding: 'utf8' }); + } +}); diff --git a/test/unit/connection/connection_config_test.js b/test/unit/connection/connection_config_test.js index 7b16499d6..e331a579a 100644 --- a/test/unit/connection/connection_config_test.js +++ b/test/unit/connection/connection_config_test.js @@ -523,7 +523,7 @@ describe('ConnectionConfig: basic', function () account: 'account', disableQueryContextCache: 1234 }, - errorCode: ErrorCodes.ERR_CONN_CREATE_INVALID_DISABLED_QUERY_CONTEXT_CACHE, + errorCode: ErrorCodes.ERR_CONN_CREATE_INVALID_DISABLED_QUERY_CONTEXT_CACHE, }, { name: 'invalid includeRetryReason', @@ -536,6 +536,16 @@ describe('ConnectionConfig: basic', function () }, errorCode: ErrorCodes.ERR_CONN_CREATE_INVALID_INCLUDE_RETRY_REASON, }, + { + name: 'invalid clientConfigFile', + options: { + account: 'account', + username: 'username', + password: 'password', + clientConfigFile: 15 + }, + errorCode: ErrorCodes.ERR_CONN_CREATE_INVALID_CLIENT_CONFIG_FILE + }, ]; var createNegativeITCallback = function (testCase) @@ -886,6 +896,24 @@ describe('ConnectionConfig: basic', function () password: 'password' } }, + { + name: 'client config file', + input: + { + account: 'account', + username: 'username', + password: 'password', + clientConfigFile: 'easy_logging_config.json' + }, + options: + { + accessUrl: 'https://account.snowflakecomputing.com', + username: 'username', + password: 'password', + account: 'account', + clientConfigFile: 'easy_logging_config.json' + } + }, ]; var createItCallback = function (testCase) diff --git a/test/unit/logger/browser_test.js b/test/unit/logger/browser_test.js index a48c492f0..405aaaa57 100644 --- a/test/unit/logger/browser_test.js +++ b/test/unit/logger/browser_test.js @@ -1,110 +1,127 @@ /* - * Copyright (c) 2015-2019 Snowflake Computing Inc. All rights reserved. + * Copyright (c) 2015-2023 Snowflake Computing Inc. All rights reserved. */ -var Logger = require('./../../../lib/logger/browser'); -var assert = require('assert'); +const Logger = require('./../../../lib/logger/browser'); +const {logTagToLevel, LOG_LEVEL_TAGS} = require('./../../../lib/logger/core'); +const assert = require('assert'); -describe('Logger - browser', function () -{ +describe('Logger - browser', function () { // TODO: negative tests // TODO: configuration tests - var ERROR = 'ERROR'; - var WARN = 'WARN'; - var INFO = 'INFO'; - var DEBUG = 'DEBUG'; - var TRACE = 'TRACE'; - - var LOG_MSG_ERROR = '0 error'; - var LOG_MSG_WARN = '1 warn'; - var LOG_MSG_INFO = '2 info'; - var LOG_MSG_DEBUG = '3 debug'; - var LOG_MSG_TRACE = '4 trace'; - - var FULL_LOG_MSG_ERROR = ERROR + ': ' + LOG_MSG_ERROR; - var FULL_LOG_MSG_WARN = WARN + ': ' + LOG_MSG_WARN; - var FULL_LOG_MSG_INFO = INFO + ': ' + LOG_MSG_INFO; - var FULL_LOG_MSG_DEBUG = DEBUG + ': ' + LOG_MSG_DEBUG; - var FULL_LOG_MSG_TRACE = TRACE + ': ' + LOG_MSG_TRACE; - - var logMessages = function (logger) - { + const ERROR = 'ERROR'; + const WARN = 'WARN'; + const INFO = 'INFO'; + const DEBUG = 'DEBUG'; + const TRACE = 'TRACE'; + + const LOG_MSG_ERROR = '0 error'; + const LOG_MSG_WARN = '1 warn'; + const LOG_MSG_INFO = '2 info'; + const LOG_MSG_DEBUG = '3 debug'; + const LOG_MSG_TRACE = '4 trace'; + + const FULL_LOG_MSG_ERROR = ERROR + ': ' + LOG_MSG_ERROR; + const FULL_LOG_MSG_WARN = WARN + ': ' + LOG_MSG_WARN; + const FULL_LOG_MSG_INFO = INFO + ': ' + LOG_MSG_INFO; + const FULL_LOG_MSG_DEBUG = DEBUG + ': ' + LOG_MSG_DEBUG; + const FULL_LOG_MSG_TRACE = TRACE + ': ' + LOG_MSG_TRACE; + + function logMessages (logger) { logger.error(LOG_MSG_ERROR); logger.warn(LOG_MSG_WARN); logger.info(LOG_MSG_INFO); logger.debug(LOG_MSG_DEBUG); logger.trace(LOG_MSG_TRACE); - }; + } - var createLogger = function (level) - { + function createLogger (level) { return new Logger( { includeTimestamp: false, level: level }); - }; + } - it('test all levels', function () - { - var logger; - var logBuffer; + it('should use info level as default', function () { + // given + const logger = createLogger(); - // default log level is 2 (Info) - logger = createLogger(); - assert.strictEqual(logger.getLevel(), 2); + // when + const level = logger.getLevel(); - // create a new logger with the log level set to error and log some messages - logger = createLogger(0); + // then + assert.strictEqual(level, logTagToLevel(LOG_LEVEL_TAGS.INFO)); + }); + + it('should log messages when a logger level is error', function () { + // given + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.ERROR)); + + // when logMessages(logger); - logBuffer = logger.getLogBuffer(); - // we should only have an error message + // then + const logBuffer = logger.getLogBuffer(); assert.strictEqual(logBuffer.length, 1); assert.strictEqual(logBuffer[0], FULL_LOG_MSG_ERROR); + }); + + it('should log messages when a logger level is warn', function () { + // given + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.WARN)); - // create a new logger with the log level set to warn and log some messages - logger = createLogger(1); + // when logMessages(logger); - logBuffer = logger.getLogBuffer(); - // we should have one error message and one warn message + // then + const logBuffer = logger.getLogBuffer(); assert.strictEqual(logBuffer.length, 2); assert.strictEqual(logBuffer[0], FULL_LOG_MSG_ERROR); assert.strictEqual(logBuffer[1], FULL_LOG_MSG_WARN); + }); - // create a new logger with the log level set to info and log some messages - logger = createLogger(2); + it('should log messages when a logger level is info', function () { + // given + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.INFO)); + + // when logMessages(logger); - logBuffer = logger.getLogBuffer(); - // we should have one error message, one warn message and one info message + // then + const logBuffer = logger.getLogBuffer(); assert.strictEqual(logBuffer.length, 3); assert.strictEqual(logBuffer[0], FULL_LOG_MSG_ERROR); assert.strictEqual(logBuffer[1], FULL_LOG_MSG_WARN); assert.strictEqual(logBuffer[2], FULL_LOG_MSG_INFO); + }); + + it('should log messages when a logger level is debug', function () { + // given + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.DEBUG)); - // create a new logger with the log level set to debug and log some messages - logger = createLogger(3); + // when logMessages(logger); - logBuffer = logger.getLogBuffer(); - // we should have one error message, one warn message, one info message and - // one debug message + // then + const logBuffer = logger.getLogBuffer(); assert.strictEqual(logBuffer.length, 4); assert.strictEqual(logBuffer[0], FULL_LOG_MSG_ERROR); assert.strictEqual(logBuffer[1], FULL_LOG_MSG_WARN); assert.strictEqual(logBuffer[2], FULL_LOG_MSG_INFO); assert.strictEqual(logBuffer[3], FULL_LOG_MSG_DEBUG); + }); + + it('should log messages when a logger level is trace', function () { + // given + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.TRACE)); - // create a new logger with the log level set to trace and log some messages - logger = createLogger(4); + // when logMessages(logger); - logBuffer = logger.getLogBuffer(); - // we should have one error message, one warn message, one info message, - // one debug message and one trace message + // then + const logBuffer = logger.getLogBuffer(); assert.strictEqual(logBuffer.length, 5); assert.strictEqual(logBuffer[0], FULL_LOG_MSG_ERROR); assert.strictEqual(logBuffer[1], FULL_LOG_MSG_WARN); @@ -112,4 +129,37 @@ describe('Logger - browser', function () assert.strictEqual(logBuffer[3], FULL_LOG_MSG_DEBUG); assert.strictEqual(logBuffer[4], FULL_LOG_MSG_TRACE); }); -}); \ No newline at end of file + + it('should not log anything when a logger level is off', function () { + // given + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.OFF)); + + // when + logMessages(logger); + + // then + const logBuffer = logger.getLogBuffer(); + assert.strictEqual(logBuffer.length, 0); + }); + + it('should change logger log level', function () { + // given + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.ERROR)); + + // when + logMessages(logger); + logger.configure({ + level: logTagToLevel(LOG_LEVEL_TAGS.INFO), + filePath: 'it does not matter' + }); + logMessages(logger); + + // then + const logBuffer = logger.getLogBuffer(); + assert.strictEqual(logBuffer.length, 4); + assert.strictEqual(logBuffer[0], FULL_LOG_MSG_ERROR); + assert.strictEqual(logBuffer[1], FULL_LOG_MSG_ERROR); + assert.strictEqual(logBuffer[2], FULL_LOG_MSG_WARN); + assert.strictEqual(logBuffer[3], FULL_LOG_MSG_INFO); + }); +}); diff --git a/test/unit/logger/easy_logging_starter_test.js b/test/unit/logger/easy_logging_starter_test.js new file mode 100644 index 000000000..6b9fde3f4 --- /dev/null +++ b/test/unit/logger/easy_logging_starter_test.js @@ -0,0 +1,125 @@ +/* + * Copyright (c) 2023 Snowflake Computing Inc. All rights reserved. + */ +const {init, reset: resetEasyLoggingModule} = require('../../../lib/logger/easy_logging_starter'); + +const assert = require('assert'); +const fsPromises = require('fs/promises'); +const path = require('path'); +const os = require('os'); +const Logger = require('../../../lib/logger'); +require('../../../lib/snowflake'); // import of it sets up node logger +const defaultConfigName = 'sf_client_config.json'; +const logLevelBefore = Logger.getInstance().getLevel(); +let tempDir = null; + +before(async function () { + tempDir = await fsPromises.mkdtemp(path.join(os.tmpdir(), 'easy_logging_starter_tests_')); +}); + +after(async function () { + await fsPromises.rm(tempDir, { recursive: true, force: true }); + Logger.getInstance().configure({ + level: logLevelBefore, + filePath: 'snowflake.log' + }); +}); + +afterEach(async function() { + await fsPromises.rm(path.join(os.tmpdir(), defaultConfigName), {force: true}); + resetEasyLoggingModule(); +}); + +describe('Easy logging starter tests', function () { + + it('should configure easy logging only once when initialized with config file path', async function () { + // given + const logLevel = 'ERROR'; + const configFilePath = await createConfigFile(logLevel, tempDir, 'config.json'); + const anotherConfigFilePath = await createConfigFile('WARN', tempDir, 'another_config.json'); + + // when + await init(configFilePath); + + // then + assert.strictEqual(Logger.getInstance().getLevelTag(), logLevel); + assert.strictEqual(Logger.getInstance().easyLoggingConfigureCounter, 1); + + // when + await init(null); + await init(configFilePath); + await init(anotherConfigFilePath); + + // then + assert.strictEqual(Logger.getInstance().getLevelTag(), logLevel); + assert.strictEqual(Logger.getInstance().easyLoggingConfigureCounter, 1); + }); + + it('should configure easy logging only once when initialized without config file path', async function () { + // given + const logLevel = 'ERROR'; + await createConfigFile(logLevel, os.tmpdir(), defaultConfigName); + + // when + await init(null); + await init(null); + + // then + assert.strictEqual(Logger.getInstance().getLevelTag(), logLevel); + assert.strictEqual(Logger.getInstance().easyLoggingConfigureCounter, 1); + }); + + it('should reconfigure easy logging if config file path is not given for the first time', async function () { + // given + const tmpDirLogLevel = 'ERROR'; + await createConfigFile(tmpDirLogLevel, os.tmpdir(), defaultConfigName); + const customLogLevel = 'DEBUG'; + const customConfigFilePath = await createConfigFile(customLogLevel, tempDir, 'config.json'); + + // when + await init(null); + + // then + assert.strictEqual(Logger.getInstance().getLevelTag(), tmpDirLogLevel); + assert.strictEqual(Logger.getInstance().easyLoggingConfigureCounter, 1); + + // when + await init(customConfigFilePath); + + // then + assert.strictEqual(Logger.getInstance().getLevelTag(), customLogLevel); + assert.strictEqual(Logger.getInstance().easyLoggingConfigureCounter, 2); + }); + + it('should fail for unknown log level', async function () { + // given + const logLevel = 'something weird'; + const configFilePath = await createConfigFile(logLevel, tempDir, defaultConfigName); + + // expect + await assert.rejects( + async () => await init(configFilePath), + (err) => { + assert.strictEqual(err.name, 'EasyLoggingError'); + assert.strictEqual(err.message, 'Failed to initialize easy logging'); + assert.match(err.cause.message, /Parsing client configuration failed/); + return true; + }); + }); + + async function createConfigFile(logLevel, configDirectory, configFileName) { + const configFilePath = path.join(configDirectory, configFileName); + const configContent = `{ + "common": { + "log_level": "${logLevel}", + "log_path": "${tempDir}" + } + }`; + await writeFile(configFilePath, configContent); + return configFilePath; + } + + async function writeFile (filePath, fileContent) { + await fsPromises.writeFile(filePath, fileContent, { encoding: 'utf8' }); + } +}); diff --git a/test/unit/logger/node_test.js b/test/unit/logger/node_test.js new file mode 100644 index 000000000..54ace0fa4 --- /dev/null +++ b/test/unit/logger/node_test.js @@ -0,0 +1,226 @@ +/* + * Copyright (c) 2023 Snowflake Computing Inc. All rights reserved. + */ + +const NodeLogger = require('./../../../lib/logger/node'); +const assert = require('assert'); +const { logTagToLevel, LOG_LEVEL_TAGS } = require('../../../lib/logger/core'); +const fsPromises = require('fs/promises'); +const path = require('path'); +const os = require('os'); +let tempDir = null; + +describe('Logger node tests', function () { + + before(async function () { + tempDir = await fsPromises.mkdtemp(path.join(os.tmpdir(), 'node_logger_tests_')); + }); + + after(async function () { + await fsPromises.rm(tempDir, { recursive: true, force: true }); + }); + + const ERROR = 'ERROR'; + const WARN = 'WARN'; + const INFO = 'INFO'; + const DEBUG = 'DEBUG'; + const TRACE = 'TRACE'; + + const LOG_MSG_ERROR = '0 error'; + const LOG_MSG_WARN = '1 warn'; + const LOG_MSG_INFO = '2 info'; + const LOG_MSG_DEBUG = '3 debug'; + const LOG_MSG_TRACE = '4 trace'; + + const OBJ_LOG_MSG_ERROR = { + level: ERROR, + message: LOG_MSG_ERROR + }; + + const OBJ_LOG_MSG_WARN = { + level: WARN, + message: LOG_MSG_WARN + }; + + const OBJ_LOG_MSG_INFO = { + level: INFO, + message: LOG_MSG_INFO + }; + + const OBJ_LOG_MSG_DEBUG = { + level: DEBUG, + message: LOG_MSG_DEBUG + }; + + const OBJ_LOG_MSG_TRACE = { + level: TRACE, + message: LOG_MSG_TRACE + }; + + const millisTimeoutToFlushLogFile = 10; + + it('should use info level as default', function () { + // given + const logger = createLogger(null, 'snowflake_default.log'); + + // when + const level = logger.getLevel(); + + // then + assert.strictEqual(level, logTagToLevel(LOG_LEVEL_TAGS.INFO)); + }); + + it('should log messages when a logger level is error', async function () { + // given + const filePath = path.join(tempDir, 'error_logs.log'); + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.ERROR), filePath); + + // when + logMessages(logger); + + // then + await logger.closeAndFlushTransports(millisTimeoutToFlushLogFile); + const logs = await readLogs(filePath); + assert.strictEqual(logs.length, 1); + assert.deepStrictEqual(logs[0], OBJ_LOG_MSG_ERROR); + }); + + it('should log messages when a logger level is warn', async function () { + // given + const filePath = path.join(tempDir, 'warn_logs.log'); + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.WARN), filePath); + + // when + logMessages(logger); + + // then + await logger.closeAndFlushTransports(millisTimeoutToFlushLogFile); + const logs = await readLogs(filePath); + assert.strictEqual(logs.length, 2); + assert.deepStrictEqual(logs[0], OBJ_LOG_MSG_ERROR); + assert.deepStrictEqual(logs[1], OBJ_LOG_MSG_WARN); + }); + + it('should log messages when a logger level is info', async function () { + // given + const filePath = path.join(tempDir, 'info_logs.log'); + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.INFO), filePath); + + // when + logMessages(logger); + + // then + await logger.closeAndFlushTransports(millisTimeoutToFlushLogFile); + const logs = await readLogs(filePath); + assert.strictEqual(logs.length, 3); + assert.deepStrictEqual(logs[0], OBJ_LOG_MSG_ERROR); + assert.deepStrictEqual(logs[1], OBJ_LOG_MSG_WARN); + assert.deepStrictEqual(logs[2], OBJ_LOG_MSG_INFO); + }); + + it('should log messages when a logger level is debug', async function () { + // given + const filePath = path.join(tempDir, 'debug_logs.log'); + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.DEBUG), filePath); + + // when + logMessages(logger); + + // then + await logger.closeAndFlushTransports(millisTimeoutToFlushLogFile); + const logs = await readLogs(filePath); + assert.strictEqual(logs.length, 4); + assert.deepStrictEqual(logs[0], OBJ_LOG_MSG_ERROR); + assert.deepStrictEqual(logs[1], OBJ_LOG_MSG_WARN); + assert.deepStrictEqual(logs[2], OBJ_LOG_MSG_INFO); + assert.deepStrictEqual(logs[3], OBJ_LOG_MSG_DEBUG); + }); + + it('should log messages when a logger level is trace', async function () { + // given + const filePath = path.join(tempDir, 'trace_logs.log'); + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.TRACE), filePath); + + // when + logMessages(logger); + + // then + await logger.closeAndFlushTransports(millisTimeoutToFlushLogFile); + const logs = await readLogs(filePath); + assert.strictEqual(logs.length, 5); + assert.deepStrictEqual(logs[0], OBJ_LOG_MSG_ERROR); + assert.deepStrictEqual(logs[1], OBJ_LOG_MSG_WARN); + assert.deepStrictEqual(logs[2], OBJ_LOG_MSG_INFO); + assert.deepStrictEqual(logs[3], OBJ_LOG_MSG_DEBUG); + assert.deepStrictEqual(logs[4], OBJ_LOG_MSG_TRACE); + }); + + it('should not log any messages when a logger level is off', async function () { + // given + const filePath = path.join(tempDir, 'off_logs.log'); + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.OFF), filePath); + + // when + logMessages(logger); + + // then + await logger.closeAndFlushTransports(millisTimeoutToFlushLogFile); + await assert.rejects( + async () => await readLogs(filePath), + (err) => { + assert.strictEqual(err.name, 'Error'); + assert.match(err.message, /ENOENT: no such file or directory./); + return true; + }); + }); + + it('should change log level and log file path', async function () { + // given + const filePath = path.join(tempDir, 'first_logs.log'); + const filePathChanged = path.join(tempDir, 'second_logs.log'); + const logger = createLogger(logTagToLevel(LOG_LEVEL_TAGS.ERROR), filePath); + + // when + logMessages(logger); + logger.configure({ + level: logTagToLevel(LOG_LEVEL_TAGS.INFO), + filePath: filePathChanged + }); + logMessages(logger); + + // then + await logger.closeAndFlushTransports(millisTimeoutToFlushLogFile); + const errorLogs = await readLogs(filePath); + assert.strictEqual(errorLogs.length, 1); + assert.deepStrictEqual(errorLogs[0], OBJ_LOG_MSG_ERROR); + const infoLogs = await readLogs(filePathChanged); + assert.strictEqual(infoLogs.length, 3); + assert.deepStrictEqual(infoLogs[0], OBJ_LOG_MSG_ERROR); + assert.deepStrictEqual(infoLogs[1], OBJ_LOG_MSG_WARN); + assert.deepStrictEqual(infoLogs[2], OBJ_LOG_MSG_INFO); + }); + + async function readLogs(filePath) { + const logs = await fsPromises.readFile(filePath, { encoding: 'utf8' }); + return logs.split('\n') + .filter(s => s) + .map(s => JSON.parse(s)); + } + + function createLogger (level, filePath) { + return new NodeLogger( + { + includeTimestamp: false, + level: level, + filePath: filePath + }); + } + + function logMessages (logger) { + logger.error(LOG_MSG_ERROR); + logger.warn(LOG_MSG_WARN); + logger.info(LOG_MSG_INFO); + logger.debug(LOG_MSG_DEBUG); + logger.trace(LOG_MSG_TRACE); + } +}); \ No newline at end of file diff --git a/test/unit/snowflake_config_test.js b/test/unit/snowflake_config_test.js index b92819af9..42a8b92d3 100644 --- a/test/unit/snowflake_config_test.js +++ b/test/unit/snowflake_config_test.js @@ -82,6 +82,13 @@ describe('Snowflake Configure Tests', function () { describe('Test valid arguments', function () { const testCases = [ + { + name: 'logLevel off', + options: + { + logLevel: LOG_LEVEL_TAGS.OFF + } + }, { name: 'logLevel error', options: diff --git a/test/unit/snowflake_test.js b/test/unit/snowflake_test.js index a50c7de05..96a1fb109 100644 --- a/test/unit/snowflake_test.js +++ b/test/unit/snowflake_test.js @@ -365,15 +365,17 @@ describe('connection.connect() asynchronous errors', function () var connection = snowflake.createConnection(connectionOptions).connect(); // try to connect again - connection.connect(function (err, conn) - { - assert.strictEqual(conn, connection, - 'the connect() callback should be invoked with the connection'); - assert.ok(err); - assert.strictEqual( - err.code, ErrorCodes.ERR_CONN_CONNECT_STATUS_CONNECTING); - done(); - }); + setTimeout(() => { + connection.connect(function (err, conn) + { + assert.strictEqual(conn, connection, + 'the connect() callback should be invoked with the connection'); + assert.ok(err); + assert.strictEqual( + err.code, ErrorCodes.ERR_CONN_CONNECT_STATUS_CONNECTING); + done(); + }); + }, 0); // when execution is on easy logging init it is not really connecting. Adding 0 timeout changes the order of code executions. }); it('connect() while already connected', function (done) @@ -1447,18 +1449,17 @@ describe('connection.destroy()', function () } }); - connection.destroy(function (err, conn) - { - assert.ok(!err, 'there should be no error'); - assert.strictEqual(conn, connection, - 'the logout() callback should be invoked with the connection'); - - context.destroycomplete = true; - if (context.connectcomplete) - { - done(); - } - }); + setTimeout(() => + connection.destroy(function (err, conn) { + assert.ok(!err, 'there should be no error'); + assert.strictEqual(conn, connection, 'the logout() callback should be invoked with the connection'); + context.destroycomplete = true; + if (context.connectcomplete) { + done(); + } + }) + , 10 // if destroy executes when connect is still in pristine state the error occurs. Destroy has to be slowed down a bit. + ); }); it('destroy after connected', function (done)