Testing / .deno /gen /https /deno.land /8d6b5cd21d4064cfdc92ce7f09807d551dcbce46a7ecc6af2532ffe73b7aba43.js
Chunte's picture
Chunte HF staff
Upload 1904 files
5de1c56 verified
// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
import { BufReader, BufWriter } from "../io/bufio.ts";
import { assert } from "../_util/assert.ts";
import { deferred, MuxAsyncIterator } from "../async/mod.ts";
import { bodyReader, chunkedBodyReader, emptyReader, readRequest, writeResponse } from "./_io.ts";
export class ServerRequest {
url;
method;
proto;
protoMinor;
protoMajor;
headers;
conn;
r;
w;
#done = deferred();
#contentLength = undefined;
#body = undefined;
#finalized = false;
get done() {
return this.#done.then((e)=>e);
}
/**
* Value of Content-Length header.
* If null, then content length is invalid or not given (e.g. chunked encoding).
*/ get contentLength() {
// undefined means not cached.
// null means invalid or not provided.
if (this.#contentLength === undefined) {
const cl = this.headers.get("content-length");
if (cl) {
this.#contentLength = parseInt(cl);
// Convert NaN to null (as NaN harder to test)
if (Number.isNaN(this.#contentLength)) {
this.#contentLength = null;
}
} else {
this.#contentLength = null;
}
}
return this.#contentLength;
}
/**
* Body of the request. The easiest way to consume the body is:
*
* const buf: Uint8Array = await readAll(req.body);
*/ get body() {
if (!this.#body) {
if (this.contentLength != null) {
this.#body = bodyReader(this.contentLength, this.r);
} else {
const transferEncoding = this.headers.get("transfer-encoding");
if (transferEncoding != null) {
const parts = transferEncoding.split(",").map((e)=>e.trim().toLowerCase());
assert(parts.includes("chunked"), 'transfer-encoding must include "chunked" if content-length is not set');
this.#body = chunkedBodyReader(this.headers, this.r);
} else {
// Neither content-length nor transfer-encoding: chunked
this.#body = emptyReader();
}
}
}
return this.#body;
}
async respond(r) {
let err;
try {
// Write our response!
await writeResponse(this.w, r);
} catch (e) {
try {
// Eagerly close on error.
this.conn.close();
} catch {
// Pass
}
err = e;
}
// Signal that this request has been processed and the next pipelined
// request on the same connection can be accepted.
this.#done.resolve(err);
if (err) {
// Error during responding, rethrow.
throw err;
}
}
async finalize() {
if (this.#finalized) return;
// Consume unread body
const body = this.body;
const buf = new Uint8Array(1024);
while(await body.read(buf) !== null){
// Pass
}
this.#finalized = true;
}
}
export class Server {
listener;
#closing;
#connections;
constructor(listener){
this.listener = listener;
this.#closing = false;
this.#connections = [];
}
close() {
this.#closing = true;
this.listener.close();
for (const conn of this.#connections){
try {
conn.close();
} catch (e) {
// Connection might have been already closed
if (!(e instanceof Deno.errors.BadResource)) {
throw e;
}
}
}
}
// Yields all HTTP requests on a single TCP connection.
async *iterateHttpRequests(conn) {
const reader = new BufReader(conn);
const writer = new BufWriter(conn);
while(!this.#closing){
let request;
try {
request = await readRequest(conn, reader);
} catch (error) {
if (error instanceof Deno.errors.InvalidData || error instanceof Deno.errors.UnexpectedEof) {
// An error was thrown while parsing request headers.
// Try to send the "400 Bad Request" before closing the connection.
try {
await writeResponse(writer, {
status: 400,
body: new TextEncoder().encode(`${error.message}\r\n\r\n`)
});
} catch {
// The connection is broken.
}
}
break;
}
if (request === null) {
break;
}
request.w = writer;
yield request;
// Wait for the request to be processed before we accept a new request on
// this connection.
const responseError = await request.done;
if (responseError) {
// Something bad happened during response.
// (likely other side closed during pipelined req)
// req.done implies this connection already closed, so we can just return.
this.untrackConnection(request.conn);
return;
}
try {
// Consume unread body and trailers if receiver didn't consume those data
await request.finalize();
} catch {
break;
}
}
this.untrackConnection(conn);
try {
conn.close();
} catch {
// might have been already closed
}
}
trackConnection(conn) {
this.#connections.push(conn);
}
untrackConnection(conn) {
const index = this.#connections.indexOf(conn);
if (index !== -1) {
this.#connections.splice(index, 1);
}
}
// Accepts a new TCP connection and yields all HTTP requests that arrive on
// it. When a connection is accepted, it also creates a new iterator of the
// same kind and adds it to the request multiplexer so that another TCP
// connection can be accepted.
async *acceptConnAndIterateHttpRequests(mux) {
if (this.#closing) return;
// Wait for a new connection.
let conn;
try {
conn = await this.listener.accept();
} catch (error) {
if (// The listener is closed:
error instanceof Deno.errors.BadResource || // TLS handshake errors:
error instanceof Deno.errors.InvalidData || error instanceof Deno.errors.UnexpectedEof || error instanceof Deno.errors.ConnectionReset) {
return mux.add(this.acceptConnAndIterateHttpRequests(mux));
}
throw error;
}
this.trackConnection(conn);
// Try to accept another connection and add it to the multiplexer.
mux.add(this.acceptConnAndIterateHttpRequests(mux));
// Yield the requests that arrive on the just-accepted connection.
yield* this.iterateHttpRequests(conn);
}
[Symbol.asyncIterator]() {
const mux = new MuxAsyncIterator();
mux.add(this.acceptConnAndIterateHttpRequests(mux));
return mux.iterate();
}
}
/**
* Parse addr from string
*
* const addr = "::1:8000";
* parseAddrFromString(addr);
*
* @param addr Address string
*/ export function _parseAddrFromStr(addr) {
let url;
try {
const host = addr.startsWith(":") ? `0.0.0.0${addr}` : addr;
url = new URL(`http://${host}`);
} catch {
throw new TypeError("Invalid address.");
}
if (url.username || url.password || url.pathname != "/" || url.search || url.hash) {
throw new TypeError("Invalid address.");
}
return {
hostname: url.hostname,
port: url.port === "" ? 80 : Number(url.port)
};
}
/**
* Create a HTTP server
*
* import { serve } from "https://deno.land/std/http/server.ts";
* const body = "Hello World\n";
* const server = serve({ port: 8000 });
* for await (const req of server) {
* req.respond({ body });
* }
*/ export function serve(addr) {
if (typeof addr === "string") {
addr = _parseAddrFromStr(addr);
}
const listener = Deno.listen(addr);
return new Server(listener);
}
/**
* Start an HTTP server with given options and request handler
*
* const body = "Hello World\n";
* const options = { port: 8000 };
* listenAndServe(options, (req) => {
* req.respond({ body });
* });
*
* @param options Server configuration
* @param handler Request handler
*/ export async function listenAndServe(addr, handler) {
const server = serve(addr);
for await (const request of server){
handler(request);
}
}
/**
* Create an HTTPS server with given options
*
* const body = "Hello HTTPS";
* const options = {
* hostname: "localhost",
* port: 443,
* certFile: "./path/to/localhost.crt",
* keyFile: "./path/to/localhost.key",
* };
* for await (const req of serveTLS(options)) {
* req.respond({ body });
* }
*
* @param options Server configuration
* @return Async iterable server instance for incoming requests
*/ export function serveTLS(options) {
const tlsOptions = {
...options,
transport: "tcp"
};
const listener = Deno.listenTls(tlsOptions);
return new Server(listener);
}
/**
* Start an HTTPS server with given options and request handler
*
* const body = "Hello HTTPS";
* const options = {
* hostname: "localhost",
* port: 443,
* certFile: "./path/to/localhost.crt",
* keyFile: "./path/to/localhost.key",
* };
* listenAndServeTLS(options, (req) => {
* req.respond({ body });
* });
*
* @param options Server configuration
* @param handler Request handler
*/ export async function listenAndServeTLS(options, handler) {
const server = serveTLS(options);
for await (const request of server){
handler(request);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["https://deno.land/std@0.92.0/http/server.ts"],"sourcesContent":["// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.\nimport { BufReader, BufWriter } from \"../io/bufio.ts\";\nimport { assert } from \"../_util/assert.ts\";\nimport { Deferred, deferred, MuxAsyncIterator } from \"../async/mod.ts\";\nimport {\n  bodyReader,\n  chunkedBodyReader,\n  emptyReader,\n  readRequest,\n  writeResponse,\n} from \"./_io.ts\";\nexport class ServerRequest {\n  url!: string;\n  method!: string;\n  proto!: string;\n  protoMinor!: number;\n  protoMajor!: number;\n  headers!: Headers;\n  conn!: Deno.Conn;\n  r!: BufReader;\n  w!: BufWriter;\n\n  #done: Deferred<Error | undefined> = deferred();\n  #contentLength?: number | null = undefined;\n  #body?: Deno.Reader = undefined;\n  #finalized = false;\n\n  get done(): Promise<Error | undefined> {\n    return this.#done.then((e) => e);\n  }\n\n  /**\n   * Value of Content-Length header.\n   * If null, then content length is invalid or not given (e.g. chunked encoding).\n   */\n  get contentLength(): number | null {\n    // undefined means not cached.\n    // null means invalid or not provided.\n    if (this.#contentLength === undefined) {\n      const cl = this.headers.get(\"content-length\");\n      if (cl) {\n        this.#contentLength = parseInt(cl);\n        // Convert NaN to null (as NaN harder to test)\n        if (Number.isNaN(this.#contentLength)) {\n          this.#contentLength = null;\n        }\n      } else {\n        this.#contentLength = null;\n      }\n    }\n    return this.#contentLength;\n  }\n\n  /**\n   * Body of the request.  The easiest way to consume the body is:\n   *\n   *     const buf: Uint8Array = await readAll(req.body);\n   */\n  get body(): Deno.Reader {\n    if (!this.#body) {\n      if (this.contentLength != null) {\n        this.#body = bodyReader(this.contentLength, this.r);\n      } else {\n        const transferEncoding = this.headers.get(\"transfer-encoding\");\n        if (transferEncoding != null) {\n          const parts = transferEncoding\n            .split(\",\")\n            .map((e): string => e.trim().toLowerCase());\n          assert(\n            parts.includes(\"chunked\"),\n            'transfer-encoding must include \"chunked\" if content-length is not set',\n          );\n          this.#body = chunkedBodyReader(this.headers, this.r);\n        } else {\n          // Neither content-length nor transfer-encoding: chunked\n          this.#body = emptyReader();\n        }\n      }\n    }\n    return this.#body;\n  }\n\n  async respond(r: Response): Promise<void> {\n    let err: Error | undefined;\n    try {\n      // Write our response!\n      await writeResponse(this.w, r);\n    } catch (e) {\n      try {\n        // Eagerly close on error.\n        this.conn.close();\n      } catch {\n        // Pass\n      }\n      err = e;\n    }\n    // Signal that this request has been processed and the next pipelined\n    // request on the same connection can be accepted.\n    this.#done.resolve(err);\n    if (err) {\n      // Error during responding, rethrow.\n      throw err;\n    }\n  }\n\n  async finalize(): Promise<void> {\n    if (this.#finalized) return;\n    // Consume unread body\n    const body = this.body;\n    const buf = new Uint8Array(1024);\n    while ((await body.read(buf)) !== null) {\n      // Pass\n    }\n    this.#finalized = true;\n  }\n}\n\nexport class Server implements AsyncIterable<ServerRequest> {\n  #closing = false;\n  #connections: Deno.Conn[] = [];\n\n  constructor(public listener: Deno.Listener) {}\n\n  close(): void {\n    this.#closing = true;\n    this.listener.close();\n    for (const conn of this.#connections) {\n      try {\n        conn.close();\n      } catch (e) {\n        // Connection might have been already closed\n        if (!(e instanceof Deno.errors.BadResource)) {\n          throw e;\n        }\n      }\n    }\n  }\n\n  // Yields all HTTP requests on a single TCP connection.\n  private async *iterateHttpRequests(\n    conn: Deno.Conn,\n  ): AsyncIterableIterator<ServerRequest> {\n    const reader = new BufReader(conn);\n    const writer = new BufWriter(conn);\n\n    while (!this.#closing) {\n      let request: ServerRequest | null;\n      try {\n        request = await readRequest(conn, reader);\n      } catch (error) {\n        if (\n          error instanceof Deno.errors.InvalidData ||\n          error instanceof Deno.errors.UnexpectedEof\n        ) {\n          // An error was thrown while parsing request headers.\n          // Try to send the \"400 Bad Request\" before closing the connection.\n          try {\n            await writeResponse(writer, {\n              status: 400,\n              body: new TextEncoder().encode(`${error.message}\\r\\n\\r\\n`),\n            });\n          } catch {\n            // The connection is broken.\n          }\n        }\n        break;\n      }\n      if (request === null) {\n        break;\n      }\n\n      request.w = writer;\n      yield request;\n\n      // Wait for the request to be processed before we accept a new request on\n      // this connection.\n      const responseError = await request.done;\n      if (responseError) {\n        // Something bad happened during response.\n        // (likely other side closed during pipelined req)\n        // req.done implies this connection already closed, so we can just return.\n        this.untrackConnection(request.conn);\n        return;\n      }\n\n      try {\n        // Consume unread body and trailers if receiver didn't consume those data\n        await request.finalize();\n      } catch {\n        // Invalid data was received or the connection was closed.\n        break;\n      }\n    }\n\n    this.untrackConnection(conn);\n    try {\n      conn.close();\n    } catch {\n      // might have been already closed\n    }\n  }\n\n  private trackConnection(conn: Deno.Conn): void {\n    this.#connections.push(conn);\n  }\n\n  private untrackConnection(conn: Deno.Conn): void {\n    const index = this.#connections.indexOf(conn);\n    if (index !== -1) {\n      this.#connections.splice(index, 1);\n    }\n  }\n\n  // Accepts a new TCP connection and yields all HTTP requests that arrive on\n  // it. When a connection is accepted, it also creates a new iterator of the\n  // same kind and adds it to the request multiplexer so that another TCP\n  // connection can be accepted.\n  private async *acceptConnAndIterateHttpRequests(\n    mux: MuxAsyncIterator<ServerRequest>,\n  ): AsyncIterableIterator<ServerRequest> {\n    if (this.#closing) return;\n    // Wait for a new connection.\n    let conn: Deno.Conn;\n    try {\n      conn = await this.listener.accept();\n    } catch (error) {\n      if (\n        // The listener is closed:\n        error instanceof Deno.errors.BadResource ||\n        // TLS handshake errors:\n        error instanceof Deno.errors.InvalidData ||\n        error instanceof Deno.errors.UnexpectedEof ||\n        error instanceof Deno.errors.ConnectionReset\n      ) {\n        return mux.add(this.acceptConnAndIterateHttpRequests(mux));\n      }\n      throw error;\n    }\n    this.trackConnection(conn);\n    // Try to accept another connection and add it to the multiplexer.\n    mux.add(this.acceptConnAndIterateHttpRequests(mux));\n    // Yield the requests that arrive on the just-accepted connection.\n    yield* this.iterateHttpRequests(conn);\n  }\n\n  [Symbol.asyncIterator](): AsyncIterableIterator<ServerRequest> {\n    const mux: MuxAsyncIterator<ServerRequest> = new MuxAsyncIterator();\n    mux.add(this.acceptConnAndIterateHttpRequests(mux));\n    return mux.iterate();\n  }\n}\n\n/** Options for creating an HTTP server. */\nexport type HTTPOptions = Omit<Deno.ListenOptions, \"transport\">;\n\n/**\n * Parse addr from string\n *\n *     const addr = \"::1:8000\";\n *     parseAddrFromString(addr);\n *\n * @param addr Address string\n */\nexport function _parseAddrFromStr(addr: string): HTTPOptions {\n  let url: URL;\n  try {\n    const host = addr.startsWith(\":\") ? `0.0.0.0${addr}` : addr;\n    url = new URL(`http://${host}`);\n  } catch {\n    throw new TypeError(\"Invalid address.\");\n  }\n  if (\n    url.username ||\n    url.password ||\n    url.pathname != \"/\" ||\n    url.search ||\n    url.hash\n  ) {\n    throw new TypeError(\"Invalid address.\");\n  }\n\n  return {\n    hostname: url.hostname,\n    port: url.port === \"\" ? 80 : Number(url.port),\n  };\n}\n\n/**\n * Create a HTTP server\n *\n *     import { serve } from \"https://deno.land/std/http/server.ts\";\n *     const body = \"Hello World\\n\";\n *     const server = serve({ port: 8000 });\n *     for await (const req of server) {\n *       req.respond({ body });\n *     }\n */\nexport function serve(addr: string | HTTPOptions): Server {\n  if (typeof addr === \"string\") {\n    addr = _parseAddrFromStr(addr);\n  }\n\n  const listener = Deno.listen(addr);\n  return new Server(listener);\n}\n\n/**\n * Start an HTTP server with given options and request handler\n *\n *     const body = \"Hello World\\n\";\n *     const options = { port: 8000 };\n *     listenAndServe(options, (req) => {\n *       req.respond({ body });\n *     });\n *\n * @param options Server configuration\n * @param handler Request handler\n */\nexport async function listenAndServe(\n  addr: string | HTTPOptions,\n  handler: (req: ServerRequest) => void,\n): Promise<void> {\n  const server = serve(addr);\n\n  for await (const request of server) {\n    handler(request);\n  }\n}\n\n/** Options for creating an HTTPS server. */\nexport type HTTPSOptions = Omit<Deno.ListenTlsOptions, \"transport\">;\n\n/**\n * Create an HTTPS server with given options\n *\n *     const body = \"Hello HTTPS\";\n *     const options = {\n *       hostname: \"localhost\",\n *       port: 443,\n *       certFile: \"./path/to/localhost.crt\",\n *       keyFile: \"./path/to/localhost.key\",\n *     };\n *     for await (const req of serveTLS(options)) {\n *       req.respond({ body });\n *     }\n *\n * @param options Server configuration\n * @return Async iterable server instance for incoming requests\n */\nexport function serveTLS(options: HTTPSOptions): Server {\n  const tlsOptions: Deno.ListenTlsOptions = {\n    ...options,\n    transport: \"tcp\",\n  };\n  const listener = Deno.listenTls(tlsOptions);\n  return new Server(listener);\n}\n\n/**\n * Start an HTTPS server with given options and request handler\n *\n *     const body = \"Hello HTTPS\";\n *     const options = {\n *       hostname: \"localhost\",\n *       port: 443,\n *       certFile: \"./path/to/localhost.crt\",\n *       keyFile: \"./path/to/localhost.key\",\n *     };\n *     listenAndServeTLS(options, (req) => {\n *       req.respond({ body });\n *     });\n *\n * @param options Server configuration\n * @param handler Request handler\n */\nexport async function listenAndServeTLS(\n  options: HTTPSOptions,\n  handler: (req: ServerRequest) => void,\n): Promise<void> {\n  const server = serveTLS(options);\n\n  for await (const request of server) {\n    handler(request);\n  }\n}\n\n/**\n * Interface of HTTP server response.\n * If body is a Reader, response would be chunked.\n * If body is a string, it would be UTF-8 encoded by default.\n */\nexport interface Response {\n  status?: number;\n  headers?: Headers;\n  body?: Uint8Array | Deno.Reader | string;\n  trailers?: () => Promise<Headers> | Headers;\n}\n"],"names":[],"mappings":"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"}