Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: brianc/node-postgres
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: v7.4.1
Choose a base ref
...
head repository: brianc/node-postgres
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: v7.4.2
Choose a head ref
  • 19 commits
  • 12 files changed
  • 12 contributors

Commits on Jan 10, 2018

  1. spelling corrections

    benhc123 authored and charmander committed Jan 10, 2018
    Copy the full SHA
    be57714 View commit details
  2. Copy the full SHA
    94d38f9 View commit details

Commits on Mar 10, 2018

  1. Remove noAssert argument

    The support for the `noAssert` argument dropped in the upcoming
    Node.js v.10.x. All input is then validated no matter if this
    is set to true or not. Just remove the argument because of that.
    BridgeAR committed Mar 10, 2018
    Copy the full SHA
    3f1d7b9 View commit details
  2. Merge pull request #1585 from BridgeAR/master

    Remove `noAssert` argument
    charmander authored Mar 10, 2018
    Copy the full SHA
    50b1221 View commit details

Commits on Mar 14, 2018

  1. Copy the full SHA
    875236f View commit details
  2. Merge pull request #1592 from chentsulin/patch-1

    use net.Socket instead of net.Stream
    charmander authored Mar 14, 2018
    Copy the full SHA
    87dd65f View commit details

Commits on Apr 4, 2018

  1. upgrade testing node versions

    Elexy authored and charmander committed Apr 4, 2018
    2
    Copy the full SHA
    8fb641e View commit details

Commits on May 4, 2018

  1. Add node v10 to travis matrix

    sehrope authored and brianc committed May 4, 2018
    Copy the full SHA
    b0c60cf View commit details
  2. Copy the full SHA
    272e9a5 View commit details
  3. Change test to use Buffer.from(...)

    sehrope authored and brianc committed May 4, 2018
    Copy the full SHA
    860928e View commit details
  4. fix: end stream connection

    Vratislav authored and brianc committed May 4, 2018
    Copy the full SHA
    0902d14 View commit details
  5. Handle SSL negotiation errors more robustly

    This commit adds some finer grained detail to handling the postmaster's
    response to SSL negotiation packets, by accounting for the possibility
    of an 'E' byte being sent back, and emitting an appropriate error.
    
    In the naive case, the postmaster will respond with either 'S' (proceed
    with an SSL connection) or 'N' (SSL is not supported). However, the
    current if statement doesn't account for an 'E' byte being returned
    by the postmaster, where an error is encountered (perhaps unable to
    fork due to being out of memory).
    
    By adding this case, we can prevent confusing error messages when SSL is
    enforced and the postmaster returns an error after successful SSL
    connections.
    
    This also brings the connection handling further in line with
    libpq, where 'E' is handled similarly as of this commit:
    
    postgres/postgres@a49fbaa
    
    Given that there are no longer pre-7.0 databases out in the wild, I
    believe this is a safe change to make, and should not break backwards
    compatibility (unless matching on error message content).
    
    * Replace if statement with switch, to catch 'S', 'E' and 'N' bytes
      returned by the postmaster
    * Return an Error for non 'S' or 'N' cases
    * Expand and restructure unit tests for SSL negotiation packets
    mble authored and brianc committed May 4, 2018
    Copy the full SHA
    5d32be4 View commit details
  6. Copy the full SHA
    7dd3b50 View commit details
  7. Copy the full SHA
    3eb7375 View commit details
  8. Copy the full SHA
    9389527 View commit details
  9. Pass through portal properly

    This happened to work before because `Query.portalName` was undefined,
    but in order to be able to set the portal explicitly it should be using
    `Query.portal`.
    Justin Jaffray authored and brianc committed May 4, 2018
    Copy the full SHA
    831dfb1 View commit details
  10. Copy the full SHA
    4905471 View commit details
  11. Copy the full SHA
    72db790 View commit details
  12. Bump version

    brianc committed May 4, 2018
    Copy the full SHA
    83ede28 View commit details
24 changes: 15 additions & 9 deletions .travis.yml
Original file line number Diff line number Diff line change
@@ -8,28 +8,34 @@ env:

