import type { Context } from '../context.ts' import { getCookie } from '../helper/cookie/index.ts' import type { Env, ValidationTargets, MiddlewareHandler } from '../types.ts' import type { BodyData } from '../utils/body.ts' import { bufferToFormData } from '../utils/buffer.ts' type ValidationTargetKeysWithBody = 'form' | 'json' type ValidationTargetByMethod = M extends 'get' | 'head' // GET and HEAD request must not have a body content. ? Exclude : keyof ValidationTargets export type ValidationFunction< InputType, OutputType, E extends Env = {}, P extends string = string > = ( value: InputType, c: Context ) => OutputType | Response | Promise | Promise export const validator = < InputType, P extends string, M extends string, U extends ValidationTargetByMethod, OutputType = ValidationTargets[U], P2 extends string = P, V extends { in: { [K in U]: unknown extends InputType ? OutputType : InputType } out: { [K in U]: OutputType } } = { in: { [K in U]: unknown extends InputType ? OutputType : InputType } out: { [K in U]: OutputType } }, // eslint-disable-next-line @typescript-eslint/no-explicit-any E extends Env = any >( target: U, validationFunc: ValidationFunction< unknown extends InputType ? ValidationTargets[U] : InputType, OutputType, E, P2 > ): MiddlewareHandler => { return async (c, next) => { let value = {} switch (target) { case 'json': try { const contentType = c.req.header('Content-Type') if (!contentType || !contentType.startsWith('application/json')) { throw new Error(`Invalid HTTP header: Content-Type=${contentType}`) } /** * Get the arrayBuffer first, create JSON object via Response, * and cache the arrayBuffer in the c.req.bodyCache. */ const arrayBuffer = c.req.bodyCache.arrayBuffer ?? (await c.req.raw.arrayBuffer()) value = await new Response(arrayBuffer).json() c.req.bodyCache.json = value c.req.bodyCache.arrayBuffer = arrayBuffer } catch { console.error('Error: Malformed JSON in request body') return c.json( { success: false, message: 'Malformed JSON in request body', }, 400 ) } break case 'form': { try { const contentType = c.req.header('Content-Type') if (contentType) { const arrayBuffer = c.req.bodyCache.arrayBuffer ?? (await c.req.raw.arrayBuffer()) const formData = await bufferToFormData(arrayBuffer, contentType) const form: BodyData = {} formData.forEach((value, key) => { form[key] = value }) value = form c.req.bodyCache.formData = formData c.req.bodyCache.arrayBuffer = arrayBuffer } } catch (e) { let message = 'Malformed FormData request.' message += e instanceof Error ? ` ${e.message}` : ` ${String(e)}` return c.json( { success: false, message, }, 400 ) } break } case 'query': value = Object.fromEntries( Object.entries(c.req.queries()).map(([k, v]) => { return v.length === 1 ? [k, v[0]] : [k, v] }) ) break case 'queries': value = c.req.queries() console.log('Warnings: Validate type `queries` is deprecated. Use `query` instead.') break case 'param': value = c.req.param() as Record break case 'header': value = c.req.header() break case 'cookie': value = getCookie(c) break } const res = await validationFunc(value as never, c as never) if (res instanceof Response) { return res } c.req.addValidatedData(target, res as never) await next() } }