mirror of
https://github.com/denoland/deno.git
synced 2025-01-12 00:54:02 -05:00
http: make server handle bad client requests properly (denoland/deno_std#419)
Original: 7620fe1a82
This commit is contained in:
parent
e2debab359
commit
3cfc1244d8
2 changed files with 97 additions and 5 deletions
|
@ -197,7 +197,7 @@ export class ServerRequest {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function readRequest(
|
export async function readRequest(
|
||||||
bufr: BufReader
|
bufr: BufReader
|
||||||
): Promise<[ServerRequest, BufState]> {
|
): Promise<[ServerRequest, BufState]> {
|
||||||
const req = new ServerRequest();
|
const req = new ServerRequest();
|
||||||
|
@ -235,7 +235,11 @@ export class Server implements AsyncIterable<ServerRequest> {
|
||||||
let req: ServerRequest;
|
let req: ServerRequest;
|
||||||
|
|
||||||
while (!this.closing) {
|
while (!this.closing) {
|
||||||
|
try {
|
||||||
[req, bufStateErr] = await readRequest(bufr);
|
[req, bufStateErr] = await readRequest(bufr);
|
||||||
|
} catch (err) {
|
||||||
|
bufStateErr = err;
|
||||||
|
}
|
||||||
if (bufStateErr) break;
|
if (bufStateErr) break;
|
||||||
req.w = w;
|
req.w = w;
|
||||||
yield req;
|
yield req;
|
||||||
|
@ -247,7 +251,11 @@ export class Server implements AsyncIterable<ServerRequest> {
|
||||||
if (bufStateErr === "EOF") {
|
if (bufStateErr === "EOF") {
|
||||||
// The connection was gracefully closed.
|
// The connection was gracefully closed.
|
||||||
} else if (bufStateErr instanceof Error) {
|
} else if (bufStateErr instanceof Error) {
|
||||||
// TODO(ry): send something back like a HTTP 500 status.
|
// An error was thrown while parsing request headers.
|
||||||
|
await writeResponse(req.w, {
|
||||||
|
status: 400,
|
||||||
|
body: new TextEncoder().encode(`${bufStateErr.message}\r\n\r\n`)
|
||||||
|
});
|
||||||
} else if (this.closing) {
|
} else if (this.closing) {
|
||||||
// There are more requests incoming but the server is closing.
|
// There are more requests incoming but the server is closing.
|
||||||
// TODO(ry): send a back a HTTP 503 Service Unavailable status.
|
// TODO(ry): send a back a HTTP 503 Service Unavailable status.
|
||||||
|
|
|
@ -7,8 +7,13 @@
|
||||||
|
|
||||||
const { Buffer } = Deno;
|
const { Buffer } = Deno;
|
||||||
import { test, runIfMain } from "../testing/mod.ts";
|
import { test, runIfMain } from "../testing/mod.ts";
|
||||||
import { assertEquals } from "../testing/asserts.ts";
|
import { assert, assertEquals } from "../testing/asserts.ts";
|
||||||
import { Response, ServerRequest, writeResponse } from "./server.ts";
|
import {
|
||||||
|
Response,
|
||||||
|
ServerRequest,
|
||||||
|
writeResponse,
|
||||||
|
readRequest
|
||||||
|
} from "./server.ts";
|
||||||
import { BufReader, BufWriter } from "../io/bufio.ts";
|
import { BufReader, BufWriter } from "../io/bufio.ts";
|
||||||
import { StringReader } from "../io/readers.ts";
|
import { StringReader } from "../io/readers.ts";
|
||||||
|
|
||||||
|
@ -283,4 +288,83 @@ test(async function writeStringReaderResponse(): Promise<void> {
|
||||||
assertEquals(decoder.decode(line), "0");
|
assertEquals(decoder.decode(line), "0");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test(async function readRequestError(): Promise<void> {
|
||||||
|
let input = `GET / HTTP/1.1
|
||||||
|
malformedHeader
|
||||||
|
`;
|
||||||
|
const reader = new BufReader(new StringReader(input));
|
||||||
|
let err;
|
||||||
|
try {
|
||||||
|
await readRequest(reader);
|
||||||
|
} catch (e) {
|
||||||
|
err = e;
|
||||||
|
}
|
||||||
|
assert(err instanceof Error);
|
||||||
|
assertEquals(err.message, "malformed MIME header line: malformedHeader");
|
||||||
|
});
|
||||||
|
|
||||||
|
// Ported from Go
|
||||||
|
// https://github.com/golang/go/blob/go1.12.5/src/net/http/request_test.go#L377-L443
|
||||||
|
// TODO(zekth) fix tests
|
||||||
|
test(async function testReadRequestError(): Promise<void> {
|
||||||
|
const testCases = {
|
||||||
|
0: {
|
||||||
|
in: "GET / HTTP/1.1\r\nheader: foo\r\n\r\n",
|
||||||
|
headers: [{ key: "header", value: "foo" }],
|
||||||
|
err: null
|
||||||
|
},
|
||||||
|
1: { in: "GET / HTTP/1.1\r\nheader:foo\r\n", err: "EOF", headers: [] },
|
||||||
|
2: { in: "", err: "EOF", headers: [] },
|
||||||
|
// 3: {
|
||||||
|
// in: "HEAD / HTTP/1.1\r\nContent-Length:4\r\n\r\n",
|
||||||
|
// err: "http: method cannot contain a Content-Length"
|
||||||
|
// },
|
||||||
|
4: {
|
||||||
|
in: "HEAD / HTTP/1.1\r\n\r\n",
|
||||||
|
headers: [],
|
||||||
|
err: null
|
||||||
|
}
|
||||||
|
// Multiple Content-Length values should either be
|
||||||
|
// deduplicated if same or reject otherwise
|
||||||
|
// See Issue 16490.
|
||||||
|
// 5: {
|
||||||
|
// in:
|
||||||
|
// "POST / HTTP/1.1\r\nContent-Length: 10\r\nContent-Length: 0\r\n\r\nGopher hey\r\n",
|
||||||
|
// err: "cannot contain multiple Content-Length headers"
|
||||||
|
// },
|
||||||
|
// 6: {
|
||||||
|
// in:
|
||||||
|
// "POST / HTTP/1.1\r\nContent-Length: 10\r\nContent-Length: 6\r\n\r\nGopher\r\n",
|
||||||
|
// err: "cannot contain multiple Content-Length headers"
|
||||||
|
// },
|
||||||
|
// 7: {
|
||||||
|
// in:
|
||||||
|
// "PUT / HTTP/1.1\r\nContent-Length: 6 \r\nContent-Length: 6\r\nContent-Length:6\r\n\r\nGopher\r\n",
|
||||||
|
// err: null,
|
||||||
|
// headers: [{ key: "Content-Length", value: "6" }]
|
||||||
|
// },
|
||||||
|
// 8: {
|
||||||
|
// in: "PUT / HTTP/1.1\r\nContent-Length: 1\r\nContent-Length: 6 \r\n\r\n",
|
||||||
|
// err: "cannot contain multiple Content-Length headers"
|
||||||
|
// },
|
||||||
|
// 9: {
|
||||||
|
// in: "POST / HTTP/1.1\r\nContent-Length:\r\nContent-Length: 3\r\n\r\n",
|
||||||
|
// err: "cannot contain multiple Content-Length headers"
|
||||||
|
// },
|
||||||
|
// 10: {
|
||||||
|
// in: "HEAD / HTTP/1.1\r\nContent-Length:0\r\nContent-Length: 0\r\n\r\n",
|
||||||
|
// headers: [{ key: "Content-Length", value: "0" }],
|
||||||
|
// err: null
|
||||||
|
// }
|
||||||
|
};
|
||||||
|
for (const p in testCases) {
|
||||||
|
const test = testCases[p];
|
||||||
|
const reader = new BufReader(new StringReader(test.in));
|
||||||
|
const [req, err] = await readRequest(reader);
|
||||||
|
assertEquals(test.err, err);
|
||||||
|
for (const h of test.headers) {
|
||||||
|
assertEquals(req.headers.get(h.key), h.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
runIfMain(import.meta);
|
runIfMain(import.meta);
|
||||||
|
|
Loading…
Reference in a new issue