Skip to content

Commit 2ed7f32

Browse files
RamIdeaspetebacondarwinmrbbot
authored
Workaround "Error: Network connection lost." in POSTs from workerd to workerd (#5106)
* ensure request.body is drained potential workaround for #5095 * regression test * fixup! regression test * fixup! regression test Use larger body and ensure request read before response sent * allow request draining to be disabled and add changeset * remove unnecessary export * fixup! ensure request.body is drained --------- Co-authored-by: Peter Bacon Darwin <[email protected]> Co-authored-by: bcoll <[email protected]>
1 parent 54f6bfc commit 2ed7f32

File tree

5 files changed

+82
-1
lines changed

5 files changed

+82
-1
lines changed

.changeset/empty-zoos-beam.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
"wrangler": patch
3+
---
4+
5+
fix: automatically drain incoming request bodies
6+
7+
Previously, requests sent to `wrangler dev` with unconsumed bodies could result in `Network connection lost` errors. This change attempts to work around the issue by ensuring incoming request bodies are drained if they're not used. This is a temporary fix whilst we try to address the underlying issue. Whilst we don't think this change will introduce any other issues, it can be disabled by setting the `WRANGLER_DISABLE_REQUEST_BODY_DRAINING=true` environment variable. Note this fix is only applied if you've enabled Wrangler's bundling—`--no-bundle` mode continues to have the previous behaviour.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
import { resolve } from "path";
2+
import { fetch } from "undici";
3+
import { afterAll, beforeAll, describe, it } from "vitest";
4+
import { runWranglerDev } from "../../shared/src/run-wrangler-long-lived";
5+
6+
describe("wrangler dev", () => {
7+
let ip: string, port: number, stop: (() => Promise<unknown>) | undefined;
8+
9+
beforeAll(async () => {
10+
({ ip, port, stop } = await runWranglerDev(resolve(__dirname, ".."), [
11+
"--port=0",
12+
"--inspector-port=0",
13+
]));
14+
});
15+
16+
afterAll(async () => {
17+
await stop?.();
18+
});
19+
20+
// https://github.com/cloudflare/workers-sdk/issues/5095
21+
it("should not fail requests if the Worker does not drain the body", async ({
22+
expect,
23+
}) => {
24+
const COUNT = 30;
25+
const requests: boolean[] = [];
26+
const errors: string[] = [];
27+
28+
const body = new Uint8Array(2_000);
29+
for (let i = 0; i < COUNT; i++) {
30+
const response = await fetch(`http://${ip}:${port}/random`, {
31+
method: "POST",
32+
body,
33+
});
34+
requests.push(response.ok);
35+
if (!response.ok) {
36+
errors.push(await response.text());
37+
}
38+
}
39+
40+
expect(requests.length).toBe(COUNT);
41+
expect(errors).toEqual([]);
42+
expect(requests).toEqual(Array.from({ length: COUNT }).map((i) => true));
43+
});
44+
});

packages/wrangler/src/deployment-bundle/bundle.ts

+11
Original file line numberDiff line numberDiff line change
@@ -138,6 +138,17 @@ export async function bundleWorker(
138138
// At this point, we take the opportunity to "wrap" the worker with middleware.
139139
const middlewareToLoad: MiddlewareLoader[] = [];
140140

141+
if (
142+
targetConsumer === "dev" &&
143+
!process.env.WRANGLER_DISABLE_REQUEST_BODY_DRAINING
144+
) {
145+
middlewareToLoad.push({
146+
name: "ensure-req-body-drained",
147+
path: "templates/middleware/middleware-ensure-req-body-drained.ts",
148+
supports: ["modules", "service-worker"],
149+
});
150+
}
151+
141152
if (targetConsumer === "dev" && !!testScheduled) {
142153
middlewareToLoad.push({
143154
name: "scheduled",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
import type { Middleware } from "./common";
2+
3+
const drainBody: Middleware = async (request, env, _ctx, middlewareCtx) => {
4+
try {
5+
return await middlewareCtx.next(request, env);
6+
} finally {
7+
try {
8+
if (request.body !== null && !request.bodyUsed) {
9+
const reader = request.body.getReader();
10+
while (!(await reader.read()).done) {}
11+
}
12+
} catch (e) {
13+
console.error("Failed to drain the unused request body.", e);
14+
}
15+
}
16+
};
17+
18+
export default drainBody;

packages/wrangler/turbo.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,8 @@
3939
"HTTP_PROXY",
4040
"CI_OS",
4141
"SENTRY_DSN",
42-
"SYSTEMROOT"
42+
"SYSTEMROOT",
43+
"WRANGLER_DISABLE_REQUEST_BODY_DRAINING"
4344
]
4445
}
4546
}

0 commit comments

Comments
 (0)