| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 | 'use strict'var dns = require('dns')var defaults = require('./defaults')var parse = require('pg-connection-string').parse // parses a connection stringvar val = function (key, config, envVar) {  if (envVar === undefined) {    envVar = process.env['PG' + key.toUpperCase()]  } else if (envVar === false) {    // do nothing ... use false  } else {    envVar = process.env[envVar]  }  return config[key] || envVar || defaults[key]}var readSSLConfigFromEnvironment = function () {  switch (process.env.PGSSLMODE) {    case 'disable':      return false    case 'prefer':    case 'require':    case 'verify-ca':    case 'verify-full':      return true    case 'no-verify':      return { rejectUnauthorized: false }  }  return defaults.ssl}// Convert arg to a string, surround in single quotes, and escape single quotes and backslashesvar quoteParamValue = function (value) {  return "'" + ('' + value).replace(/\\/g, '\\\\').replace(/'/g, "\\'") + "'"}var add = function (params, config, paramName) {  var value = config[paramName]  if (value !== undefined && value !== null) {    params.push(paramName + '=' + quoteParamValue(value))  }}class ConnectionParameters {  constructor(config) {    // if a string is passed, it is a raw connection string so we parse it into a config    config = typeof config === 'string' ? parse(config) : config || {}    // if the config has a connectionString defined, parse IT into the config we use    // this will override other default values with what is stored in connectionString    if (config.connectionString) {      config = Object.assign({}, config, parse(config.connectionString))    }    this.user = val('user', config)    this.database = val('database', config)    if (this.database === undefined) {      this.database = this.user    }    this.port = parseInt(val('port', config), 10)    this.host = val('host', config)    // "hiding" the password so it doesn't show up in stack traces    // or if the client is console.logged    Object.defineProperty(this, 'password', {      configurable: true,      enumerable: false,      writable: true,      value: val('password', config),    })    this.binary = val('binary', config)    this.options = val('options', config)    this.ssl = typeof config.ssl === 'undefined' ? readSSLConfigFromEnvironment() : config.ssl    if (typeof this.ssl === 'string') {      if (this.ssl === 'true') {        this.ssl = true      }    }    // support passing in ssl=no-verify via connection string    if (this.ssl === 'no-verify') {      this.ssl = { rejectUnauthorized: false }    }    if (this.ssl && this.ssl.key) {      Object.defineProperty(this.ssl, 'key', {        enumerable: false,      })    }    this.client_encoding = val('client_encoding', config)    this.replication = val('replication', config)    // a domain socket begins with '/'    this.isDomainSocket = !(this.host || '').indexOf('/')    this.application_name = val('application_name', config, 'PGAPPNAME')    this.fallback_application_name = val('fallback_application_name', config, false)    this.statement_timeout = val('statement_timeout', config, false)    this.lock_timeout = val('lock_timeout', config, false)    this.idle_in_transaction_session_timeout = val('idle_in_transaction_session_timeout', config, false)    this.query_timeout = val('query_timeout', config, false)    if (config.connectionTimeoutMillis === undefined) {      this.connect_timeout = process.env.PGCONNECT_TIMEOUT || 0    } else {      this.connect_timeout = Math.floor(config.connectionTimeoutMillis / 1000)    }    if (config.keepAlive === false) {      this.keepalives = 0    } else if (config.keepAlive === true) {      this.keepalives = 1    }    if (typeof config.keepAliveInitialDelayMillis === 'number') {      this.keepalives_idle = Math.floor(config.keepAliveInitialDelayMillis / 1000)    }  }  getLibpqConnectionString(cb) {    var params = []    add(params, this, 'user')    add(params, this, 'password')    add(params, this, 'port')    add(params, this, 'application_name')    add(params, this, 'fallback_application_name')    add(params, this, 'connect_timeout')    add(params, this, 'options')    var ssl = typeof this.ssl === 'object' ? this.ssl : this.ssl ? { sslmode: this.ssl } : {}    add(params, ssl, 'sslmode')    add(params, ssl, 'sslca')    add(params, ssl, 'sslkey')    add(params, ssl, 'sslcert')    add(params, ssl, 'sslrootcert')    if (this.database) {      params.push('dbname=' + quoteParamValue(this.database))    }    if (this.replication) {      params.push('replication=' + quoteParamValue(this.replication))    }    if (this.host) {      params.push('host=' + quoteParamValue(this.host))    }    if (this.isDomainSocket) {      return cb(null, params.join(' '))    }    if (this.client_encoding) {      params.push('client_encoding=' + quoteParamValue(this.client_encoding))    }    dns.lookup(this.host, function (err, address) {      if (err) return cb(err, null)      params.push('hostaddr=' + quoteParamValue(address))      return cb(null, params.join(' '))    })  }}module.exports = ConnectionParameters
 |