// Loaded from https://deno.land/x/oak/context.ts // 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 { 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 { ServerSentEventTarget, ServerSentEventTargetOptions, } from "./server_sent_event.ts"; import type { ErrorStatus, ServerRequest } 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. */ // deno-lint-ignore no-explicit-any export class Context> { #socket?: WebSocket; #sse?: ServerSentEventTarget; /** A reference to the current application. */ app: Application; /** 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" } }); * ``` */ state: S; constructor( app: Application, serverRequest: ServerRequest, secure = false, ) { this.app = app; this.state = 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, ): 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 { 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) { return this.#sse; } this.respond = false; return this.#sse = new ServerSentEventTarget( this.app, this.request.serverRequest, options, ); } /** 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, ): 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 { if (this.#socket) { return this.#socket; } const { conn, r: bufReader, w: bufWriter, headers } = this.request.serverRequest; this.#socket = await acceptWebSocket( { conn, bufReader, bufWriter, headers }, ); this.respond = false; return this.#socket; } }