Skip to content

Commit 6d711a7

Browse files
committed
introduce payloadAsStream option #154
1 parent 013990b commit 6d711a7

File tree

2 files changed

+186
-3
lines changed

2 files changed

+186
-3
lines changed

index.js

+12-3
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ module.exports = (app, options) => {
1818
options.retainStage = options.retainStage !== undefined ? options.retainStage : false
1919
options.pathParameterUsedAsPath = options.pathParameterUsedAsPath !== undefined ? options.pathParameterUsedAsPath : false
2020
options.parseCommaSeparatedQueryParams = options.parseCommaSeparatedQueryParams !== undefined ? options.parseCommaSeparatedQueryParams : true
21+
options.payloadAsStream = options.payloadAsStream !== undefined ? options.payloadAsStream : false
2122
let currentAwsArguments = {}
2223
if (options.decorateRequest) {
2324
options.decorationPropertyName = options.decorationPropertyName || 'awsLambda'
@@ -110,7 +111,7 @@ module.exports = (app, options) => {
110111
}
111112

112113
const prom = new Promise((resolve) => {
113-
app.inject({ method, url, query, payload, headers, remoteAddress }, (err, res) => {
114+
app.inject({ method, url, query, payload, headers, remoteAddress, payloadAsStream: options.payloadAsStream }, (err, res) => {
114115
currentAwsArguments = {}
115116
if (err) {
116117
console.error(err)
@@ -149,14 +150,22 @@ module.exports = (app, options) => {
149150

150151
const ret = {
151152
statusCode: res.statusCode,
152-
body: isBase64Encoded ? res.rawPayload.toString('base64') : res.payload,
153153
headers: res.headers,
154154
isBase64Encoded
155155
}
156156

157157
if (cookies && event.version === '2.0') ret.cookies = cookies
158158
if (multiValueHeaders && (!event.version || event.version === '1.0')) ret.multiValueHeaders = multiValueHeaders
159-
resolve(ret)
159+
160+
if (!options.payloadAsStream) {
161+
ret.body = isBase64Encoded ? res.rawPayload.toString('base64') : res.payload
162+
return resolve(ret)
163+
}
164+
165+
resolve({
166+
meta: ret,
167+
stream: res.stream()
168+
})
160169
})
161170
})
162171
if (!callback) return prom

test/stream.test.js

+174
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,174 @@
1+
'use strict'
2+
3+
const { describe, it } = require('node:test')
4+
const assert = require('node:assert')
5+
const { promisify } = require('node:util')
6+
const { Readable } = require('node:stream')
7+
const fastify = require('fastify')
8+
const awsLambdaFastify = require('../index')
9+
10+
const accumulate = promisify(function (stream, cb) {
11+
const chunks = []
12+
stream.on('error', cb)
13+
stream.on('data', (chunk) => {
14+
chunks.push(chunk)
15+
})
16+
stream.on('end', () => {
17+
cb(null, Buffer.concat(chunks))
18+
})
19+
})
20+
21+
describe('Basic Stream Tests', () => {
22+
it('GET a normal response as stream', async () => {
23+
const app = fastify()
24+
25+
const evt = {
26+
version: '2.0',
27+
httpMethod: 'GET',
28+
path: '/test',
29+
headers: {
30+
'X-My-Header': 'wuuusaaa'
31+
},
32+
cookies: ['foo=bar'],
33+
queryStringParameters: ''
34+
}
35+
36+
app.get('/test', async (request, reply) => {
37+
assert.equal(
38+
request.headers['x-my-header'],
39+
'wuuusaaa',
40+
'Header "X-My-Header" should match'
41+
)
42+
43+
assert.equal(
44+
request.headers.cookie,
45+
'foo=bar',
46+
'Cookie header should match'
47+
)
48+
assert.equal(
49+
request.headers['user-agent'],
50+
'lightMyRequest',
51+
'User-agent header should match'
52+
)
53+
assert.equal(
54+
request.headers.host,
55+
'localhost:80',
56+
'Host header should match'
57+
)
58+
59+
reply.header('Set-Cookie', 'qwerty=one')
60+
reply.header('Set-Cookie', 'qwerty=two')
61+
reply.send({ hello: 'world' })
62+
})
63+
64+
const proxy = awsLambdaFastify(app, { payloadAsStream: true })
65+
66+
const { meta, stream } = await proxy(evt)
67+
68+
assert.equal(meta.statusCode, 200, 'Status code should be 200')
69+
70+
const data = await accumulate(stream)
71+
assert.equal(data.toString(), '{"hello":"world"}', 'Response body should match')
72+
73+
assert.equal(meta.isBase64Encoded, false, 'isBase64Encoded should be false')
74+
assert.ok(meta.headers, 'Headers should be defined')
75+
assert.equal(
76+
meta.headers['content-type'],
77+
'application/json; charset=utf-8',
78+
'Content-Type should match'
79+
)
80+
assert.equal(
81+
meta.headers['content-length'],
82+
'17',
83+
'Content-Length should match'
84+
)
85+
assert.ok(meta.headers.date, 'Date header should be defined')
86+
assert.equal(
87+
meta.headers.connection,
88+
'keep-alive',
89+
'Connection header should match'
90+
)
91+
assert.deepEqual(
92+
meta.cookies,
93+
['qwerty=one', 'qwerty=two'],
94+
'Cookies should match'
95+
)
96+
})
97+
98+
it('GET a streamed response as stream', async () => {
99+
const app = fastify()
100+
101+
const evt = {
102+
version: '2.0',
103+
httpMethod: 'GET',
104+
path: '/test',
105+
headers: {
106+
'X-My-Header': 'wuuusaaa'
107+
},
108+
cookies: ['foo=bar'],
109+
queryStringParameters: ''
110+
}
111+
112+
app.get('/test', async (request, reply) => {
113+
assert.equal(
114+
request.headers['x-my-header'],
115+
'wuuusaaa',
116+
'Header "X-My-Header" should match'
117+
)
118+
119+
assert.equal(
120+
request.headers.cookie,
121+
'foo=bar',
122+
'Cookie header should match'
123+
)
124+
assert.equal(
125+
request.headers['user-agent'],
126+
'lightMyRequest',
127+
'User-agent header should match'
128+
)
129+
assert.equal(
130+
request.headers.host,
131+
'localhost:80',
132+
'Host header should match'
133+
)
134+
135+
reply.header('Set-Cookie', 'qwerty=one')
136+
reply.header('Set-Cookie', 'qwerty=two')
137+
reply.header('content-type', 'application/json; charset=utf-8')
138+
return reply.send(Readable.from(JSON.stringify({ hello: 'world' })))
139+
})
140+
141+
const proxy = awsLambdaFastify(app, { payloadAsStream: true })
142+
143+
const { meta, stream } = await proxy(evt)
144+
145+
assert.equal(meta.statusCode, 200, 'Status code should be 200')
146+
147+
const data = await accumulate(stream)
148+
assert.equal(data.toString(), '{"hello":"world"}', 'Response body should match')
149+
150+
assert.equal(meta.isBase64Encoded, false, 'isBase64Encoded should be false')
151+
assert.ok(meta.headers, 'Headers should be defined')
152+
assert.equal(
153+
meta.headers['content-type'],
154+
'application/json; charset=utf-8',
155+
'Content-Type should match'
156+
)
157+
assert.equal(
158+
meta.headers['content-length'],
159+
undefined,
160+
'Content-Length should not be present'
161+
)
162+
assert.ok(meta.headers.date, 'Date header should be defined')
163+
assert.equal(
164+
meta.headers.connection,
165+
'keep-alive',
166+
'Connection header should match'
167+
)
168+
assert.deepEqual(
169+
meta.cookies,
170+
['qwerty=one', 'qwerty=two'],
171+
'Cookies should match'
172+
)
173+
})
174+
})

0 commit comments

Comments
 (0)