Skip to content

Commit b2695c4

Browse files
authored
Merge commit from fork
* feat: add option to customize the depth Main Changes: - Update documentation to reflect the new features and errors - Update the changelog - Upgrade to `[email protected]` - Add the `depth` option to define the depth of parsing while parsing the query string - Enable the `strictDepth` option by default in `qs.parse` - Add a 400 status code when the depth of the query string exceeds the limit defined by the `depth` option * chore: reduce the default depth limit to 32 Main Changes: - Breaking Change: Set default depth limit to 32 - Update documentation - Update the HISTORY
1 parent ade0f3f commit b2695c4

File tree

5 files changed

+124
-12
lines changed

5 files changed

+124
-12
lines changed

HISTORY.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
11
unreleased
2-
==========
3-
2+
===================
3+
4+
5+
* add `depth` option to customize the depth level in the parser
6+
* IMPORTANT: The default `depth` level for parsing URL-encoded data is now `32` (previously was `Infinity`)
47

58
1.20.2 / 2023-02-21
69
===================

README.md

+8
Original file line numberDiff line numberDiff line change
@@ -278,6 +278,10 @@ The `verify` option, if supplied, is called as `verify(req, res, buf, encoding)`
278278
where `buf` is a `Buffer` of the raw request body and `encoding` is the
279279
encoding of the request. The parsing can be aborted by throwing an error.
280280

281+
#### depth
282+
283+
The `depth` option is used to configure the maximum depth of the `qs` library when `extended` is `true`. This allows you to limit the amount of keys that are parsed and can be useful to prevent certain types of abuse. Defaults to `32`. It is recommended to keep this value as low as possible.
284+
281285
## Errors
282286

283287
The middlewares provided by this module create errors using the
@@ -374,6 +378,10 @@ as well as in the `encoding` property. The `status` property is set to `415`,
374378
the `type` property is set to `'encoding.unsupported'`, and the `encoding`
375379
property is set to the encoding that is unsupported.
376380

381+
### The input exceeded the depth
382+
383+
This error occurs when using `bodyParser.urlencoded` with the `extended` property set to `true` and the input exceeds the configured `depth` option. The `status` property is set to `400`. It is recommended to review the `depth` option and evaluate if it requires a higher value. When the `depth` option is set to `32` (default value), the error will not be thrown.
384+
377385
## Examples
378386

379387
### Express/Connect top-level generic

lib/types/urlencoded.js

+30-7
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,9 @@ function urlencoded (options) {
5555
: opts.limit
5656
var type = opts.type || 'application/x-www-form-urlencoded'
5757
var verify = opts.verify || false
58+
var depth = typeof opts.depth !== 'number'
59+
? Number(opts.depth || 32)
60+
: opts.depth
5861

5962
if (verify !== false && typeof verify !== 'function') {
6063
throw new TypeError('option verify must be function')
@@ -118,7 +121,8 @@ function urlencoded (options) {
118121
encoding: charset,
119122
inflate: inflate,
120123
limit: limit,
121-
verify: verify
124+
verify: verify,
125+
depth: depth
122126
})
123127
}
124128
}
@@ -133,12 +137,20 @@ function extendedparser (options) {
133137
var parameterLimit = options.parameterLimit !== undefined
134138
? options.parameterLimit
135139
: 1000
140+
141+
var depth = typeof options.depth !== 'number'
142+
? Number(options.depth || 32)
143+
: options.depth
136144
var parse = parser('qs')
137145

138146
if (isNaN(parameterLimit) || parameterLimit < 1) {
139147
throw new TypeError('option parameterLimit must be a positive number')
140148
}
141149

150+
if(isNaN(depth) || depth < 0) {
151+
throw new TypeError('option depth must be a zero or a positive number')
152+
}
153+
142154
if (isFinite(parameterLimit)) {
143155
parameterLimit = parameterLimit | 0
144156
}
@@ -156,12 +168,23 @@ function extendedparser (options) {
156168
var arrayLimit = Math.max(100, paramCount)
157169

158170
debug('parse extended urlencoding')
159-
return parse(body, {
160-
allowPrototypes: true,
161-
arrayLimit: arrayLimit,
162-
depth: Infinity,
163-
parameterLimit: parameterLimit
164-
})
171+
try {
172+
return parse(body, {
173+
allowPrototypes: true,
174+
arrayLimit: arrayLimit,
175+
depth: depth,
176+
strictDepth: true,
177+
parameterLimit: parameterLimit
178+
})
179+
} catch (err) {
180+
if (err instanceof RangeError) {
181+
throw createError(400, 'The input exceeded the depth', {
182+
type: 'querystring.parse.rangeError'
183+
})
184+
} else {
185+
throw err
186+
}
187+
}
165188
}
166189
}
167190

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@
1717
"http-errors": "2.0.0",
1818
"iconv-lite": "0.4.24",
1919
"on-finished": "2.4.1",
20-
"qs": "6.12.3",
20+
"qs": "6.13.0",
2121
"raw-body": "2.5.2",
2222
"type-is": "~1.6.18",
2323
"unpipe": "1.0.0"

