0
0
mirror of https://github.com/honojs/hono.git synced 2024-11-25 13:19:30 +01:00
hono/deno_dist/validator/validator.ts
Robert Sasak 79dc187dbb
fix(validator): Invalid HTTP header error is hidden (#1694)
* Invalid HTTP header error is hidden

This error is captured and never shown which may by misleading. This PR suggest to throw this error directly to the reponse.

* fixed the linter error

* add a test

* denoify

---------

Co-authored-by: Yusuke Wada <yusuke@kamawada.com>
2023-11-21 19:42:27 +09:00

144 lines
4.2 KiB
TypeScript

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> = M extends 'get' | 'head' // GET and HEAD request must not have a body content.
? Exclude<keyof ValidationTargets, ValidationTargetKeysWithBody>
: keyof ValidationTargets
export type ValidationFunction<
InputType,
OutputType,
E extends Env = {},
P extends string = string
> = (
value: InputType,
c: Context<E, P>
) => OutputType | Response | Promise<OutputType> | Promise<Response>
export const validator = <
InputType,
P extends string,
M extends string,
U extends ValidationTargetByMethod<M>,
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<E, P, V> => {
return async (c, next) => {
let value = {}
const contentType = c.req.header('Content-Type')
switch (target) {
case 'json':
if (!contentType || !contentType.startsWith('application/json')) {
const message = `Invalid HTTP header: Content-Type=${contentType}`
console.error(message)
return c.json(
{
success: false,
message,
},
400
)
}
/**
* Get the arrayBuffer first, create JSON object via Response,
* and cache the arrayBuffer in the c.req.bodyCache.
*/
try {
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<string, string>
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()
}
}