123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- 'use strict'
- var dns = require('dns')
- var defaults = require('./defaults')
- var parse = require('pg-connection-string').parse // parses a connection string
- var 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 backslashes
- var 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.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
|