0
0
mirror of https://github.com/honojs/hono.git synced 2024-12-01 10:51:01 +00:00
hono/deno_dist/request.ts

245 lines
6.3 KiB
TypeScript
Raw Normal View History

/* eslint-disable @typescript-eslint/no-explicit-any */
/* eslint-disable @typescript-eslint/no-non-null-assertion */
import type {
Input,
InputToDataByTarget,
ParamKeys,
ParamKeyToRecord,
RemoveQuestion,
UndefinedIfHavingQuestion,
ValidationTargets,
} from './types.ts'
2022-07-12 14:25:47 +00:00
import { parseBody } from './utils/body.ts'
import type { BodyData, ParseBodyOptions } from './utils/body.ts'
2022-07-10 15:17:29 +00:00
import type { Cookie } from './utils/cookie.ts'
import { parse } from './utils/cookie.ts'
import type { UnionToIntersection } from './utils/types.ts'
2023-04-30 12:07:00 +00:00
import { getQueryParam, getQueryParams, decodeURIComponent_ } from './utils/url.ts'
2022-07-10 15:17:29 +00:00
type Body = {
json: any
text: string
arrayBuffer: ArrayBuffer
blob: Blob
formData: FormData
}
type BodyCache = Partial<Body & { parsedBody: BodyData }>
export class HonoRequest<P extends string = '/', I extends Input['out'] = {}> {
raw: Request
2022-07-02 06:09:45 +00:00
private paramData: Record<string, string> | undefined
2023-04-30 12:07:00 +00:00
private vData: { [K in keyof ValidationTargets]?: {} } // Short name of validatedData
path: string
bodyCache: BodyCache = {}
constructor(
request: Request,
path: string = '/',
paramData?: Record<string, string> | undefined
) {
this.raw = request
this.path = path
this.paramData = paramData
2023-04-30 12:07:00 +00:00
this.vData = {}
2022-07-02 06:09:45 +00:00
}
2023-08-14 14:34:56 +00:00
param<P2 extends string = P>(
key: RemoveQuestion<ParamKeys<P2>>
): UndefinedIfHavingQuestion<ParamKeys<P2>>
param<P2 extends string = P>(): UnionToIntersection<ParamKeyToRecord<ParamKeys<P2>>>
2023-01-22 05:28:13 +00:00
param(key?: string): unknown {
2022-07-02 06:09:45 +00:00
if (this.paramData) {
if (key) {
const param = this.paramData[key]
2023-04-30 12:07:00 +00:00
return param ? (/\%/.test(param) ? decodeURIComponent_(param) : param) : undefined
2022-07-02 06:09:45 +00:00
} else {
const decoded: Record<string, string> = {}
for (const [key, value] of Object.entries(this.paramData)) {
if (value && typeof value === 'string') {
2023-04-30 12:07:00 +00:00
decoded[key] = /\%/.test(value) ? decodeURIComponent_(value) : value
}
}
return decoded
2022-07-02 06:09:45 +00:00
}
}
return null
}
2022-07-02 06:09:45 +00:00
query(key: string): string | undefined
query(): Record<string, string>
query(key?: string) {
return getQueryParam(this.url, key)
}
2022-07-02 06:09:45 +00:00
queries(key: string): string[] | undefined
queries(): Record<string, string[]>
queries(key?: string) {
return getQueryParams(this.url, key)
}
2023-02-12 21:47:05 +00:00
header(name: string): string | undefined
header(): Record<string, string>
header(name?: string) {
if (name) return this.raw.headers.get(name.toLowerCase()) ?? undefined
const headerData: Record<string, string | undefined> = {}
2023-02-01 15:25:58 +00:00
this.raw.headers.forEach((value, key) => {
headerData[key] = value
})
return headerData
}
2022-07-10 15:17:29 +00:00
/** @deprecated
* Use Cookie Middleware instead of `c.req.cookie()`. The `c.req.cookie()` will be removed in v4.
*
* @example
*
* import { getCookie } from 'hono/cookie'
* // ...
* app.get('/', (c) => c.text(getCookie(c, 'cookie-name')))
*/
cookie(key: string): string | undefined
/** @deprecated
* Use Cookie Middleware instead of `c.req.cookie()`. The `c.req.cookie()` will be removed in v4.
*
* @example
*
* import { getCookie } from 'hono/cookie'
* // ...
* app.get('/', (c) => c.json(getCookie(c)))
*/
cookie(): Cookie
cookie(key?: string) {
const cookie = this.raw.headers.get('Cookie')
if (!cookie) return
2022-07-10 15:17:29 +00:00
const obj = parse(cookie)
if (key) {
const value = obj[key]
return value
} else {
return obj
}
}
2022-07-12 14:25:47 +00:00
async parseBody<T extends BodyData = BodyData>(options?: ParseBodyOptions): Promise<T> {
if (this.bodyCache.parsedBody) return this.bodyCache.parsedBody as T
const parsedBody = await parseBody<T>(this, options)
this.bodyCache.parsedBody = parsedBody
return parsedBody
}
private cachedBody = (key: keyof Body) => {
const { bodyCache, raw } = this
const cachedBody = bodyCache[key]
if (cachedBody) return cachedBody
/**
* If an arrayBuffer cache is exist,
* use it for creating a text, json, and others.
*/
if (bodyCache.arrayBuffer) {
return (async () => {
return await new Response(bodyCache.arrayBuffer)[key]()
})()
}
return (bodyCache[key] = raw[key]())
}
2023-02-16 09:08:50 +00:00
json<T = any>(): Promise<T> {
return this.cachedBody('json')
}
text(): Promise<string> {
return this.cachedBody('text')
}
arrayBuffer(): Promise<ArrayBuffer> {
return this.cachedBody('arrayBuffer')
}
blob(): Promise<Blob> {
return this.cachedBody('blob')
}
formData(): Promise<FormData> {
return this.cachedBody('formData')
}
addValidatedData(target: keyof ValidationTargets, data: {}) {
2023-04-30 12:07:00 +00:00
this.vData[target] = data
}
valid<T extends keyof I & keyof ValidationTargets>(target: T): InputToDataByTarget<I, T>
valid(target: keyof ValidationTargets) {
return this.vData[target] as unknown
}
get url() {
return this.raw.url
}
get method() {
return this.raw.method
}
/** @deprecated
* Use `c.req.raw.headers` instead of `c.req.headers`. The `c.req.headers` will be removed in v4.
* Or you can get the header values with using `c.req.header`.
* @example
*
* app.get('/', (c) => {
* const userAgent = c.req.header('User-Agent')
* //...
* })
*/
get headers() {
return this.raw.headers
}
/** @deprecated
* Use `c.req.raw.body` instead of `c.req.body`. The `c.req.body` will be removed in v4.
*/
get body() {
return this.raw.body
}
/** @deprecated
* Use `c.req.raw.bodyUsed` instead of `c.req.bodyUsed`. The `c.req.bodyUsed` will be removed in v4.
*/
get bodyUsed() {
return this.raw.bodyUsed
}
/** @deprecated
* Use `c.req.raw.integrity` instead of `c.req.integrity`. The `c.req.integrity` will be removed in v4.
*/
get integrity() {
return this.raw.integrity
}
/** @deprecated
* Use `c.req.raw.keepalive` instead of `c.req.keepalive`. The `c.req.keepalive` will be removed in v4.
*/
get keepalive() {
return this.raw.keepalive
}
/** @deprecated
* Use `c.req.raw.referrer` instead of `c.req.referrer`. The `c.req.referrer` will be removed in v4.
*/
get referrer() {
return this.raw.referrer
}
/** @deprecated
* Use `c.req.raw.signal` instead of `c.req.signal`. The `c.req.signal` will be removed in v4.
*/
get signal() {
return this.raw.signal
}
2022-07-02 06:09:45 +00:00
}