Skip to content

Commit 4080e1c

Browse files
Skn0ttpiehascorbic
authoredMar 11, 2024
fix: disable transfer-encoding leading to problems in fastly (#333)
* fix: disregard accept-encoding at origin * chore: format with prettier * fix: disable buggy transfer-encoding behaviour in fastly * chore: format with prettier * chore: add e2e test * fix: rename util * fix: types * chore: update comment * refactor: rename function * Update src/run/handlers/server.ts Co-authored-by: Matt Kane <m@mk.gg> --------- Co-authored-by: Skn0tt <Skn0tt@users.noreply.github.com> Co-authored-by: Michal Piechowiak <misiek.piechowiak@gmail.com> Co-authored-by: Matt Kane <m@mk.gg>
1 parent 72101f3 commit 4080e1c

File tree

8 files changed

+82
-18
lines changed

8 files changed

+82
-18
lines changed
 

‎src/run/handlers/server.ts

+28-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
1-
import { toComputeResponse, toReqRes } from '@fastly/http-compute-js'
1+
import type { OutgoingHttpHeaders } from 'http'
2+
3+
import { toComputeResponse, toReqRes, ComputeJsOutgoingMessage } from '@fastly/http-compute-js'
24
import { SpanStatusCode, trace } from '@opentelemetry/api'
35
import type { NextConfigComplete } from 'next/dist/server/config-shared.js'
46
import type { WorkerRequestHandler } from 'next/dist/server/lib/types.js'
@@ -18,6 +20,29 @@ import { createRequestContext, runWithRequestContext } from './request-context.c
1820

1921
let nextHandler: WorkerRequestHandler, nextConfig: NextConfigComplete, tagsManifest: TagsManifest
2022

23+
/**
24+
* When Next.js proxies requests externally, it writes the response back as-is.
25+
* In some cases, this includes Transfer-Encoding: chunked.
26+
* This triggers behaviour in @fastly/http-compute-js to separate chunks with chunk delimiters, which is not what we want at this level.
27+
* We want Lambda to control the behaviour around chunking, not this.
28+
* This workaround removes the Transfer-Encoding header, which makes the library send the response as-is.
29+
*/
30+
const disableFaultyTransferEncodingHandling = (res: ComputeJsOutgoingMessage) => {
31+
const originalStoreHeader = res._storeHeader
32+
res._storeHeader = function _storeHeader(firstLine, headers) {
33+
if (headers) {
34+
if (Array.isArray(headers)) {
35+
// eslint-disable-next-line no-param-reassign
36+
headers = headers.filter(([header]) => header.toLowerCase() !== 'transfer-encoding')
37+
} else {
38+
delete (headers as OutgoingHttpHeaders)['transfer-encoding']
39+
}
40+
}
41+
42+
return originalStoreHeader.call(this, firstLine, headers)
43+
}
44+
}
45+
2146
export default async (request: Request) => {
2247
const tracer = trace.getTracer('Next.js Runtime')
2348

@@ -53,6 +78,8 @@ export default async (request: Request) => {
5378

5479
const requestContext = createRequestContext()
5580

81+
disableFaultyTransferEncodingHandling(res as unknown as ComputeJsOutgoingMessage)
82+
5683
const resProxy = nextResponseProxy(res, requestContext)
5784

5885
// We don't await this here, because it won't resolve until the response is finished.

‎tests/e2e/simple-app.test.ts

+8
Original file line numberDiff line numberDiff line change
@@ -205,3 +205,11 @@ test('requesting a non existing page route that needs to be fetched from the blo
205205
)
206206
expect(headers['cache-control']).toBe('public,max-age=0,must-revalidate')
207207
})
208+
209+
test('Compressed rewrites are readable', async ({ simpleNextApp }) => {
210+
const resp = await fetch(`${simpleNextApp.url}/rewrite-no-basepath`)
211+
expect(resp.headers.get('content-length')).toBeNull()
212+
expect(resp.headers.get('transfer-encoding')).toEqual('chunked')
213+
expect(resp.headers.get('content-encoding')).toEqual('br')
214+
expect(await resp.text()).toContain('<title>Example Domain</title>')
215+
})

‎tests/fixtures/simple-next-app/next.config.js

+9
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,15 @@ const nextConfig = {
1717
],
1818
domains: ['images.pexels.com'],
1919
},
20+
async rewrites() {
21+
return [
22+
{
23+
source: '/rewrite-no-basepath',
24+
destination: 'https://example.vercel.sh',
25+
basePath: false,
26+
},
27+
]
28+
},
2029
}
2130

