1
0
Fork 0
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:
Vincent LE GOFF 2019-05-23 01:28:03 +02:00 committed by Bert Belder
parent e2debab359
commit 3cfc1244d8
2 changed files with 97 additions and 5 deletions

View file

@ -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.

View file

@ -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);