Skip to content

Commit ff47a97

Browse files
alxndrsnbrianc
andauthored
Add option to force use of Extended Queries (#3214)
This feature can be used as follows: ``` client.query({ text: 'SELECT 1', queryMode: 'extended' }) ``` This will force the query to be sent with parse/bind/execute even when it has no parameters and disallows multiple statements being executed. This can be useful in scenarios where you want to enforce more security & help prevent sql injection attacks...particularly by library authors. --------- Co-authored-by: alxndrsn <alxndrsn> Co-authored-by: Brian Carlson <[email protected]>
1 parent fe88e82 commit ff47a97

File tree

5 files changed

+37
-1
lines changed

5 files changed

+37
-1
lines changed

docs/pages/apis/client.mdx

+3
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,9 @@ type QueryConfig {
7575

7676
// custom type parsers just for this query result
7777
types?: Types;
78+
79+
// TODO: document
80+
queryMode?: string;
7881
}
7982
```
8083

packages/pg-native/index.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ Client.prototype.query = function (text, values, cb) {
5757
cb = values
5858
}
5959

60-
if (Array.isArray(values) && values.length > 0) {
60+
if (Array.isArray(values)) {
6161
queryFn = function () {
6262
return self.pq.sendQueryParams(text, values)
6363
}

packages/pg/lib/native/query.js

+3
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ var NativeQuery = (module.exports = function (config, values, callback) {
1010
this.text = config.text
1111
this.values = config.values
1212
this.name = config.name
13+
this.queryMode = config.queryMode
1314
this.callback = config.callback
1415
this.state = 'new'
1516
this._arrayMode = config.rowMode === 'array'
@@ -159,6 +160,8 @@ NativeQuery.prototype.submit = function (client) {
159160
}
160161
var vals = this.values.map(utils.prepareValue)
161162
client.native.query(this.text, vals, after)
163+
} else if (this.queryMode === 'extended') {
164+
client.native.query(this.text, [], after)
162165
} else {
163166
client.native.query(this.text, after)
164167
}

packages/pg/lib/query.js

+5
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ class Query extends EventEmitter {
1616
this.rows = config.rows
1717
this.types = config.types
1818
this.name = config.name
19+
this.queryMode = config.queryMode
1920
this.binary = config.binary
2021
// use unique portal name each time
2122
this.portal = config.portal || ''
@@ -32,6 +33,10 @@ class Query extends EventEmitter {
3233
}
3334

3435
requiresPreparation() {
36+
if (this.queryMode === 'extended') {
37+
return true
38+
}
39+
3540
// named queries must always be prepared
3641
if (this.name) {
3742
return true

packages/pg/test/integration/client/multiple-results-tests.js

+25
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,31 @@ suite.test(
2525
})
2626
)
2727

28+
suite.test(
29+
'throws if queryMode set to "extended"',
30+
co.wrap(function* () {
31+
const client = new helper.Client()
32+
yield client.connect()
33+
34+
// TODO should be text or sql?
35+
try {
36+
const results = yield client.query({
37+
text: `SELECT 'foo'::text as name; SELECT 'bar'::text as baz`,
38+
queryMode: 'extended',
39+
})
40+
assert.fail('Should have thrown')
41+
} catch (err) {
42+
if (err instanceof assert.AssertionError) throw err
43+
44+
assert.equal(err.severity, 'ERROR')
45+
assert.equal(err.code, '42601')
46+
assert.equal(err.message, 'cannot insert multiple commands into a prepared statement')
47+
}
48+
49+
return client.end()
50+
})
51+
)
52+
2853
suite.test(
2954
'multiple selects work',
3055
co.wrap(function* () {

0 commit comments

Comments
 (0)