Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add configurable query timeout #1760

Merged
merged 11 commits into from Nov 29, 2018
39 changes: 39 additions & 0 deletions lib/client.js
Expand Up @@ -399,15 +399,20 @@ Client.prototype.query = function (config, values, callback) {
// can take in strings, config object or query object
var query
var result
var readTimeout
var readTimeoutTimer
var queryCallback

if (config === null || config === undefined) {
throw new TypeError('Client was passed a null or undefined query')
} else if (typeof config.submit === 'function') {
readTimeout = config.query_timeout || this.connectionParameters.query_timeout
result = query = config
if (typeof values === 'function') {
query.callback = query.callback || values
}
} else {
readTimeout = this.connectionParameters.query_timeout
query = new Query(config, values, callback)
if (!query.callback) {
result = new this._Promise((resolve, reject) => {
Expand All @@ -416,6 +421,40 @@ Client.prototype.query = function (config, values, callback) {
}
}

if (readTimeout) {
queryCallback = query.callback

readTimeoutTimer = setTimeout(() => {
var error = new Error('Query read timeout')

process.nextTick(() => {
query.handleError(error, this.connection)
})

if (typeof queryCallback === 'function') {
queryCallback(error)

// we already returned an error,
// just do nothing if query completes
query.callback = () => {
}
}

// Remove from queue
var index = this.queryQueue.indexOf(query)
if (index > -1) {
this.queryQueue.splice(index, 1)
}

this._pulseQueryQueue()
}, readTimeout)

query.callback = (err, res) => {
clearTimeout(readTimeoutTimer)
queryCallback(err, res)
edevil marked this conversation as resolved.
Show resolved Hide resolved
}
}

if (this.binary && !query.binary) {
query.binary = true
}
Expand Down
1 change: 1 addition & 0 deletions lib/connection-parameters.js
Expand Up @@ -65,6 +65,7 @@ var ConnectionParameters = function (config) {
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.query_timeout = val('query_timeout', config, false)
}

// Convert arg to a string, surround in single quotes, and escape single quotes and backslashes
Expand Down
5 changes: 4 additions & 1 deletion lib/defaults.js
Expand Up @@ -55,7 +55,10 @@ module.exports = {
parseInputDatesAsUTC: false,

// max milliseconds any query using this connection will execute for before timing out in error. false=unlimited
statement_timeout: false
statement_timeout: false,

// max miliseconds to wait for query to complete (client side)
query_timeout: false
}

var pgTypes = require('pg-types')
Expand Down
45 changes: 43 additions & 2 deletions lib/native/client.js
Expand Up @@ -146,14 +146,21 @@ Client.prototype.connect = function (callback) {
Client.prototype.query = function (config, values, callback) {
var query
var result

if (typeof config.submit === 'function') {
var readTimeout
var readTimeoutTimer
var queryCallback

if (config === null || config === undefined) {
throw new TypeError('Client was passed a null or undefined query')
} else if (typeof config.submit === 'function') {
readTimeout = config.query_timeout || this.connectionParameters.query_timeout
result = query = config
// accept query(new Query(...), (err, res) => { }) style
if (typeof values === 'function') {
config.callback = values
}
} else {
readTimeout = this.connectionParameters.query_timeout
query = new NativeQuery(config, values, callback)
if (!query.callback) {
let resolveOut, rejectOut
Expand All @@ -165,6 +172,40 @@ Client.prototype.query = function (config, values, callback) {
}
}

if (readTimeout) {
queryCallback = query.callback

readTimeoutTimer = setTimeout(() => {
var error = new Error('Query read timeout')

process.nextTick(() => {
query.handleError(error, this.connection)
})

if (typeof queryCallback === 'function') {
queryCallback(error)

// we already returned an error,
// just do nothing if query completes
query.callback = () => {
}
}

// Remove from queue
var index = this._queryQueue.indexOf(query)
if (index > -1) {
this._queryQueue.splice(index, 1)
}

this._pulseQueryQueue()
}, readTimeout)

query.callback = (err, res) => {
clearTimeout(readTimeoutTimer)
queryCallback(err, res)
}
}

if (!this._queryable) {
query.native = this.native
process.nextTick(() => {
Expand Down
23 changes: 23 additions & 0 deletions test/integration/client/api-tests.js
Expand Up @@ -15,6 +15,29 @@ suite.test('pool callback behavior', done => {
})
})

suite.test('query timeout', (cb) => {
const pool = new pg.Pool({query_timeout: 1000})
pool.connect().then((client) => {
client.query('SELECT pg_sleep(2)', assert.calls(function (err, result) {
assert(err)
edevil marked this conversation as resolved.
Show resolved Hide resolved
assert(err.message === 'Query read timeout')
client.release()
pool.end(cb)
}))
})
})

suite.test('query no timeout', (cb) => {
const pool = new pg.Pool({query_timeout: 10000})
pool.connect().then((client) => {
client.query('SELECT pg_sleep(1)', assert.calls(function (err, result) {
assert(!err)
client.release()
pool.end(cb)
}))
})
})

suite.test('callback API', done => {
const client = new helper.Client()
client.query('CREATE TEMP TABLE peep(name text)')
Expand Down