| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 |
- const { RedisStore: RateLimitRedisStore } = require("rate-limit-redis");
- const { rateLimit: expressRateLimit } = require("express-rate-limit");
- const { validationResult } = require("express-validator");
- const { CustomError } = require("../utils");
- const query = require("../queries");
- const redis = require("../redis");
- const env = require("../env");
- function error(error, req, res, _next) {
- if (!(error instanceof CustomError)) {
- console.error(error);
- } else if (env.isDev) {
- console.error(error.message);
- }
- const message = error instanceof CustomError ? error.message : "An error occurred.";
- const statusCode = error.statusCode ?? 500;
- if (req.isHTML && req.viewTemplate) {
- res.locals.error = message;
- res.render(req.viewTemplate);
- return;
- }
- if (req.isHTML) {
- res.render("error", {
- message: "An error occurred. Please try again later."
- });
- return;
- }
- return res.status(statusCode).json({ error: message });
- };
- function verify(req, res, next) {
- const result = validationResult(req);
- if (result.isEmpty()) return next();
- const errors = result.array();
- const error = errors[0].msg;
-
- res.locals.errors = {};
- errors.forEach(e => {
- if (res.locals.errors[e.param]) return;
- res.locals.errors[e.param] = e.msg;
- });
- throw new CustomError(error, 400);
- }
- function parseQuery(req, res, next) {
- const { admin } = req.user || {};
- if (
- typeof req.query.limit !== "undefined" &&
- typeof req.query.limit !== "string"
- ) {
- return res.status(400).json({ error: "limit query is not valid." });
- }
- if (
- typeof req.query.skip !== "undefined" &&
- typeof req.query.skip !== "string"
- ) {
- return res.status(400).json({ error: "skip query is not valid." });
- }
- if (
- typeof req.query.search !== "undefined" &&
- typeof req.query.search !== "string"
- ) {
- return res.status(400).json({ error: "search query is not valid." });
- }
- const limit = parseInt(req.query.limit) || 10;
- req.context = {
- limit: limit > 50 ? 50 : limit,
- skip: parseInt(req.query.skip) || 0,
- };
- next();
- };
- function rateLimit(params) {
- if (!env.ENABLE_RATE_LIMIT) {
- return function(req, res, next) {
- return next();
- }
- }
-
- let store = undefined;
- if (env.REDIS_ENABLED) {
- store = new RateLimitRedisStore({
- sendCommand: (...args) => redis.client.call(...args),
- })
- }
-
- return expressRateLimit({
- windowMs: params.window * 1000,
- validate: { trustProxy: false },
- skipSuccessfulRequests: !!params.skipSuccess,
- skipFailedRequests: !!params.skipFailed,
- ...(store && { store }),
- limit: function (req, res) {
- if (params.user && req.user) {
- return params.user;
- }
- return params.limit;
- },
- keyGenerator: function(req, res) {
- return "rl:" + req.method + req.baseUrl + req.path + ":" + req.ip;
- },
- requestWasSuccessful: function(req, res) {
- return !res.locals.error && res.statusCode < 400;
- },
- handler: function (req, res, next, options) {
- throw new CustomError(options.message, options.statusCode);
- },
- });
- }
- // redirect to create admin page if the kutt instance is ran for the first time
- async function adminSetup(req, res, next) {
- const isThereAUser = req.user || (await query.user.findAny());
- if (isThereAUser) {
- next();
- return;
- }
- res.redirect("/create-admin");
- }
- module.exports = {
- adminSetup,
- error,
- parseQuery,
- rateLimit,
- verify,
- }
|