helpers.handler.js 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. const { RedisStore: RateLimitRedisStore } = require("rate-limit-redis");
  2. const { rateLimit: expressRateLimit } = require("express-rate-limit");
  3. const { validationResult } = require("express-validator");
  4. const { CustomError } = require("../utils");
  5. const query = require("../queries");
  6. const redis = require("../redis");
  7. const env = require("../env");
  8. function error(error, req, res, _next) {
  9. if (!(error instanceof CustomError)) {
  10. console.error(error);
  11. } else if (env.isDev) {
  12. console.error(error.message);
  13. }
  14. const message = error instanceof CustomError ? error.message : "An error occurred.";
  15. const statusCode = error.statusCode ?? 500;
  16. if (req.isHTML && req.viewTemplate) {
  17. res.locals.error = message;
  18. res.render(req.viewTemplate);
  19. return;
  20. }
  21. if (req.isHTML) {
  22. res.render("error", {
  23. message: "An error occurred. Please try again later."
  24. });
  25. return;
  26. }
  27. return res.status(statusCode).json({ error: message });
  28. };
  29. function verify(req, res, next) {
  30. const result = validationResult(req);
  31. if (result.isEmpty()) return next();
  32. const errors = result.array();
  33. const error = errors[0].msg;
  34. res.locals.errors = {};
  35. errors.forEach(e => {
  36. if (res.locals.errors[e.param]) return;
  37. res.locals.errors[e.param] = e.msg;
  38. });
  39. throw new CustomError(error, 400);
  40. }
  41. function parseQuery(req, res, next) {
  42. const { admin } = req.user || {};
  43. if (
  44. typeof req.query.limit !== "undefined" &&
  45. typeof req.query.limit !== "string"
  46. ) {
  47. return res.status(400).json({ error: "limit query is not valid." });
  48. }
  49. if (
  50. typeof req.query.skip !== "undefined" &&
  51. typeof req.query.skip !== "string"
  52. ) {
  53. return res.status(400).json({ error: "skip query is not valid." });
  54. }
  55. if (
  56. typeof req.query.search !== "undefined" &&
  57. typeof req.query.search !== "string"
  58. ) {
  59. return res.status(400).json({ error: "search query is not valid." });
  60. }
  61. const limit = parseInt(req.query.limit) || 10;
  62. req.context = {
  63. limit: limit > 50 ? 50 : limit,
  64. skip: parseInt(req.query.skip) || 0,
  65. };
  66. next();
  67. };
  68. function rateLimit(params) {
  69. if (!env.ENABLE_RATE_LIMIT) {
  70. return function(req, res, next) {
  71. return next();
  72. }
  73. }
  74. let store = undefined;
  75. if (env.REDIS_ENABLED) {
  76. store = new RateLimitRedisStore({
  77. sendCommand: (...args) => redis.client.call(...args),
  78. })
  79. }
  80. return expressRateLimit({
  81. windowMs: params.window * 1000,
  82. validate: { trustProxy: false },
  83. skipSuccessfulRequests: !!params.skipSuccess,
  84. skipFailedRequests: !!params.skipFailed,
  85. ...(store && { store }),
  86. limit: function (req, res) {
  87. if (params.user && req.user) {
  88. return params.user;
  89. }
  90. return params.limit;
  91. },
  92. keyGenerator: function(req, res) {
  93. return "rl:" + req.method + req.baseUrl + req.path + ":" + req.ip;
  94. },
  95. requestWasSuccessful: function(req, res) {
  96. return !res.locals.error && res.statusCode < 400;
  97. },
  98. handler: function (req, res, next, options) {
  99. throw new CustomError(options.message, options.statusCode);
  100. },
  101. });
  102. }
  103. // redirect to create admin page if the kutt instance is ran for the first time
  104. async function adminSetup(req, res, next) {
  105. const isThereAUser = req.user || (await query.user.findAny());
  106. if (isThereAUser) {
  107. next();
  108. return;
  109. }
  110. res.redirect("/create-admin");
  111. }
  112. module.exports = {
  113. adminSetup,
  114. error,
  115. parseQuery,
  116. rateLimit,
  117. verify,
  118. }