@nestia/core
Version:
Super-fast validation decorators of NestJS
98 lines (90 loc) • 3.43 kB
text/typescript
import { AesPkcs5 } from "@nestia/fetcher/lib/AesPkcs5";
import { IEncryptionPassword } from "@nestia/fetcher/lib/IEncryptionPassword";
import {
BadRequestException,
ExecutionContext,
createParamDecorator,
} from "@nestjs/common";
import type express from "express";
import type { FastifyRequest } from "fastify";
import { IRequestBodyValidator } from "../options/IRequestBodyValidator";
import { Singleton } from "../utils/Singleton";
import { ENCRYPTION_METADATA_KEY } from "./internal/EncryptedConstant";
import { get_text_body } from "./internal/get_text_body";
import { headers_to_object } from "./internal/headers_to_object";
import { validate_request_body } from "./internal/validate_request_body";
/**
* Encrypted body decorator.
*
* `EncryptedBody` is a decorator function getting `application/json` typed data
* from request body which has been encrypted by AES-128/256 algorithm. Also,
* `EncryptedBody` validates the request body data type through
* [typia](https://github.com/samchon/typia) ad the validation speed is maximum
* 15,000x times faster than `class-validator`.
*
* For reference, when the request body data is not following the promised type
* `T`, `BadRequestException` error (status code: 400) would be thrown. Also,
* `EncryptedRoute` decrypts request body using those options.
*
* - AES-128/256
* - CBC mode
* - PKCS #5 Padding
* - Base64 Encoding
*
* @author Jeongho Nam - https://github.com/samchon
* @returns Parameter decorator
*/
export function EncryptedBody<T>(
validator?: IRequestBodyValidator<T>,
): ParameterDecorator {
const checker = validate_request_body("EncryptedBody")(validator);
return createParamDecorator(async function EncryptedBody(
_unknown: any,
context: ExecutionContext,
) {
const request: express.Request | FastifyRequest = context
.switchToHttp()
.getRequest();
if (isTextPlain(request.headers["content-type"]) === false)
throw new BadRequestException(`Request body type is not "text/plain".`);
const param: IEncryptionPassword | IEncryptionPassword.Closure | undefined =
Reflect.getMetadata(ENCRYPTION_METADATA_KEY, context.getClass());
if (!param)
throw new Error(
"Error on nestia.core.EncryptedBody(): no encryption password is given.",
);
// GET BODY DATA
const headers: Singleton<Record<string, string>> = new Singleton(() =>
headers_to_object(request.headers),
);
const body: string = await get_text_body(request);
const password: IEncryptionPassword =
typeof param === "function"
? param({ headers: headers.get(), body, direction: "decode" })
: param;
// PARSE AND VALIDATE DATA
const data: any = JSON.parse(decrypt(body, password.key, password.iv));
const error: Error | null = checker(data);
if (error !== null) throw error;
return data;
})();
}
/** @internal */
const decrypt = (body: string, key: string, iv: string): string => {
try {
return AesPkcs5.decrypt(body, key, iv);
} catch (exp) {
if (exp instanceof Error)
throw new BadRequestException(
"Failed to decrypt the request body. Check your body content or encryption password.",
);
else throw exp;
}
};
/** @internal */
const isTextPlain = (text?: string): boolean =>
text !== undefined &&
text
.split(";")
.map((str) => str.trim())
.some((str) => str === "text/plain");