1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-15 18:38:53 -05:00

Revert "Redesign of http server module (#188)"

We need to consider the API changes here more carefully.

This reverts commit da188a7d30.
and commit 8569f15207.
This commit is contained in:
Ryan Dahl 2019-02-19 12:38:19 -05:00
parent 37a6bca8d0
commit 57c9176b19
11 changed files with 384 additions and 718 deletions

View file

@ -5,22 +5,13 @@ A framework for creating HTTP/HTTPS server.
## Example ## Example
```typescript ```typescript
import { createServer } from "https://deno.land/x/http/server.ts"; import { serve } from "https://deno.land/x/http/server.ts";
import { encode } from "https://deno.land/x/strings/strings.ts"; const s = serve("0.0.0.0:8000");
async function main() { async function main() {
const server = createServer(); for await (const req of s) {
server.handle("/", async (req, res) => { req.respond({ body: new TextEncoder().encode("Hello World\n") });
await res.respond({ }
status: 200,
body: encode("ok")
});
});
server.handle(new RegExp("/foo/(?<id>.+)"), async (req, res) => {
const { id } = req.match.groups;
await res.respondJson({ id });
});
server.listen("127.0.0.1:8080");
} }
main(); main();

View file

@ -10,7 +10,7 @@ import {
listenAndServe, listenAndServe,
ServerRequest, ServerRequest,
setContentLength, setContentLength,
ServerResponse Response
} from "./server.ts"; } from "./server.ts";
import { cwd, DenoError, ErrorKind, args, stat, readDir, open } from "deno"; import { cwd, DenoError, ErrorKind, args, stat, readDir, open } from "deno";
import { extname } from "../fs/path.ts"; import { extname } from "../fs/path.ts";
@ -195,14 +195,14 @@ async function serveFallback(req: ServerRequest, e: Error) {
} }
} }
function serverLog(req: ServerRequest, res: ServerResponse) { function serverLog(req: ServerRequest, res: Response) {
const d = new Date().toISOString(); const d = new Date().toISOString();
const dateFmt = `[${d.slice(0, 10)} ${d.slice(11, 19)}]`; const dateFmt = `[${d.slice(0, 10)} ${d.slice(11, 19)}]`;
const s = `${dateFmt} "${req.method} ${req.url} ${req.proto}" ${res.status}`; const s = `${dateFmt} "${req.method} ${req.url} ${req.proto}" ${res.status}`;
console.log(s); console.log(s);
} }
function setCORS(res: ServerResponse) { function setCORS(res: Response) {
if (!res.headers) { if (!res.headers) {
res.headers = new Headers(); res.headers = new Headers();
} }
@ -213,11 +213,11 @@ function setCORS(res: ServerResponse) {
); );
} }
listenAndServe(addr, async (req, res) => { listenAndServe(addr, async req => {
const fileName = req.url.replace(/\/$/, ""); const fileName = req.url.replace(/\/$/, "");
const filePath = currentDir + fileName; const filePath = currentDir + fileName;
let response: ServerResponse; let response: Response;
try { try {
const fileInfo = await stat(filePath); const fileInfo = await stat(filePath);
@ -235,7 +235,7 @@ listenAndServe(addr, async (req, res) => {
setCORS(response); setCORS(response);
} }
serverLog(req, response); serverLog(req, response);
res.respond(response); req.respond(response);
} }
}); });

View file

@ -1,6 +1,6 @@
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license. // Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
import * as deno from "deno"; import * as deno from "deno";
import { serve } from "./server.ts"; import { serve } from "./mod.ts";
const addr = deno.args[1] || "127.0.0.1:4500"; const addr = deno.args[1] || "127.0.0.1:4500";
const server = serve(addr); const server = serve(addr);
@ -8,13 +8,8 @@ const server = serve(addr);
const body = new TextEncoder().encode("Hello World"); const body = new TextEncoder().encode("Hello World");
async function main(): Promise<void> { async function main(): Promise<void> {
try {
for await (const request of server) { for await (const request of server) {
await request.responder.respond({ status: 200, body }); await request.respond({ status: 200, body });
}
} catch (e) {
console.log(e.stack);
console.error(e);
} }
} }

View file

