velocious 1.0.158 → 1.0.160

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -8,6 +8,7 @@ const DEFAULT_LOGGING_CONFIGURATION = {
8
8
  /** @type {LogLevel[]} */
9
9
  levels: ["info", "warn", "error"]
10
10
  };
11
+ /** @type {LogLevel[]} */
11
12
  const LEVEL_ORDER = ["debug-low-level", "debug", "info", "warn", "error"];
12
13
  /**
13
14
  * @param {string} message - Message text.
@@ -95,8 +96,24 @@ function messagesToMessage(...messages) {
95
96
  * @returns {Required<Pick<import("./configuration-types.js").LoggingConfiguration, "console" | "file" | "levels">> & Partial<Pick<import("./configuration-types.js").LoggingConfiguration, "filePath">>} - The logging configuration.
96
97
  */
97
98
  function resolveLoggingConfiguration(configuration) {
99
+ const debugEnabled = configuration?.debug === true;
98
100
  if (configuration && typeof configuration.getLoggingConfiguration === "function") {
99
- return configuration.getLoggingConfiguration();
101
+ const resolved = configuration.getLoggingConfiguration();
102
+ if (debugEnabled) {
103
+ return {
104
+ ...resolved,
105
+ console: true,
106
+ levels: LEVEL_ORDER
107
+ };
108
+ }
109
+ return resolved;
110
+ }
111
+ if (debugEnabled) {
112
+ return {
113
+ ...DEFAULT_LOGGING_CONFIGURATION,
114
+ console: true,
115
+ levels: LEVEL_ORDER
116
+ };
100
117
  }
101
118
  return DEFAULT_LOGGING_CONFIGURATION;
102
119
  }
@@ -298,4 +315,4 @@ export default async function logger(object, ...messages) {
298
315
  await Promise.all(writes);
299
316
  }
300
317
  }
