mirror of
https://github.com/monkeytypegame/monkeytype.git
synced 2025-02-27 00:04:37 +08:00
113 lines
3 KiB
TypeScript
113 lines
3 KiB
TypeScript
import _ from "lodash";
|
|
import joi from "joi";
|
|
import MonkeyError from "../handlers/error";
|
|
import { Response, NextFunction, RequestHandler } from "express";
|
|
import {
|
|
handleMonkeyResponse,
|
|
MonkeyResponse,
|
|
} from "../handlers/monkey-response";
|
|
|
|
interface ConfigurationValidationOptions {
|
|
criteria: (configuration: MonkeyTypes.Configuration) => boolean;
|
|
invalidMessage?: string;
|
|
}
|
|
|
|
/**
|
|
* This utility checks that the server's configuration matches
|
|
* the criteria.
|
|
*/
|
|
function validateConfiguration(
|
|
options: ConfigurationValidationOptions
|
|
): RequestHandler {
|
|
const { criteria, invalidMessage } = options;
|
|
|
|
return (req: MonkeyTypes.Request, _res: Response, next: NextFunction) => {
|
|
const configuration = req.ctx.configuration;
|
|
|
|
const validated = criteria(configuration);
|
|
if (!validated) {
|
|
throw new MonkeyError(
|
|
503,
|
|
invalidMessage ?? "This service is currently unavailable."
|
|
);
|
|
}
|
|
|
|
next();
|
|
};
|
|
}
|
|
|
|
type AsyncHandler = (
|
|
req: MonkeyTypes.Request,
|
|
res?: Response
|
|
) => Promise<MonkeyResponse>;
|
|
|
|
/**
|
|
* This utility serves as an alternative to wrapping express handlers with try/catch statements.
|
|
* Any routes that use an async handler function should wrap the handler with this function.
|
|
* Without this, any errors thrown will not be caught by the error handling middleware, and
|
|
* the app will hang!
|
|
*/
|
|
function asyncHandler(handler: AsyncHandler): RequestHandler {
|
|
return async (
|
|
req: MonkeyTypes.Request,
|
|
res: Response,
|
|
next: NextFunction
|
|
) => {
|
|
try {
|
|
const handlerData = await handler(req, res);
|
|
return handleMonkeyResponse(handlerData, res);
|
|
} catch (error) {
|
|
next(error);
|
|
}
|
|
};
|
|
}
|
|
|
|
interface ValidationSchema {
|
|
body?: object;
|
|
query?: object;
|
|
params?: object;
|
|
validationErrorMessage?: string;
|
|
}
|
|
|
|
function validateRequest(validationSchema: ValidationSchema): RequestHandler {
|
|
/**
|
|
* In dev environments, as an alternative to token authentication,
|
|
* you can pass the authentication middleware by having a user id in the body.
|
|
* Inject the user id into the schema so that validation will not fail.
|
|
*/
|
|
if (process.env.MODE === "dev") {
|
|
validationSchema.body = {
|
|
uid: joi.any(),
|
|
...(validationSchema.body ?? {}),
|
|
};
|
|
}
|
|
|
|
const { validationErrorMessage } = validationSchema;
|
|
const normalizedValidationSchema: ValidationSchema = _.omit(
|
|
validationSchema,
|
|
"validationErrorMessage"
|
|
);
|
|
|
|
return (req: MonkeyTypes.Request, _res: Response, next: NextFunction) => {
|
|
_.each(
|
|
normalizedValidationSchema,
|
|
(schema: object, key: keyof ValidationSchema) => {
|
|
const joiSchema = joi.object().keys(schema);
|
|
|
|
const { error } = joiSchema.validate(req[key] ?? {});
|
|
if (error) {
|
|
const errorMessage = error.details[0].message;
|
|
throw new MonkeyError(
|
|
500,
|
|
validationErrorMessage ??
|
|
`${errorMessage} (${error.details[0].context.value})`
|
|
);
|
|
}
|
|
}
|
|
);
|
|
|
|
next();
|
|
};
|
|
}
|
|
|
|
export { validateConfiguration, asyncHandler, validateRequest };
|