matrix:
include:
- node_js: "lts/boron"
addons:
postgresql: "9.6"
- node_js: "lts/argon"
addons:
postgresql: "9.6"
- node_js: "lts/boron"
- node_js: "9"
addons:
postgresql: "9.6"
- node_js: "10"
addons:
postgresql: "9.6"
- node_js: "lts/carbon"
addons:
postgresql: "9.1"
dist: precise
- node_js: "lts/boron"
- node_js: "lts/carbon"
addons:
postgresql: "9.2"
- node_js: "lts/boron"
- node_js: "lts/carbon"
addons:
postgresql: "9.3"
- node_js: "lts/boron"
- node_js: "lts/carbon"
addons:
postgresql: "9.4"
- node_js: "lts/boron"
- node_js: "lts/carbon"
addons:
postgresql: "9.5"
- node_js: "lts/boron"
addons:
postgresql: "9.6"
- node_js: "8"
- node_js: "lts/carbon"
addons:
postgresql: "9.6"
4 changes: 2 additions & 2 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -66,7 +66,7 @@ var pg = require('pg')
var pool = new pg.Pool()
// your friendly neighboorhood pool interface, without the singleton
// your friendly neighborhood pool interface, without the singleton
pool.connect(function(err, client, done) {
// ...
})
@@ -113,7 +113,7 @@ client.query('SELECT $1::text as name', ['brianc'])
- Add option to parse JS date objects in query parameters as [UTC](https://github.com/brianc/node-postgres/pull/943)

### v4.4.0
- Warn to `stderr` if a named query exceeds 63 characters which is the max lenght supported by postgres.
- Warn to `stderr` if a named query exceeds 63 characters which is the max length supported by postgres.

### v4.3.0
- Unpin `pg-types` semver. Allow it to float against `pg-types@1.x`.
16 changes: 8 additions & 8 deletions README.md
Original file line number Diff line number Diff line change
@@ -5,7 +5,7 @@
<span class="badge-npmversion"><a href="https://npmjs.org/package/pg" title="View this project on NPM"><img src="https://img.shields.io/npm/v/pg.svg" alt="NPM version" /></a></span>
<span class="badge-npmdownloads"><a href="https://npmjs.org/package/pg" title="View this project on NPM"><img src="https://img.shields.io/npm/dm/pg.svg" alt="NPM downloads" /></a></span>

Non-blocking PostgreSQL client for node.js. Pure JavaScript and optional native libpq bindings.
Non-blocking PostgreSQL client for Node.js. Pure JavaScript and optional native libpq bindings.

## Install

@@ -19,9 +19,9 @@ $ npm install pg

### Features

* Pure JavaScript client and native libpq bindings share _the same api_
* Pure JavaScript client and native libpq bindings share _the same API_
* Connection pooling
* Extensible js<->postgresql data-type coercion
* Extensible JS<->PostgreSQL data-type coercion
* Supported PostgreSQL features
* Parameterized queries
* Named statements with query plan caching
@@ -35,18 +35,18 @@ The entire list can be found on our [wiki](https://github.com/brianc/node-postgr

## Support

node-postgres is free software. If you encounter a bug with the library please open an issue on the [github repo](https://github.com/brianc/node-postgres). If you have questions unanswered by the documentation please open an issue pointing out how the documentation was unclear & I will do my best to make it better!
node-postgres is free software. If you encounter a bug with the library please open an issue on the [GitHub repo](https://github.com/brianc/node-postgres). If you have questions unanswered by the documentation please open an issue pointing out how the documentation was unclear & I will do my best to make it better!

When you open an issue please provide:
- version of node
- version of postgres
- version of Node
- version of Postgres
- smallest possible snippet of code to reproduce the problem

You can also follow me [@briancarlson](https://twitter.com/briancarlson) if that's your thing. I try to always announce noteworthy changes & developments with node-postgres on Twitter.

### Professional Support

I offer professional support for node-postgres. I provide implementation, training, and many years of expertise on how to build applications with node, express, PostgreSQL, and react/redux. Please contact me at [brian.m.carlson@gmail.com](mailto:brian.m.carlson@gmail.com) to discuss how I can help your company be more successful!
I offer professional support for node-postgres. I provide implementation, training, and many years of expertise on how to build applications with Node, Express, PostgreSQL, and React/Redux. Please contact me at [brian.m.carlson@gmail.com](mailto:brian.m.carlson@gmail.com) to discuss how I can help your company be more successful!

### Sponsorship :star:

@@ -63,7 +63,7 @@ I will __happily__ accept your pull request if it:

## Troubleshooting and FAQ

The causes and solutions to common errors can be found among the [Frequently Asked Questions(FAQ)](https://github.com/brianc/node-postgres/wiki/FAQ)
The causes and solutions to common errors can be found among the [Frequently Asked Questions (FAQ)](https://github.com/brianc/node-postgres/wiki/FAQ)

## License

20 changes: 14 additions & 6 deletions lib/connection.js
Original file line number Diff line number Diff line change
@@ -19,7 +19,7 @@ var BINARY_MODE = 1
var Connection = function (config) {
EventEmitter.call(this)
config = config || {}
this.stream = config.stream || new net.Stream()
this.stream = config.stream || new net.Socket()
this._keepAlive = config.keepAlive
this.lastBuffer = false
this.lastOffset = 0
@@ -82,13 +82,19 @@ Connection.prototype.connect = function (port, host) {

this.stream.once('data', function (buffer) {
var responseCode = buffer.toString('utf8')
if (responseCode !== 'S') {
return self.emit('error', new Error('The server does not support SSL connections'))
switch (responseCode) {
case 'N': // Server does not support SSL connections
return self.emit('error', new Error('The server does not support SSL connections'))
case 'S': // Server supports SSL connections, continue with a secure connection
break
default: // Any other response byte, including 'E' (ErrorResponse) indicating a server error
return self.emit('error', new Error('There was an error establishing an SSL connection'))
}
var tls = require('tls')
self.stream = tls.connect({
socket: self.stream,
servername: host,
checkServerIdentity: self.ssl.checkServerIdentity,
rejectUnauthorized: self.ssl.rejectUnauthorized,
ca: self.ssl.ca,
pfx: self.ssl.pfx,
@@ -309,7 +315,9 @@ Connection.prototype.end = function () {
// 0x58 = 'X'
this.writer.add(emptyBuffer)
this._ending = true
return this.stream.write(END_BUFFER)
return this.stream.write(END_BUFFER, () => {
this.stream.end()
})
}

Connection.prototype.close = function (msg, more) {
@@ -617,13 +625,13 @@ Connection.prototype.parsed = function (buffer, length) {
}

Connection.prototype.parseInt32 = function (buffer) {
var value = buffer.readInt32BE(this.offset, true)
var value = buffer.readInt32BE(this.offset)
this.offset += 4
return value
}

Connection.prototype.parseInt16 = function (buffer) {
var value = buffer.readInt16BE(this.offset, true)
var value = buffer.readInt16BE(this.offset)
this.offset += 2
return value
}
6 changes: 3 additions & 3 deletions lib/query.js
Original file line number Diff line number Diff line change
@@ -175,7 +175,7 @@ Query.prototype.handlePortalSuspended = function (connection) {

Query.prototype._getRows = function (connection, rows) {
connection.execute({
portal: this.portalName,
portal: this.portal,
rows: rows
}, true)
connection.flush()
@@ -201,15 +201,15 @@ Query.prototype.prepare = function (connection) {

// http://developer.postgresql.org/pgdocs/postgres/protocol-flow.html#PROTOCOL-FLOW-EXT-QUERY
connection.bind({
portal: self.portalName,
portal: self.portal,
statement: self.name,
values: self.values,
binary: self.binary
}, true)

connection.describe({
type: 'P',
name: self.portalName || ''
name: self.portal || ''
}, true)

this._getRows(connection, this.rows)
35 changes: 11 additions & 24 deletions lib/result.js
Original file line number Diff line number Diff line change
@@ -8,7 +8,6 @@
*/

var types = require('pg-types')
var escape = require('js-string-escape')

// result object returned from query
// in the 'end' event and also
@@ -65,29 +64,24 @@ Result.prototype._parseRowAsArray = function (rowData) {
return row
}

// rowData is an array of text or binary values
// this turns the row into a JavaScript object
Result.prototype.parseRow = function (rowData) {
return new this.RowCtor(this._parsers, rowData)
var row = {}
for (var i = 0, len = rowData.length; i < len; i++) {
var rawValue = rowData[i]
var field = this.fields[i].name
if (rawValue !== null) {
row[field] = this._parsers[i](rawValue)
} else {
row[field] = null
}
}
return row
}

Result.prototype.addRow = function (row) {
this.rows.push(row)
}

var inlineParser = function (fieldName, i) {
return "\nthis['" +
// fields containing single quotes will break
// the evaluated javascript unless they are escaped
// see https://github.com/brianc/node-postgres/issues/507
// Addendum: However, we need to make sure to replace all
// occurences of apostrophes, not just the first one.
// See https://github.com/brianc/node-postgres/issues/934
escape(fieldName) +
"'] = " +
'rowData[' + i + '] == null ? null : parsers[' + i + '](rowData[' + i + ']);'
}

Result.prototype.addFields = function (fieldDescriptions) {
// clears field definitions
// multiple query statements in 1 action can result in multiple sets
@@ -97,18 +91,11 @@ Result.prototype.addFields = function (fieldDescriptions) {
this.fields = []
this._parsers = []
}
var ctorBody = ''
for (var i = 0; i < fieldDescriptions.length; i++) {
var desc = fieldDescriptions[i]
this.fields.push(desc)
var parser = this._getTypeParser(desc.dataTypeID, desc.format || 'text')
this._parsers.push(parser)
// this is some craziness to compile the row result parsing
// results in ~60% speedup on large query result sets
ctorBody += inlineParser(desc.name, i)
}
if (!this.rowAsArray) {
this.RowCtor = Function('parsers', 'rowData', ctorBody)
}
}

3 changes: 1 addition & 2 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "pg",
"version": "7.4.1",
"version": "7.4.2",
"description": "PostgreSQL client - pure javascript & libpq with the same API",
"keywords": [
"database",
@@ -20,7 +20,6 @@
"main": "./lib",
"dependencies": {
"buffer-writer": "1.0.1",
"js-string-escape": "1.0.1",
"packet-reader": "0.3.1",
"pg-connection-string": "0.1.3",
"pg-pool": "~2.0.3",
4 changes: 2 additions & 2 deletions test/integration/client/network-partition-tests.js
Original file line number Diff line number Diff line change
@@ -22,7 +22,7 @@ Server.prototype.start = function (cb) {
this.socket.on('data', function (data) {
// deny request for SSL
if (data.length == 8) {
this.socket.write(new Buffer('N', 'utf8'))
this.socket.write(Buffer.from('N', 'utf8'))
// consider all authentication requests as good
} else if (!data[0]) {
this.socket.write(buffers.authenticationOk())
@@ -46,7 +46,7 @@ Server.prototype.start = function (cb) {
}

Server.prototype.drop = function () {
this.socket.end()
this.socket.destroy()
}

Server.prototype.close = function (cb) {
71 changes: 69 additions & 2 deletions test/unit/client/prepared-statement-tests.js
Original file line number Diff line number Diff line change
@@ -65,7 +65,7 @@ test('bound command', function () {

test('bind argument', function () {
assert.equal(bindArg.statement, null)
assert.equal(bindArg.portal, null)
assert.equal(bindArg.portal, '')
assert.lengthIs(bindArg.values, 1)
assert.equal(bindArg.values[0], 'hi')
})
@@ -76,7 +76,7 @@ test('bound command', function () {
})

test('execute argument', function () {
assert.equal(executeArg.portal, null)
assert.equal(executeArg.portal, '')
assert.equal(executeArg.rows, null)
})

@@ -86,3 +86,70 @@ test('bound command', function () {
})
})
})

var portalClient = helper.client()
var portalCon = portalClient.connection
var portalParseArg = null
portalCon.parse = function (arg) {
portalParseArg = arg
process.nextTick(function () {
portalCon.emit('parseComplete')
})
}

var portalBindArg = null
portalCon.bind = function (arg) {
portalBindArg = arg
process.nextTick(function () {
portalCon.emit('bindComplete')
})
}

var portalExecuteArg = null
portalCon.execute = function (arg) {
portalExecuteArg = arg
process.nextTick(function () {
portalCon.emit('rowData', { fields: [] })
portalCon.emit('commandComplete', { text: '' })
})
}

var portalDescribeArg = null
portalCon.describe = function (arg) {
portalDescribeArg = arg
process.nextTick(function () {
portalCon.emit('rowDescription', { fields: [] })
})
}

portalCon.flush = function () {
}
portalCon.sync = function () {
process.nextTick(function () {
portalCon.emit('readyForQuery')
})
}

test('prepared statement with explicit portal', function () {
assert.ok(portalClient.connection.emit('readyForQuery'))

var query = portalClient.query(new Query({
text: 'select * from X where name = $1',
portal: 'myportal',
values: ['hi']
}))

assert.emits(query, 'end', function () {
test('bind argument', function () {
assert.equal(portalBindArg.portal, 'myportal')
})

test('describe argument', function () {
assert.equal(portalDescribeArg.name, 'myportal')
})

test('execute argument', function () {
assert.equal(portalExecuteArg.portal, 'myportal')
})
})
})
Loading