@ -1,78 +0,0 @@
import { Reader, ReadResult } from "deno";
import { BufReader } from "../io/bufio.ts";
import { TextProtoReader } from "../textproto/mod.ts";
import { assert } from "../testing/mod.ts";
export class BodyReader implements Reader {
total: number;
bufReader: BufReader;
constructor(reader: Reader, private contentLength: number) {
this.total = 0;
this.bufReader = new BufReader(reader);
}
async read(p: Uint8Array): Promise<ReadResult> {
if (p.length > this.contentLength - this.total) {
const buf = new Uint8Array(this.contentLength - this.total);
const [nread, err] = await this.bufReader.readFull(buf);
if (err && err !== "EOF") {
throw err;
}
p.set(buf);
this.total += nread;
assert.assert(
this.total === this.contentLength,
`${this.total}, ${this.contentLength}`
);
return { nread, eof: true };
} else {
const { nread } = await this.bufReader.read(p);
this.total += nread;
return { nread, eof: false };
}
}
}
export class ChunkedBodyReader implements Reader {
bufReader = new BufReader(this.reader);
tpReader = new TextProtoReader(this.bufReader);
constructor(private reader: Reader) {}
chunks: Uint8Array[] = [];
crlfBuf = new Uint8Array(2);
finished: boolean = false;
async read(p: Uint8Array): Promise<ReadResult> {
const [line, sizeErr] = await this.tpReader.readLine();
if (sizeErr) {
throw sizeErr;
}
const len = parseInt(line, 16);
if (len === 0) {
this.finished = true;
await this.bufReader.readFull(this.crlfBuf);
return { nread: 0, eof: true };
} else {
const buf = new Uint8Array(len);
await this.bufReader.readFull(buf);
await this.bufReader.readFull(this.crlfBuf);
this.chunks.push(buf);
}
const buf = this.chunks[0];
if (buf) {
if (buf.byteLength <= p.byteLength) {
p.set(buf);
this.chunks.shift();
return { nread: buf.byteLength, eof: false };
} else {
p.set(buf.slice(0, p.byteLength));
this.chunks[0] = buf.slice(p.byteLength, buf.byteLength);
return { nread: p.byteLength, eof: false };
}
} else {
return { nread: 0, eof: true };
}
}
}

View file

@ -1,12 +0,0 @@
import { assert, runTests, test } from "../testing/mod.ts";
import { ChunkedBodyReader } from "./readers.ts";
import { StringReader } from "../io/readers.ts";
import { Buffer, copy } from "deno";
test(async function httpChunkedBodyReader() {
const chunked = "3\r\nabc\r\n5\r\ndefgh\r\n0\r\n\r\n";
const r = new ChunkedBodyReader(new StringReader(chunked));
const w = new Buffer();
await copy(w, r);
assert.equal(w.toString(), "abcdefgh");
});

View file

