Skip to content

Commit 5ccea91

Browse files
authored
Allow users to opt out of literal comma "," query parameters parsing (#215)
* feat: add parseCommaSeparatedQueryParams option to opt-out of parsing multi-value query params with api gateway v2.0 format, fixes #202 * Chrore: Update readme with the new parseCommaSeparatedQueryParams option
1 parent c5c945a commit 5ccea91

File tree

4 files changed

+81
-6
lines changed

4 files changed

+81
-6
lines changed

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ $ npm i @fastify/aws-lambda
3030
| callbackWaitsForEmptyEventLoop | See: [Official Documentation](https://docs.aws.amazon.com/lambda/latest/dg/nodejs-context.html#nodejs-prog-model-context-properties) | `undefined` |
3131
| retainStage | Retain the stage part of the API Gateway URL | `false` |
3232
| pathParameterUsedAsPath | Use a defined pathParameter as path (i.e. `'proxy'`) | `false` |
33-
33+
| parseCommaSeparatedQueryParams | Parse querystring with commas into an array of values. Affects the behavior of the querystring parser with commas while using [Payload Format Version 2.0](https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-develop-integrations-lambda.html#http-api-develop-integrations-lambda.proxy-format) | `true` |
3434
## 📖Example
3535

3636
### lambda.js

index.js

+7-5
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ module.exports = (app, options) => {
1717
options.decorateRequest = options.decorateRequest !== undefined ? options.decorateRequest : true
1818
options.retainStage = options.retainStage !== undefined ? options.retainStage : false
1919
options.pathParameterUsedAsPath = options.pathParameterUsedAsPath !== undefined ? options.pathParameterUsedAsPath : false
20+
options.parseCommaSeparatedQueryParams = options.parseCommaSeparatedQueryParams !== undefined ? options.parseCommaSeparatedQueryParams : true
2021
let currentAwsArguments = {}
2122
if (options.decorateRequest) {
2223
options.decorationPropertyName = options.decorationPropertyName || 'awsLambda'
@@ -48,6 +49,7 @@ module.exports = (app, options) => {
4849
url = url.substring(event.requestContext.stage.length + 1)
4950
}
5051
const query = {}
52+
const parsedCommaSeparatedQuery = {}
5153
if (event.requestContext && event.requestContext.elb) {
5254
if (event.multiValueQueryStringParameters) {
5355
Object.keys(event.multiValueQueryStringParameters).forEach((q) => {
@@ -56,20 +58,20 @@ module.exports = (app, options) => {
5658
} else if (event.queryStringParameters) {
5759
Object.keys(event.queryStringParameters).forEach((q) => {
5860
query[decodeURIComponent(q)] = decodeURIComponent(event.queryStringParameters[q])
59-
if (event.version === '2.0' && typeof query[decodeURIComponent(q)] === 'string' && query[decodeURIComponent(q)].indexOf(',') > 0) {
60-
query[decodeURIComponent(q)] = query[decodeURIComponent(q)].split(',')
61+
if (options.parseCommaSeparatedQueryParams && event.version === '2.0' && typeof query[decodeURIComponent(q)] === 'string' && query[decodeURIComponent(q)].indexOf(',') > 0) {
62+
parsedCommaSeparatedQuery[decodeURIComponent(q)] = query[decodeURIComponent(q)].split(',')
6163
}
6264
})
6365
}
6466
} else {
65-
if (event.queryStringParameters && event.version === '2.0') {
67+
if (event.queryStringParameters && options.parseCommaSeparatedQueryParams && event.version === '2.0') {
6668
Object.keys(event.queryStringParameters).forEach((k) => {
6769
if (typeof event.queryStringParameters[k] === 'string' && event.queryStringParameters[k].indexOf(',') > 0) {
68-
event.queryStringParameters[k] = event.queryStringParameters[k].split(',')
70+
parsedCommaSeparatedQuery[decodeURIComponent(k)] = event.queryStringParameters[k].split(',')
6971
}
7072
})
7173
}
72-
Object.assign(query, event.multiValueQueryStringParameters || event.queryStringParameters)
74+
Object.assign(query, event.multiValueQueryStringParameters || event.queryStringParameters, parsedCommaSeparatedQuery)
7375
}
7476
const headers = Object.assign({}, event.headers)
7577
if (event.multiValueHeaders) {

test/basic.test.js

+63
Original file line numberDiff line numberDiff line change
@@ -207,6 +207,69 @@ test('GET with multi-value query params (queryStringParameters)', async (t) => {
207207
t.equal(ret.body, '{"foo":["qux","bar"]}')
208208
})
209209

210+
test('GET with multi-value query params (queryStringParameters) with parseCommaSeparatedQueryParams disabled', async (t) => {
211+
t.plan(2)
212+
213+
const app = fastify()
214+
app.get('/test', async (request, reply) => {
215+
reply.send(request.query)
216+
})
217+
const proxy = awsLambdaFastify(app, { parseCommaSeparatedQueryParams: false })
218+
219+
const ret = await proxy({
220+
version: '2.0',
221+
httpMethod: 'GET',
222+
path: '/test',
223+
queryStringParameters: {
224+
foo: 'qux,bar'
225+
}
226+
})
227+
t.equal(ret.statusCode, 200)
228+
t.equal(ret.body, '{"foo":"qux,bar"}')
229+
})
230+
231+
test('GET Retains queryStringParameters in the original awsLambda onRequest hook when parseCommaSeparatedQueryParams is enabled by default', async (t) => {
232+
t.plan(2)
233+
234+
const app = fastify()
235+
app.get('/test', async (request, reply) => {
236+
t.same(request.awsLambda.event.queryStringParameters, { foo: 'qux,bar' })
237+
reply.send(request.query)
238+
})
239+
const proxy = awsLambdaFastify(app)
240+
241+
const ret = await proxy({
242+
version: '2.0',
243+
httpMethod: 'GET',
244+
path: '/test',
245+
queryStringParameters: {
246+
foo: 'qux,bar'
247+
}
248+
})
249+
t.equal(ret.statusCode, 200)
250+
})
251+
252+
test('GET Retains queryStringParameters in the original awsLambda onRequest hook when parseCommaSeparatedQueryParams is disabled', async (t) => {
253+
t.plan(2)
254+
255+
const app = fastify()
256+
app.get('/test', async (request, reply) => {
257+
t.same(request.awsLambda.event.queryStringParameters, { foo: 'qux,bar' })
258+
reply.send(request.query)
259+
})
260+
const proxy = awsLambdaFastify(app, { parseCommaSeparatedQueryParams: false })
261+
262+
const ret = await proxy({
263+
version: '2.0',
264+
httpMethod: 'GET',
265+
path: '/test',
266+
queryStringParameters: {
267+
foo: 'qux,bar'
268+
}
269+
})
270+
t.equal(ret.statusCode, 200)
271+
})
272+
210273
test('GET with double encoded query value', async (t) => {
211274
t.plan(2)
212275

types/index.d.ts

+10
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,16 @@ declare namespace awsLambdaFastify {
1616
* usually set to 'proxy', if used
1717
*/
1818
pathParameterUsedAsPath?: string;
19+
/**
20+
* Parse querystring with commas into an array of values.
21+
* Affects the behavior of the querystring parser with commas while using [Payload Format Version 2.0](https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-develop-integrations-lambda.html#http-api-develop-integrations-lambda.proxy-format)
22+
*
23+
* e.g. when set to `true` (default) `?foo=qux,bar` => `{ foo: ['qux', 'bar'] }`
24+
*
25+
* e.g. when set to `false` `?foo=qux,bar` => `{ foo: 'qux,bar' }`
26+
* @default true
27+
*/
28+
parseCommaSeparatedQueryParams?: boolean;
1929
}
2030

2131
export interface LambdaResponse {

0 commit comments

Comments
 (0)