test/urlencoded.js

+80-2
Original file line numberDiff line numberDiff line change
@@ -195,7 +195,7 @@ describe('bodyParser.urlencoded()', function () {
195195
it('should parse deep object', function (done) {
196196
var str = 'foo'
197197

198-
for (var i = 0; i < 500; i++) {
198+
for (var i = 0; i < 32; i++) {
199199
str += '[p]'
200200
}
201201

@@ -213,13 +213,91 @@ describe('bodyParser.urlencoded()', function () {
213213
var depth = 0
214214
var ref = obj.foo
215215
while ((ref = ref.p)) { depth++ }
216-
assert.strictEqual(depth, 500)
216+
assert.strictEqual(depth, 32)
217217
})
218218
.expect(200, done)
219219
})
220220
})
221221
})
222222

223+
224+
describe('with depth option', function () {
225+
describe('when custom value set', function () {
226+
227+
it('should reject non possitive numbers', function () {
228+
assert.throws(createServer.bind(null, { extended: true, depth: -1 }),
229+
/TypeError: option depth must be a zero or a positive number/)
230+
assert.throws(createServer.bind(null, { extended: true, depth: NaN }),
231+
/TypeError: option depth must be a zero or a positive number/)
232+
assert.throws(createServer.bind(null, { extended: true, depth: "beep" }),
233+
/TypeError: option depth must be a zero or a positive number/)
234+
})
235+
236+
237+
it('should parse up to the specified depth', function (done) {
238+
this.server = createServer({ extended:true, depth: 10 })
239+
request(this.server)
240+
.post('/')
241+
.set('Content-Type', 'application/x-www-form-urlencoded')
242+
.send('a[b][c][d]=value')
243+
.expect(200, '{"a":{"b":{"c":{"d":"value"}}}}', done)
244+
})
245+
246+
it('should not parse beyond the specified depth', function (done) {
247+
this.server = createServer({ extended:true, depth: 1 })
248+
request(this.server)
249+
.post('/')
250+
.set('Content-Type', 'application/x-www-form-urlencoded')
251+
.send('a[b][c][d][e]=value')
252+
.expect(400, '[querystring.parse.rangeError] The input exceeded the depth', done)
253+
})
254+
})
255+
256+
257+
describe('when default value', function () {
258+
before(function () {
259+
this.server = createServer({ })
260+
})
261+
262+
it('should parse deeply nested objects', function (done) {
263+
var deepObject = 'a'
264+
for (var i = 0; i < 32; i++) {
265+
deepObject += '[p]'
266+
}
267+
deepObject += '=value'
268+
269+
request(this.server)
270+
.post('/')
271+
.set('Content-Type', 'application/x-www-form-urlencoded')
272+
.send(deepObject)
273+
.expect(function (res) {
274+
var obj = JSON.parse(res.text)
275+
var depth = 0
276+
var ref = obj.a
277+
while ((ref = ref.p)) { depth++ }
278+
assert.strictEqual(depth, 32)
279+
})
280+
.expect(200, done)
281+
})
282+
283+
it('should not parse beyond the specified depth', function (done) {
284+
var deepObject = 'a';
285+
for (var i = 0; i < 33; i++) {
286+
deepObject += '[p]';
287+
}
288+
deepObject += '=value';
289+
290+
request(this.server)
291+
.post('/')
292+
.set('Content-Type', 'application/x-www-form-urlencoded')
293+
.send(deepObject)
294+
.expect(400, '[querystring.parse.rangeError] The input exceeded the depth', done);
295+
});
296+
297+
})
298+
299+
})
300+
223301
describe('with inflate option', function () {
224302
describe('when false', function () {
225303
before(function () {

0 commit comments

Comments
 (0)