From f19ffd1f6b2da4cccbd2be6e48429c566719ade6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Vin=C3=ADcius=20Louren=C3=A7o?= Date: Tue, 2 Jan 2024 23:10:56 -0300 Subject: [PATCH] feat(network): support buffering transfer-encoding: chunked --- src/adapters/aws/alb.adapter.ts | 9 ++ src/adapters/aws/api-gateway-v1.adapter.ts | 31 ++++--- src/adapters/aws/api-gateway-v2.adapter.ts | 32 +++++--- src/network/response.ts | 20 ++++- test/adapters/aws/alb.adapter.spec.ts | 40 +++++++++ .../aws/api-gateway-v1.adapter.spec.ts | 65 +++++++++++++++ .../aws/api-gateway-v2.adapter.spec.ts | 80 ++++++++++++++++++ .../transfer-encoding-chunked-support.spec.ts | 82 +++++++++++++++++++ www/docs/main/adapters/aws/alb.mdx | 4 + www/docs/main/adapters/aws/api-gateway-v1.mdx | 12 ++- www/docs/main/adapters/aws/api-gateway-v2.mdx | 13 ++- 11 files changed, 358 insertions(+), 30 deletions(-) create mode 100644 test/issues/issue-165/transfer-encoding-chunked-support.spec.ts diff --git a/src/adapters/aws/alb.adapter.ts b/src/adapters/aws/alb.adapter.ts index 4f0fe99a..1ffed4c1 100644 --- a/src/adapters/aws/alb.adapter.ts +++ b/src/adapters/aws/alb.adapter.ts @@ -145,6 +145,15 @@ export class AlbAdapter ? getFlattenedHeadersMap(responseHeaders) : undefined; + if (headers && headers['transfer-encoding'] === 'chunked') + delete headers['transfer-encoding']; + + if ( + multiValueHeaders && + multiValueHeaders['transfer-encoding']?.includes('chunked') + ) + delete multiValueHeaders['transfer-encoding']; + return { statusCode, body, diff --git a/src/adapters/aws/api-gateway-v1.adapter.ts b/src/adapters/aws/api-gateway-v1.adapter.ts index 9eb6a9f3..f0175511 100644 --- a/src/adapters/aws/api-gateway-v1.adapter.ts +++ b/src/adapters/aws/api-gateway-v1.adapter.ts @@ -11,6 +11,7 @@ import type { import { type StripBasePathFn, buildStripBasePath, + getDefaultIfUndefined, getEventBodyAsBuffer, getMultiValueHeadersMap, getPathWithQueryStringParams, @@ -31,6 +32,16 @@ export interface ApiGatewayV1Options { * @defaultValue '' */ stripBasePath?: string; + + /** + * Throw an exception when you send the `transfer-encoding=chunked`, currently, API Gateway doesn't support chunked transfer. + * If this is set to `false`, we will remove the `transfer-encoding` header from the response and buffer the response body + * while we remove the special characters inserted by the chunked encoding. + * + * @remarks To learn more https://github.com/H4ad/serverless-adapter/issues/165 + * @defaultValue true + */ + throwOnChunkedTransferEncoding?: boolean; } /** @@ -165,21 +176,21 @@ export class ApiGatewayV1Adapter }: GetResponseAdapterProps): APIGatewayProxyResult { const multiValueHeaders = getMultiValueHeadersMap(responseHeaders); + const shouldThrowOnChunkedTransferEncoding = getDefaultIfUndefined( + this.options?.throwOnChunkedTransferEncoding, + true, + ); const transferEncodingHeader = multiValueHeaders['transfer-encoding']; const hasTransferEncodingChunked = transferEncodingHeader?.some(value => value.includes('chunked'), ); - if (hasTransferEncodingChunked) { - throw new Error( - 'chunked encoding in headers is not supported by API Gateway V1', - ); - } - - if (response?.chunkedEncoding) { - throw new Error( - 'chunked encoding in response is not supported by API Gateway V1', - ); + if (hasTransferEncodingChunked || response?.chunkedEncoding) { + if (shouldThrowOnChunkedTransferEncoding) { + throw new Error( + 'chunked encoding in headers is not supported by API Gateway V1', + ); + } else delete multiValueHeaders['transfer-encoding']; } return { diff --git a/src/adapters/aws/api-gateway-v2.adapter.ts b/src/adapters/aws/api-gateway-v2.adapter.ts index ba3923cc..4282a197 100644 --- a/src/adapters/aws/api-gateway-v2.adapter.ts +++ b/src/adapters/aws/api-gateway-v2.adapter.ts @@ -11,6 +11,7 @@ import type { import { type StripBasePathFn, buildStripBasePath, + getDefaultIfUndefined, getEventBodyAsBuffer, getFlattenedHeadersMapAndCookies, getPathWithQueryStringParams, @@ -31,6 +32,16 @@ export interface ApiGatewayV2Options { * @defaultValue '' */ stripBasePath?: string; + + /** + * Throw an exception when you send the `transfer-encoding=chunked`, currently, API Gateway doesn't support chunked transfer. + * If this is set to `false`, we will remove the `transfer-encoding` header from the response and buffer the response body + * while we remove the special characters inserted by the chunked encoding. + * + * @remarks To learn more https://github.com/H4ad/serverless-adapter/issues/165 + * @defaultValue true + */ + throwOnChunkedTransferEncoding?: boolean; } /** @@ -152,22 +163,23 @@ export class ApiGatewayV2Adapter const { cookies, headers } = getFlattenedHeadersMapAndCookies(responseHeaders); + const shouldThrowOnChunkedTransferEncoding = getDefaultIfUndefined( + this.options?.throwOnChunkedTransferEncoding, + true, + ); + const transferEncodingHeader: string | undefined = headers['transfer-encoding']; const hasTransferEncodingChunked = transferEncodingHeader && transferEncodingHeader.includes('chunked'); - if (hasTransferEncodingChunked) { - throw new Error( - 'chunked encoding in headers is not supported by API Gateway V2', - ); - } - - if (response?.chunkedEncoding) { - throw new Error( - 'chunked encoding in response is not supported by API Gateway V2', - ); + if (hasTransferEncodingChunked || response?.chunkedEncoding) { + if (shouldThrowOnChunkedTransferEncoding) { + throw new Error( + 'chunked encoding in headers is not supported by API Gateway V2', + ); + } else delete headers['transfer-encoding']; } return { diff --git a/src/network/response.ts b/src/network/response.ts index 2cf0e883..452d85a6 100644 --- a/src/network/response.ts +++ b/src/network/response.ts @@ -5,6 +5,7 @@ import { NO_OP } from '../core'; import { getString } from './utils'; const headerEnd = '\r\n\r\n'; +const endChunked = '0\r\n\r\n'; const BODY = Symbol('Response body'); const HEADERS = Symbol('Response headers'); @@ -50,6 +51,11 @@ export class ServerlessResponse extends ServerResponse { this.chunkedEncoding = false; this._header = ''; + // this ignore is used because I need to ignore these write calls: + // https://github.com/nodejs/node/blob/main/lib/_http_outgoing.js#L934-L935 + // https://github.com/nodejs/node/blob/main/lib/_http_outgoing.js#L937 + let writesToIgnore = 1; + const socket: Partial & { _writableState: any } = { _writableState: {}, writable: true, @@ -68,15 +74,23 @@ export class ServerlessResponse extends ServerResponse { encoding = null; } - if (this._header === '' || this._wroteHeader) addData(this, data); - else { + if (this._header === '' || this._wroteHeader) { + if (!this.chunkedEncoding) addData(this, data); + else { + if (writesToIgnore > 0) writesToIgnore--; + else if (data !== endChunked) { + addData(this, data); + writesToIgnore = 3; + } + } + } else { const string = getString(data); const index = string.indexOf(headerEnd); if (index !== -1) { const remainder = string.slice(index + headerEnd.length); - if (remainder) addData(this, remainder); + if (remainder && !this.chunkedEncoding) addData(this, remainder); this._wroteHeader = true; } diff --git a/test/adapters/aws/alb.adapter.spec.ts b/test/adapters/aws/alb.adapter.spec.ts index 94823cc4..e003d0a4 100644 --- a/test/adapters/aws/alb.adapter.spec.ts +++ b/test/adapters/aws/alb.adapter.spec.ts @@ -256,6 +256,46 @@ describe(AlbAdapter.name, () => { ); expect(result).toHaveProperty('isBase64Encoded', resultIsBase64Encoded); }); + + it('should remove the transfer-encoding header if it is chunked', () => { + const event = createAlbEventWithMultiValueHeaders('GET', '/events'); + const responseHeaders = getFlattenedHeadersMap(event.multiValueHeaders!); + const responseMultiValueHeaders = + getMultiValueHeadersMap(responseHeaders); + + responseHeaders['transfer-encoding'] = 'chunked'; + + const result = adapter.getResponse({ + event, + headers: responseHeaders, + body: '', + log: {} as ILogger, + isBase64Encoded: false, + statusCode: 200, + }); + + expect(result).toHaveProperty('headers', undefined); + expect(result).toHaveProperty( + 'multiValueHeaders', + responseMultiValueHeaders, + ); + + responseMultiValueHeaders['transfer-encoding'] = ['chunked']; + + const event2 = createAlbEvent('GET', '/events'); + const responseHeaders2 = getFlattenedHeadersMap(event2.headers!); + + const result2 = adapter.getResponse({ + event: event2, + headers: responseHeaders2, + body: '', + log: {} as ILogger, + isBase64Encoded: false, + statusCode: 200, + }); + + expect(result2).toHaveProperty('headers', responseHeaders2); + }); }); describe('onErrorWhileForwarding', () => { diff --git a/test/adapters/aws/api-gateway-v1.adapter.spec.ts b/test/adapters/aws/api-gateway-v1.adapter.spec.ts index 4858d865..9832fa67 100644 --- a/test/adapters/aws/api-gateway-v1.adapter.spec.ts +++ b/test/adapters/aws/api-gateway-v1.adapter.spec.ts @@ -217,6 +217,71 @@ describe(ApiGatewayV1Adapter.name, () => { }), ).toThrowError('is not supported'); }); + + describe('when throwOnChunkedTransferEncoding=false', () => { + it('should NOT throw an error when framework send chunkedEncoding=true in response', () => { + const customAdapter = new ApiGatewayV1Adapter({ + throwOnChunkedTransferEncoding: false, + }); + + const method = 'GET'; + const path = '/events/stream'; + const requestBody = undefined; + + const resultBody = '{"success":true}'; + const resultStatusCode = 200; + const resultIsBase64Encoded = false; + + const event = createApiGatewayV1(method, path, requestBody); + const resultHeaders = getFlattenedHeadersMap(event.headers); + + const fakeChunkedResponse = new ServerlessResponse({ method }); + + fakeChunkedResponse.chunkedEncoding = true; + + const result = customAdapter.getResponse({ + event, + log: {} as ILogger, + body: resultBody, + isBase64Encoded: resultIsBase64Encoded, + statusCode: resultStatusCode, + headers: resultHeaders, + response: fakeChunkedResponse, + }); + + expect(result.multiValueHeaders!['transfer-encoding']).toBeUndefined(); + }); + + it('should NOT throw an error when framework send transfer-encoding=chunked in headers', () => { + const customAdapter = new ApiGatewayV1Adapter({ + throwOnChunkedTransferEncoding: false, + }); + + const method = 'GET'; + const path = '/events/stream'; + const requestBody = undefined; + + const resultBody = '{"success":true}'; + const resultStatusCode = 200; + const resultIsBase64Encoded = false; + + const event = createApiGatewayV1(method, path, requestBody); + const resultHeaders = getFlattenedHeadersMap(event.headers); + + resultHeaders['transfer-encoding'] = 'gzip,chunked'; + + const result = customAdapter.getResponse({ + event, + log: {} as ILogger, + body: resultBody, + isBase64Encoded: resultIsBase64Encoded, + statusCode: resultStatusCode, + headers: resultHeaders, + }); + + expect(result.multiValueHeaders!['transfer-encoding']).toBeUndefined(); + }); + }); }); describe('onErrorWhileForwarding', () => { diff --git a/test/adapters/aws/api-gateway-v2.adapter.spec.ts b/test/adapters/aws/api-gateway-v2.adapter.spec.ts index bdbc6688..cf465a07 100644 --- a/test/adapters/aws/api-gateway-v2.adapter.spec.ts +++ b/test/adapters/aws/api-gateway-v2.adapter.spec.ts @@ -287,6 +287,86 @@ describe(ApiGatewayV2Adapter.name, () => { }), ).toThrowError('is not supported'); }); + + describe('when throwOnChunkedTransferEncoding=false', () => { + it('should NOT throw an error when framework send transfer-encoding=chunked in headers', () => { + const customAdapter = new ApiGatewayV2Adapter({ + throwOnChunkedTransferEncoding: false, + }); + + const method = 'GET'; + const path = '/collaborators/stream'; + const requestBody = undefined; + + const resultBody = '{"success":true}'; + const resultStatusCode = 200; + const resultIsBase64Encoded = false; + + const event = createApiGatewayV2(method, path, requestBody); + const resultHeaders = getFlattenedHeadersMap(event.headers); + + resultHeaders['transfer-encoding'] = 'gzip,chunked'; + + const result1 = customAdapter.getResponse({ + event, + log: {} as ILogger, + body: resultBody, + isBase64Encoded: resultIsBase64Encoded, + statusCode: resultStatusCode, + headers: resultHeaders, + }); + + expect(result1.headers!['transfer-encoding']).toBeUndefined(); + + const resultMultiValueHeaders = getMultiValueHeadersMap(event.headers); + + resultMultiValueHeaders['transfer-encoding'] = ['gzip', 'chunked']; + + const result2 = customAdapter.getResponse({ + event, + log: {} as ILogger, + body: resultBody, + isBase64Encoded: resultIsBase64Encoded, + statusCode: resultStatusCode, + headers: resultMultiValueHeaders, + }); + + expect(result2.headers!['transfer-encoding']).toBeUndefined(); + }); + + it('should NOT throw an error when framework send chunkedEncoding=true in response', () => { + const customAdapter = new ApiGatewayV2Adapter({ + throwOnChunkedTransferEncoding: false, + }); + + const method = 'GET'; + const path = '/collaborators/stream'; + const requestBody = undefined; + + const resultBody = '{"success":true}'; + const resultStatusCode = 200; + const resultIsBase64Encoded = false; + + const event = createApiGatewayV2(method, path, requestBody); + const resultHeaders = getFlattenedHeadersMap(event.headers); + + const fakeChunkedResponse = new ServerlessResponse({ method }); + + fakeChunkedResponse.chunkedEncoding = true; + + const result = customAdapter.getResponse({ + event, + log: {} as ILogger, + body: resultBody, + isBase64Encoded: resultIsBase64Encoded, + statusCode: resultStatusCode, + headers: resultHeaders, + response: fakeChunkedResponse, + }); + + expect(result.headers!['transfer-encoding']).toBeUndefined(); + }); + }); }); describe('onErrorWhileForwarding', () => { diff --git a/test/issues/issue-165/transfer-encoding-chunked-support.spec.ts b/test/issues/issue-165/transfer-encoding-chunked-support.spec.ts new file mode 100644 index 00000000..a5804428 --- /dev/null +++ b/test/issues/issue-165/transfer-encoding-chunked-support.spec.ts @@ -0,0 +1,82 @@ +import { setTimeout } from 'timers/promises'; +import { Readable } from 'node:stream'; +import { describe, expect, it } from 'vitest'; +import express from 'express'; +import fastify from 'fastify'; +import { ServerlessAdapter } from '../../../src'; +import { DefaultHandler } from '../../../src/handlers/default'; +import { PromiseResolver } from '../../../src/resolvers/promise'; +import { ExpressFramework } from '../../../src/frameworks/express'; +import { AlbAdapter } from '../../../src/adapters/aws'; +import { createAlbEvent } from '../../adapters/aws/utils/alb-event'; +import { FastifyFramework } from '../../../src/frameworks/fastify'; + +const expectedResult = 'INITIAL PAYLOAD RESPONSE\nFINAL PAYLOAD RESPONSE\n'; + +describe('Issue 165: cannot handle transfer-encoding: chunked', () => { + it('express: should handle transfer-encoding: chunked', async () => { + const app = express(); + + // eslint-disable-next-line @typescript-eslint/no-misused-promises + app.get('/chunked-response', async (_req, res) => { + // Send headers right away + res.setHeader('Content-Type', 'text/plain'); + res.setHeader('Transfer-Encoding', 'chunked'); + res.status(200); + + res.write('INITIAL PAYLOAD RESPONSE\n'); + await setTimeout(50); + res.end('FINAL PAYLOAD RESPONSE\n'); + }); + + const albEvent = createAlbEvent('GET', '/chunked-response'); + + const handler = ServerlessAdapter.new(app) + .setHandler(new DefaultHandler()) + .setFramework(new ExpressFramework()) + .setResolver(new PromiseResolver()) + .addAdapter(new AlbAdapter()) + .build(); + + const result = await handler(albEvent, {}); + + expect(result.body).toEqual(expectedResult); + expect(result.headers['content-length'], expectedResult.length.toString()); + }); + + it('fastify: should handle transfer-encoding: chunked', async () => { + const app = fastify(); + + // eslint-disable-next-line @typescript-eslint/no-misused-promises + app.get('/chunked-response', async (_req, res) => { + // Send headers right away + res.type('text/plain'); + res.header('Transfer-Encoding', 'chunked'); + res.status(200); + + const buffer = new Readable(); + buffer._read = () => {}; + + res.send(buffer); + + buffer.push('INITIAL PAYLOAD RESPONSE\n'); + await setTimeout(50); + buffer.push('FINAL PAYLOAD RESPONSE\n'); + buffer.push(null); + }); + + const albEvent = createAlbEvent('GET', '/chunked-response'); + + const handler = ServerlessAdapter.new(app) + .setHandler(new DefaultHandler()) + .setFramework(new FastifyFramework()) + .setResolver(new PromiseResolver()) + .addAdapter(new AlbAdapter()) + .build(); + + const result = await handler(albEvent, {}); + + expect(result.body).toEqual(expectedResult); + expect(result.headers['content-length'], expectedResult.length.toString()); + }); +}); diff --git a/www/docs/main/adapters/aws/alb.mdx b/www/docs/main/adapters/aws/alb.mdx index 873c7da7..f31c76bc 100644 --- a/www/docs/main/adapters/aws/alb.mdx +++ b/www/docs/main/adapters/aws/alb.mdx @@ -87,3 +87,7 @@ export const handler = ServerlessAdapter.new(app) .build(); ``` +### Transfer Encoding Chunked + +ALB currently didn't support chunked transfer, so the response body will be buffered +without the special characters introduced by the chunked transfer keeping the body complete. diff --git a/www/docs/main/adapters/aws/api-gateway-v1.mdx b/www/docs/main/adapters/aws/api-gateway-v1.mdx index 1f18f98f..99ba18a5 100644 --- a/www/docs/main/adapters/aws/api-gateway-v1.mdx +++ b/www/docs/main/adapters/aws/api-gateway-v1.mdx @@ -139,8 +139,14 @@ export const handler = ServerlessAdapter.new(app) .build(); ``` -:::danger +### Transfer Encoding Chunked -As per [know issues](https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-known-issues.html), we throw an exception when you send the `transfer-encoding=chunked`, currently, API Gateway doesn't support chunked transfer. +API Gateway V1 currently didn't support chunked transfer, so we throw an exception when you send the `transfer-encoding=chunked`. -::: +But, you can disable the exception by setting the `throwOnChunkedTransfer` to `false` in the [ApiGatewayV1Options](/docs/api/Adapters/AWS/ApiGatewayV1Adapter/ApiGatewayV1Options). + +```ts title="index.ts" +new ApiGatewayV1Adapter({ throwOnChunkedTransfer: false }) +``` + +The response body will be buffered without the special characters introduced by the chunked transfer keeping the body complete. diff --git a/www/docs/main/adapters/aws/api-gateway-v2.mdx b/www/docs/main/adapters/aws/api-gateway-v2.mdx index 0276f10b..bdeb9c8d 100644 --- a/www/docs/main/adapters/aws/api-gateway-v2.mdx +++ b/www/docs/main/adapters/aws/api-gateway-v2.mdx @@ -129,9 +129,14 @@ export const handler = ServerlessAdapter.new(app) .build(); ``` -:::danger +### Transfer Encoding Chunked -As per [know issues](https://docs.aws.amazon.com/apigateway/latest/developerguide/api-gateway-known-issues.html), we throw an exception when you send the `transfer-encoding=chunked`, currently, API Gateway doesn't support chunked transfer. -If you use Function URL, you can send the `transfer-encoding=chunked` when you configure the invocation method as `RESPONSE_STREAM`. +API Gateway V2 currently didn't support chunked transfer, so we throw an exception when you send the `transfer-encoding=chunked`. -::: +But, you can disable the exception by setting the `throwOnChunkedTransfer` to `false` in the [ApiGatewayV2Options](/docs/api/Adapters/AWS/ApiGatewayV2Adapter/ApiGatewayV2Options). + +```ts title="index.ts" +new ApiGatewayV1Adapter({ throwOnChunkedTransfer: false }) +``` + +The response body will be buffered without the special characters introduced by the chunked transfer keeping the body complete.