301
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/logger.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Configuration from \"./configuration.js\"\nimport restArgsError from \"./utils/rest-args-error.js\"\n\n/** @typedef {\"debug-low-level\" | \"debug\" | \"info\" | \"warn\" | \"error\"} LogLevel */\n\nconst DEFAULT_LOGGING_CONFIGURATION = {\n  console: true,\n  file: false,\n  /** @type {LogLevel[]} */\n  levels: [\"info\", \"warn\", \"error\"]\n}\n\nconst LEVEL_ORDER = [\"debug-low-level\", \"debug\", \"info\", \"warn\", \"error\"]\n\n/**\n * @param {string} message - Message text.\n * @returns {Promise<void>} - Resolves when complete.\n */\nfunction consoleLog(message) {\n  return new Promise((resolve) => {\n    if (process.stdout) {\n      process.stdout.write(`${message}\\n`, \"utf8\", () => resolve())\n    } else {\n      console.log(message)\n      resolve()\n    }\n  })\n}\n\n/**\n * @param {string} message - Message text.\n * @returns {Promise<void>} - Resolves when complete.\n */\nfunction consoleError(message) {\n  return new Promise((resolve) => {\n    if (process.stderr) {\n      process.stderr.write(`${message}\\n`, \"utf8\", () => resolve())\n    } else {\n      console.error(message)\n      resolve()\n    }\n  })\n}\n\n/**\n * @param {string} message - Message text.\n * @returns {Promise<void>} - Resolves when complete.\n */\nfunction consoleWarn(message) {\n  return new Promise((resolve) => {\n    if (process.stderr) {\n      process.stderr.write(`${message}\\n`, \"utf8\", () => resolve())\n    } else {\n      console.warn(message)\n      resolve()\n    }\n  })\n}\n\n/**\n * @param {...any|function() : Array<any>} messages - Messages.\n * @returns {Array<any>} - Either the function result or the messages\n */\nfunction functionOrMessages(...messages) {\n  if (messages.length === 1 && typeof messages[0] == \"function\") {\n    const result = messages[0]()\n    messages = Array.isArray(result) ? result : [result]\n  }\n\n  return messages\n}\n\n/**\n * Converts multiple message parts into a single string.\n * @param {...any} messages - Parts to combine into a message\n * @returns {string} - The messages to message.\n */\nfunction messagesToMessage(...messages) {\n  let message = \"\"\n\n  for (const messagePartIndex in messages) {\n    const messagePart = messages[messagePartIndex]\n\n    if (Number(messagePartIndex) > 0) {\n      message += \" \"\n    }\n\n    if (typeof messagePart == \"object\") {\n      message += JSON.stringify(messagePart)\n    } else {\n      message += messagePart\n    }\n  }\n\n  return message\n}\n\n/**\n * @param {import(\"./configuration.js\").default | undefined} configuration - Configuration instance.\n * @returns {Required<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"console\" | \"file\" | \"levels\">> & Partial<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"filePath\">>} - The logging configuration.\n */\n/**\n * @param {import(\"./configuration.js\").default | undefined} configuration - Configuration instance.\n * @returns {Required<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"console\" | \"file\" | \"levels\">> & Partial<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"filePath\">>} - The logging configuration.\n */\nfunction resolveLoggingConfiguration(configuration) {\n  if (configuration && typeof configuration.getLoggingConfiguration === \"function\") {\n    return configuration.getLoggingConfiguration()\n  }\n\n  return DEFAULT_LOGGING_CONFIGURATION\n}\n\n/**\n * @param {object} args - Options object.\n * @param {LogLevel} args.level - Level.\n * @param {Required<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"console\" | \"file\" | \"levels\">> & Partial<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"filePath\">>} args.loggingConfiguration - Logging configuration.\n * @param {boolean} [args.debugFlag] - Whether debug flag.\n * @returns {boolean} - Whether level allowed.\n */\nfunction isLevelAllowed({level, loggingConfiguration, debugFlag}) {\n  const allowedLevels = loggingConfiguration.levels || DEFAULT_LOGGING_CONFIGURATION.levels\n\n  if (allowedLevels.includes(level)) return true\n\n  if (debugFlag && LEVEL_ORDER.indexOf(level) >= LEVEL_ORDER.indexOf(\"debug\")) return true\n\n  return false\n}\n\nclass Logger {\n  /**\n   * @param {string | object} object - Object.\n   * @param {object} args - Options object.\n   * @param {import(\"./configuration.js\").default} [args.configuration] - Configuration instance.\n   * @param {boolean} [args.debug] - Whether debug.\n   * @param {import(\"./configuration-types.js\").LoggingConfiguration} [args.loggingConfiguration] - Logging configuration.\n   */\n  constructor(object, {configuration, debug = false, loggingConfiguration, ...restArgs} = {}) {\n    restArgsError(restArgs)\n\n    this._debug = debug\n    this._configuration = configuration\n    this._loggingConfiguration = loggingConfiguration\n\n    if (typeof object == \"string\") {\n      this._subject = object\n    } else {\n      this._object = object\n      this._subject = object.constructor.name\n    }\n\n    if (!this._subject) {\n      throw new Error(`No subject given`)\n    }\n  }\n\n  /**\n   * @returns {import(\"./configuration.js\").default} - The configuration.\n   */\n  getConfiguration() {\n    if (!this._configuration) {\n      const objectWithConfig = /** @type {{configuration?: import(\"./configuration.js\").default}} */ (this._object)\n      this._configuration = objectWithConfig?.configuration || Configuration.current()\n    }\n\n    return this._configuration\n  }\n\n  /**\n   * @returns {import(\"./configuration.js\").default | undefined} - The safe configuration.\n   */\n  _safeConfiguration() {\n    try {\n      return this.getConfiguration()\n    } catch {\n      return undefined\n    }\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async debug(...messages) {\n    await this._write({level: \"debug\", messages})\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async info(...messages) {\n    await this._write({level: \"info\", messages})\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async debugLowLevel(...messages) {\n    await this._write({level: \"debug-low-level\", messages})\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async log(...messages) {\n    await this._write({level: \"info\", messages})\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async error(...messages) {\n    await this._write({level: \"error\", messages})\n  }\n\n  /**\n   * @param {boolean} newValue - New value.\n   * @returns {void} - No return value.\n   */\n  setDebug(newValue) {\n    this._debug = newValue\n  }\n\n  /**\n   * @type {(...args: Parameters<typeof functionOrMessages>) => Promise<void>}\n   */\n  async warn(...messages) {\n    await this._write({level: \"warn\", messages})\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {LogLevel} args.level - Level.\n   * @param {Parameters<typeof functionOrMessages>} args.messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _write({level, messages}) {\n    const resolvedMessages = functionOrMessages(...messages)\n    const message = messagesToMessage(this._subject, ...resolvedMessages)\n    const configuration = this._safeConfiguration()\n    const loggingConfiguration = /** @type {Required<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"console\" | \"file\" | \"levels\">> & Partial<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"filePath\">>} */ (\n      this._loggingConfiguration || resolveLoggingConfiguration(configuration)\n    )\n    const shouldLog = isLevelAllowed({level, loggingConfiguration, debugFlag: this._debug})\n\n    if (!shouldLog) return\n    const writes = []\n\n    if (loggingConfiguration.console !== false) {\n      if (level === \"error\") {\n        writes.push(consoleError(message))\n      } else if (level === \"warn\") {\n        writes.push(consoleWarn(message))\n      } else {\n        writes.push(consoleLog(message))\n      }\n    }\n\n    if (loggingConfiguration.file !== false && loggingConfiguration.filePath && configuration) {\n      const environmentHandler = configuration.getEnvironmentHandler?.()\n\n      if (environmentHandler?.writeLogToFile) {\n        writes.push(environmentHandler.writeLogToFile({\n          filePath: loggingConfiguration.filePath,\n          message\n        }))\n      }\n    }\n\n    if (writes.length === 1) {\n      await writes[0]\n    } else if (writes.length > 1) {\n      await Promise.all(writes)\n    }\n  }\n}\n\nexport {Logger}\n\n/**\n * @param {any} object - Object.\n * @param {...Parameters<typeof functionOrMessages>} messages - forwarded args\n */\n/**\n * @param {object} object - Log subject.\n * @param {...Parameters<typeof functionOrMessages>} messages - forwarded args\n */\nexport default async function logger(object, ...messages) {\n  const objectWithConfig = /** @type {{constructor: {name: string}, configuration?: import(\"./configuration.js\").default}} */ (object)\n  const className = objectWithConfig.constructor.name\n  let configuration = objectWithConfig.configuration\n\n  if (!configuration) {\n    try {\n      configuration = Configuration.current()\n    } catch {\n      // Ignore missing configuration\n    }\n  }\n\n  const loggingConfiguration = resolveLoggingConfiguration(configuration)\n  /** @type {LogLevel} */\n  const level = \"debug\"\n\n  if (!isLevelAllowed({level, loggingConfiguration, debugFlag: configuration?.debug || false})) return\n\n  const message = messagesToMessage(className, ...functionOrMessages(...messages))\n  const writes = []\n\n  if (loggingConfiguration.console !== false) {\n    writes.push(consoleLog(message))\n  }\n\n  if (loggingConfiguration.file !== false && loggingConfiguration.filePath && configuration) {\n    const environmentHandler = configuration.getEnvironmentHandler?.()\n\n    if (environmentHandler?.writeLogToFile) {\n      writes.push(environmentHandler.writeLogToFile({\n        filePath: loggingConfiguration.filePath,\n        message\n      }))\n    }\n  }\n\n  if (writes.length === 1) {\n    await writes[0]\n  } else if (writes.length > 1) {\n    await Promise.all(writes)\n  }\n}\n\n"]}
318
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/logger.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Configuration from \"./configuration.js\"\nimport restArgsError from \"./utils/rest-args-error.js\"\n\n/** @typedef {\"debug-low-level\" | \"debug\" | \"info\" | \"warn\" | \"error\"} LogLevel */\n\nconst DEFAULT_LOGGING_CONFIGURATION = {\n  console: true,\n  file: false,\n  /** @type {LogLevel[]} */\n  levels: [\"info\", \"warn\", \"error\"]\n}\n\n/** @type {LogLevel[]} */\nconst LEVEL_ORDER = [\"debug-low-level\", \"debug\", \"info\", \"warn\", \"error\"]\n\n/**\n * @param {string} message - Message text.\n * @returns {Promise<void>} - Resolves when complete.\n */\nfunction consoleLog(message) {\n  return new Promise((resolve) => {\n    if (process.stdout) {\n      process.stdout.write(`${message}\\n`, \"utf8\", () => resolve())\n    } else {\n      console.log(message)\n      resolve()\n    }\n  })\n}\n\n/**\n * @param {string} message - Message text.\n * @returns {Promise<void>} - Resolves when complete.\n */\nfunction consoleError(message) {\n  return new Promise((resolve) => {\n    if (process.stderr) {\n      process.stderr.write(`${message}\\n`, \"utf8\", () => resolve())\n    } else {\n      console.error(message)\n      resolve()\n    }\n  })\n}\n\n/**\n * @param {string} message - Message text.\n * @returns {Promise<void>} - Resolves when complete.\n */\nfunction consoleWarn(message) {\n  return new Promise((resolve) => {\n    if (process.stderr) {\n      process.stderr.write(`${message}\\n`, \"utf8\", () => resolve())\n    } else {\n      console.warn(message)\n      resolve()\n    }\n  })\n}\n\n/**\n * @param {...any|function() : Array<any>} messages - Messages.\n * @returns {Array<any>} - Either the function result or the messages\n */\nfunction functionOrMessages(...messages) {\n  if (messages.length === 1 && typeof messages[0] == \"function\") {\n    const result = messages[0]()\n    messages = Array.isArray(result) ? result : [result]\n  }\n\n  return messages\n}\n\n/**\n * Converts multiple message parts into a single string.\n * @param {...any} messages - Parts to combine into a message\n * @returns {string} - The messages to message.\n */\nfunction messagesToMessage(...messages) {\n  let message = \"\"\n\n  for (const messagePartIndex in messages) {\n    const messagePart = messages[messagePartIndex]\n\n    if (Number(messagePartIndex) > 0) {\n      message += \" \"\n    }\n\n    if (typeof messagePart == \"object\") {\n      message += JSON.stringify(messagePart)\n    } else {\n      message += messagePart\n    }\n  }\n\n  return message\n}\n\n/**\n * @param {import(\"./configuration.js\").default | undefined} configuration - Configuration instance.\n * @returns {Required<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"console\" | \"file\" | \"levels\">> & Partial<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"filePath\">>} - The logging configuration.\n */\n/**\n * @param {import(\"./configuration.js\").default | undefined} configuration - Configuration instance.\n * @returns {Required<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"console\" | \"file\" | \"levels\">> & Partial<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"filePath\">>} - The logging configuration.\n */\nfunction resolveLoggingConfiguration(configuration) {\n  const debugEnabled = configuration?.debug === true\n  if (configuration && typeof configuration.getLoggingConfiguration === \"function\") {\n    const resolved = configuration.getLoggingConfiguration()\n\n    if (debugEnabled) {\n      return {\n        ...resolved,\n        console: true,\n        levels: LEVEL_ORDER\n      }\n    }\n\n    return resolved\n  }\n\n  if (debugEnabled) {\n    return {\n      ...DEFAULT_LOGGING_CONFIGURATION,\n      console: true,\n      levels: LEVEL_ORDER\n    }\n  }\n\n  return DEFAULT_LOGGING_CONFIGURATION\n}\n\n/**\n * @param {object} args - Options object.\n * @param {LogLevel} args.level - Level.\n * @param {Required<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"console\" | \"file\" | \"levels\">> & Partial<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"filePath\">>} args.loggingConfiguration - Logging configuration.\n * @param {boolean} [args.debugFlag] - Whether debug flag.\n * @returns {boolean} - Whether level allowed.\n */\nfunction isLevelAllowed({level, loggingConfiguration, debugFlag}) {\n  const allowedLevels = loggingConfiguration.levels || DEFAULT_LOGGING_CONFIGURATION.levels\n\n  if (allowedLevels.includes(level)) return true\n\n  if (debugFlag && LEVEL_ORDER.indexOf(level) >= LEVEL_ORDER.indexOf(\"debug\")) return true\n\n  return false\n}\n\nclass Logger {\n  /**\n   * @param {string | object} object - Object.\n   * @param {object} args - Options object.\n   * @param {import(\"./configuration.js\").default} [args.configuration] - Configuration instance.\n   * @param {boolean} [args.debug] - Whether debug.\n   * @param {import(\"./configuration-types.js\").LoggingConfiguration} [args.loggingConfiguration] - Logging configuration.\n   */\n  constructor(object, {configuration, debug = false, loggingConfiguration, ...restArgs} = {}) {\n    restArgsError(restArgs)\n\n    this._debug = debug\n    this._configuration = configuration\n    this._loggingConfiguration = loggingConfiguration\n\n    if (typeof object == \"string\") {\n      this._subject = object\n    } else {\n      this._object = object\n      this._subject = object.constructor.name\n    }\n\n    if (!this._subject) {\n      throw new Error(`No subject given`)\n    }\n  }\n\n  /**\n   * @returns {import(\"./configuration.js\").default} - The configuration.\n   */\n  getConfiguration() {\n    if (!this._configuration) {\n      const objectWithConfig = /** @type {{configuration?: import(\"./configuration.js\").default}} */ (this._object)\n      this._configuration = objectWithConfig?.configuration || Configuration.current()\n    }\n\n    return this._configuration\n  }\n\n  /**\n   * @returns {import(\"./configuration.js\").default | undefined} - The safe configuration.\n   */\n  _safeConfiguration() {\n    try {\n      return this.getConfiguration()\n    } catch {\n      return undefined\n    }\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async debug(...messages) {\n    await this._write({level: \"debug\", messages})\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async info(...messages) {\n    await this._write({level: \"info\", messages})\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async debugLowLevel(...messages) {\n    await this._write({level: \"debug-low-level\", messages})\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async log(...messages) {\n    await this._write({level: \"info\", messages})\n  }\n\n  /**\n   * @param {any[]} messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async error(...messages) {\n    await this._write({level: \"error\", messages})\n  }\n\n  /**\n   * @param {boolean} newValue - New value.\n   * @returns {void} - No return value.\n   */\n  setDebug(newValue) {\n    this._debug = newValue\n  }\n\n  /**\n   * @type {(...args: Parameters<typeof functionOrMessages>) => Promise<void>}\n   */\n  async warn(...messages) {\n    await this._write({level: \"warn\", messages})\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {LogLevel} args.level - Level.\n   * @param {Parameters<typeof functionOrMessages>} args.messages - Messages.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _write({level, messages}) {\n    const resolvedMessages = functionOrMessages(...messages)\n    const message = messagesToMessage(this._subject, ...resolvedMessages)\n    const configuration = this._safeConfiguration()\n    const loggingConfiguration = /** @type {Required<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"console\" | \"file\" | \"levels\">> & Partial<Pick<import(\"./configuration-types.js\").LoggingConfiguration, \"filePath\">>} */ (\n      this._loggingConfiguration || resolveLoggingConfiguration(configuration)\n    )\n    const shouldLog = isLevelAllowed({level, loggingConfiguration, debugFlag: this._debug})\n\n    if (!shouldLog) return\n    const writes = []\n\n    if (loggingConfiguration.console !== false) {\n      if (level === \"error\") {\n        writes.push(consoleError(message))\n      } else if (level === \"warn\") {\n        writes.push(consoleWarn(message))\n      } else {\n        writes.push(consoleLog(message))\n      }\n    }\n\n    if (loggingConfiguration.file !== false && loggingConfiguration.filePath && configuration) {\n      const environmentHandler = configuration.getEnvironmentHandler?.()\n\n      if (environmentHandler?.writeLogToFile) {\n        writes.push(environmentHandler.writeLogToFile({\n          filePath: loggingConfiguration.filePath,\n          message\n        }))\n      }\n    }\n\n    if (writes.length === 1) {\n      await writes[0]\n    } else if (writes.length > 1) {\n      await Promise.all(writes)\n    }\n  }\n}\n\nexport {Logger}\n\n/**\n * @param {any} object - Object.\n * @param {...Parameters<typeof functionOrMessages>} messages - forwarded args\n */\n/**\n * @param {object} object - Log subject.\n * @param {...Parameters<typeof functionOrMessages>} messages - forwarded args\n */\nexport default async function logger(object, ...messages) {\n  const objectWithConfig = /** @type {{constructor: {name: string}, configuration?: import(\"./configuration.js\").default}} */ (object)\n  const className = objectWithConfig.constructor.name\n  let configuration = objectWithConfig.configuration\n\n  if (!configuration) {\n    try {\n      configuration = Configuration.current()\n    } catch {\n      // Ignore missing configuration\n    }\n  }\n\n  const loggingConfiguration = resolveLoggingConfiguration(configuration)\n  /** @type {LogLevel} */\n  const level = \"debug\"\n\n  if (!isLevelAllowed({level, loggingConfiguration, debugFlag: configuration?.debug || false})) return\n\n  const message = messagesToMessage(className, ...functionOrMessages(...messages))\n  const writes = []\n\n  if (loggingConfiguration.console !== false) {\n    writes.push(consoleLog(message))\n  }\n\n  if (loggingConfiguration.file !== false && loggingConfiguration.filePath && configuration) {\n    const environmentHandler = configuration.getEnvironmentHandler?.()\n\n    if (environmentHandler?.writeLogToFile) {\n      writes.push(environmentHandler.writeLogToFile({\n        filePath: loggingConfiguration.filePath,\n        message\n      }))\n    }\n  }\n\n  if (writes.length === 1) {\n    await writes[0]\n  } else if (writes.length > 1) {\n    await Promise.all(writes)\n  }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"resolver.d.ts","sourceRoot":"","sources":["../../../src/routes/resolver.js"],"names":[],"mappings":"AAUA;IAIE;;;;;OAKG;IACH,kDAJG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACmF,OAAO,EAA/H,OAAO,kCAAkC,EAAE,OAAO,GAAG,OAAO,4CAA4C,EAAE,OAAO;QACvD,QAAQ,EAAlE,OAAO,mCAAmC,EAAE,OAAO;KAC7D,EAUA;IAlBD,iCAAiC;IACjC,QADW,MAAM,GAAG,SAAS,CACvB;IAaJ,qDAAkC;IAClC,YAA8B;IAC9B,2HAAsB;IACtB,8DAAwB;IAG1B,yBA+EC;IAED;;;;OAIG;IACH,2BAJW,OAAO,iBAAiB,EAAE,OAAO,QACjC,MAAM,GACJ;QAAC,QAAQ,EAAE,MAAM,CAAA;KAAC,GAAG,SAAS,CAsB1C;IAED;;;;;OAKG;IACH,6CAJG;QAAqB,MAAM,EAAnB,MAAM;QAC0C,eAAe,EAA/D,cAAc,kBAAkB,EAAE,OAAO;KACjD,GAAU,OAAO,CAAC,IAAI,CAAC,CAczB;IAED;;;OAGG;IACH,uBAHW,IAAI,GACF,MAAM,CAqBlB;IAED;;;OAGG;IACH,iCAHW,GAAG,GACD,GAAG,CA2Bf;CACF;uBAhNoB,cAAc"}
1
+ {"version":3,"file":"resolver.d.ts","sourceRoot":"","sources":["../../../src/routes/resolver.js"],"names":[],"mappings":"AAUA;IAIE;;;;;OAKG;IACH,kDAJG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACmF,OAAO,EAA/H,OAAO,kCAAkC,EAAE,OAAO,GAAG,OAAO,4CAA4C,EAAE,OAAO;QACvD,QAAQ,EAAlE,OAAO,mCAAmC,EAAE,OAAO;KAC7D,EAUA;IAlBD,iCAAiC;IACjC,QADW,MAAM,GAAG,SAAS,CACvB;IAaJ,qDAAkC;IAClC,YAA8B;IAC9B,2HAAsB;IACtB,8DAAwB;IAG1B,yBA+EC;IAED;;;;OAIG;IACH,2BAJW,OAAO,iBAAiB,EAAE,OAAO,QACjC,MAAM,GACJ;QAAC,QAAQ,EAAE,MAAM,CAAA;KAAC,GAAG,SAAS,CAsB1C;IAED;;;;;OAKG;IACH,6CAJG;QAAqB,MAAM,EAAnB,MAAM;QAC0C,eAAe,EAA/D,cAAc,kBAAkB,EAAE,OAAO;KACjD,GAAU,OAAO,CAAC,IAAI,CAAC,CAezB;IAED;;;OAGG;IACH,uBAHW,IAAI,GACF,MAAM,CAqBlB;IAED;;;OAGG;IACH,iCAHW,GAAG,GACD,GAAG,CA2Bf;CACF;uBAjNoB,cAAc"}
@@ -131,11 +131,12 @@ export default class VelociousRoutesResolver {
131
131
  const timestamp = this._formatTimestamp(new Date());
132
132
  const remoteAddress = request.remoteAddress?.() || request.header("x-forwarded-for") || "unknown";
133
133
  const loggedParams = /** @type {Record<string, unknown>} */ (this._sanitizeParamsForLogging(this.params));
134
+ const logMethod = this.configuration.getEnvironment() === "test" ? "debug" : "info";
134
135
  delete loggedParams.action;
135
136
  delete loggedParams.controller;
136
- await this.logger.info(() => `Started ${request.httpMethod()} "${request.path()}" for ${remoteAddress} at ${timestamp}`);
137
- await this.logger.info(() => `Processing by ${controllerClass.name}#${action}`);
138
- await this.logger.info(() => [` Parameters:`, loggedParams]);
137
+ await this.logger[logMethod](() => `Started ${request.httpMethod()} "${request.path()}" for ${remoteAddress} at ${timestamp}`);
138
+ await this.logger[logMethod](() => `Processing by ${controllerClass.name}#${action}`);
139
+ await this.logger[logMethod](() => [` Parameters:`, loggedParams]);
139
140
  }
