Skip to content

Commit 7b305cf

Browse files
Skn0tteduardoboucas
andauthoredMay 12, 2023
feat: add stream helper (#395)
This PR adds a helper to support streaming responses in Netlify Functions. The decorator handles all the `awslambda` things under the hood, all that devs have to do is return a `NodeJS.Readable` or a Web Stream as the `body`. It also updates the Node.js version to v14, so that `pipeline` is available, which makes this a breaking change technically - but as @ascorbic notes, not an actual one. --------- Co-authored-by: Eduardo Bouças <mail@eduardoboucas.com>
1 parent 88274d8 commit 7b305cf

File tree

9 files changed

+91
-9
lines changed

9 files changed

+91
-9
lines changed
 

‎.github/workflows/workflow.yml

+3-3
Original file line numberDiff line numberDiff line change
@@ -13,12 +13,12 @@ jobs:
1313
strategy:
1414
matrix:
1515
os: [ubuntu-latest, macOS-latest, windows-latest]
16-
node-version: [8.17.0, '*']
16+
node-version: [14.0.0, '*']
1717
exclude:
1818
- os: macOS-latest
19-
node-version: 8.17.0
19+
node-version: 14.0.0
2020
- os: windows-latest
21-
node-version: 8.17.0
21+
node-version: 14.0.0
2222
fail-fast: false
2323
steps:
2424
- name: Git checkout

‎package-lock.json

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,6 @@
6666
"typescript": "^4.4.4"
6767
},
6868
"engines": {
69-
"node": ">=8.3.0"
69+
"node": ">=14.0.0"
7070
}
7171
}

‎src/function/handler.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import type { Context } from './context.js'
22
import type { Event } from './event.js'
3-
import type { Response, BuilderResponse } from './response.js'
3+
import type { Response, BuilderResponse, StreamingResponse } from './response.js'
44

55
export interface HandlerCallback<ResponseType extends Response = Response> {
66
// eslint-disable-next-line @typescript-eslint/no-explicit-any
@@ -17,3 +17,7 @@ export interface BackgroundHandler<C extends Context = Context> {
1717

1818
export type Handler = BaseHandler<Response, Context>
1919
export type BuilderHandler = BaseHandler<BuilderResponse, Context>
20+
21+
export interface StreamingHandler {
22+
(event: Event, context: Context): Promise<StreamingResponse>
23+
}

‎src/function/index.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
11
export { Context as HandlerContext } from './context.js'
22
export { Event as HandlerEvent } from './event.js'
3-
export { BuilderHandler, Handler, BackgroundHandler, HandlerCallback } from './handler.js'
4-
export { BuilderResponse, Response as HandlerResponse } from './response.js'
3+
export { BuilderHandler, Handler, BackgroundHandler, HandlerCallback, StreamingHandler } from './handler.js'
4+
export { BuilderResponse, Response as HandlerResponse, StreamingResponse } from './response.js'

‎src/function/response.ts

+7
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
1+
import type { PipelineSource } from 'node:stream'
2+
13
export interface Response {
24
statusCode: number
35
headers?: {
@@ -12,3 +14,8 @@ export interface Response {
1214
export interface BuilderResponse extends Response {
1315
ttl?: number
1416
}
17+
18+
export interface StreamingResponse extends Omit<Response, 'body'> {
19+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
20+
body?: string | PipelineSource<any>
21+
}

‎src/lib/stream.ts

+70
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
import { pipeline as pipelineSync } from 'node:stream'
2+
import { promisify } from 'node:util'
3+
4+
import type { Handler, HandlerEvent, HandlerContext, StreamingHandler, StreamingResponse } from '../function/index.js'
5+
6+
// Node v14 doesn't have node:stream/promises
7+
const pipeline = promisify(pipelineSync)
8+
9+
declare global {
10+
// eslint-disable-next-line @typescript-eslint/no-namespace
11+
namespace awslambda {
12+
function streamifyResponse(
13+
handler: (event: HandlerEvent, responseStream: NodeJS.WritableStream, context: HandlerContext) => Promise<void>,
14+
): Handler
15+
16+
// eslint-disable-next-line @typescript-eslint/no-namespace
17+
namespace HttpResponseStream {
18+
function from(stream: NodeJS.WritableStream, metadata: Omit<StreamingResponse, 'body'>): NodeJS.WritableStream
19+
}
20+
}
21+
}
22+
23+
/**
24+
* Enables streaming responses. `body` accepts a Node.js `Readable` stream or a WHATWG `ReadableStream`.
25+
*
26+
* @example
27+
* ```
28+
* const { Readable } = require('stream');
29+
*
30+
* export const handler = stream(async (event, context) => {
31+
* const stream = Readable.from(Buffer.from(JSON.stringify(event)))
32+
* return {
33+
* statusCode: 200,
34+
* body: stream,
35+
* }
36+
* })
37+
* ```
38+
*
39+
* @example
40+
* ```
41+
* export const handler = stream(async (event, context) => {
42+
* const response = await fetch('https://api.openai.com/', { ... })
43+
* // ...
44+
* return {
45+
* statusCode: 200,
46+
* body: response.body, // Web stream
47+
* }
48+
* })
49+
* ```
50+
*
51+
* @param handler
52+
* @see https://ntl.fyi/streaming-func
53+
*/
54+
const stream = (handler: StreamingHandler): Handler =>
55+
awslambda.streamifyResponse(async (event, responseStream, context) => {
56+
const { body, ...httpResponseMetadata } = await handler(event, context)
57+
58+
const responseBody = awslambda.HttpResponseStream.from(responseStream, httpResponseMetadata)
59+
60+
if (typeof body === 'undefined') {
61+
responseBody.end()
62+
} else if (typeof body === 'string') {
63+
responseBody.write(body)
64+
responseBody.end()
65+
} else {
66+
await pipeline(body, responseBody)
67+
}
68+
})
69+
70+
export { stream }

‎src/main.ts

+1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
11
export { builder } from './lib/builder.js'
22
export { schedule } from './lib/schedule.js'
3+
export { stream } from './lib/stream.js'
34
export * from './function/index.js'

‎tsconfig.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111
// "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */
1212

1313
/* Language and Environment */
14-
"target": "es5" /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */,
14+
"target": "ES2020" /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */,
1515
// "lib": [], /* Specify a set of bundled library declaration files that describe the target runtime environment. */
1616
// "jsx": "preserve", /* Specify what JSX code is generated. */
1717
// "experimentalDecorators": true, /* Enable experimental support for TC39 stage 2 draft decorators. */

0 commit comments

Comments
 (0)