2231
module.exports = nextConfig

‎tests/integration/simple-app.test.ts

+17
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ import {
1414
getBlobEntries,
1515
startMockBlobStore,
1616
} from '../utils/helpers.js'
17+
import { gunzipSync } from 'node:zlib'
1718

1819
// Disable the verbose logging of the lambda-local runtime
1920
getLogger().level = 'alert'
@@ -139,3 +140,19 @@ test<FixtureTestContext>('handlers can add cookies in route handlers with the co
139140
expect(setCookieHeader).toContain('test1=value1; Path=/; Secure')
140141
expect(setCookieHeader).toContain('test2=value2; Path=/handler; HttpOnly')
141142
})
143+
144+
// there's a bug where requests accept-encoding header
145+
// result in corrupted bodies
146+
// while that bug stands, we want to ignore accept-encoding
147+
test<FixtureTestContext>('rewrites to external addresses dont use compression', async (ctx) => {
148+
await createFixture('simple-next-app', ctx)
149+
await runPlugin(ctx)
150+
const page = await invokeFunction(ctx, {
151+
url: '/rewrite-no-basepath',
152+
headers: { 'accept-encoding': 'gzip' },
153+
})
154+
expect(page.headers).not.toHaveProperty('content-length')
155+
expect(page.headers).not.toHaveProperty('transfer-encoding')
156+
expect(page.headers['content-encoding']).toBe('gzip')
157+
expect(gunzipSync(page.bodyBuffer).toString('utf-8')).toContain('<title>Example Domain</title>')
158+
})

‎tests/utils/fixture.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ import { env } from 'node:process'
1717
import { fileURLToPath } from 'node:url'
1818
import { v4 } from 'uuid'
1919
import { LocalServer } from './local-server.js'
20-
import { streamToString } from './stream-to-string.js'
20+
import { streamToBuffer } from './stream-to-buffer.js'
2121

2222
import { glob } from 'fast-glob'
2323
import {
@@ -393,9 +393,12 @@ export async function invokeFunction(
393393
response.headers || {},
394394
)
395395

396+
const bodyBuffer = await streamToBuffer(response.body)
397+
396398
return {
397399
statusCode: response.statusCode,
398-
body: await streamToString(response.body),
400+
bodyBuffer,
401+
body: bodyBuffer.toString('utf-8'),
399402
headers: responseHeaders,
400403
isBase64Encoded: response.isBase64Encoded,
401404
}

‎tests/utils/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
export * from './helpers.js'
22
export * from './mock-file-system.js'
3-
export * from './stream-to-string.js'
3+
export * from './stream-to-buffer.js'

‎tests/utils/stream-to-buffer.ts

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
/**
2+
* Converts a readable stream to a buffer
3+
* @param stream Node.js Readable stream
4+
* @returns
5+
*/
6+
export function streamToBuffer(stream: NodeJS.ReadableStream) {
7+
const chunks: Buffer[] = []
8+
9+
return new Promise<Buffer>((resolve, reject) => {
10+
stream.on('data', (chunk) => chunks.push(Buffer.from(chunk)))
11+
stream.on('error', (err) => reject(err))
12+
stream.on('end', () => resolve(Buffer.concat(chunks)))
13+
})
14+
}

‎tests/utils/stream-to-string.ts

-14
This file was deleted.

0 commit comments

Comments
 (0)
Please sign in to comment.