Skip to main content
Module

x/oak/context.ts

A middleware framework for handling HTTP with Deno 🐿️ 🦕
Extremely Popular
Go to Latest
File
// Copyright 2018-2021 the oak authors. All rights reserved. MIT license.
import type { Application, State } from "./application.ts";import { Cookies } from "./cookies.ts";import { acceptable, acceptWebSocket, WebSocket } from "./deps.ts";import { NativeRequest } from "./http_server_native.ts";import type { ServerRequest } from "./http_server_std.ts";import { createHttpError } from "./httpError.ts";import type { KeyStack } from "./keyStack.ts";import { Request } from "./request.ts";import { Response } from "./response.ts";import { send, SendOptions } from "./send.ts";import { ServerSentEventTargetOptions, SSEStdLibTarget, SSEStreamTarget,} from "./server_sent_event.ts";import type { ServerSentEventTarget } from "./server_sent_event.ts";import { structuredClone } from "./structured_clone.ts";import type { ErrorStatus } from "./types.d.ts";
export interface ContextSendOptions extends SendOptions { /** The filename to send, which will be resolved based on the other options. * If this property is omitted, the current context's `.request.url.pathname` * will be used. */ path?: string;}
/** Provides context about the current request and response to middleware * functions. */export class Context< S extends AS = State, // deno-lint-ignore no-explicit-any AS extends State = Record<string, any>,> { #socket?: WebSocket; #sse?: ServerSentEventTarget;
/** A reference to the current application. */ app: Application<AS>;
/** An object which allows access to cookies, mediating both the request and * response. */ cookies: Cookies;
/** Is `true` if the current connection is upgradeable to a web socket. * Otherwise the value is `false`. Use `.upgrade()` to upgrade the connection * and return the web socket. */ get isUpgradable(): boolean { return acceptable(this.request); }
/** Determines if the request should be responded to. If `false` when the * middleware completes processing, the response will not be sent back to the * requestor. Typically this is used if the middleware will take over low * level processing of requests and responses, for example if using web * sockets. This automatically gets set to `false` when the context is * upgraded to a web socket via the `.upgrade()` method. * * The default is `true`. */ respond: boolean;
/** An object which contains information about the current request. */ request: Request;
/** An object which contains information about the response that will be sent * when the middleware finishes processing. */ response: Response;
/** If the the current context has been upgraded, then this will be set to * with the web socket, otherwise it is `undefined`. */ get socket(): WebSocket | undefined { return this.#socket; }
/** The object to pass state to front-end views. This can be typed by * supplying the generic state argument when creating a new app. For * example: * * ```ts * const app = new Application<{ foo: string }>(); * ``` * * Or can be contextually inferred based on setting an initial state object: * * ```ts * const app = new Application({ state: { foo: "bar" } }); * ``` * * On each request/response cycle, the context's state is cloned from the * application state. This means changes to the context's `.state` will be * dropped when the request drops, but "defaults" can be applied to the * application's state. Changes to the application's state though won't be * reflected until the next request in the context's state. */ state: S;
constructor( app: Application<AS>, serverRequest: ServerRequest | NativeRequest, secure = false, ) { this.app = app; this.state = structuredClone(app.state); this.request = new Request(serverRequest, app.proxy, secure); this.respond = true; this.response = new Response(this.request); this.cookies = new Cookies(this.request, this.response, { keys: this.app.keys as KeyStack | undefined, secure: this.request.secure, }); }
/** Asserts the condition and if the condition fails, creates an HTTP error * with the provided status (which defaults to `500`). The error status by * default will be set on the `.response.status`. */ assert( // deno-lint-ignore no-explicit-any condition: any, errorStatus: ErrorStatus = 500, message?: string, props?: Record<string, unknown>, ): asserts condition { if (condition) { return; } const err = createHttpError(errorStatus, message); if (props) { Object.assign(err, props); } throw err; }
/** Asynchronously fulfill a response with a file from the local file * system. * * If the `options.path` is not supplied, the file to be sent will default * to this `.request.url.pathname`. * * Requires Deno read permission. */ send(options: ContextSendOptions): Promise<string | undefined> { const { path = this.request.url.pathname, ...sendOptions } = options; return send(this, path, sendOptions); }
/** Convert the connection to stream events, returning an event target for * sending server sent events. Events dispatched on the returned target will * be sent to the client and be available in the client's `EventSource` that * initiated the connection. * * This will set `.respond` to `false`. */ sendEvents(options?: ServerSentEventTargetOptions): ServerSentEventTarget { if (!this.#sse) { if (this.request.originalRequest instanceof NativeRequest) { this.#sse = new SSEStreamTarget(this, options); } else { this.respond = false; this.#sse = new SSEStdLibTarget(this, options); } } return this.#sse; }
/** Create and throw an HTTP Error, which can be used to pass status * information which can be caught by other middleware to send more * meaningful error messages back to the client. The passed error status will * be set on the `.response.status` by default as well. */ throw( errorStatus: ErrorStatus, message?: string, props?: Record<string, unknown>, ): never { const err = createHttpError(errorStatus, message); if (props) { Object.assign(err, props); } throw err; }
/** Take the current request and upgrade it to a web socket, resolving with * the web socket object. This will set `.respond` to `false`. */ async upgrade(): Promise<WebSocket> { if (this.#socket) { return this.#socket; } if (this.request.originalRequest instanceof NativeRequest) { throw new TypeError( "Socket upgrades are not yet supported on native Deno requests.", ); } const { conn, r: bufReader, w: bufWriter, headers } = this.request.originalRequest; this.#socket = await acceptWebSocket( { conn, bufReader, bufWriter, headers }, ); this.respond = false; return this.#socket; }
[Symbol.for("Deno.customInspect")](inspect: (value: unknown) => string) { const { app, cookies, isUpgradable, respond, request, response, socket, state, } = this; return `${this.constructor.name} ${ inspect({ app, cookies, isUpgradable, respond, request, response, socket, state, }) }`; }}