@ -1,90 +1,63 @@
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license. // Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
import { listen, Conn, toAsyncIterator, Reader, Writer, copy } from "deno";
import { Conn, copy, listen, Reader, toAsyncIterator, Writer } from "deno"; import { BufReader, BufState, BufWriter } from "../io/bufio.ts";
import { BufReader, BufWriter } from "../io/bufio.ts";
import { TextProtoReader } from "../textproto/mod.ts"; import { TextProtoReader } from "../textproto/mod.ts";
import { STATUS_TEXT } from "./http_status.ts"; import { STATUS_TEXT } from "./http_status.ts";
import { assert } from "../testing/mod.ts"; import { assert } from "../testing/mod.ts";
import { defer, Deferred } from "../util/deferred.ts";
import { BodyReader, ChunkedBodyReader } from "./readers.ts";
import { encode } from "../strings/strings.ts";
/** basic handler for http request */ interface Deferred {
export type HttpHandler = (req: ServerRequest, res: ServerResponder) => unknown; promise: Promise<{}>;
resolve: () => void;
export type ServerRequest = { reject: () => void;
/** request path with queries. always begin with / */
url: string;
/** HTTP method */
method: string;
/** requested protocol. like HTTP/1.1 */
proto: string;
/** HTTP Headers */
headers: Headers;
/** matched result for path pattern */
match: RegExpMatchArray;
/** body stream. body with "transfer-encoding: chunked" will automatically be combined into original data */
body: Reader;
};
/** basic responder for http response */
export interface ServerResponder {
respond(response: ServerResponse): Promise<void>;
respondJson(obj: any, headers?: Headers): Promise<void>;
respondText(text: string, headers?: Headers): Promise<void>;
readonly isResponded: boolean;
} }
export interface ServerResponse { function deferred(): Deferred {
/** let resolve, reject;
* HTTP status code const promise = new Promise((res, rej) => {
* @default 200 */ resolve = res;
status?: number; reject = rej;
headers?: Headers; });
body?: Uint8Array | Reader; return {
promise,
resolve,
reject
};
} }
interface ServeEnv { interface ServeEnv {
reqQueue: { req: ServerRequest; conn: Conn }[]; reqQueue: ServerRequest[];
serveDeferred: Deferred; serveDeferred: Deferred;
} }
/** Continuously read more requests from conn until EOF /** Continuously read more requests from conn until EOF
* Calls maybeHandleReq. * Calls maybeHandleReq.
* bufr is empty on a fresh TCP connection.
* Would be passed around and reused for later request on same conn
* TODO: make them async function after this change is done * TODO: make them async function after this change is done
* https://github.com/tc39/ecma262/pull/1250 * https://github.com/tc39/ecma262/pull/1250
* See https://v8.dev/blog/fast-async * See https://v8.dev/blog/fast-async
*/ */
function serveConn(env: ServeEnv, conn: Conn) { function serveConn(env: ServeEnv, conn: Conn, bufr?: BufReader) {
readRequest(conn) readRequest(conn, bufr).then(maybeHandleReq.bind(null, env, conn));
.then(maybeHandleReq.bind(null, env, conn))
.catch(e => {
conn.close();
});
} }
function maybeHandleReq(env: ServeEnv, conn: Conn, req: ServerRequest) { function maybeHandleReq(env: ServeEnv, conn: Conn, maybeReq: any) {
env.reqQueue.push({ conn, req }); // push req to queue const [req, _err] = maybeReq;
if (_err) {
conn.close(); // assume EOF for now...
return;
}
env.reqQueue.push(req); // push req to queue
env.serveDeferred.resolve(); // signal while loop to process it env.serveDeferred.resolve(); // signal while loop to process it
} }
/** export async function* serve(addr: string) {
* iterate new http request asynchronously
* @param addr listening address. like 127.0.0.1:80
* @param cancel deferred object for cancellation of serving
* */
export async function* serve(
addr: string,
cancel: Deferred = defer()
): AsyncIterableIterator<{ req: ServerRequest; res: ServerResponder }> {
const listener = listen("tcp", addr); const listener = listen("tcp", addr);
const env: ServeEnv = { const env: ServeEnv = {
reqQueue: [], // in case multiple promises are ready reqQueue: [], // in case multiple promises are ready
serveDeferred: defer() serveDeferred: deferred()
}; };
// Routine that keeps calling accept // Routine that keeps calling accept
const acceptRoutine = () => { const acceptRoutine = () => {
const handleConn = (conn: Conn) => { const handleConn = (conn: Conn) => {
@ -92,168 +65,47 @@ export async function* serve(
scheduleAccept(); // schedule next accept scheduleAccept(); // schedule next accept
}; };
const scheduleAccept = () => { const scheduleAccept = () => {
Promise.race([cancel.promise, listener.accept().then(handleConn)]); listener.accept().then(handleConn);
}; };
scheduleAccept(); scheduleAccept();
}; };
acceptRoutine(); acceptRoutine();
// Loop hack to allow yield (yield won't work in callbacks)
while (true) { while (true) {
// do race between accept, serveDeferred and cancel await env.serveDeferred.promise;
await Promise.race([env.serveDeferred.promise, cancel.promise]); env.serveDeferred = deferred(); // use a new deferred
// cancellation deferred resolved let queueToProcess = env.reqQueue;
if (cancel.handled) {
break;
}
// next serve deferred
env.serveDeferred = defer();
const queueToProcess = env.reqQueue;
env.reqQueue = []; env.reqQueue = [];
for (const { req, conn } of queueToProcess) { for (const result of queueToProcess) {
if (req) { yield result;
const res = createResponder(conn); // Continue read more from conn when user is done with the current req
yield { req, res }; // Moving this here makes it easier to manage
} serveConn(env, result.conn, result.r);
serveConn(env, conn);
} }
} }
listener.close(); listener.close();
} }
export async function listenAndServe(addr: string, handler: HttpHandler) { export async function listenAndServe(
addr: string,
handler: (req: ServerRequest) => void
) {
const server = serve(addr); const server = serve(addr);
for await (const { req, res } of server) { for await (const request of server) {
await handler(req, res); await handler(request);
} }
} }
export interface HttpServer { export interface Response {
handle(pattern: string | RegExp, handler: HttpHandler); status?: number;
headers?: Headers;
listen(addr: string, cancel?: Deferred): Promise<void>; body?: Uint8Array | Reader;
} }
/** create HttpServer object */ export function setContentLength(r: Response): void {
export function createServer(): HttpServer {
return new HttpServerImpl();
}
/** create ServerResponder object */
export function createResponder(w: Writer): ServerResponder {
return new ServerResponderImpl(w);
}
class HttpServerImpl implements HttpServer {
private handlers: { pattern: string | RegExp; handler: HttpHandler }[] = [];
handle(pattern: string | RegExp, handler: HttpHandler) {
this.handlers.push({ pattern, handler });
}
async listen(addr: string, cancel: Deferred = defer()) {
for await (const { req, res } of serve(addr, cancel)) {
let { pathname } = new URL(req.url, addr);
const { index, match } = findLongestAndNearestMatch(
pathname,
this.handlers.map(v => v.pattern)
);
req.match = match;
if (index > -1) {
const { handler } = this.handlers[index];
await handler(req, res);
if (!res.isResponded) {
await res.respond({
status: 500,
body: encode("Not Responded")
});
}
} else {
await res.respond({
status: 404,
body: encode("Not Found")
});
}
}
}
}
/**
* Find the match that appeared in the nearest position to the beginning of word.
* If positions are same, the longest one will be picked.
* Return -1 and null if no match found.
* */
export function findLongestAndNearestMatch(
pathname: string,
patterns: (string | RegExp)[]
): { index: number; match: RegExpMatchArray } {
let lastMatchIndex = pathname.length;
let lastMatchLength = 0;
let match: RegExpMatchArray = null;
let index = -1;
for (let i = 0; i < patterns.length; i++) {
const pattern = patterns[i];
const m = pathname.match(pattern);
if (!m) continue;
if (
m.index < lastMatchIndex ||
(m.index === lastMatchIndex && m[0].length > lastMatchLength)
) {
index = i;
match = m;
lastMatchIndex = m.index;
lastMatchLength = m[0].length;
}
}
return { index, match };
}
class ServerResponderImpl implements ServerResponder {
constructor(private w: Writer) {}
private _responded: boolean = false;
get isResponded() {
return this._responded;
}
private checkIfResponded() {
if (this.isResponded) {
throw new Error("http: already responded");
}
}
respond(response: ServerResponse): Promise<void> {
this.checkIfResponded();
this._responded = true;
return writeResponse(this.w, response);
}
respondJson(obj: any, headers: Headers = new Headers()): Promise<void> {
const body = encode(JSON.stringify(obj));
if (!headers.has("content-type")) {
headers.set("content-type", "application/json");
}
return this.respond({
status: 200,
body,
headers
});
}
respondText(text: string, headers: Headers = new Headers()): Promise<void> {
const body = encode(text);
if (!headers.has("content-type")) {
headers.set("content-type", "text/plain");
}
return this.respond({
status: 200,
headers,
body
});
}
}
export function setContentLength(r: ServerResponse): void {
if (!r.headers) { if (!r.headers) {
r.headers = new Headers(); r.headers = new Headers();
} }
@ -270,6 +122,100 @@ export function setContentLength(r: ServerResponse): void {
} }
} }
export class ServerRequest {
url: string;
method: string;
proto: string;
headers: Headers;
conn: Conn;
r: BufReader;
w: BufWriter;
public async *bodyStream() {
if (this.headers.has("content-length")) {
const len = +this.headers.get("content-length");
if (Number.isNaN(len)) {
return new Uint8Array(0);
}
let buf = new Uint8Array(1024);
let rr = await this.r.read(buf);
let nread = rr.nread;
while (!rr.eof && nread < len) {
yield buf.subarray(0, rr.nread);
buf = new Uint8Array(1024);
rr = await this.r.read(buf);
nread += rr.nread;
}
yield buf.subarray(0, rr.nread);
} else {
if (this.headers.has("transfer-encoding")) {
const transferEncodings = this.headers
.get("transfer-encoding")
.split(",")
.map(e => e.trim().toLowerCase());
if (transferEncodings.includes("chunked")) {
// Based on https://tools.ietf.org/html/rfc2616#section-19.4.6
const tp = new TextProtoReader(this.r);
let [line, _] = await tp.readLine();
// TODO: handle chunk extension
let [chunkSizeString, optExt] = line.split(";");
let chunkSize = parseInt(chunkSizeString, 16);
if (Number.isNaN(chunkSize) || chunkSize < 0) {
throw new Error("Invalid chunk size");
}
while (chunkSize > 0) {
let data = new Uint8Array(chunkSize);
let [nread, err] = await this.r.readFull(data);
if (nread !== chunkSize) {
throw new Error("Chunk data does not match size");
}
yield data;
await this.r.readLine(); // Consume \r\n
[line, _] = await tp.readLine();
chunkSize = parseInt(line, 16);
}
const [entityHeaders, err] = await tp.readMIMEHeader();
if (!err) {
for (let [k, v] of entityHeaders) {
this.headers.set(k, v);
}
}
/* Pseudo code from https://tools.ietf.org/html/rfc2616#section-19.4.6
length := 0
read chunk-size, chunk-extension (if any) and CRLF
while (chunk-size > 0) {
read chunk-data and CRLF
append chunk-data to entity-body
length := length + chunk-size
read chunk-size and CRLF
}
read entity-header
while (entity-header not empty) {
append entity-header to existing header fields
read entity-header
}
Content-Length := length
Remove "chunked" from Transfer-Encoding
*/
return; // Must return here to avoid fall through
}
// TODO: handle other transfer-encoding types
}
// Otherwise...
yield new Uint8Array(0);
}
}
// Read the body of the request into a single Uint8Array
public async body(): Promise<Uint8Array> {
return readAllIterator(this.bodyStream());
}
async respond(r: Response): Promise<void> {
return writeResponse(this.w, r);
}
}
function bufWriter(w: Writer): BufWriter { function bufWriter(w: Writer): BufWriter {
if (w instanceof BufWriter) { if (w instanceof BufWriter) {
return w; return w;
@ -278,10 +224,7 @@ function bufWriter(w: Writer): BufWriter {
} }
} }
export async function writeResponse( export async function writeResponse(w: Writer, r: Response): Promise<void> {
w: Writer,
r: ServerResponse
): Promise<void> {
const protoMajor = 1; const protoMajor = 1;
const protoMinor = 1; const protoMinor = 1;
const statusCode = r.status || 200; const statusCode = r.status || 200;
@ -339,52 +282,53 @@ async function writeChunkedBody(w: Writer, r: Reader) {
await writer.write(endChunk); await writer.write(endChunk);
} }
export async function readRequest(conn: Reader): Promise<ServerRequest> { async function readRequest(
const bufr = new BufReader(conn); c: Conn,
bufr?: BufReader
): Promise<[ServerRequest, BufState]> {
if (!bufr) {
bufr = new BufReader(c);
}
const bufw = new BufWriter(c);
const req = new ServerRequest();
req.conn = c;
req.r = bufr!;
req.w = bufw;
const tp = new TextProtoReader(bufr!); const tp = new TextProtoReader(bufr!);
let s: string;
let err: BufState;
// First line: GET /index.html HTTP/1.0 // First line: GET /index.html HTTP/1.0
const [line, lineErr] = await tp.readLine(); [s, err] = await tp.readLine();
if (lineErr) { if (err) {
throw lineErr; return [null, err];
} }
const [method, url, proto] = line.split(" ", 3); [req.method, req.url, req.proto] = s.split(" ", 3);
const [headers, headersErr] = await tp.readMIMEHeader();
if (headersErr) { [req.headers, err] = await tp.readMIMEHeader();
throw headersErr;
} return [req, err];
const contentLength = headers.get("content-length");
const body =
headers.get("transfer-encoding") === "chunked"
? new ChunkedBodyReader(bufr)
: new BodyReader(bufr, parseInt(contentLength));
return {
method,
url,
proto,
headers,
body,
match: null
};
} }
export async function readResponse(conn: Reader): Promise<ServerResponse> { async function readAllIterator(
const bufr = new BufReader(conn); it: AsyncIterableIterator<Uint8Array>
const tp = new TextProtoReader(bufr!); ): Promise<Uint8Array> {
// First line: HTTP/1,1 200 OK const chunks = [];
const [line, lineErr] = await tp.readLine(); let len = 0;
if (lineErr) { for await (const chunk of it) {
throw lineErr; chunks.push(chunk);
len += chunk.length;
} }
const [proto, status, statusText] = line.split(" ", 3); if (chunks.length === 0) {
const [headers, headersErr] = await tp.readMIMEHeader(); // No need for copy
if (headersErr) { return chunks[0];
throw headersErr;
} }
const contentLength = headers.get("content-length"); const collected = new Uint8Array(len);
const body = let offset = 0;
headers.get("transfer-encoding") === "chunked" for (let chunk of chunks) {
? new ChunkedBodyReader(bufr) collected.set(chunk, offset);
: new BodyReader(bufr, parseInt(contentLength)); offset += chunk.length;
return { status: parseInt(status), headers, body }; }
return collected;
} }

View file

@ -5,27 +5,19 @@
// Ported from // Ported from
// https://github.com/golang/go/blob/master/src/net/http/responsewrite_test.go // https://github.com/golang/go/blob/master/src/net/http/responsewrite_test.go
import { Buffer, copy, Reader } from "deno"; import { Buffer } from "deno";
import { assert, assertEqual, runTests, test } from "../testing/mod.ts"; import { assertEqual, test } from "../testing/mod.ts";
import { import { Response, ServerRequest } from "./server.ts";
createResponder, import { BufReader, BufWriter } from "../io/bufio.ts";
createServer,
findLongestAndNearestMatch,
readRequest,
readResponse,
ServerResponse,
writeResponse
} from "./server.ts";
import { encode } from "../strings/strings.ts";
import { StringReader } from "../io/readers.ts";
import { StringWriter } from "../io/writers.ts";
import { defer } from "../util/deferred.ts";
interface ResponseTest { interface ResponseTest {
response: ServerResponse; response: Response;
raw: string; raw: string;
} }
const enc = new TextEncoder();
const dec = new TextDecoder();
const responseTests: ResponseTest[] = [ const responseTests: ResponseTest[] = [
// Default response // Default response
{ {
@ -36,7 +28,7 @@ const responseTests: ResponseTest[] = [
{ {
response: { response: {
status: 200, status: 200,
body: new Buffer(encode("abcdef")) body: new Buffer(new TextEncoder().encode("abcdef"))
}, },
raw: raw:
@ -46,284 +38,181 @@ const responseTests: ResponseTest[] = [
} }
]; ];
test(async function httpWriteResponse() { test(async function responseWrite() {
for (const { raw, response } of responseTests) { for (const testCase of responseTests) {
const buf = new Buffer(); const buf = new Buffer();
await writeResponse(buf, response); const bufw = new BufWriter(buf);
assertEqual(buf.toString(), raw); const request = new ServerRequest();
request.w = bufw;
await request.respond(testCase.response);
assertEqual(buf.toString(), testCase.raw);
} }
}); });
test(async function httpReadRequest() { test(async function requestBodyWithContentLength() {
const body = "0123456789";
const lines = [
"GET /index.html?deno=land HTTP/1.1",
"Host: deno.land",
"Content-Type: text/plain",
`Content-Length: ${body.length}`,
"\r\n"
];
let msg = lines.join("\r\n");
msg += body;
const req = await readRequest(new StringReader(`${msg}`));
assert.equal(req.url, "/index.html?deno=land");
assert.equal(req.method, "GET");
assert.equal(req.proto, "HTTP/1.1");
assert.equal(req.headers.get("host"), "deno.land");
assert.equal(req.headers.get("content-type"), "text/plain");
assert.equal(req.headers.get("content-length"), `${body.length}`);
const w = new StringWriter();
await copy(w, req.body);
assert.equal(w.toString(), body);
});
test(async function httpReadRequestChunkedBody() {
const lines = [
"GET /index.html?deno=land HTTP/1.1",
"Host: deno.land",
"Content-Type: text/plain",
`Transfer-Encoding: chunked`,
"\r\n"
];
const hd = lines.join("\r\n");
const buf = new Buffer();
await buf.write(encode(hd));
await buf.write(encode("4\r\ndeno\r\n"));
await buf.write(encode("5\r\n.land\r\n"));
await buf.write(encode("0\r\n\r\n"));
const req = await readRequest(buf);
assert.equal(req.url, "/index.html?deno=land");
assert.equal(req.method, "GET");
assert.equal(req.proto, "HTTP/1.1");
assert.equal(req.headers.get("host"), "deno.land");
assert.equal(req.headers.get("content-type"), "text/plain");
assert.equal(req.headers.get("transfer-encoding"), `chunked`);
const dest = new Buffer();
await copy(dest, req.body);
assert.equal(dest.toString(), "deno.land");
});
test(function httpMatchNearest() {
assert.equal(
findLongestAndNearestMatch("/foo", ["/foo", "/bar", "/f"]).index,
0
);
assert.equal(
findLongestAndNearestMatch("/foo", ["/foo", "/foo/bar"]).index,
0
);
assert.equal(
findLongestAndNearestMatch("/foo/bar", [
"/",
"/foo",
"/hoo",
"/hoo/foo/bar",
"/foo/bar"
]).index,
4
);
assert.equal(
findLongestAndNearestMatch("/foo/bar/foo", ["/foo", "/foo/bar", "/bar/foo"])
.index,
1
);
assert.equal(
findLongestAndNearestMatch("/foo", ["/", "/hoo", "/hoo/foo"]).index,
0
);
assert.equal(
findLongestAndNearestMatch("/deno/land", [/d(.+?)o/, /d(.+?)d/]).index,
1
);
assert.equal(findLongestAndNearestMatch("/foo", ["/", "/a/foo"]).index, 0);
assert.equal(
findLongestAndNearestMatch("/foo", [/\/foo/, /\/bar\/foo/]).index,
0
);
assert.equal(
findLongestAndNearestMatch("/foo", [/\/a\/foo/, /\/foo/]).index,
1
);
});
test(async function httpServer() {
const server = createServer();
server.handle("/index", async (req, res) => {
await res.respond({
status: 200,
body: encode("ok")
});
});
server.handle(new RegExp("/foo/(?<id>.+)"), async (req, res) => {
const { id } = req.match.groups;
await res.respond({
status: 200,
headers: new Headers({
"content-type": "application/json"
}),
body: encode(JSON.stringify({ id }))
});
});
server.handle("/no-response", async (req, res) => {});
const cancel = defer<void>();
try {
server.listen("127.0.0.1:8080", cancel);
{ {
const res1 = await fetch("http://127.0.0.1:8080/index"); const req = new ServerRequest();
const text = await res1.body.text(); req.headers = new Headers();
assert.equal(res1.status, 200); req.headers.set("content-length", "5");
assert.equal(text, "ok"); const buf = new Buffer(enc.encode("Hello"));
req.r = new BufReader(buf);
const body = dec.decode(await req.body());
assertEqual(body, "Hello");
} }
// Larger than internal buf
{ {
const res2 = await fetch("http://127.0.0.1:8080/foo/123"); const longText = "1234\n".repeat(1000);
const json = await res2.body.json(); const req = new ServerRequest();
assert.equal(res2.status, 200); req.headers = new Headers();
assert.equal(res2.headers.get("content-type"), "application/json"); req.headers.set("Content-Length", "5000");
assert.equal(json["id"], "123"); const buf = new Buffer(enc.encode(longText));
} req.r = new BufReader(buf);
{ const body = dec.decode(await req.body());
const res = await fetch("http://127.0.0.1:8080/no-response"); assertEqual(body, longText);
assert.equal(res.status, 500);
const text = await res.body.text();
assert.assert(!!text.match("Not Responded"));
}
{
const res = await fetch("http://127.0.0.1:8080/not-found");
const text = await res.body.text();
assert.equal(res.status, 404);
assert.assert(!!text.match("Not Found"));
}
} finally {
cancel.resolve();
} }
}); });
test(async function httpServerResponder() { test(async function requestBodyWithTransferEncoding() {
const w = new Buffer(); {
const res = createResponder(w); const shortText = "Hello";
assert.assert(!res.isResponded); const req = new ServerRequest();
await res.respond({ req.headers = new Headers();
status: 200, req.headers.set("transfer-encoding", "chunked");
headers: new Headers({ let chunksData = "";
"content-type": "text/plain" let chunkOffset = 0;
}), const maxChunkSize = 70;
body: encode("ok") while (chunkOffset < shortText.length) {
}); const chunkSize = Math.min(maxChunkSize, shortText.length - chunkOffset);
assert.assert(res.isResponded); chunksData += `${chunkSize.toString(16)}\r\n${shortText.substr(
const resp = await readResponse(w); chunkOffset,
assert.equal(resp.status, 200); chunkSize
assert.equal(resp.headers.get("content-type"), "text/plain"); )}\r\n`;
const sw = new StringWriter(); chunkOffset += chunkSize;
await copy(sw, resp.body as Reader); }
assert.equal(sw.toString(), "ok"); chunksData += "0\r\n\r\n";
const buf = new Buffer(enc.encode(chunksData));
req.r = new BufReader(buf);
const body = dec.decode(await req.body());
assertEqual(body, shortText);
}
// Larger than internal buf
{
const longText = "1234\n".repeat(1000);
const req = new ServerRequest();
req.headers = new Headers();
req.headers.set("transfer-encoding", "chunked");
let chunksData = "";
let chunkOffset = 0;
const maxChunkSize = 70;
while (chunkOffset < longText.length) {
const chunkSize = Math.min(maxChunkSize, longText.length - chunkOffset);
chunksData += `${chunkSize.toString(16)}\r\n${longText.substr(
chunkOffset,
chunkSize
)}\r\n`;
chunkOffset += chunkSize;
}
chunksData += "0\r\n\r\n";
const buf = new Buffer(enc.encode(chunksData));
req.r = new BufReader(buf);
const body = dec.decode(await req.body());
assertEqual(body, longText);
}
}); });
test(async function httpServerResponderRespondJson() { test(async function requestBodyStreamWithContentLength() {
const w = new Buffer(); {
const res = createResponder(w); const shortText = "Hello";
const json = { const req = new ServerRequest();
id: 1, req.headers = new Headers();
deno: { req.headers.set("content-length", "" + shortText.length);
is: "land" const buf = new Buffer(enc.encode(shortText));
req.r = new BufReader(buf);
const it = await req.bodyStream();
let offset = 0;
for await (const chunk of it) {
const s = dec.decode(chunk);
assertEqual(shortText.substr(offset, s.length), s);
offset += s.length;
}
}
// Larger than internal buf
{
const longText = "1234\n".repeat(1000);
const req = new ServerRequest();
req.headers = new Headers();
req.headers.set("Content-Length", "5000");
const buf = new Buffer(enc.encode(longText));
req.r = new BufReader(buf);
const it = await req.bodyStream();
let offset = 0;
for await (const chunk of it) {
const s = dec.decode(chunk);
assertEqual(longText.substr(offset, s.length), s);
offset += s.length;
}
} }
};
assert.assert(!res.isResponded);
await res.respondJson(
json,
new Headers({
deno: "land"
})
);
assert.assert(res.isResponded);
const resp = await readResponse(w);
assert.equal(resp.status, 200);
assert.equal(resp.headers.get("content-type"), "application/json");
const sw = new StringWriter();
await copy(sw, resp.body as Reader);
const resJson = JSON.parse(sw.toString());
assert.equal(resJson, json);
assert.equal(resp.headers.get("deno"), "land");
}); });
test(async function httpServerResponderRespondText() { test(async function requestBodyStreamWithTransferEncoding() {
const w = new Buffer(); {
const res = createResponder(w); const shortText = "Hello";
assert.assert(!res.isResponded); const req = new ServerRequest();
await res.respondText( req.headers = new Headers();
"deno.land", req.headers.set("transfer-encoding", "chunked");
new Headers({ let chunksData = "";
deno: "land" let chunkOffset = 0;
}) const maxChunkSize = 70;
); while (chunkOffset < shortText.length) {
assert.assert(res.isResponded); const chunkSize = Math.min(maxChunkSize, shortText.length - chunkOffset);
const resp = await readResponse(w); chunksData += `${chunkSize.toString(16)}\r\n${shortText.substr(
assert.equal(resp.status, 200); chunkOffset,
assert.equal(resp.headers.get("content-type"), "text/plain"); chunkSize
const sw = new StringWriter(); )}\r\n`;
await copy(sw, resp.body as Reader); chunkOffset += chunkSize;
assert.equal(sw.toString(), "deno.land"); }
assert.equal(resp.headers.get("deno"), "land"); chunksData += "0\r\n\r\n";
}); const buf = new Buffer(enc.encode(chunksData));
req.r = new BufReader(buf);
const it = await req.bodyStream();
let offset = 0;
for await (const chunk of it) {
const s = dec.decode(chunk);
assertEqual(shortText.substr(offset, s.length), s);
offset += s.length;
}
}
test(async function httpServerResponderShouldThrow() { // Larger than internal buf
const w = new Buffer();
{ {
const res = createResponder(w); const longText = "1234\n".repeat(1000);
await res.respond({ const req = new ServerRequest();
body: null req.headers = new Headers();
}); req.headers.set("transfer-encoding", "chunked");
await assert.throwsAsync( let chunksData = "";
async () => res.respond({ body: null }), let chunkOffset = 0;
Error, const maxChunkSize = 70;
"responded" while (chunkOffset < longText.length) {
); const chunkSize = Math.min(maxChunkSize, longText.length - chunkOffset);
await assert.throwsAsync( chunksData += `${chunkSize.toString(16)}\r\n${longText.substr(
async () => res.respondJson({}), chunkOffset,
Error, chunkSize
"responded" )}\r\n`;
); chunkOffset += chunkSize;
await assert.throwsAsync(
async () => res.respondText(""),
Error,
"responded"
);
} }
{ chunksData += "0\r\n\r\n";
const res = createResponder(w); const buf = new Buffer(enc.encode(chunksData));
await res.respondText(""); req.r = new BufReader(buf);
await assert.throwsAsync( const it = await req.bodyStream();
async () => res.respond({ body: null }), let offset = 0;
Error, for await (const chunk of it) {
"responded" const s = dec.decode(chunk);
); assertEqual(longText.substr(offset, s.length), s);
await assert.throwsAsync( offset += s.length;
async () => res.respondJson({}),
Error,
"responded"
);
await assert.throwsAsync(
async () => res.respondText(""),
Error,
"responded"
);
} }
{
const res = createResponder(w);
await res.respondJson({});
await assert.throwsAsync(
async () => res.respond({ body: null }),
Error,
"responded"
);
await assert.throwsAsync(
async () => res.respondJson({}),
Error,
"responded"
);
await assert.throwsAsync(
async () => res.respondText(""),
Error,
"responded"
);
} }
}); });

View file

@ -7,11 +7,9 @@ import { decode } from "../strings/strings.ts";
test(async function ioStringReader() { test(async function ioStringReader() {
const r = new StringReader("abcdef"); const r = new StringReader("abcdef");
const buf = new Uint8Array(6); const { nread, eof } = await r.read(new Uint8Array(6));
const { nread, eof } = await r.read(buf);
assert.equal(nread, 6); assert.equal(nread, 6);
assert.equal(eof, true); assert.equal(eof, true);
assert.equal(decode(buf), "abcdef");
}); });
test(async function ioStringReader() { test(async function ioStringReader() {

View file

@ -1,8 +1,6 @@
#!/usr/bin/env deno -A #!/usr/bin/env deno -A
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license. // Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
import "benching/test.ts"; import "benching/test.ts";
import "util/deferred_test.ts";
import "colors/test.ts"; import "colors/test.ts";
import "datetime/test.ts"; import "datetime/test.ts";
import "examples/test.ts"; import "examples/test.ts";
@ -17,7 +15,6 @@ import "fs/walk_test.ts";
import "io/test.ts"; import "io/test.ts";
import "http/server_test.ts"; import "http/server_test.ts";
import "http/file_server_test.ts"; import "http/file_server_test.ts";
import "http/readers_test.ts";
import "log/test.ts"; import "log/test.ts";
import "media_types/test.ts"; import "media_types/test.ts";
import "multipart/formfile_test.ts"; import "multipart/formfile_test.ts";

View file

@ -1,42 +0,0 @@
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
export type Deferred<T = any, R = Error> = {
promise: Promise<T>;
resolve: (t?: T) => void;
reject: (r?: R) => void;
readonly handled: boolean;
};
/** Create deferred promise that can be resolved and rejected by outside */
export function defer<T>(): Deferred<T> {
let handled = false;
let resolve;
let reject;
const promise = new Promise<T>((res, rej) => {
resolve = r => {
handled = true;
res(r);
};
reject = r => {
handled = true;
rej(r);
};
});
return {
promise,
resolve,
reject,
get handled() {
return handled;
}
};
}
export function isDeferred(x): x is Deferred {
return (
typeof x === "object" &&
x.promise instanceof Promise &&
typeof x["resolve"] === "function" &&
typeof x["reject"] === "function"
);
}

View file

@ -1,16 +0,0 @@
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
import { assert, test } from "../testing/mod.ts";
import { defer, isDeferred } from "./deferred.ts";
test(async function asyncIsDeferred() {
const d = defer();
assert.assert(isDeferred(d));
assert.assert(
isDeferred({
promise: null,
resolve: () => {},
reject: () => {}
}) === false
);
});