140
141
  /**
141
142
  * @param {Date} date - Date value.
@@ -186,4 +187,4 @@ export default class VelociousRoutesResolver {
186
187
  return value;
187
188
  }
188
189
  }
189
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resolver.js","sourceRoot":"","sources":["../../../src/routes/resolver.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {dirname} from \"path\"\nimport {fileURLToPath} from \"url\"\nimport fs from \"fs/promises\"\nimport * as inflection from \"inflection\"\nimport {Logger} from \"../logger.js\"\nimport UploadedFile from \"../http-server/client/uploaded-file/uploaded-file.js\"\nimport ensureError from \"../utils/ensure-error.js\"\n\nexport default class VelociousRoutesResolver {\n  /** @type {Logger | undefined} */\n  logger\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {import(\"../http-server/client/request.js\").default | import(\"../http-server/client/websocket-request.js\").default} args.request - Request object.\n   * @param {import(\"../http-server/client/response.js\").default} args.response - Response object.\n   */\n  constructor({configuration, request, response}) {\n    if (!configuration) throw new Error(\"No configuration given\")\n    if (!request) throw new Error(\"No request given\")\n    if (!response) throw new Error(\"No response given\")\n\n    this.configuration = configuration\n    this.params = request.params()\n    this.request = request\n    this.response = response\n  }\n\n  async resolve() {\n    let controllerPath\n    let currentRoute = this.configuration.routes.rootRoute\n    const rawPath = this.request.path()\n    const currentPath = rawPath.split(\"?\")[0]\n    let viewPath\n\n    const matchResult = this.matchPathWithRoutes(currentRoute, currentPath)\n    const actionParam = this.params.action\n    const controllerParam = this.params.controller\n    const actionValue = typeof actionParam == \"string\" ? actionParam : (Array.isArray(actionParam) ? actionParam[0] : undefined)\n    let action = typeof actionValue == \"string\" ? inflection.camelize(actionValue.replaceAll(\"-\", \"_\"), true) : undefined\n    let controller = typeof controllerParam == \"string\" ? controllerParam : (Array.isArray(controllerParam) ? controllerParam[0] : undefined)\n\n    if (!matchResult) {\n      const __filename = fileURLToPath(import.meta.url)\n      const __dirname = dirname(__filename)\n      const requestedPath = currentPath.replace(/^\\//, \"\") || \"_root\"\n      const attemptedControllerPath = `${this.configuration.getDirectory()}/src/routes/${requestedPath}/controller.js`\n\n      await (this.logger || new Logger(\"RoutesResolver\", {configuration: this.configuration})).warn(`No route matched for ${rawPath}. Tried controller at ${attemptedControllerPath}`)\n\n      controller = \"errors\"\n      controllerPath = \"./built-in/errors/controller.js\"\n      action = \"notFound\"\n      viewPath = await fs.realpath(`${__dirname}/built-in/errors`)\n    } else if (action) {\n      if (!controller) controller = \"_root\"\n\n      controllerPath = `${this.configuration.getDirectory()}/src/routes/${controller}/controller.js`\n      viewPath = `${this.configuration.getDirectory()}/src/routes/${controller}`\n    } else {\n      throw new Error(`Matched the route but didn't know what to do with it: ${rawPath} (action: ${action}, controller: ${controller}, params: ${JSON.stringify(this.params)})`)\n    }\n\n    const controllerClassImport = await import(controllerPath)\n    const controllerClass = controllerClassImport.default\n    const controllerInstance = new controllerClass({\n      action,\n      configuration: this.configuration,\n      controller,\n      params: this.params,\n      request: this.request,\n      response: this.response,\n      viewPath\n    })\n\n    if (!(action in controllerInstance)) {\n      throw new Error(`Missing action on controller: ${controller}#${action}`)\n    }\n\n    this.logger ||= new Logger(controllerClass.name, {configuration: this.configuration})\n\n    await this._logActionStart({action, controllerClass})\n\n    try {\n      await this.configuration.ensureConnections(async () => {\n        await controllerInstance._runBeforeCallbacks()\n        await controllerInstance[action]()\n      })\n    } catch (error) {\n      const ensuredError = ensureError(error)\n      const errorContext = {\n        action,\n        controller,\n        httpMethod: this.request.httpMethod(),\n        path: this.request.path(),\n        stage: \"controller-action\"\n      }\n\n      const errorWithContext = /** @type {{velociousContext?: object}} */ (ensuredError)\n\n      errorWithContext.velociousContext = {\n        ...(errorWithContext.velociousContext || {}),\n        controllerAction: errorContext\n      }\n\n      throw ensuredError\n    }\n  }\n\n  /**\n   * @param {import(\"./base-route.js\").default} route - Route.\n   * @param {string} path - Path.\n   * @returns {{restPath: string} | undefined} - REST path metadata for this route.\n   */\n  matchPathWithRoutes(route, path) {\n    const pathWithoutSlash = path.replace(/^\\//, \"\").split(\"?\")[0]\n\n    for (const subRoute of route.routes) {\n      const matchResult = subRoute.matchWithPath({\n        params: this.params,\n        path: pathWithoutSlash,\n        request: this.request\n      })\n\n      if (!matchResult) continue\n\n      const {restPath} = matchResult\n\n      if (restPath) {\n        return this.matchPathWithRoutes(subRoute, restPath)\n      }\n\n      return matchResult\n    }\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.action - Action.\n   * @param {typeof import(\"../controller.js\").default} args.controllerClass - Controller class.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _logActionStart({action, controllerClass}) {\n    const request = this.request\n    const timestamp = this._formatTimestamp(new Date())\n    const remoteAddress = request.remoteAddress?.() || request.header(\"x-forwarded-for\") || \"unknown\"\n    const loggedParams = /** @type {Record<string, unknown>} */ (this._sanitizeParamsForLogging(this.params))\n\n    delete loggedParams.action\n    delete loggedParams.controller\n\n    await this.logger.info(() => `Started ${request.httpMethod()} \"${request.path()}\" for ${remoteAddress} at ${timestamp}`)\n    await this.logger.info(() => `Processing by ${controllerClass.name}#${action}`)\n    await this.logger.info(() => [`  Parameters:`, loggedParams])\n  }\n\n  /**\n   * @param {Date} date - Date value.\n   * @returns {string} - The timestamp.\n   */\n  _formatTimestamp(date) {\n    /**\n     * @param {number} num - Num.\n     * @returns {string} - The pad.\n     */\n    const pad = (num) => String(num).padStart(2, \"0\")\n    const year = date.getFullYear()\n    const month = pad(date.getMonth() + 1)\n    const day = pad(date.getDate())\n    const hours = pad(date.getHours())\n    const minutes = pad(date.getMinutes())\n    const seconds = pad(date.getSeconds())\n    const offsetMinutes = date.getTimezoneOffset()\n    const offsetSign = offsetMinutes > 0 ? \"-\" : \"+\"\n    const offsetTotalMinutes = Math.abs(offsetMinutes)\n    const offsetHours = pad(Math.floor(offsetTotalMinutes / 60))\n    const offsetRemainingMinutes = pad(offsetTotalMinutes % 60)\n\n    return `${year}-${month}-${day} ${hours}:${minutes}:${seconds} ${offsetSign}${offsetHours}${offsetRemainingMinutes}`\n  }\n\n  /**\n   * @param {any} value - Value to use.\n   * @returns {any} - The sanitize params for logging.\n   */\n  _sanitizeParamsForLogging(value) {\n    if (value instanceof UploadedFile) {\n      return {\n        className: value.constructor.name,\n        filename: value.filename(),\n        size: value.size()\n      }\n    }\n\n    if (Array.isArray(value)) {\n      return value.map((item) => this._sanitizeParamsForLogging(item))\n    }\n\n    if (value && typeof value === \"object\") {\n      /** @type {Record<string, any>} */\n      const result = {}\n\n      for (const key of Object.keys(value)) {\n        result[key] = this._sanitizeParamsForLogging(value[key])\n      }\n\n      return result\n    }\n\n    return value\n  }\n}\n\n"]}
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resolver.js","sourceRoot":"","sources":["../../../src/routes/resolver.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {dirname} from \"path\"\nimport {fileURLToPath} from \"url\"\nimport fs from \"fs/promises\"\nimport * as inflection from \"inflection\"\nimport {Logger} from \"../logger.js\"\nimport UploadedFile from \"../http-server/client/uploaded-file/uploaded-file.js\"\nimport ensureError from \"../utils/ensure-error.js\"\n\nexport default class VelociousRoutesResolver {\n  /** @type {Logger | undefined} */\n  logger\n\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {import(\"../http-server/client/request.js\").default | import(\"../http-server/client/websocket-request.js\").default} args.request - Request object.\n   * @param {import(\"../http-server/client/response.js\").default} args.response - Response object.\n   */\n  constructor({configuration, request, response}) {\n    if (!configuration) throw new Error(\"No configuration given\")\n    if (!request) throw new Error(\"No request given\")\n    if (!response) throw new Error(\"No response given\")\n\n    this.configuration = configuration\n    this.params = request.params()\n    this.request = request\n    this.response = response\n  }\n\n  async resolve() {\n    let controllerPath\n    let currentRoute = this.configuration.routes.rootRoute\n    const rawPath = this.request.path()\n    const currentPath = rawPath.split(\"?\")[0]\n    let viewPath\n\n    const matchResult = this.matchPathWithRoutes(currentRoute, currentPath)\n    const actionParam = this.params.action\n    const controllerParam = this.params.controller\n    const actionValue = typeof actionParam == \"string\" ? actionParam : (Array.isArray(actionParam) ? actionParam[0] : undefined)\n    let action = typeof actionValue == \"string\" ? inflection.camelize(actionValue.replaceAll(\"-\", \"_\"), true) : undefined\n    let controller = typeof controllerParam == \"string\" ? controllerParam : (Array.isArray(controllerParam) ? controllerParam[0] : undefined)\n\n    if (!matchResult) {\n      const __filename = fileURLToPath(import.meta.url)\n      const __dirname = dirname(__filename)\n      const requestedPath = currentPath.replace(/^\\//, \"\") || \"_root\"\n      const attemptedControllerPath = `${this.configuration.getDirectory()}/src/routes/${requestedPath}/controller.js`\n\n      await (this.logger || new Logger(\"RoutesResolver\", {configuration: this.configuration})).warn(`No route matched for ${rawPath}. Tried controller at ${attemptedControllerPath}`)\n\n      controller = \"errors\"\n      controllerPath = \"./built-in/errors/controller.js\"\n      action = \"notFound\"\n      viewPath = await fs.realpath(`${__dirname}/built-in/errors`)\n    } else if (action) {\n      if (!controller) controller = \"_root\"\n\n      controllerPath = `${this.configuration.getDirectory()}/src/routes/${controller}/controller.js`\n      viewPath = `${this.configuration.getDirectory()}/src/routes/${controller}`\n    } else {\n      throw new Error(`Matched the route but didn't know what to do with it: ${rawPath} (action: ${action}, controller: ${controller}, params: ${JSON.stringify(this.params)})`)\n    }\n\n    const controllerClassImport = await import(controllerPath)\n    const controllerClass = controllerClassImport.default\n    const controllerInstance = new controllerClass({\n      action,\n      configuration: this.configuration,\n      controller,\n      params: this.params,\n      request: this.request,\n      response: this.response,\n      viewPath\n    })\n\n    if (!(action in controllerInstance)) {\n      throw new Error(`Missing action on controller: ${controller}#${action}`)\n    }\n\n    this.logger ||= new Logger(controllerClass.name, {configuration: this.configuration})\n\n    await this._logActionStart({action, controllerClass})\n\n    try {\n      await this.configuration.ensureConnections(async () => {\n        await controllerInstance._runBeforeCallbacks()\n        await controllerInstance[action]()\n      })\n    } catch (error) {\n      const ensuredError = ensureError(error)\n      const errorContext = {\n        action,\n        controller,\n        httpMethod: this.request.httpMethod(),\n        path: this.request.path(),\n        stage: \"controller-action\"\n      }\n\n      const errorWithContext = /** @type {{velociousContext?: object}} */ (ensuredError)\n\n      errorWithContext.velociousContext = {\n        ...(errorWithContext.velociousContext || {}),\n        controllerAction: errorContext\n      }\n\n      throw ensuredError\n    }\n  }\n\n  /**\n   * @param {import(\"./base-route.js\").default} route - Route.\n   * @param {string} path - Path.\n   * @returns {{restPath: string} | undefined} - REST path metadata for this route.\n   */\n  matchPathWithRoutes(route, path) {\n    const pathWithoutSlash = path.replace(/^\\//, \"\").split(\"?\")[0]\n\n    for (const subRoute of route.routes) {\n      const matchResult = subRoute.matchWithPath({\n        params: this.params,\n        path: pathWithoutSlash,\n        request: this.request\n      })\n\n      if (!matchResult) continue\n\n      const {restPath} = matchResult\n\n      if (restPath) {\n        return this.matchPathWithRoutes(subRoute, restPath)\n      }\n\n      return matchResult\n    }\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.action - Action.\n   * @param {typeof import(\"../controller.js\").default} args.controllerClass - Controller class.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _logActionStart({action, controllerClass}) {\n    const request = this.request\n    const timestamp = this._formatTimestamp(new Date())\n    const remoteAddress = request.remoteAddress?.() || request.header(\"x-forwarded-for\") || \"unknown\"\n    const loggedParams = /** @type {Record<string, unknown>} */ (this._sanitizeParamsForLogging(this.params))\n    const logMethod = this.configuration.getEnvironment() === \"test\" ? \"debug\" : \"info\"\n\n    delete loggedParams.action\n    delete loggedParams.controller\n\n    await this.logger[logMethod](() => `Started ${request.httpMethod()} \"${request.path()}\" for ${remoteAddress} at ${timestamp}`)\n    await this.logger[logMethod](() => `Processing by ${controllerClass.name}#${action}`)\n    await this.logger[logMethod](() => [`  Parameters:`, loggedParams])\n  }\n\n  /**\n   * @param {Date} date - Date value.\n   * @returns {string} - The timestamp.\n   */\n  _formatTimestamp(date) {\n    /**\n     * @param {number} num - Num.\n     * @returns {string} - The pad.\n     */\n    const pad = (num) => String(num).padStart(2, \"0\")\n    const year = date.getFullYear()\n    const month = pad(date.getMonth() + 1)\n    const day = pad(date.getDate())\n    const hours = pad(date.getHours())\n    const minutes = pad(date.getMinutes())\n    const seconds = pad(date.getSeconds())\n    const offsetMinutes = date.getTimezoneOffset()\n    const offsetSign = offsetMinutes > 0 ? \"-\" : \"+\"\n    const offsetTotalMinutes = Math.abs(offsetMinutes)\n    const offsetHours = pad(Math.floor(offsetTotalMinutes / 60))\n    const offsetRemainingMinutes = pad(offsetTotalMinutes % 60)\n\n    return `${year}-${month}-${day} ${hours}:${minutes}:${seconds} ${offsetSign}${offsetHours}${offsetRemainingMinutes}`\n  }\n\n  /**\n   * @param {any} value - Value to use.\n   * @returns {any} - The sanitize params for logging.\n   */\n  _sanitizeParamsForLogging(value) {\n    if (value instanceof UploadedFile) {\n      return {\n        className: value.constructor.name,\n        filename: value.filename(),\n        size: value.size()\n      }\n    }\n\n    if (Array.isArray(value)) {\n      return value.map((item) => this._sanitizeParamsForLogging(item))\n    }\n\n    if (value && typeof value === \"object\") {\n      /** @type {Record<string, any>} */\n      const result = {}\n\n      for (const key of Object.keys(value)) {\n        result[key] = this._sanitizeParamsForLogging(value[key])\n      }\n\n      return result\n    }\n\n    return value\n  }\n}\n"]}
@@ -26,6 +26,7 @@ export default class TestFilesFinder {
26
26
  directoryArgs: string[];
27
27
  /** @type {string[]} */
28
28
  fileArgs: string[];
29
+ _argsPrepared: boolean;
29
30
  /**
30
31
  * @returns {Promise<string[]>} - Resolves with the test files.
31
32
  */
@@ -60,6 +61,20 @@ export default class TestFilesFinder {
60
61
  * @returns {boolean} - Whether looks like test file.
61
62
  */
62
63
  looksLikeTestFile(file: string): boolean;
64
+ /**
65
+ * @returns {Promise<void>} - Resolves when test args are prepared.
66
+ */
67
+ prepareArgs(): Promise<void>;
68
+ /**
69
+ * @param {string} localPath - Local path.
70
+ * @returns {string} - Normalized local path with trailing slash.
71
+ */
72
+ ensureTrailingSlash(localPath: string): string;
73
+ /**
74
+ * @param {string} fullPath - Full path.
75
+ * @returns {string} - Local path relative to the base directory.
76
+ */
77
+ toLocalPath(fullPath: string): string;
63
78
  }
64
79
  import { Logger } from "../logger.js";
65
80
  //# sourceMappingURL=test-files-finder.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"test-files-finder.d.ts","sourceRoot":"","sources":["../../../src/testing/test-files-finder.js"],"names":[],"mappings":"AASA;IACE,+BAA+C;IAE/C;;;;;OAKG;IACH,kEAJG;QAAqB,SAAS,EAAtB,MAAM;QACU,WAAW,GAA3B,MAAM,EAAE;QACO,WAAW,EAA1B,MAAM,EAAE;KAClB,EA0CA;IAtCC,kBAA0B;IAC1B,eAA8B;IAG5B,sBAA8B;IAShC,qBAAqB;IACrB,sBAA8B;IAE9B,uBAAuB;IACvB,YADW,MAAM,EAAE,CACC;IAEpB,4CAA4C;IAC5C,iBADW,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC,CACf;IAEzB,uBAAuB;IACvB,UADW,MAAM,EAAE,CACuD;IAE1E,uBAAuB;IACvB,eADW,MAAM,EAAE,CACI;IAEvB,uBAAuB;IACvB,UADW,MAAM,EAAE,CACD;IAWpB;;OAEG;IACH,iBAFa,OAAO,CAAC,MAAM,EAAE,CAAC,CAc7B;IAED;;OAEG;IACH,yBAFa,MAAM,CAEwD;IAE3E,wCAIC;IAED;;OAEG;IACH,mCAFa,OAAO,CAAC,IAAI,CAAC,CAYzB;IAED;;;OAGG;IACH,2BAHW,MAAa,OAAO,CAAC,IAAI,CAAC,GACxB,OAAO,CAAC,IAAI,CAAC,CAgBzB;IAED;;;OAGG;IACH,wBAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAwBzB;IAED;;;;OAIG;IACH,iCAJW,MAAM,aACN,MAAM,GACJ,OAAO,CA2BnB;IAED;;;OAGG;IACH,wBAHW,MAAM,GACJ,OAAO,CAInB;CACF;uBA1LoB,cAAc"}
1
+ {"version":3,"file":"test-files-finder.d.ts","sourceRoot":"","sources":["../../../src/testing/test-files-finder.js"],"names":[],"mappings":"AAUA;IACE,+BAA+C;IAE/C;;;;;OAKG;IACH,kEAJG;QAAqB,SAAS,EAAtB,MAAM;QACU,WAAW,GAA3B,MAAM,EAAE;QACO,WAAW,EAA1B,MAAM,EAAE;KAClB,EAoCA;IAhCC,kBAAwC;IACxC,eAA8B;IAG5B,sBAAkE;IASpE,qBAAqB;IACrB,sBAA8B;IAE9B,uBAAuB;IACvB,YADW,MAAM,EAAE,CACC;IAEpB,4CAA4C;IAC5C,iBADW,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC,CACf;IAEzB,uBAAuB;IACvB,UADW,MAAM,EAAE,CACuD;IAE1E,uBAAuB;IACvB,eADW,MAAM,EAAE,CACI;IAEvB,uBAAuB;IACvB,UADW,MAAM,EAAE,CACD;IAElB,uBAA0B;IAG5B;;OAEG;IACH,iBAFa,OAAO,CAAC,MAAM,EAAE,CAAC,CAgB7B;IAED;;OAEG;IACH,yBAFa,MAAM,CAEwD;IAE3E,wCAIC;IAED;;OAEG;IACH,mCAFa,OAAO,CAAC,IAAI,CAAC,CAYzB;IAED;;;OAGG;IACH,2BAHW,MAAa,OAAO,CAAC,IAAI,CAAC,GACxB,OAAO,CAAC,IAAI,CAAC,CAgBzB;IAED;;;OAGG;IACH,wBAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAwBzB;IAED;;;;OAIG;IACH,iCAJW,MAAM,aACN,MAAM,GACJ,OAAO,CA2BnB;IAED;;;OAGG;IACH,wBAHW,MAAM,GACJ,OAAO,CAInB;IAED;;OAEG;IACH,eAFa,OAAO,CAAC,IAAI,CAAC,CAgDzB;IAED;;;OAGG;IACH,+BAHW,MAAM,GACJ,MAAM,CAKlB;IAED;;;OAGG;IACH,sBAHW,MAAM,GACJ,MAAM,CAKlB;CACF;uBA3PoB,cAAc"}
@@ -1,5 +1,6 @@
1
1
  // @ts-check
2
2
  import fs from "fs/promises";
3
+ import path from "path";
3
4
  import fileExists from "../utils/file-exists.js";
4
5
  import { Logger } from "../logger.js";
5
6
  import restArgsError from "../utils/rest-args-error.js";
@@ -14,10 +15,10 @@ export default class TestFilesFinder {
14
15
  */
15
16
  constructor({ directory, directories, processArgs, ...restArgs }) {
16
17
  restArgsError(restArgs);
17
- this.directory = directory;
18
+ this.directory = path.resolve(directory);
18
19
  this.logger = new Logger(this);
19
20
  if (directories) {
20
- this.directories = directories;
21
+ this.directories = directories.map((entry) => path.resolve(entry));
21
22
  }
22
23
  else {
23
24
  this.directories = [
@@ -38,19 +39,13 @@ export default class TestFilesFinder {
38
39
  this.directoryArgs = [];
39
40
  /** @type {string[]} */
40
41
  this.fileArgs = [];
41
- for (const testArg of this.testArgs) {
42
- if (testArg.endsWith("/")) {
43
- this.directoryArgs.push(testArg);
44
- }
45
- else {
46
- this.fileArgs.push(testArg);
47
- }
48
- }
42
+ this._argsPrepared = false;
49
43
  }
50
44
  /**
51
45
  * @returns {Promise<string[]>} - Resolves with the test files.
52
46
  */
53
47
  async findTestFiles() {
48
+ await this.prepareArgs();
54
49
  await this.withFindingCount(async () => {
55
50
  for (const directory of this.directories) {
56
51
  if (await fileExists(directory)) {
@@ -157,5 +152,70 @@ export default class TestFilesFinder {
157
152
  looksLikeTestFile(file) {
158
153
  return Boolean(file.match(/-(spec|test)\.(m|)js$/));
159
154
  }
155
+ /**
156
+ * @returns {Promise<void>} - Resolves when test args are prepared.
157
+ */
158
+ async prepareArgs() {
159
+ if (this._argsPrepared)
160
+ return;
161
+ for (const testArg of this.testArgs) {
162
+ const forceDirectory = testArg.endsWith("/") || testArg.endsWith(path.sep);
163
+ const fullPath = path.isAbsolute(testArg) ? testArg : path.resolve(this.directory, testArg);
164
+ const baseName = path.basename(this.directory);
165
+ const hasBasePrefix = testArg === baseName || testArg.startsWith(`${baseName}/`) || testArg.startsWith(`${baseName}${path.sep}`);
166
+ const basePrefixedFullPath = (!path.isAbsolute(testArg) && hasBasePrefix) ? path.resolve(path.dirname(this.directory), testArg) : null;
167
+ const fullPathCandidates = basePrefixedFullPath ? [basePrefixedFullPath] : [fullPath];
168
+ if (forceDirectory) {
169
+ const preferredLocalPath = this.toLocalPath(basePrefixedFullPath || fullPath);
170
+ this.directoryArgs.push(this.ensureTrailingSlash(preferredLocalPath));
171
+ continue;
172
+ }
173
+ try {
174
+ let stats;
175
+ let resolvedFullPath;
176
+ for (const candidatePath of fullPathCandidates) {
177
+ try {
178
+ stats = await fs.stat(candidatePath);
179
+ resolvedFullPath = candidatePath;
180
+ break;
181
+ }
182
+ catch {
183
+ // Keep searching
184
+ }
185
+ }
186
+ if (!stats || !resolvedFullPath)
187
+ throw new Error("Path not found");
188
+ const localPath = this.toLocalPath(resolvedFullPath);
189
+ if (stats.isDirectory()) {
190
+ this.directoryArgs.push(this.ensureTrailingSlash(localPath));
191
+ }
192
+ else if (stats.isFile()) {
193
+ this.fileArgs.push(localPath);
194
+ }
195
+ }
196
+ catch {
197
+ const fallbackLocalPath = this.toLocalPath(basePrefixedFullPath || fullPath);
198
+ this.fileArgs.push(fallbackLocalPath);
199
+ }
200
+ }
201
+ this._argsPrepared = true;
202
+ }
203
+ /**
204
+ * @param {string} localPath - Local path.
205
+ * @returns {string} - Normalized local path with trailing slash.
206
+ */
207
+ ensureTrailingSlash(localPath) {
208
+ if (localPath === "")
209
+ return localPath;
210
+ return localPath.endsWith("/") ? localPath : `${localPath}/`;
211
+ }
212
+ /**
213
+ * @param {string} fullPath - Full path.
214
+ * @returns {string} - Local path relative to the base directory.
215
+ */
216
+ toLocalPath(fullPath) {
217
+ const relativePath = path.relative(this.directory, fullPath);
218
+ return relativePath.split(path.sep).join("/");
219
+ }
160
220
  }
161
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-files-finder.js","sourceRoot":"","sources":["../../../src/testing/test-files-finder.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport fs from \"fs/promises\"\n\nimport fileExists from \"../utils/file-exists.js\"\nimport {Logger} from \"../logger.js\"\nimport restArgsError from \"../utils/rest-args-error.js\"\n\n// Incredibly complex class to find files in multiple simultanious running promises to do it as fast as possible.\nexport default class TestFilesFinder {\n  static IGNORED_NAMES = [\".git\", \"node_modules\"]\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.directory - Directory path.\n   * @param {string[]} [args.directories] - Directories.\n   * @param {string[]} args.processArgs - Process args.\n   */\n  constructor({directory, directories, processArgs, ...restArgs}) {\n    restArgsError(restArgs)\n\n    this.directory = directory\n    this.logger = new Logger(this)\n\n    if (directories) {\n      this.directories = directories\n    } else {\n      this.directories = [\n        `${this.directory}/__tests__`,\n        `${this.directory}/tests`,\n        `${this.directory}/spec`\n      ]\n    }\n\n    this.findingCount = 0\n    this.processArgs = processArgs\n\n    /** @type {string[]} */\n    this.foundFiles = []\n\n    /** @type {Record<number, Promise<void>>} */\n    this.findingPromises = {}\n\n    /** @type {string[]} */\n    this.testArgs = this.processArgs.filter((processArg, index) => index != 0)\n\n    /** @type {string[]} */\n    this.directoryArgs = []\n\n    /** @type {string[]} */\n    this.fileArgs = []\n\n    for (const testArg of this.testArgs) {\n      if (testArg.endsWith(\"/\")) {\n        this.directoryArgs.push(testArg)\n      } else {\n        this.fileArgs.push(testArg)\n      }\n    }\n  }\n\n  /**\n   * @returns {Promise<string[]>} - Resolves with the test files.\n   */\n  async findTestFiles() {\n    await this.withFindingCount(async () => {\n      for (const directory of this.directories) {\n        if (await fileExists(directory)) {\n          await this.findTestFilesInDir(directory)\n        }\n      }\n    })\n\n    await this.waitForFindingPromises()\n\n    return this.foundFiles\n  }\n\n  /**\n   * @returns {number} - The ing promises length.\n   */\n  findingPromisesLength() { return Object.keys(this.findingPromises).length }\n\n  async waitForFindingPromises() {\n    while (this.findingPromisesLength() > 0) {\n      await this.waitForFindingPromisesIteration()\n    }\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async waitForFindingPromisesIteration() {\n    const unfinishedPromises = []\n\n    for (const findingPromiseId in this.findingPromises) {\n      const findingPromise = this.findingPromises[findingPromiseId]\n\n      unfinishedPromises.push(findingPromise)\n    }\n\n    await Promise.all(unfinishedPromises)\n  }\n\n  /**\n   * @param {function() : Promise<void>} callback - Callback function.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  withFindingCount(callback) {\n    return new Promise((resolve) => {\n      const findingPromise = callback()\n      const findingCount = this.findingCount\n\n      this.findingCount += 1\n      this.findingPromises[findingCount] = findingPromise\n\n      findingPromise.finally(() => {\n        delete this.findingPromises[findingCount]\n\n        resolve(undefined)\n      })\n    })\n  }\n\n  /**\n   * @param {string} dir - Dir.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async findTestFilesInDir(dir) {\n    await this.withFindingCount(async () => {\n      const files = await fs.readdir(dir)\n\n      for (const file of files) {\n        if (TestFilesFinder.IGNORED_NAMES.includes(file)) {\n          continue\n        }\n\n        const fullPath = `${dir}/${file}`\n        const localPath = fullPath.replace(`${this.directory}/`, \"\")\n        const isDir = (await fs.stat(fullPath)).isDirectory()\n\n        if (isDir) {\n          this.findTestFilesInDir(fullPath)\n        } else {\n          if (this.isFileMatchingRequirements(file, localPath)) {\n            this.foundFiles.push(fullPath)\n          }\n        }\n      }\n    })\n  }\n\n  /**\n   * @param {string} file - File.\n   * @param {string} localPath - Local path.\n   * @returns {boolean} - Whether file matching requirements.\n   */\n  isFileMatchingRequirements(file, localPath) {\n    if (this.directoryArgs.length > 0) {\n      for (const directoryArg of this.directoryArgs) {\n        if (localPath.startsWith(directoryArg) && this.looksLikeTestFile(file)) {\n          this.logger.debug(\"Found test file because matching dir and looks like this file:\", file)\n          return true\n        }\n      }\n    }\n\n    if (this.fileArgs.length > 0) {\n      for (const fileArg of this.fileArgs) {\n        if (fileArg == localPath) {\n          this.logger.debug(\"Found test file because matching file arg:\", file)\n          return true\n        }\n      }\n    }\n\n    if (this.fileArgs.length == 0 && this.directoryArgs.length == 0 && this.looksLikeTestFile(file)) {\n      this.logger.debug(\"Found test file because looks like this file:\", file)\n      return true\n    }\n\n    return false\n  }\n\n  /**\n   * @param {string} file - File.\n   * @returns {boolean} - Whether looks like test file.\n   */\n  looksLikeTestFile(file) {\n    return Boolean(file.match(/-(spec|test)\\.(m|)js$/))\n  }\n}\n"]}
221
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-files-finder.js","sourceRoot":"","sources":["../../../src/testing/test-files-finder.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,OAAO,EAAE,MAAM,aAAa,CAAA;AAC5B,OAAO,IAAI,MAAM,MAAM,CAAA;AAEvB,OAAO,UAAU,MAAM,yBAAyB,CAAA;AAChD,OAAO,EAAC,MAAM,EAAC,MAAM,cAAc,CAAA;AACnC,OAAO,aAAa,MAAM,6BAA6B,CAAA;AAEvD,iHAAiH;AACjH,MAAM,CAAC,OAAO,OAAO,eAAe;IAClC,MAAM,CAAC,aAAa,GAAG,CAAC,MAAM,EAAE,cAAc,CAAC,CAAA;IAE/C;;;;;OAKG;IACH,YAAY,EAAC,SAAS,EAAE,WAAW,EAAE,WAAW,EAAE,GAAG,QAAQ,EAAC;QAC5D,aAAa,CAAC,QAAQ,CAAC,CAAA;QAEvB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAA;QACxC,IAAI,CAAC,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,CAAA;QAE9B,IAAI,WAAW,EAAE,CAAC;YAChB,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAA;QACpE,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,WAAW,GAAG;gBACjB,GAAG,IAAI,CAAC,SAAS,YAAY;gBAC7B,GAAG,IAAI,CAAC,SAAS,QAAQ;gBACzB,GAAG,IAAI,CAAC,SAAS,OAAO;aACzB,CAAA;QACH,CAAC;QAED,IAAI,CAAC,YAAY,GAAG,CAAC,CAAA;QACrB,IAAI,CAAC,WAAW,GAAG,WAAW,CAAA;QAE9B,uBAAuB;QACvB,IAAI,CAAC,UAAU,GAAG,EAAE,CAAA;QAEpB,4CAA4C;QAC5C,IAAI,CAAC,eAAe,GAAG,EAAE,CAAA;QAEzB,uBAAuB;QACvB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,UAAU,EAAE,KAAK,EAAE,EAAE,CAAC,KAAK,IAAI,CAAC,CAAC,CAAA;QAE1E,uBAAuB;QACvB,IAAI,CAAC,aAAa,GAAG,EAAE,CAAA;QAEvB,uBAAuB;QACvB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAA;QAElB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAA;IAC5B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,aAAa;QACjB,MAAM,IAAI,CAAC,WAAW,EAAE,CAAA;QAExB,MAAM,IAAI,CAAC,gBAAgB,CAAC,KAAK,IAAI,EAAE;YACrC,KAAK,MAAM,SAAS,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;gBACzC,IAAI,MAAM,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;oBAChC,MAAM,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAA;gBAC1C,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,MAAM,IAAI,CAAC,sBAAsB,EAAE,CAAA;QAEnC,OAAO,IAAI,CAAC,UAAU,CAAA;IACxB,CAAC;IAED;;OAEG;IACH,qBAAqB,KAAK,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,MAAM,CAAA,CAAC,CAAC;IAE3E,KAAK,CAAC,sBAAsB;QAC1B,OAAO,IAAI,CAAC,qBAAqB,EAAE,GAAG,CAAC,EAAE,CAAC;YACxC,MAAM,IAAI,CAAC,+BAA+B,EAAE,CAAA;QAC9C,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,+BAA+B;QACnC,MAAM,kBAAkB,GAAG,EAAE,CAAA;QAE7B,KAAK,MAAM,gBAAgB,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACpD,MAAM,cAAc,GAAG,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,CAAA;YAE7D,kBAAkB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAA;QACzC,CAAC;QAED,MAAM,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAA;IACvC,CAAC;IAED;;;OAGG;IACH,gBAAgB,CAAC,QAAQ;QACvB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7B,MAAM,cAAc,GAAG,QAAQ,EAAE,CAAA;YACjC,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAA;YAEtC,IAAI,CAAC,YAAY,IAAI,CAAC,CAAA;YACtB,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,GAAG,cAAc,CAAA;YAEnD,cAAc,CAAC,OAAO,CAAC,GAAG,EAAE;gBAC1B,OAAO,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,CAAA;gBAEzC,OAAO,CAAC,SAAS,CAAC,CAAA;YACpB,CAAC,CAAC,CAAA;QACJ,CAAC,CAAC,CAAA;IACJ,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,kBAAkB,CAAC,GAAG;QAC1B,MAAM,IAAI,CAAC,gBAAgB,CAAC,KAAK,IAAI,EAAE;YACrC,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,CAAA;YAEnC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,eAAe,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;oBACjD,SAAQ;gBACV,CAAC;gBAED,MAAM,QAAQ,GAAG,GAAG,GAAG,IAAI,IAAI,EAAE,CAAA;gBACjC,MAAM,SAAS,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,IAAI,CAAC,SAAS,GAAG,EAAE,EAAE,CAAC,CAAA;gBAC5D,MAAM,KAAK,GAAG,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,WAAW,EAAE,CAAA;gBAErD,IAAI,KAAK,EAAE,CAAC;oBACV,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAA;gBACnC,CAAC;qBAAM,CAAC;oBACN,IAAI,IAAI,CAAC,0BAA0B,CAAC,IAAI,EAAE,SAAS,CAAC,EAAE,CAAC;wBACrD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;oBAChC,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC;IAED;;;;OAIG;IACH,0BAA0B,CAAC,IAAI,EAAE,SAAS;QACxC,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAClC,KAAK,MAAM,YAAY,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;gBAC9C,IAAI,SAAS,CAAC,UAAU,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC;oBACvE,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,gEAAgE,EAAE,IAAI,CAAC,CAAA;oBACzF,OAAO,IAAI,CAAA;gBACb,CAAC;YACH,CAAC;QACH,CAAC;QAED,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC7B,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACpC,IAAI,OAAO,IAAI,SAAS,EAAE,CAAC;oBACzB,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,4CAA4C,EAAE,IAAI,CAAC,CAAA;oBACrE,OAAO,IAAI,CAAA;gBACb,CAAC;YACH,CAAC;QACH,CAAC;QAED,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,IAAI,CAAC,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,IAAI,CAAC,IAAI,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC;YAChG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,+CAA+C,EAAE,IAAI,CAAC,CAAA;YACxE,OAAO,IAAI,CAAA;QACb,CAAC;QAED,OAAO,KAAK,CAAA;IACd,CAAC;IAED;;;OAGG;IACH,iBAAiB,CAAC,IAAI;QACpB,OAAO,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC,CAAA;IACrD,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW;QACf,IAAI,IAAI,CAAC,aAAa;YAAE,OAAM;QAE9B,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YACpC,MAAM,cAAc,GAAG,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;YAC1E,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,EAAE,OAAO,CAAC,CAAA;YAC3F,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,CAAA;YAC9C,MAAM,aAAa,GAAG,OAAO,KAAK,QAAQ,IAAI,OAAO,CAAC,UAAU,CAAC,GAAG,QAAQ,GAAG,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,GAAG,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC,CAAA;YAChI,MAAM,oBAAoB,GAAG,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,CAAA;YACtI,MAAM,kBAAkB,GAAG,oBAAoB,CAAC,CAAC,CAAC,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAA;YAErF,IAAI,cAAc,EAAE,CAAC;gBACnB,MAAM,kBAAkB,GAAG,IAAI,CAAC,WAAW,CAAC,oBAAoB,IAAI,QAAQ,CAAC,CAAA;gBAC7E,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,CAAC,CAAC,CAAA;gBACrE,SAAQ;YACV,CAAC;YAED,IAAI,CAAC;gBACH,IAAI,KAAK,CAAA;gBACT,IAAI,gBAAgB,CAAA;gBAEpB,KAAK,MAAM,aAAa,IAAI,kBAAkB,EAAE,CAAC;oBAC/C,IAAI,CAAC;wBACH,KAAK,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,CAAA;wBACpC,gBAAgB,GAAG,aAAa,CAAA;wBAChC,MAAK;oBACP,CAAC;oBAAC,MAAM,CAAC;wBACP,iBAAiB;oBACnB,CAAC;gBACH,CAAC;gBAED,IAAI,CAAC,KAAK,IAAI,CAAC,gBAAgB;oBAAE,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAA;gBAClE,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,gBAAgB,CAAC,CAAA;gBAEpD,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;oBACxB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC,CAAA;gBAC9D,CAAC;qBAAM,IAAI,KAAK,CAAC,MAAM,EAAE,EAAE,CAAC;oBAC1B,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,CAAA;gBAC/B,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,MAAM,iBAAiB,GAAG,IAAI,CAAC,WAAW,CAAC,oBAAoB,IAAI,QAAQ,CAAC,CAAA;gBAC5E,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAA;YACvC,CAAC;QACH,CAAC;QAED,IAAI,CAAC,aAAa,GAAG,IAAI,CAAA;IAC3B,CAAC;IAED;;;OAGG;IACH,mBAAmB,CAAC,SAAS;QAC3B,IAAI,SAAS,KAAK,EAAE;YAAE,OAAO,SAAS,CAAA;QACtC,OAAO,SAAS,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,GAAG,CAAA;IAC9D,CAAC;IAED;;;OAGG;IACH,WAAW,CAAC,QAAQ;QAClB,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAA;QAC5D,OAAO,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;IAC/C,CAAC","sourcesContent":["// @ts-check\n\nimport fs from \"fs/promises\"\nimport path from \"path\"\n\nimport fileExists from \"../utils/file-exists.js\"\nimport {Logger} from \"../logger.js\"\nimport restArgsError from \"../utils/rest-args-error.js\"\n\n// Incredibly complex class to find files in multiple simultanious running promises to do it as fast as possible.\nexport default class TestFilesFinder {\n  static IGNORED_NAMES = [\".git\", \"node_modules\"]\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.directory - Directory path.\n   * @param {string[]} [args.directories] - Directories.\n   * @param {string[]} args.processArgs - Process args.\n   */\n  constructor({directory, directories, processArgs, ...restArgs}) {\n    restArgsError(restArgs)\n\n    this.directory = path.resolve(directory)\n    this.logger = new Logger(this)\n\n    if (directories) {\n      this.directories = directories.map((entry) => path.resolve(entry))\n    } else {\n      this.directories = [\n        `${this.directory}/__tests__`,\n        `${this.directory}/tests`,\n        `${this.directory}/spec`\n      ]\n    }\n\n    this.findingCount = 0\n    this.processArgs = processArgs\n\n    /** @type {string[]} */\n    this.foundFiles = []\n\n    /** @type {Record<number, Promise<void>>} */\n    this.findingPromises = {}\n\n    /** @type {string[]} */\n    this.testArgs = this.processArgs.filter((processArg, index) => index != 0)\n\n    /** @type {string[]} */\n    this.directoryArgs = []\n\n    /** @type {string[]} */\n    this.fileArgs = []\n\n    this._argsPrepared = false\n  }\n\n  /**\n   * @returns {Promise<string[]>} - Resolves with the test files.\n   */\n  async findTestFiles() {\n    await this.prepareArgs()\n\n    await this.withFindingCount(async () => {\n      for (const directory of this.directories) {\n        if (await fileExists(directory)) {\n          await this.findTestFilesInDir(directory)\n        }\n      }\n    })\n\n    await this.waitForFindingPromises()\n\n    return this.foundFiles\n  }\n\n  /**\n   * @returns {number} - The ing promises length.\n   */\n  findingPromisesLength() { return Object.keys(this.findingPromises).length }\n\n  async waitForFindingPromises() {\n    while (this.findingPromisesLength() > 0) {\n      await this.waitForFindingPromisesIteration()\n    }\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async waitForFindingPromisesIteration() {\n    const unfinishedPromises = []\n\n    for (const findingPromiseId in this.findingPromises) {\n      const findingPromise = this.findingPromises[findingPromiseId]\n\n      unfinishedPromises.push(findingPromise)\n    }\n\n    await Promise.all(unfinishedPromises)\n  }\n\n  /**\n   * @param {function() : Promise<void>} callback - Callback function.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  withFindingCount(callback) {\n    return new Promise((resolve) => {\n      const findingPromise = callback()\n      const findingCount = this.findingCount\n\n      this.findingCount += 1\n      this.findingPromises[findingCount] = findingPromise\n\n      findingPromise.finally(() => {\n        delete this.findingPromises[findingCount]\n\n        resolve(undefined)\n      })\n    })\n  }\n\n  /**\n   * @param {string} dir - Dir.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async findTestFilesInDir(dir) {\n    await this.withFindingCount(async () => {\n      const files = await fs.readdir(dir)\n\n      for (const file of files) {\n        if (TestFilesFinder.IGNORED_NAMES.includes(file)) {\n          continue\n        }\n\n        const fullPath = `${dir}/${file}`\n        const localPath = fullPath.replace(`${this.directory}/`, \"\")\n        const isDir = (await fs.stat(fullPath)).isDirectory()\n\n        if (isDir) {\n          this.findTestFilesInDir(fullPath)\n        } else {\n          if (this.isFileMatchingRequirements(file, localPath)) {\n            this.foundFiles.push(fullPath)\n          }\n        }\n      }\n    })\n  }\n\n  /**\n   * @param {string} file - File.\n   * @param {string} localPath - Local path.\n   * @returns {boolean} - Whether file matching requirements.\n   */\n  isFileMatchingRequirements(file, localPath) {\n    if (this.directoryArgs.length > 0) {\n      for (const directoryArg of this.directoryArgs) {\n        if (localPath.startsWith(directoryArg) && this.looksLikeTestFile(file)) {\n          this.logger.debug(\"Found test file because matching dir and looks like this file:\", file)\n          return true\n        }\n      }\n    }\n\n    if (this.fileArgs.length > 0) {\n      for (const fileArg of this.fileArgs) {\n        if (fileArg == localPath) {\n          this.logger.debug(\"Found test file because matching file arg:\", file)\n          return true\n        }\n      }\n    }\n\n    if (this.fileArgs.length == 0 && this.directoryArgs.length == 0 && this.looksLikeTestFile(file)) {\n      this.logger.debug(\"Found test file because looks like this file:\", file)\n      return true\n    }\n\n    return false\n  }\n\n  /**\n   * @param {string} file - File.\n   * @returns {boolean} - Whether looks like test file.\n   */\n  looksLikeTestFile(file) {\n    return Boolean(file.match(/-(spec|test)\\.(m|)js$/))\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when test args are prepared.\n   */\n  async prepareArgs() {\n    if (this._argsPrepared) return\n\n    for (const testArg of this.testArgs) {\n      const forceDirectory = testArg.endsWith(\"/\") || testArg.endsWith(path.sep)\n      const fullPath = path.isAbsolute(testArg) ? testArg : path.resolve(this.directory, testArg)\n      const baseName = path.basename(this.directory)\n      const hasBasePrefix = testArg === baseName || testArg.startsWith(`${baseName}/`) || testArg.startsWith(`${baseName}${path.sep}`)\n      const basePrefixedFullPath = (!path.isAbsolute(testArg) && hasBasePrefix) ? path.resolve(path.dirname(this.directory), testArg) : null\n      const fullPathCandidates = basePrefixedFullPath ? [basePrefixedFullPath] : [fullPath]\n\n      if (forceDirectory) {\n        const preferredLocalPath = this.toLocalPath(basePrefixedFullPath || fullPath)\n        this.directoryArgs.push(this.ensureTrailingSlash(preferredLocalPath))\n        continue\n      }\n\n      try {\n        let stats\n        let resolvedFullPath\n\n        for (const candidatePath of fullPathCandidates) {\n          try {\n            stats = await fs.stat(candidatePath)\n            resolvedFullPath = candidatePath\n            break\n          } catch {\n            // Keep searching\n          }\n        }\n\n        if (!stats || !resolvedFullPath) throw new Error(\"Path not found\")\n        const localPath = this.toLocalPath(resolvedFullPath)\n\n        if (stats.isDirectory()) {\n          this.directoryArgs.push(this.ensureTrailingSlash(localPath))\n        } else if (stats.isFile()) {\n          this.fileArgs.push(localPath)\n        }\n      } catch {\n        const fallbackLocalPath = this.toLocalPath(basePrefixedFullPath || fullPath)\n        this.fileArgs.push(fallbackLocalPath)\n      }\n    }\n\n    this._argsPrepared = true\n  }\n\n  /**\n   * @param {string} localPath - Local path.\n   * @returns {string} - Normalized local path with trailing slash.\n   */\n  ensureTrailingSlash(localPath) {\n    if (localPath === \"\") return localPath\n    return localPath.endsWith(\"/\") ? localPath : `${localPath}/`\n  }\n\n  /**\n   * @param {string} fullPath - Full path.\n   * @returns {string} - Local path relative to the base directory.\n   */\n  toLocalPath(fullPath) {\n    const relativePath = path.relative(this.directory, fullPath)\n    return relativePath.split(path.sep).join(\"/\")\n  }\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  // @ts-check
2
- import { AsyncLocalStorage } from "async_hooks";
3
- /** @type {AsyncLocalStorage<Array<string[]>> | undefined} */
2
+ import { AsyncLocalStorage } from "node:async_hooks";
3
+ /** @type {import("node:async_hooks").AsyncLocalStorage<Array<string[]>> | undefined} */
4
4
  let asyncLocalStorage;
5
5
  if (AsyncLocalStorage) {
6
6
  asyncLocalStorage = new AsyncLocalStorage();
@@ -67,4 +67,4 @@ else {
67
67
  globalThis.withTrackedStack = { addTrackedStackToError, withTrackedStack };
68
68
  }
69
69
  export { addTrackedStackToError, withTrackedStack };
70
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2l0aC10cmFja2VkLXN0YWNrLWFzeW5jLWhvb2tzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3V0aWxzL3dpdGgtdHJhY2tlZC1zdGFjay1hc3luYy1ob29rcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxZQUFZO0FBRVosT0FBTyxFQUFDLGlCQUFpQixFQUFDLE1BQU0sYUFBYSxDQUFBO0FBRTdDLDZEQUE2RDtBQUM3RCxJQUFJLGlCQUFpQixDQUFBO0FBRXJCLElBQUksaUJBQWlCLEVBQUUsQ0FBQztJQUN0QixpQkFBaUIsR0FBRyxJQUFJLGlCQUFpQixFQUFFLENBQUE7QUFDN0MsQ0FBQztBQUVELHFFQUFxRTtBQUNyRSxTQUFTLHNCQUFzQixDQUFDLEtBQUs7SUFDbkMsZ0JBQWdCO0lBQ2hCLElBQUksQ0FBQyxpQkFBaUI7UUFBRSxPQUFNO0lBRTlCLE1BQU0sWUFBWSxHQUFHLGlCQUFpQixDQUFDLFFBQVEsRUFBRSxJQUFJLEVBQUUsQ0FBQTtJQUN2RCxNQUFNLG9CQUFvQixHQUFHLEVBQUUsQ0FBQTtJQUUvQixLQUFLLE1BQU0sV0FBVyxJQUFJLFlBQVksRUFBRSxDQUFDO1FBQ3ZDLEtBQUssTUFBTSxlQUFlLElBQUksV0FBVyxFQUFFLENBQUM7WUFDMUMsb0JBQW9CLENBQUMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxDQUFBO1FBQzVDLENBQUM7SUFDSCxDQUFDO0lBRUQsK0RBQStEO0lBQy9ELEtBQUssQ0FBQyxLQUFLLElBQUksSUFBSSxHQUFHLG9CQUFvQixDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQTtBQUN2RCxDQUFDO0FBRUQ7Ozs7O0dBS0c7QUFDSCxLQUFLLFVBQVUsZ0JBQWdCLENBQUMsSUFBSSxFQUFFLElBQUk7SUFDeEMsa0NBQWtDO0lBQ2xDLElBQUksUUFBUSxDQUFBO0lBRVoscUJBQXFCO0lBQ3JCLElBQUksS0FBSyxDQUFBO0lBRVQsSUFBSSxPQUFPLElBQUksSUFBSSxVQUFVLElBQUksT0FBTyxJQUFJLElBQUksUUFBUSxFQUFFLENBQUM7UUFDekQsbUJBQW1CO1FBQ25CLFFBQVEsR0FBRyxJQUFJLENBQUE7UUFDZixLQUFLLEdBQUcsSUFBSSxDQUFBO0lBQ2QsQ0FBQztTQUFNLENBQUM7UUFDTixtQkFBbUI7UUFDbkIsUUFBUSxHQUFHLElBQUksQ0FBQTtRQUNmLEtBQUssR0FBRyxLQUFLLEVBQUUsQ0FBQyxLQUFLLElBQUksRUFBRSxDQUFBO0lBQzdCLENBQUM7SUFFRCxnQkFBZ0I7SUFDaEIsSUFBSSxDQUFDLGlCQUFpQjtRQUFFLE9BQU8sTUFBTSxRQUFRLEVBQUUsQ0FBQTtJQUUvQyxNQUFNLFlBQVksR0FBRyxpQkFBaUIsQ0FBQyxRQUFRLEVBQUUsSUFBSSxFQUFFLENBQUE7SUFDdkQsTUFBTSxvQkFBb0IsR0FBRyxFQUFFLENBQUE7SUFDL0IsTUFBTSxpQkFBaUIsR0FBRyxLQUFLLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFBO0lBRTNDLGlCQUFpQixDQUFDLENBQUMsQ0FBQyxHQUFHLDBCQUEwQixDQUFBO0lBRWpELEtBQUssSUFBSSxDQUFDLEdBQUcsaUJBQWlCLENBQUMsTUFBTSxFQUFFLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQztRQUNuRCxNQUFNLFNBQVMsR0FBRyxpQkFBaUIsQ0FBQyxDQUFDLENBQUMsQ0FBQTtRQUV0QyxvQkFBb0IsQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLENBQUE7UUFFdkMsSUFBSSxTQUFTLElBQUksMEJBQTBCLEVBQUUsQ0FBQztZQUM1QyxNQUFLO1FBQ1AsQ0FBQztJQUNILENBQUM7SUFFRCxNQUFNLFNBQVMsR0FBRyxDQUFDLG9CQUFvQixFQUFFLEdBQUcsWUFBWSxDQUFDLENBQUE7SUFFekQsTUFBTSxpQkFBaUIsQ0FBQyxHQUFHLENBQUMsU0FBUyxFQUFFLEtBQUssSUFBSSxFQUFFO1FBQ2hELE1BQU0sUUFBUSxFQUFFLENBQUE7SUFDbEIsQ0FBQyxDQUFDLENBQUE7QUFDSixDQUFDO0FBRUQsSUFBSSxVQUFVLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQztJQUNoQyxPQUFPLENBQUMsSUFBSSxDQUFDLGlEQUFpRCxDQUFDLENBQUE7QUFDakUsQ0FBQztLQUFNLENBQUM7SUFDTixVQUFVLENBQUMsZ0JBQWdCLEdBQUcsRUFBQyxzQkFBc0IsRUFBRSxnQkFBZ0IsRUFBQyxDQUFBO0FBQzFFLENBQUM7QUFFRCxPQUFPLEVBQUMsc0JBQXNCLEVBQUUsZ0JBQWdCLEVBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIEB0cy1jaGVja1xuXG5pbXBvcnQge0FzeW5jTG9jYWxTdG9yYWdlfSBmcm9tIFwiYXN5bmNfaG9va3NcIlxuXG4vKiogQHR5cGUge0FzeW5jTG9jYWxTdG9yYWdlPEFycmF5PHN0cmluZ1tdPj4gfCB1bmRlZmluZWR9ICovXG5sZXQgYXN5bmNMb2NhbFN0b3JhZ2VcblxuaWYgKEFzeW5jTG9jYWxTdG9yYWdlKSB7XG4gIGFzeW5jTG9jYWxTdG9yYWdlID0gbmV3IEFzeW5jTG9jYWxTdG9yYWdlKClcbn1cblxuLyoqIEBwYXJhbSB7RXJyb3J9IGVycm9yIC0gRXJyb3IgdG8gYW5ub3RhdGUgd2l0aCBhIHRyYWNrZWQgc3RhY2suICovXG5mdW5jdGlvbiBhZGRUcmFja2VkU3RhY2tUb0Vycm9yKGVycm9yKSB7XG4gIC8vIE5vdCBzdXBwb3J0ZWRcbiAgaWYgKCFhc3luY0xvY2FsU3RvcmFnZSkgcmV0dXJuXG5cbiAgY29uc3QgcGFyZW50U3RhY2tzID0gYXN5bmNMb2NhbFN0b3JhZ2UuZ2V0U3RvcmUoKSB8fCBbXVxuICBjb25zdCBhZGRpdGlvbmFsU3RhY2tMaW5lcyA9IFtdXG5cbiAgZm9yIChjb25zdCBwYXJlbnRTdGFjayBvZiBwYXJlbnRTdGFja3MpIHtcbiAgICBmb3IgKGNvbnN0IHBhcmVudFN0YWNrTGluZSBvZiBwYXJlbnRTdGFjaykge1xuICAgICAgYWRkaXRpb25hbFN0YWNrTGluZXMucHVzaChwYXJlbnRTdGFja0xpbmUpXG4gICAgfVxuICB9XG5cbiAgLy8gUmVwbGFjZSB0aGUgZXJyb3IgbWVzc2FnZSBvbiB0aGUgZmlyc3QgbGluZSB3aXRoIHRoaXMgc3RyaW5nXG4gIGVycm9yLnN0YWNrICs9IFwiXFxuXCIgKyBhZGRpdGlvbmFsU3RhY2tMaW5lcy5qb2luKFwiXFxuXCIpXG59XG5cbi8qKlxuICpcbiAqIEBwYXJhbSB7KCkgPT4gUHJvbWlzZTx2b2lkPiB8IHN0cmluZ30gYXJnMSAtIEFyZzEuXG4gKiBAcGFyYW0geygpID0+IFByb21pc2U8dm9pZD4gfCBFcnJvcn0gW2FyZzJdIC0gQXJnMi5cbiAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gY29tcGxldGUuXG4gKi9cbmFzeW5jIGZ1bmN0aW9uIHdpdGhUcmFja2VkU3RhY2soYXJnMSwgYXJnMikge1xuICAvKiogQHR5cGUgeygpID0+IFByb21pc2U8dm9pZD59ICovXG4gIGxldCBjYWxsYmFja1xuXG4gIC8qKiBAdHlwZSB7c3RyaW5nfSAqL1xuICBsZXQgc3RhY2tcblxuICBpZiAodHlwZW9mIGFyZzIgPT0gXCJmdW5jdGlvblwiICYmIHR5cGVvZiBhcmcxID09IFwic3RyaW5nXCIpIHtcbiAgICAvLyBAdHMtZXhwZWN0LWVycm9yXG4gICAgY2FsbGJhY2sgPSBhcmcyXG4gICAgc3RhY2sgPSBhcmcxXG4gIH0gZWxzZSB7XG4gICAgLy8gQHRzLWV4cGVjdC1lcnJvclxuICAgIGNhbGxiYWNrID0gYXJnMVxuICAgIHN0YWNrID0gRXJyb3IoKS5zdGFjayB8fCBcIlwiXG4gIH1cblxuICAvLyBOb3Qgc3VwcG9ydGVkXG4gIGlmICghYXN5bmNMb2NhbFN0b3JhZ2UpIHJldHVybiBhd2FpdCBjYWxsYmFjaygpXG5cbiAgY29uc3QgcGFyZW50U3RhY2tzID0gYXN5bmNMb2NhbFN0b3JhZ2UuZ2V0U3RvcmUoKSB8fCBbXVxuICBjb25zdCBhZGRpdGlvbmFsU3RhY2tMaW5lcyA9IFtdXG4gIGNvbnN0IGN1cnJlbnRTdGFja0xpbmVzID0gc3RhY2suc3BsaXQoXCJcXG5cIilcblxuICBjdXJyZW50U3RhY2tMaW5lc1swXSA9IFwiICAgIFtXSVRIIFRSQUNLRUQgU1RBQ0tdXCJcblxuICBmb3IgKGxldCBpID0gY3VycmVudFN0YWNrTGluZXMubGVuZ3RoOyBpID49IDA7IGktLSkge1xuICAgIGNvbnN0IHN0YWNrTGluZSA9IGN1cnJlbnRTdGFja0xpbmVzW2ldXG5cbiAgICBhZGRpdGlvbmFsU3RhY2tMaW5lcy51bnNoaWZ0KHN0YWNrTGluZSlcblxuICAgIGlmIChzdGFja0xpbmUgPT0gXCIgICAgW1dJVEggVFJBQ0tFRCBTVEFDS11cIikge1xuICAgICAgYnJlYWtcbiAgICB9XG4gIH1cblxuICBjb25zdCBuZXdTdGFja3MgPSBbYWRkaXRpb25hbFN0YWNrTGluZXMsIC4uLnBhcmVudFN0YWNrc11cblxuICBhd2FpdCBhc3luY0xvY2FsU3RvcmFnZS5ydW4obmV3U3RhY2tzLCBhc3luYyAoKSA9PiB7XG4gICAgYXdhaXQgY2FsbGJhY2soKVxuICB9KVxufVxuXG5pZiAoZ2xvYmFsVGhpcy53aXRoVHJhY2tlZFN0YWNrKSB7XG4gIGNvbnNvbGUud2FybihcImdsb2JhbFRoaXMud2l0aFRyYWNrZWRTdGFjayB3YXMgYWxyZWFkeSBkZWZpbmVkXCIpXG59IGVsc2Uge1xuICBnbG9iYWxUaGlzLndpdGhUcmFja2VkU3RhY2sgPSB7YWRkVHJhY2tlZFN0YWNrVG9FcnJvciwgd2l0aFRyYWNrZWRTdGFja31cbn1cblxuZXhwb3J0IHthZGRUcmFja2VkU3RhY2tUb0Vycm9yLCB3aXRoVHJhY2tlZFN0YWNrfVxuIl19
70
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2l0aC10cmFja2VkLXN0YWNrLWFzeW5jLWhvb2tzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3V0aWxzL3dpdGgtdHJhY2tlZC1zdGFjay1hc3luYy1ob29rcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxZQUFZO0FBRVosT0FBTyxFQUFDLGlCQUFpQixFQUFDLE1BQU0sa0JBQWtCLENBQUE7QUFFbEQsd0ZBQXdGO0FBQ3hGLElBQUksaUJBQWlCLENBQUE7QUFFckIsSUFBSSxpQkFBaUIsRUFBRSxDQUFDO0lBQ3RCLGlCQUFpQixHQUFHLElBQUksaUJBQWlCLEVBQUUsQ0FBQTtBQUM3QyxDQUFDO0FBRUQscUVBQXFFO0FBQ3JFLFNBQVMsc0JBQXNCLENBQUMsS0FBSztJQUNuQyxnQkFBZ0I7SUFDaEIsSUFBSSxDQUFDLGlCQUFpQjtRQUFFLE9BQU07SUFFOUIsTUFBTSxZQUFZLEdBQUcsaUJBQWlCLENBQUMsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFBO0lBQ3ZELE1BQU0sb0JBQW9CLEdBQUcsRUFBRSxDQUFBO0lBRS9CLEtBQUssTUFBTSxXQUFXLElBQUksWUFBWSxFQUFFLENBQUM7UUFDdkMsS0FBSyxNQUFNLGVBQWUsSUFBSSxXQUFXLEVBQUUsQ0FBQztZQUMxQyxvQkFBb0IsQ0FBQyxJQUFJLENBQUMsZUFBZSxDQUFDLENBQUE7UUFDNUMsQ0FBQztJQUNILENBQUM7SUFFRCwrREFBK0Q7SUFDL0QsS0FBSyxDQUFDLEtBQUssSUFBSSxJQUFJLEdBQUcsb0JBQW9CLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFBO0FBQ3ZELENBQUM7QUFFRDs7Ozs7R0FLRztBQUNILEtBQUssVUFBVSxnQkFBZ0IsQ0FBQyxJQUFJLEVBQUUsSUFBSTtJQUN4QyxrQ0FBa0M7SUFDbEMsSUFBSSxRQUFRLENBQUE7SUFFWixxQkFBcUI7SUFDckIsSUFBSSxLQUFLLENBQUE7SUFFVCxJQUFJLE9BQU8sSUFBSSxJQUFJLFVBQVUsSUFBSSxPQUFPLElBQUksSUFBSSxRQUFRLEVBQUUsQ0FBQztRQUN6RCxtQkFBbUI7UUFDbkIsUUFBUSxHQUFHLElBQUksQ0FBQTtRQUNmLEtBQUssR0FBRyxJQUFJLENBQUE7SUFDZCxDQUFDO1NBQU0sQ0FBQztRQUNOLG1CQUFtQjtRQUNuQixRQUFRLEdBQUcsSUFBSSxDQUFBO1FBQ2YsS0FBSyxHQUFHLEtBQUssRUFBRSxDQUFDLEtBQUssSUFBSSxFQUFFLENBQUE7SUFDN0IsQ0FBQztJQUVELGdCQUFnQjtJQUNoQixJQUFJLENBQUMsaUJBQWlCO1FBQUUsT0FBTyxNQUFNLFFBQVEsRUFBRSxDQUFBO0lBRS9DLE1BQU0sWUFBWSxHQUFHLGlCQUFpQixDQUFDLFFBQVEsRUFBRSxJQUFJLEVBQUUsQ0FBQTtJQUN2RCxNQUFNLG9CQUFvQixHQUFHLEVBQUUsQ0FBQTtJQUMvQixNQUFNLGlCQUFpQixHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUE7SUFFM0MsaUJBQWlCLENBQUMsQ0FBQyxDQUFDLEdBQUcsMEJBQTBCLENBQUE7SUFFakQsS0FBSyxJQUFJLENBQUMsR0FBRyxpQkFBaUIsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDO1FBQ25ELE1BQU0sU0FBUyxHQUFHLGlCQUFpQixDQUFDLENBQUMsQ0FBQyxDQUFBO1FBRXRDLG9CQUFvQixDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQTtRQUV2QyxJQUFJLFNBQVMsSUFBSSwwQkFBMEIsRUFBRSxDQUFDO1lBQzVDLE1BQUs7UUFDUCxDQUFDO0lBQ0gsQ0FBQztJQUVELE1BQU0sU0FBUyxHQUFHLENBQUMsb0JBQW9CLEVBQUUsR0FBRyxZQUFZLENBQUMsQ0FBQTtJQUV6RCxNQUFNLGlCQUFpQixDQUFDLEdBQUcsQ0FBQyxTQUFTLEVBQUUsS0FBSyxJQUFJLEVBQUU7UUFDaEQsTUFBTSxRQUFRLEVBQUUsQ0FBQTtJQUNsQixDQUFDLENBQUMsQ0FBQTtBQUNKLENBQUM7QUFFRCxJQUFJLFVBQVUsQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO0lBQ2hDLE9BQU8sQ0FBQyxJQUFJLENBQUMsaURBQWlELENBQUMsQ0FBQTtBQUNqRSxDQUFDO0tBQU0sQ0FBQztJQUNOLFVBQVUsQ0FBQyxnQkFBZ0IsR0FBRyxFQUFDLHNCQUFzQixFQUFFLGdCQUFnQixFQUFDLENBQUE7QUFDMUUsQ0FBQztBQUVELE9BQU8sRUFBQyxzQkFBc0IsRUFBRSxnQkFBZ0IsRUFBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQHRzLWNoZWNrXG5cbmltcG9ydCB7QXN5bmNMb2NhbFN0b3JhZ2V9IGZyb20gXCJub2RlOmFzeW5jX2hvb2tzXCJcblxuLyoqIEB0eXBlIHtpbXBvcnQoXCJub2RlOmFzeW5jX2hvb2tzXCIpLkFzeW5jTG9jYWxTdG9yYWdlPEFycmF5PHN0cmluZ1tdPj4gfCB1bmRlZmluZWR9ICovXG5sZXQgYXN5bmNMb2NhbFN0b3JhZ2VcblxuaWYgKEFzeW5jTG9jYWxTdG9yYWdlKSB7XG4gIGFzeW5jTG9jYWxTdG9yYWdlID0gbmV3IEFzeW5jTG9jYWxTdG9yYWdlKClcbn1cblxuLyoqIEBwYXJhbSB7RXJyb3J9IGVycm9yIC0gRXJyb3IgdG8gYW5ub3RhdGUgd2l0aCBhIHRyYWNrZWQgc3RhY2suICovXG5mdW5jdGlvbiBhZGRUcmFja2VkU3RhY2tUb0Vycm9yKGVycm9yKSB7XG4gIC8vIE5vdCBzdXBwb3J0ZWRcbiAgaWYgKCFhc3luY0xvY2FsU3RvcmFnZSkgcmV0dXJuXG5cbiAgY29uc3QgcGFyZW50U3RhY2tzID0gYXN5bmNMb2NhbFN0b3JhZ2UuZ2V0U3RvcmUoKSB8fCBbXVxuICBjb25zdCBhZGRpdGlvbmFsU3RhY2tMaW5lcyA9IFtdXG5cbiAgZm9yIChjb25zdCBwYXJlbnRTdGFjayBvZiBwYXJlbnRTdGFja3MpIHtcbiAgICBmb3IgKGNvbnN0IHBhcmVudFN0YWNrTGluZSBvZiBwYXJlbnRTdGFjaykge1xuICAgICAgYWRkaXRpb25hbFN0YWNrTGluZXMucHVzaChwYXJlbnRTdGFja0xpbmUpXG4gICAgfVxuICB9XG5cbiAgLy8gUmVwbGFjZSB0aGUgZXJyb3IgbWVzc2FnZSBvbiB0aGUgZmlyc3QgbGluZSB3aXRoIHRoaXMgc3RyaW5nXG4gIGVycm9yLnN0YWNrICs9IFwiXFxuXCIgKyBhZGRpdGlvbmFsU3RhY2tMaW5lcy5qb2luKFwiXFxuXCIpXG59XG5cbi8qKlxuICpcbiAqIEBwYXJhbSB7KCkgPT4gUHJvbWlzZTx2b2lkPiB8IHN0cmluZ30gYXJnMSAtIEFyZzEuXG4gKiBAcGFyYW0geygpID0+IFByb21pc2U8dm9pZD4gfCBFcnJvcn0gW2FyZzJdIC0gQXJnMi5cbiAqIEByZXR1cm5zIHtQcm9taXNlPHZvaWQ+fSAtIFJlc29sdmVzIHdoZW4gY29tcGxldGUuXG4gKi9cbmFzeW5jIGZ1bmN0aW9uIHdpdGhUcmFja2VkU3RhY2soYXJnMSwgYXJnMikge1xuICAvKiogQHR5cGUgeygpID0+IFByb21pc2U8dm9pZD59ICovXG4gIGxldCBjYWxsYmFja1xuXG4gIC8qKiBAdHlwZSB7c3RyaW5nfSAqL1xuICBsZXQgc3RhY2tcblxuICBpZiAodHlwZW9mIGFyZzIgPT0gXCJmdW5jdGlvblwiICYmIHR5cGVvZiBhcmcxID09IFwic3RyaW5nXCIpIHtcbiAgICAvLyBAdHMtZXhwZWN0LWVycm9yXG4gICAgY2FsbGJhY2sgPSBhcmcyXG4gICAgc3RhY2sgPSBhcmcxXG4gIH0gZWxzZSB7XG4gICAgLy8gQHRzLWV4cGVjdC1lcnJvclxuICAgIGNhbGxiYWNrID0gYXJnMVxuICAgIHN0YWNrID0gRXJyb3IoKS5zdGFjayB8fCBcIlwiXG4gIH1cblxuICAvLyBOb3Qgc3VwcG9ydGVkXG4gIGlmICghYXN5bmNMb2NhbFN0b3JhZ2UpIHJldHVybiBhd2FpdCBjYWxsYmFjaygpXG5cbiAgY29uc3QgcGFyZW50U3RhY2tzID0gYXN5bmNMb2NhbFN0b3JhZ2UuZ2V0U3RvcmUoKSB8fCBbXVxuICBjb25zdCBhZGRpdGlvbmFsU3RhY2tMaW5lcyA9IFtdXG4gIGNvbnN0IGN1cnJlbnRTdGFja0xpbmVzID0gc3RhY2suc3BsaXQoXCJcXG5cIilcblxuICBjdXJyZW50U3RhY2tMaW5lc1swXSA9IFwiICAgIFtXSVRIIFRSQUNLRUQgU1RBQ0tdXCJcblxuICBmb3IgKGxldCBpID0gY3VycmVudFN0YWNrTGluZXMubGVuZ3RoOyBpID49IDA7IGktLSkge1xuICAgIGNvbnN0IHN0YWNrTGluZSA9IGN1cnJlbnRTdGFja0xpbmVzW2ldXG5cbiAgICBhZGRpdGlvbmFsU3RhY2tMaW5lcy51bnNoaWZ0KHN0YWNrTGluZSlcblxuICAgIGlmIChzdGFja0xpbmUgPT0gXCIgICAgW1dJVEggVFJBQ0tFRCBTVEFDS11cIikge1xuICAgICAgYnJlYWtcbiAgICB9XG4gIH1cblxuICBjb25zdCBuZXdTdGFja3MgPSBbYWRkaXRpb25hbFN0YWNrTGluZXMsIC4uLnBhcmVudFN0YWNrc11cblxuICBhd2FpdCBhc3luY0xvY2FsU3RvcmFnZS5ydW4obmV3U3RhY2tzLCBhc3luYyAoKSA9PiB7XG4gICAgYXdhaXQgY2FsbGJhY2soKVxuICB9KVxufVxuXG5pZiAoZ2xvYmFsVGhpcy53aXRoVHJhY2tlZFN0YWNrKSB7XG4gIGNvbnNvbGUud2FybihcImdsb2JhbFRoaXMud2l0aFRyYWNrZWRTdGFjayB3YXMgYWxyZWFkeSBkZWZpbmVkXCIpXG59IGVsc2Uge1xuICBnbG9iYWxUaGlzLndpdGhUcmFja2VkU3RhY2sgPSB7YWRkVHJhY2tlZFN0YWNrVG9FcnJvciwgd2l0aFRyYWNrZWRTdGFja31cbn1cblxuZXhwb3J0IHthZGRUcmFja2VkU3RhY2tUb0Vycm9yLCB3aXRoVHJhY2tlZFN0YWNrfVxuIl19