mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-22 13:08:07 +00:00
fix(e2e): cancel timed out response reads
This commit is contained in:
@@ -5,6 +5,12 @@ function bodyTooLargeError(label, byteLimit) {
|
||||
});
|
||||
}
|
||||
|
||||
function cancelReaderSoon(reader) {
|
||||
void Promise.resolve()
|
||||
.then(() => reader.cancel())
|
||||
.catch(() => {});
|
||||
}
|
||||
|
||||
export async function readBoundedResponseText(response, label, byteLimit, timeoutPromise) {
|
||||
const contentLength = response.headers.get("content-length");
|
||||
if (contentLength) {
|
||||
@@ -22,22 +28,35 @@ export async function readBoundedResponseText(response, label, byteLimit, timeou
|
||||
const decoder = new TextDecoder();
|
||||
let byteCount = 0;
|
||||
let text = "";
|
||||
let canceled = false;
|
||||
try {
|
||||
while (true) {
|
||||
const { done, value } = await (timeoutPromise
|
||||
? Promise.race([reader.read(), timeoutPromise])
|
||||
: reader.read());
|
||||
const read = reader.read();
|
||||
const readWithTimeout = timeoutPromise
|
||||
? Promise.race([
|
||||
read,
|
||||
timeoutPromise.catch((error) => {
|
||||
canceled = true;
|
||||
cancelReaderSoon(reader);
|
||||
throw error;
|
||||
}),
|
||||
])
|
||||
: read;
|
||||
const { done, value } = await readWithTimeout;
|
||||
if (done) {
|
||||
return text + decoder.decode();
|
||||
}
|
||||
byteCount += value.byteLength;
|
||||
if (byteCount > byteLimit) {
|
||||
canceled = true;
|
||||
await reader.cancel().catch(() => {});
|
||||
throw bodyTooLargeError(label, byteLimit);
|
||||
}
|
||||
text += decoder.decode(value, { stream: true });
|
||||
}
|
||||
} finally {
|
||||
reader.releaseLock();
|
||||
if (!canceled) {
|
||||
reader.releaseLock();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
85
test/scripts/bounded-response-text.test.ts
Normal file
85
test/scripts/bounded-response-text.test.ts
Normal file
@@ -0,0 +1,85 @@
|
||||
// E2E bounded response text tests cover shared E2E HTTP body limits.
|
||||
import { describe, expect, it } from "vitest";
|
||||
import { readBoundedResponseText } from "../../scripts/e2e/lib/bounded-response-text.mjs";
|
||||
|
||||
describe("scripts/e2e/lib/bounded-response-text.mjs", () => {
|
||||
it("cancels pending response body reads when the timeout wins", async () => {
|
||||
let canceled = false;
|
||||
const response = {
|
||||
headers: new Headers(),
|
||||
body: {
|
||||
getReader() {
|
||||
return {
|
||||
read() {
|
||||
return new Promise<ReadableStreamReadResult<Uint8Array>>(() => {});
|
||||
},
|
||||
async cancel() {
|
||||
canceled = true;
|
||||
},
|
||||
releaseLock() {
|
||||
throw new Error("releaseLock should not run while a read is pending");
|
||||
},
|
||||
};
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
await expect(
|
||||
readBoundedResponseText(
|
||||
response,
|
||||
"probe",
|
||||
1024,
|
||||
Promise.reject(new Error("probe timed out")),
|
||||
),
|
||||
).rejects.toThrow("probe timed out");
|
||||
|
||||
expect(canceled).toBe(true);
|
||||
});
|
||||
|
||||
it("keeps timeout rejection ahead of cancel-unblocked stream reads", async () => {
|
||||
let canceled = false;
|
||||
const response = new Response(
|
||||
new ReadableStream({
|
||||
pull() {
|
||||
return new Promise(() => {});
|
||||
},
|
||||
cancel() {
|
||||
canceled = true;
|
||||
},
|
||||
}),
|
||||
{ headers: new Headers() },
|
||||
);
|
||||
|
||||
await expect(
|
||||
readBoundedResponseText(
|
||||
response,
|
||||
"probe",
|
||||
1024,
|
||||
Promise.reject(new Error("probe timed out")),
|
||||
),
|
||||
).rejects.toThrow("probe timed out");
|
||||
|
||||
expect(canceled).toBe(true);
|
||||
});
|
||||
|
||||
it("cancels oversized streamed response bodies", async () => {
|
||||
let canceled = false;
|
||||
const response = new Response(
|
||||
new ReadableStream({
|
||||
start(controller) {
|
||||
controller.enqueue(new Uint8Array(17));
|
||||
},
|
||||
cancel() {
|
||||
canceled = true;
|
||||
},
|
||||
}),
|
||||
{ headers: new Headers() },
|
||||
);
|
||||
|
||||
await expect(readBoundedResponseText(response, "probe", 16)).rejects.toMatchObject({
|
||||
code: "ETOOBIG",
|
||||
message: "probe response body exceeded 16 bytes",
|
||||
});
|
||||
expect(canceled).toBe(true);
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user