Utils.js 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. "use strict";
  2. function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
  3. function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
  4. function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
  5. function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; }
  6. function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
  7. /**
  8. * utils.js
  9. * @file General purpose utilities
  10. * @description General purpose utilities.
  11. */
  12. const path = require('path');
  13. const fs = require('fs').promises;
  14. /**
  15. * The general purpose utilities.
  16. */
  17. class Utils {
  18. /**
  19. * @function getLocalizedPath
  20. * @description Returns a localized file path accoring to the locale.
  21. *
  22. * Localized files are searched in subfolders of a given path, e.g.
  23. *
  24. * root/
  25. * ├── base/ // base path to files
  26. * │ ├── example.html // default file
  27. * │ └── de/ // de language folder
  28. * │ │ └── example.html // de localized file
  29. * │ └── de-AT/ // de-AT locale folder
  30. * │ │ └── example.html // de-AT localized file
  31. *
  32. * Files are matched with the locale in the following order:
  33. * 1. Locale match, e.g. locale `de-AT` matches file in folder `de-AT`.
  34. * 2. Language match, e.g. locale `de-AT` matches file in folder `de`.
  35. * 3. Default; file in base folder is returned.
  36. *
  37. * @param {String} defaultPath The absolute file path, which is also
  38. * the default path returned if localization is not available.
  39. * @param {String} locale The locale.
  40. * @returns {Promise<Object>} The object contains:
  41. * - `path`: The path to the localized file, or the original path if
  42. * localization is not available.
  43. * - `subdir`: The subdirectory of the localized file, or undefined if
  44. * there is no matching localized file.
  45. */
  46. static async getLocalizedPath(defaultPath, locale) {
  47. // Get file name and paths
  48. const file = path.basename(defaultPath);
  49. const basePath = path.dirname(defaultPath);
  50. // If locale is not set return default file
  51. if (!locale) {
  52. return {
  53. path: defaultPath
  54. };
  55. }
  56. // Check file for locale exists
  57. const localePath = path.join(basePath, locale, file);
  58. const localeFileExists = await Utils.fileExists(localePath);
  59. // If file for locale exists return file
  60. if (localeFileExists) {
  61. return {
  62. path: localePath,
  63. subdir: locale
  64. };
  65. }
  66. // Check file for language exists
  67. const language = locale.split('-')[0];
  68. const languagePath = path.join(basePath, language, file);
  69. const languageFileExists = await Utils.fileExists(languagePath);
  70. // If file for language exists return file
  71. if (languageFileExists) {
  72. return {
  73. path: languagePath,
  74. subdir: language
  75. };
  76. }
  77. // Return default file
  78. return {
  79. path: defaultPath
  80. };
  81. }
  82. /**
  83. * @function fileExists
  84. * @description Checks whether a file exists.
  85. * @param {String} path The file path.
  86. * @returns {Promise<Boolean>} Is true if the file can be accessed, false otherwise.
  87. */
  88. static async fileExists(path) {
  89. try {
  90. await fs.access(path);
  91. return true;
  92. } catch (e) {
  93. return false;
  94. }
  95. }
  96. /**
  97. * @function isPath
  98. * @description Evaluates whether a string is a file path (as opposed to a URL for example).
  99. * @param {String} s The string to evaluate.
  100. * @returns {Boolean} Returns true if the evaluated string is a path.
  101. */
  102. static isPath(s) {
  103. return /(^\/)|(^\.\/)|(^\.\.\/)/.test(s);
  104. }
  105. /**
  106. * Flattens an object and crates new keys with custom delimiters.
  107. * @param {Object} obj The object to flatten.
  108. * @param {String} [delimiter='.'] The delimiter of the newly generated keys.
  109. * @param {Object} result
  110. * @returns {Object} The flattened object.
  111. **/
  112. static flattenObject(obj, parentKey, delimiter = '.', result = {}) {
  113. for (const key in obj) {
  114. if (Object.prototype.hasOwnProperty.call(obj, key)) {
  115. const newKey = parentKey ? parentKey + delimiter + key : key;
  116. if (typeof obj[key] === 'object' && obj[key] !== null) {
  117. this.flattenObject(obj[key], newKey, delimiter, result);
  118. } else {
  119. result[newKey] = obj[key];
  120. }
  121. }
  122. }
  123. return result;
  124. }
  125. /**
  126. * Determines whether an object is a Promise.
  127. * @param {any} object The object to validate.
  128. * @returns {Boolean} Returns true if the object is a promise.
  129. */
  130. static isPromise(object) {
  131. return object instanceof Promise;
  132. }
  133. /**
  134. * Creates an object with all permutations of the original keys.
  135. * For example, this definition:
  136. * ```
  137. * {
  138. * a: [true, false],
  139. * b: [1, 2],
  140. * c: ['x']
  141. * }
  142. * ```
  143. * permutates to:
  144. * ```
  145. * [
  146. * { a: true, b: 1, c: 'x' },
  147. * { a: true, b: 2, c: 'x' },
  148. * { a: false, b: 1, c: 'x' },
  149. * { a: false, b: 2, c: 'x' }
  150. * ]
  151. * ```
  152. * @param {Object} object The object to permutate.
  153. * @param {Integer} [index=0] The current key index.
  154. * @param {Object} [current={}] The current result entry being composed.
  155. * @param {Array} [results=[]] The resulting array of permutations.
  156. */
  157. static getObjectKeyPermutations(object, index = 0, current = {}, results = []) {
  158. const keys = Object.keys(object);
  159. const key = keys[index];
  160. const values = object[key];
  161. for (const value of values) {
  162. current[key] = value;
  163. const nextIndex = index + 1;
  164. if (nextIndex < keys.length) {
  165. Utils.getObjectKeyPermutations(object, nextIndex, current, results);
  166. } else {
  167. const result = Object.assign({}, current);
  168. results.push(result);
  169. }
  170. }
  171. return results;
  172. }
  173. /**
  174. * Validates parameters and throws if a parameter is invalid.
  175. * Example parameter types syntax:
  176. * ```
  177. * {
  178. * parameterName: {
  179. * t: 'boolean',
  180. * v: isBoolean,
  181. * o: true
  182. * },
  183. * ...
  184. * }
  185. * ```
  186. * @param {Object} params The parameters to validate.
  187. * @param {Array<Object>} types The parameter types used for validation.
  188. * @param {Object} types.t The parameter type; used for error message, not for validation.
  189. * @param {Object} types.v The function to validate the parameter value.
  190. * @param {Boolean} [types.o=false] Is true if the parameter is optional.
  191. */
  192. static validateParams(params, types) {
  193. for (const key of Object.keys(params)) {
  194. const type = types[key];
  195. const isOptional = !!type.o;
  196. const param = params[key];
  197. if (!(isOptional && param == null) && !type.v(param)) {
  198. throw `Invalid parameter ${key} must be of type ${type.t} but is ${typeof param}`;
  199. }
  200. }
  201. }
  202. /**
  203. * Computes the relative date based on a string.
  204. * @param {String} text The string to interpret the date from.
  205. * @param {Date} now The date the string is comparing against.
  206. * @returns {Object} The relative date object.
  207. **/
  208. static relativeTimeToDate(text, now = new Date()) {
  209. text = text.toLowerCase();
  210. let parts = text.split(' ');
  211. // Filter out whitespace
  212. parts = parts.filter(part => part !== '');
  213. const future = parts[0] === 'in';
  214. const past = parts[parts.length - 1] === 'ago';
  215. if (!future && !past && text !== 'now') {
  216. return {
  217. status: 'error',
  218. info: "Time should either start with 'in' or end with 'ago'"
  219. };
  220. }
  221. if (future && past) {
  222. return {
  223. status: 'error',
  224. info: "Time cannot have both 'in' and 'ago'"
  225. };
  226. }
  227. // strip the 'ago' or 'in'
  228. if (future) {
  229. parts = parts.slice(1);
  230. } else {
  231. // past
  232. parts = parts.slice(0, parts.length - 1);
  233. }
  234. if (parts.length % 2 !== 0 && text !== 'now') {
  235. return {
  236. status: 'error',
  237. info: 'Invalid time string. Dangling unit or number.'
  238. };
  239. }
  240. const pairs = [];
  241. while (parts.length) {
  242. pairs.push([parts.shift(), parts.shift()]);
  243. }
  244. let seconds = 0;
  245. for (const [num, interval] of pairs) {
  246. const val = Number(num);
  247. if (!Number.isInteger(val)) {
  248. return {
  249. status: 'error',
  250. info: `'${num}' is not an integer.`
  251. };
  252. }
  253. switch (interval) {
  254. case 'yr':
  255. case 'yrs':
  256. case 'year':
  257. case 'years':
  258. seconds += val * 31536000; // 365 * 24 * 60 * 60
  259. break;
  260. case 'wk':
  261. case 'wks':
  262. case 'week':
  263. case 'weeks':
  264. seconds += val * 604800; // 7 * 24 * 60 * 60
  265. break;
  266. case 'd':
  267. case 'day':
  268. case 'days':
  269. seconds += val * 86400; // 24 * 60 * 60
  270. break;
  271. case 'hr':
  272. case 'hrs':
  273. case 'hour':
  274. case 'hours':
  275. seconds += val * 3600; // 60 * 60
  276. break;
  277. case 'min':
  278. case 'mins':
  279. case 'minute':
  280. case 'minutes':
  281. seconds += val * 60;
  282. break;
  283. case 'sec':
  284. case 'secs':
  285. case 'second':
  286. case 'seconds':
  287. seconds += val;
  288. break;
  289. default:
  290. return {
  291. status: 'error',
  292. info: `Invalid interval: '${interval}'`
  293. };
  294. }
  295. }
  296. const milliseconds = seconds * 1000;
  297. if (future) {
  298. return {
  299. status: 'success',
  300. info: 'future',
  301. result: new Date(now.valueOf() + milliseconds)
  302. };
  303. } else if (past) {
  304. return {
  305. status: 'success',
  306. info: 'past',
  307. result: new Date(now.valueOf() - milliseconds)
  308. };
  309. } else {
  310. return {
  311. status: 'success',
  312. info: 'present',
  313. result: new Date(now.valueOf())
  314. };
  315. }
  316. }
  317. /**
  318. * Deep-scans an object for a matching key/value definition.
  319. * @param {Object} obj The object to scan.
  320. * @param {String | undefined} key The key to match, or undefined if only the value should be matched.
  321. * @param {any | undefined} value The value to match, or undefined if only the key should be matched.
  322. * @returns {Boolean} True if a match was found, false otherwise.
  323. */
  324. static objectContainsKeyValue(obj, key, value) {
  325. const isMatch = (a, b) => typeof a === 'string' && new RegExp(b).test(a) || a === b;
  326. const isKeyMatch = k => isMatch(k, key);
  327. const isValueMatch = v => isMatch(v, value);
  328. for (const [k, v] of Object.entries(obj)) {
  329. if (key !== undefined && value === undefined && isKeyMatch(k)) {
  330. return true;
  331. } else if (key === undefined && value !== undefined && isValueMatch(v)) {
  332. return true;
  333. } else if (key !== undefined && value !== undefined && isKeyMatch(k) && isValueMatch(v)) {
  334. return true;
  335. }
  336. if (['[object Object]', '[object Array]'].includes(Object.prototype.toString.call(v))) {
  337. return Utils.objectContainsKeyValue(v, key, value);
  338. }
  339. }
  340. return false;
  341. }
  342. static checkProhibitedKeywords(config, data) {
  343. if (config !== null && config !== void 0 && config.requestKeywordDenylist) {
  344. // Scan request data for denied keywords
  345. for (const keyword of config.requestKeywordDenylist) {
  346. const match = Utils.objectContainsKeyValue(data, keyword.key, keyword.value);
  347. if (match) {
  348. throw `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`;
  349. }
  350. }
  351. }
  352. }
  353. /**
  354. * Moves the nested keys of a specified key in an object to the root of the object.
  355. *
  356. * @param {Object} obj The object to modify.
  357. * @param {String} key The key whose nested keys will be moved to root.
  358. * @returns {Object} The modified object, or the original object if no modification happened.
  359. * @example
  360. * const obj = {
  361. * a: 1,
  362. * b: {
  363. * c: 2,
  364. * d: 3
  365. * },
  366. * e: 4
  367. * };
  368. * addNestedKeysToRoot(obj, 'b');
  369. * console.log(obj);
  370. * // Output: { a: 1, e: 4, c: 2, d: 3 }
  371. */
  372. static addNestedKeysToRoot(obj, key) {
  373. if (obj[key] && typeof obj[key] === 'object') {
  374. // Add nested keys to root
  375. Object.assign(obj, _objectSpread({}, obj[key]));
  376. // Delete original nested key
  377. delete obj[key];
  378. }
  379. return obj;
  380. }
  381. }
  382. module.exports = Utils;
  383. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["path","require","fs","promises","Utils","getLocalizedPath","defaultPath","locale","file","basename","basePath","dirname","localePath","join","localeFileExists","fileExists","subdir","language","split","languagePath","languageFileExists","access","e","isPath","s","test","flattenObject","obj","parentKey","delimiter","result","key","Object","prototype","hasOwnProperty","call","newKey","isPromise","object","Promise","getObjectKeyPermutations","index","current","results","keys","values","value","nextIndex","length","assign","push","validateParams","params","types","type","isOptional","o","param","v","t","relativeTimeToDate","text","now","Date","toLowerCase","parts","filter","part","future","past","status","info","slice","pairs","shift","seconds","num","interval","val","Number","isInteger","milliseconds","valueOf","objectContainsKeyValue","isMatch","a","b","RegExp","isKeyMatch","k","isValueMatch","entries","undefined","includes","toString","checkProhibitedKeywords","config","data","requestKeywordDenylist","keyword","match","JSON","stringify","addNestedKeysToRoot","_objectSpread","module","exports"],"sources":["../src/Utils.js"],"sourcesContent":["/**\n * utils.js\n * @file General purpose utilities\n * @description General purpose utilities.\n */\n\nconst path = require('path');\nconst fs = require('fs').promises;\n\n/**\n * The general purpose utilities.\n */\nclass Utils {\n  /**\n   * @function getLocalizedPath\n   * @description Returns a localized file path accoring to the locale.\n   *\n   * Localized files are searched in subfolders of a given path, e.g.\n   *\n   * root/\n   * ├── base/                    // base path to files\n   * │   ├── example.html         // default file\n   * │   └── de/                  // de language folder\n   * │   │   └── example.html     // de localized file\n   * │   └── de-AT/               // de-AT locale folder\n   * │   │   └── example.html     // de-AT localized file\n   *\n   * Files are matched with the locale in the following order:\n   * 1. Locale match, e.g. locale `de-AT` matches file in folder `de-AT`.\n   * 2. Language match, e.g. locale `de-AT` matches file in folder `de`.\n   * 3. Default; file in base folder is returned.\n   *\n   * @param {String} defaultPath The absolute file path, which is also\n   * the default path returned if localization is not available.\n   * @param {String} locale The locale.\n   * @returns {Promise<Object>} The object contains:\n   * - `path`: The path to the localized file, or the original path if\n   *   localization is not available.\n   * - `subdir`: The subdirectory of the localized file, or undefined if\n   *   there is no matching localized file.\n   */\n  static async getLocalizedPath(defaultPath, locale) {\n    // Get file name and paths\n    const file = path.basename(defaultPath);\n    const basePath = path.dirname(defaultPath);\n\n    // If locale is not set return default file\n    if (!locale) {\n      return { path: defaultPath };\n    }\n\n    // Check file for locale exists\n    const localePath = path.join(basePath, locale, file);\n    const localeFileExists = await Utils.fileExists(localePath);\n\n    // If file for locale exists return file\n    if (localeFileExists) {\n      return { path: localePath, subdir: locale };\n    }\n\n    // Check file for language exists\n    const language = locale.split('-')[0];\n    const languagePath = path.join(basePath, language, file);\n    const languageFileExists = await Utils.fileExists(languagePath);\n\n    // If file for language exists return file\n    if (languageFileExists) {\n      return { path: languagePath, subdir: language };\n    }\n\n    // Return default file\n    return { path: defaultPath };\n  }\n\n  /**\n   * @function fileExists\n   * @description Checks whether a file exists.\n   * @param {String} path The file path.\n   * @returns {Promise<Boolean>} Is true if the file can be accessed, false otherwise.\n   */\n  static async fileExists(path) {\n    try {\n      await fs.access(path);\n      return true;\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * @function isPath\n   * @description Evaluates whether a string is a file path (as opposed to a URL for example).\n   * @param {String} s The string to evaluate.\n   * @returns {Boolean} Returns true if the evaluated string is a path.\n   */\n  static isPath(s) {\n    return /(^\\/)|(^\\.\\/)|(^\\.\\.\\/)/.test(s);\n  }\n\n  /**\n   * Flattens an object and crates new keys with custom delimiters.\n   * @param {Object} obj The object to flatten.\n   * @param {String} [delimiter='.'] The delimiter of the newly generated keys.\n   * @param {Object} result\n   * @returns {Object} The flattened object.\n   **/\n  static flattenObject(obj, parentKey, delimiter = '.', result = {}) {\n    for (const key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        const newKey = parentKey ? parentKey + delimiter + key : key;\n\n        if (typeof obj[key] === 'object' && obj[key] !== null) {\n          this.flattenObject(obj[key], newKey, delimiter, result);\n        } else {\n          result[newKey] = obj[key];\n        }\n      }\n    }\n    return result;\n  }\n\n  /**\n   * Determines whether an object is a Promise.\n   * @param {any} object The object to validate.\n   * @returns {Boolean} Returns true if the object is a promise.\n   */\n  static isPromise(object) {\n    return object instanceof Promise;\n  }\n\n  /**\n   * Creates an object with all permutations of the original keys.\n   * For example, this definition:\n   * ```\n   * {\n   *   a: [true, false],\n   *   b: [1, 2],\n   *   c: ['x']\n   * }\n   * ```\n   * permutates to:\n   * ```\n   * [\n   *   { a: true, b: 1, c: 'x' },\n   *   { a: true, b: 2, c: 'x' },\n   *   { a: false, b: 1, c: 'x' },\n   *   { a: false, b: 2, c: 'x' }\n   * ]\n   * ```\n   * @param {Object} object The object to permutate.\n   * @param {Integer} [index=0] The current key index.\n   * @param {Object} [current={}] The current result entry being composed.\n   * @param {Array} [results=[]] The resulting array of permutations.\n   */\n  static getObjectKeyPermutations(object, index = 0, current = {}, results = []) {\n    const keys = Object.keys(object);\n    const key = keys[index];\n    const values = object[key];\n\n    for (const value of values) {\n      current[key] = value;\n      const nextIndex = index + 1;\n\n      if (nextIndex < keys.length) {\n        Utils.getObjectKeyPermutations(object, nextIndex, current, results);\n      } else {\n        const result = Object.assign({}, current);\n        results.push(result);\n      }\n    }\n    return results;\n  }\n\n  /**\n   * Validates parameters and throws if a parameter is invalid.\n   * Example parameter types syntax:\n   * ```\n   * {\n   *   parameterName: {\n   *      t: 'boolean',\n   *      v: isBoolean,\n   *      o: true\n   *   },\n   *   ...\n   * }\n   * ```\n   * @param {Object} params The parameters to validate.\n   * @param {Array<Object>} types The parameter types used for validation.\n   * @param {Object} types.t The parameter type; used for error message, not for validation.\n   * @param {Object} types.v The function to validate the parameter value.\n   * @param {Boolean} [types.o=false] Is true if the parameter is optional.\n   */\n  static validateParams(params, types) {\n    for (const key of Object.keys(params)) {\n      const type = types[key];\n      const isOptional = !!type.o;\n      const param = params[key];\n      if (!(isOptional && param == null) && !type.v(param)) {\n        throw `Invalid parameter ${key} must be of type ${type.t} but is ${typeof param}`;\n      }\n    }\n  }\n\n  /**\n   * Computes the relative date based on a string.\n   * @param {String} text The string to interpret the date from.\n   * @param {Date} now The date the string is comparing against.\n   * @returns {Object} The relative date object.\n   **/\n  static relativeTimeToDate(text, now = new Date()) {\n    text = text.toLowerCase();\n    let parts = text.split(' ');\n\n    // Filter out whitespace\n    parts = parts.filter(part => part !== '');\n\n    const future = parts[0] === 'in';\n    const past = parts[parts.length - 1] === 'ago';\n\n    if (!future && !past && text !== 'now') {\n      return {\n        status: 'error',\n        info: \"Time should either start with 'in' or end with 'ago'\",\n      };\n    }\n\n    if (future && past) {\n      return {\n        status: 'error',\n        info: \"Time cannot have both 'in' and 'ago'\",\n      };\n    }\n\n    // strip the 'ago' or 'in'\n    if (future) {\n      parts = parts.slice(1);\n    } else {\n      // past\n      parts = parts.slice(0, parts.length - 1);\n    }\n\n    if (parts.length % 2 !== 0 && text !== 'now') {\n      return {\n        status: 'error',\n        info: 'Invalid time string. Dangling unit or number.',\n      };\n    }\n\n    const pairs = [];\n    while (parts.length) {\n      pairs.push([parts.shift(), parts.shift()]);\n    }\n\n    let seconds = 0;\n    for (const [num, interval] of pairs) {\n      const val = Number(num);\n      if (!Number.isInteger(val)) {\n        return {\n          status: 'error',\n          info: `'${num}' is not an integer.`,\n        };\n      }\n\n      switch (interval) {\n        case 'yr':\n        case 'yrs':\n        case 'year':\n        case 'years':\n          seconds += val * 31536000; // 365 * 24 * 60 * 60\n          break;\n\n        case 'wk':\n        case 'wks':\n        case 'week':\n        case 'weeks':\n          seconds += val * 604800; // 7 * 24 * 60 * 60\n          break;\n\n        case 'd':\n        case 'day':\n        case 'days':\n          seconds += val * 86400; // 24 * 60 * 60\n          break;\n\n        case 'hr':\n        case 'hrs':\n        case 'hour':\n        case 'hours':\n          seconds += val * 3600; // 60 * 60\n          break;\n\n        case 'min':\n        case 'mins':\n        case 'minute':\n        case 'minutes':\n          seconds += val * 60;\n          break;\n\n        case 'sec':\n        case 'secs':\n        case 'second':\n        case 'seconds':\n          seconds += val;\n          break;\n\n        default:\n          return {\n            status: 'error',\n            info: `Invalid interval: '${interval}'`,\n          };\n      }\n    }\n\n    const milliseconds = seconds * 1000;\n    if (future) {\n      return {\n        status: 'success',\n        info: 'future',\n        result: new Date(now.valueOf() + milliseconds),\n      };\n    } else if (past) {\n      return {\n        status: 'success',\n        info: 'past',\n        result: new Date(now.valueOf() - milliseconds),\n      };\n    } else {\n      return {\n        status: 'success',\n        info: 'present',\n        result: new Date(now.valueOf()),\n      };\n    }\n  }\n\n  /**\n   * Deep-scans an object for a matching key/value definition.\n   * @param {Object} obj The object to scan.\n   * @param {String | undefined} key The key to match, or undefined if only the value should be matched.\n   * @param {any | undefined} value The value to match, or undefined if only the key should be matched.\n   * @returns {Boolean} True if a match was found, false otherwise.\n   */\n  static objectContainsKeyValue(obj, key, value) {\n    const isMatch = (a, b) => (typeof a === 'string' && new RegExp(b).test(a)) || a === b;\n    const isKeyMatch = k => isMatch(k, key);\n    const isValueMatch = v => isMatch(v, value);\n    for (const [k, v] of Object.entries(obj)) {\n      if (key !== undefined && value === undefined && isKeyMatch(k)) {\n        return true;\n      } else if (key === undefined && value !== undefined && isValueMatch(v)) {\n        return true;\n      } else if (key !== undefined && value !== undefined && isKeyMatch(k) && isValueMatch(v)) {\n        return true;\n      }\n      if (['[object Object]', '[object Array]'].includes(Object.prototype.toString.call(v))) {\n        return Utils.objectContainsKeyValue(v, key, value);\n      }\n    }\n    return false;\n  }\n\n  static checkProhibitedKeywords(config, data) {\n    if (config?.requestKeywordDenylist) {\n      // Scan request data for denied keywords\n      for (const keyword of config.requestKeywordDenylist) {\n        const match = Utils.objectContainsKeyValue(data, keyword.key, keyword.value);\n        if (match) {\n          throw `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`;\n        }\n      }\n    }\n  }\n\n  /**\n   * Moves the nested keys of a specified key in an object to the root of the object.\n   *\n   * @param {Object} obj The object to modify.\n   * @param {String} key The key whose nested keys will be moved to root.\n   * @returns {Object} The modified object, or the original object if no modification happened.\n   * @example\n   * const obj = {\n   *   a: 1,\n   *   b: {\n   *     c: 2,\n   *     d: 3\n   *   },\n   *   e: 4\n   * };\n   * addNestedKeysToRoot(obj, 'b');\n   * console.log(obj);\n   * // Output: { a: 1, e: 4, c: 2, d: 3 }\n  */\n  static addNestedKeysToRoot(obj, key) {\n    if (obj[key] && typeof obj[key] === 'object') {\n      // Add nested keys to root\n      Object.assign(obj, { ...obj[key] });\n      // Delete original nested key\n      delete obj[key];\n    }\n    return obj;\n  }\n}\n\nmodule.exports = Utils;\n"],"mappings":"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","ignoreList":[]}