helpers.handler.js 3.1 KB

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