velocious 1.0.240 → 1.0.242

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.
package/README.md CHANGED
@@ -71,10 +71,12 @@ Retry flaky tests by setting a retry count on the test args.
71
71
 
72
72
  ```js
73
73
  describe("Tasks", () => {
74
- it("retries a flaky check", {retry: 2}, async () => {})
74
+ it("retries a flaky check", {retry: 2}, async () => {})
75
75
  })
76
76
  ```
77
77
 
78
+ During a failing run, Velocious captures all console output emitted while each test executes. At the end of a failed run, those logs are saved under `tmp/screenshots` next to failure screenshots/browser logs/HTML, and each failed test summary prints the saved console log path.
79
+
78
80
  Listen for retry events if you need to restart services between attempts.
79
81
 
80
82
  ```js
@@ -1 +1 @@
1
- {"version":3,"file":"controller.d.ts","sourceRoot":"","sources":["../../src/controller.js"],"names":[],"mappings":"AAYA;IACE;;;OAGG;IACH,gCAHW,MAAM,GACJ,IAAI,CAShB;IAED;;;;;;;;;OASG;IACH,wFARG;QAAqB,MAAM,EAAnB,MAAM;QACqC,aAAa,EAAxD,OAAO,oBAAoB,EAAE,OAAO;QACvB,UAAU,EAAvB,MAAM;QACO,MAAM,EAAnB,MAAM;QACkD,OAAO,EAA/D,OAAO,iCAAiC,EAAE,OAAO;QACQ,QAAQ,EAAjE,OAAO,kCAAkC,EAAE,OAAO;QACrC,QAAQ,EAArB,MAAM;KAChB,EAmBA;IATC,gBAAqB;IACrB,oBAA6B;IAC7B,qDAAmC;IACnC,eAA8B;IAC9B,aAAqB;IACrB,4DAAuB;IACvB,8DAAyB;IACzB,eAAoB;IACpB,kBAAyB;IAG3B,uCAAuC;IACvC,aADc,MAAM,CACe;IAEnC,4EAA4E;IAC5E,oBADc,OAAO,oBAAoB,EAAE,OAAO,CACD;IAEjD,oDAAoD;IACpD,aADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CACE;IAEnC,mFAAmF;IACnF,cADc,OAAO,iCAAiC,EAAE,OAAO,CAC1B;IAErC;;;;;;;;;;;;;OAaG;IACH,gBAbW,MAAM,SACN,OAAO,SAEf;QAAsB,MAAM,GAApB,MAAM;QACM,OAAO,GAAnB,IAAI;QACW,QAAQ,GAAvB,OAAO;QACO,MAAM,GAApB,MAAM;QACQ,IAAI,GAAlB,MAAM;QACS,MAAM,GAArB,OAAO;QAC0B,QAAQ,GAAzC,KAAK,GAAG,QAAQ,GAAG,MAAM;QACV,SAAS,GAAxB,OAAO;KACf,GAAU,MAAM,CAWlB;IAED,sDAAsD;IACtD,cADc,MAAM,EAAE,CAUrB;IAJG,mBAAuD;IAM3D;;;OAGG;IACH,4BAIC;IAED,qCA4BC;IAED,oDAAoD;IACpD,UADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAQhC;IAED,8DAA8D;IAC9D,mBADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAWhC;IAED;;;;;OAKG;IACH,uCAJG;QAAsB,IAAI,GAAlB,MAAM;QACiB,MAAM,GAA7B,MAAM,GAAG,MAAM;KACvB,GAAU,OAAO,CAAC,IAAI,CAAC,CAczB;IAED,2CAA2C;IAC3C,oBADY,MAAM,QAMjB;IAED,0DAA0D;IAC1D,cADc,OAAO,CAAC,IAAI,CAAC,CA8B1B;IAED,0CAA0C;IAC1C,cADc,IAAI,CAGjB;IAED;;;;;;;OAOG;IACH,mBANW,MAAM,SAEd;QAAsB,WAAW,GAAzB,MAAM;QACiB,MAAM,GAA7B,MAAM,GAAG,MAAM;KACvB,GAAU,IAAI,CAsBhB;IAED;;;OAGG;IACH,8BAHW,MAAM,GACJ,MAAM,CAiBlB;IAED,+GAA+G;IAC/G,kBADc,OAAO,4BAA4B,EAAE,OAAO,GAAG,SAAS,CAGrE;IAED,mFAAmF;IACnF,WADc,OAAO,iCAAiC,EAAE,OAAO,CAC7B;IAElC,qFAAqF;IACrF,YADc,OAAO,kCAAkC,EAAE,OAAO,CAC5B;CACrC;mBAjSkB,aAAa;mBACb,yBAAyB"}
1
+ {"version":3,"file":"controller.d.ts","sourceRoot":"","sources":["../../src/controller.js"],"names":[],"mappings":"AAYA;IACE;;;OAGG;IACH,gCAHW,MAAM,GACJ,IAAI,CAShB;IAED;;;;;;;;;OASG;IACH,wFARG;QAAqB,MAAM,EAAnB,MAAM;QACqC,aAAa,EAAxD,OAAO,oBAAoB,EAAE,OAAO;QACvB,UAAU,EAAvB,MAAM;QACO,MAAM,EAAnB,MAAM;QACkD,OAAO,EAA/D,OAAO,iCAAiC,EAAE,OAAO;QACQ,QAAQ,EAAjE,OAAO,kCAAkC,EAAE,OAAO;QACrC,QAAQ,EAArB,MAAM;KAChB,EAmBA;IATC,gBAAqB;IACrB,oBAA6B;IAC7B,qDAAmC;IACnC,eAA8B;IAC9B,aAAqB;IACrB,4DAAuB;IACvB,8DAAyB;IACzB,eAAoB;IACpB,kBAAyB;IAG3B,uCAAuC;IACvC,aADc,MAAM,CACe;IAEnC,4EAA4E;IAC5E,oBADc,OAAO,oBAAoB,EAAE,OAAO,CACD;IAEjD,oDAAoD;IACpD,aADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CACE;IAEnC,mFAAmF;IACnF,cADc,OAAO,iCAAiC,EAAE,OAAO,CAC1B;IAErC;;;;;;;;;;;;;OAaG;IACH,gBAbW,MAAM,SACN,OAAO,SAEf;QAAsB,MAAM,GAApB,MAAM;QACM,OAAO,GAAnB,IAAI;QACW,QAAQ,GAAvB,OAAO;QACO,MAAM,GAApB,MAAM;QACQ,IAAI,GAAlB,MAAM;QACS,MAAM,GAArB,OAAO;QAC0B,QAAQ,GAAzC,KAAK,GAAG,QAAQ,GAAG,MAAM;QACV,SAAS,GAAxB,OAAO;KACf,GAAU,MAAM,CAWlB;IAED,sDAAsD;IACtD,cADc,MAAM,EAAE,CAUrB;IAJG,mBAAuD;IAM3D;;;OAGG;IACH,4BAIC;IAED,qCA4BC;IAED,oDAAoD;IACpD,UADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAQhC;IAED,8DAA8D;IAC9D,mBADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CA4BhC;IAED;;;;;OAKG;IACH,uCAJG;QAAsB,IAAI,GAAlB,MAAM;QACiB,MAAM,GAA7B,MAAM,GAAG,MAAM;KACvB,GAAU,OAAO,CAAC,IAAI,CAAC,CAczB;IAED,2CAA2C;IAC3C,oBADY,MAAM,QAMjB;IAED,0DAA0D;IAC1D,cADc,OAAO,CAAC,IAAI,CAAC,CA8B1B;IAED,0CAA0C;IAC1C,cADc,IAAI,CAGjB;IAED;;;;;;;OAOG;IACH,mBANW,MAAM,SAEd;QAAsB,WAAW,GAAzB,MAAM;QACiB,MAAM,GAA7B,MAAM,GAAG,MAAM;KACvB,GAAU,IAAI,CAsBhB;IAED;;;OAGG;IACH,8BAHW,MAAM,GACJ,MAAM,CAiBlB;IAED,+GAA+G;IAC/G,kBADc,OAAO,4BAA4B,EAAE,OAAO,GAAG,SAAS,CAGrE;IAED,mFAAmF;IACnF,WADc,OAAO,iCAAiC,EAAE,OAAO,CAC7B;IAElC,qFAAqF;IACrF,YADc,OAAO,kCAAkC,EAAE,OAAO,CAC5B;CACrC;mBAlTkB,aAAa;mBACb,yBAAyB"}
@@ -136,10 +136,26 @@ export default class VelociousController {
136
136
  const query = this._request.path().split("?")[1];
137
137
  if (!query)
138
138
  return {};
139
- /** @type {Record<string, any>} */
140
- const unparsedParams = querystring.parse(query);
141
- const paramsToObject = new ParamsToObject(unparsedParams);
142
- return paramsToObject.toObject();
139
+ try {
140
+ /** @type {Record<string, any>} */
141
+ const unparsedParams = querystring.parse(query);
142
+ const paramsToObject = new ParamsToObject(unparsedParams);
143
+ return paramsToObject.toObject();
144
+ }
145
+ catch (error) {
146
+ const ensuredError = /** @type {Error & {velociousContext?: Record<string, unknown>}} */ (error);
147
+ ensuredError.velociousContext = {
148
+ ...(ensuredError.velociousContext || {}),
149
+ requestParsing: {
150
+ httpMethod: this._request.httpMethod(),
151
+ parameterKeys: Object.keys(querystring.parse(query)),
152
+ path: this._request.path(),
153
+ queryPreview: query.length > 300 ? `${query.slice(0, 300)}...` : query,
154
+ stage: "query-parameters"
155
+ }
156
+ };
157
+ throw ensuredError;
158
+ }
143
159
  }
144
160
  /**
145
161
  * @param {object} [args] - Options object.
@@ -255,4 +271,4 @@ export default class VelociousController {
255
271
  /** @returns {import("./http-server/client/response.js").default} - The response. */
256
272
  response() { return this._response; }
257
273
  }
258
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"controller.js","sourceRoot":"","sources":["../../src/controller.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport ejs from \"ejs\"\nimport {incorporate} from \"incorporator\"\nimport * as inflection from \"inflection\"\nimport Logger from \"./logger.js\"\nimport Cookie from \"./http-server/cookie.js\"\nimport ParamsToObject from \"./http-server/client/params-to-object.js\"\nimport path from \"node:path\"\nimport restArgsError from \"./utils/rest-args-error.js\"\nimport querystring from \"querystring\"\n\nexport default class VelociousController {\n  /**\n   * @param {string} methodName - Method name.\n   * @returns {void} - No return value.\n   */\n  static beforeAction(methodName) {\n    if (!this._beforeActions) {\n      /** @type {Array<string>}  */\n      this._beforeActions = []\n    }\n\n    this._beforeActions.push(methodName)\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.action - Action.\n   * @param {import(\"./configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} args.controller - Controller.\n   * @param {object} args.params - Parameters object.\n   * @param {import(\"./http-server/client/request.js\").default} args.request - Request object.\n   * @param {import(\"./http-server/client/response.js\").default} args.response - Response object.\n   * @param {string} args.viewPath - View path.\n   */\n  constructor({action, configuration, controller, params, request, response, viewPath}) {\n    if (!action) throw new Error(\"No action given\")\n    if (!configuration) throw new Error(\"No configuration given\")\n    if (!controller) throw new Error(\"No controller given\")\n    if (!params) throw new Error(\"No params given\")\n    if (!request) throw new Error(\"No request given\")\n    if (!response) throw new Error(\"No response given\")\n    if (!viewPath) throw new Error(\"No viewPath given\")\n\n    this._action = action\n    this._controller = controller\n    this._configuration = configuration\n    this.logger = new Logger(this)\n    this._params = params\n    this._request = request\n    this._response = response\n    this.viewParams = {}\n    this._viewPath = viewPath\n  }\n\n  /** @returns {string} - The action.  */\n  getAction() { return this._action }\n\n  /** @returns {import(\"./configuration.js\").default} - The configuration.  */\n  getConfiguration() { return this._configuration }\n\n  /** @returns {Record<string, any>} - The params.  */\n  getParams() { return this._params }\n\n  /** @returns {import(\"./http-server/client/request.js\").default} - The request.  */\n  getRequest() { return this._request }\n\n  /**\n   * @param {string} name - Cookie name.\n   * @param {unknown} value - Cookie value.\n   * @param {object} [args] - Options object.\n   * @param {string} [args.domain] - Domain.\n   * @param {Date} [args.expires] - Expires date.\n   * @param {boolean} [args.httpOnly] - HttpOnly flag.\n   * @param {number} [args.maxAge] - Max-Age in seconds.\n   * @param {string} [args.path] - Path.\n   * @param {boolean} [args.secure] - Secure flag.\n   * @param {\"Lax\" | \"Strict\" | \"None\"} [args.sameSite] - SameSite value.\n   * @param {boolean} [args.encrypted] - Whether to encrypt the cookie value.\n   * @returns {Cookie} - Cookie instance.\n   */\n  setCookie(name, value, args = {}) {\n    const {encrypted = false, ...options} = args\n    const secret = encrypted ? this.getConfiguration().getCookieSecret() : undefined\n    const cookieValue = encrypted ? Cookie.encryptValue(value, secret) : String(value ?? \"\")\n    const cookie = new Cookie({name, value: cookieValue, options, encrypted})\n\n    this._response.addHeader(\"Set-Cookie\", cookie.toHeader())\n\n    return cookie\n  }\n\n  /** @returns {Cookie[]} - Cookies from the request. */\n  getCookies() {\n    if (!this._cookies) {\n      const secret = this.getConfiguration().getCookieSecret()\n      const headerValue = this._request.header(\"cookie\")\n\n      this._cookies = Cookie.parseHeader(headerValue, secret)\n    }\n\n    return this._cookies\n  }\n\n  /**\n   * @private\n   * @returns {typeof VelociousController} - The controller class.\n   */\n  _getControllerClass() {\n    const controllerClass = /** @type {typeof VelociousController} */ (this.constructor)\n\n    return controllerClass\n  }\n\n  async _runBeforeCallbacks() {\n    await this.logger.debug(\"_runBeforeCallbacks\")\n\n    let currentControllerClass = this._getControllerClass()\n\n    while (currentControllerClass) {\n      await this.logger.debug(`Running callbacks for ${currentControllerClass.name}`)\n\n      const beforeActions = currentControllerClass._beforeActions\n\n      if (beforeActions) {\n        for (const beforeActionName of beforeActions) {\n          const beforeAction = currentControllerClass.prototype[beforeActionName]\n\n          if (!beforeAction) throw new Error(`No such before action: ${beforeActionName}`)\n\n          const boundBeforeAction = beforeAction.bind(this)\n\n          await boundBeforeAction()\n        }\n      }\n\n      currentControllerClass = Object.getPrototypeOf(currentControllerClass)\n\n      if (!currentControllerClass?.name?.endsWith(\"Controller\")) break\n    }\n\n    await this.logger.debug(\"After runBeforeCallbacks\")\n  }\n\n  /** @returns {Record<string, any>} - The params.  */\n  params() {\n    // Merge query parameters so controllers can read them via params()\n    const mergedParams = {...this.queryParameters(), ...this._params}\n\n    if (!mergedParams.controller) mergedParams.controller = this._controller\n\n    return mergedParams\n  }\n\n  /** @returns {Record<string, any>} - The query parameters.  */\n  queryParameters() {\n    const query = this._request.path().split(\"?\")[1]\n\n    if (!query) return {}\n\n    /** @type {Record<string, any>} */\n    const unparsedParams = querystring.parse(query)\n    const paramsToObject = new ParamsToObject(unparsedParams)\n\n    return paramsToObject.toObject()\n  }\n\n  /**\n   * @param {object} [args] - Options object.\n   * @param {object} [args.json] - Json.\n   * @param {number | string} [args.status] - Status.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async render({json, status, ...restArgs} = {}) {\n    restArgsError(restArgs)\n\n    if (json) {\n      return this.renderJsonArg(json)\n    }\n\n    if (status) {\n      this._response.setStatus(status)\n    }\n\n    return await this.renderView()\n  }\n\n  /** @param {object} json - JSON payload. */\n  renderJsonArg(json) {\n    const body = JSON.stringify(json)\n\n    this._response.setHeader(\"Content-Type\", \"application/json; charset=UTF-8\")\n    this._response.setBody(body)\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  renderView() {\n    return new Promise((resolve, reject) => {\n      const viewPath = `${this._viewPath}/${inflection.dasherize(inflection.underscore(this._action))}.ejs`\n      const actualViewParams = incorporate({controller: this}, this.viewParams)\n\n      ejs.renderFile(viewPath, actualViewParams, {}, (err, str) => {\n        if (err) {\n          if (err.code === \"ENOENT\") {\n            this.logger.warn(`Missing view file: ${viewPath}`)\n\n            if (this._response.getStatusCode() === 200) {\n              this._response.setStatus(\"internal-server-error\")\n            }\n\n            this._response.setHeader(\"Content-Type\", \"text/plain; charset=UTF-8\")\n            this._response.setBody(`Missing view file: ${viewPath}`)\n\n            resolve(null)\n          } else {\n            reject(err)\n          }\n        } else {\n          this._response.setHeader(\"Content-Type\", \"text/html; charset=UTF-8\")\n          this._response.setBody(str)\n\n          resolve(null)\n        }\n      })\n    })\n  }\n\n  /** @returns {void} - No return value.  */\n  renderText() {\n    throw new Error(\"renderText stub\")\n  }\n\n  /**\n   * Streams a file response from disk without loading the full file into controller memory.\n   * @param {string} filePath - File path.\n   * @param {object} [args] - Options object.\n   * @param {string} [args.contentType] - Content type.\n   * @param {number | string} [args.status] - Status.\n   * @returns {void} - No return value.\n   */\n  sendFile(filePath, args = {}) {\n    const {contentType, status, ...restArgs} = args\n\n    restArgsError(restArgs)\n\n    if (typeof filePath !== \"string\" || filePath.length < 1) {\n      throw new Error(`Expected file path to be a non-empty string, got: ${String(filePath)}`)\n    }\n\n    const detectedContentType = contentType || this.sendFileContentType(filePath)\n\n    if (detectedContentType) {\n      this._response.setHeader(\"Content-Type\", detectedContentType)\n    }\n\n    if (status) {\n      this._response.setStatus(status)\n    }\n\n    this._response.setFilePath(filePath)\n  }\n\n  /**\n   * @param {string} filePath - File path.\n   * @returns {string} - Content type value.\n   */\n  sendFileContentType(filePath) {\n    const extension = path.extname(filePath).toLowerCase()\n\n    if (extension === \".wasm\") return \"application/wasm\"\n    if (extension === \".js\") return \"text/javascript; charset=UTF-8\"\n    if (extension === \".json\") return \"application/json; charset=UTF-8\"\n    if (extension === \".css\") return \"text/css; charset=UTF-8\"\n    if (extension === \".html\") return \"text/html; charset=UTF-8\"\n    if (extension === \".txt\") return \"text/plain; charset=UTF-8\"\n    if (extension === \".svg\") return \"image/svg+xml\"\n    if (extension === \".png\") return \"image/png\"\n    if (extension === \".jpg\" || extension === \".jpeg\") return \"image/jpeg\"\n    if (extension === \".gif\") return \"image/gif\"\n\n    return \"application/octet-stream\"\n  }\n\n  /** @returns {import(\"./authorization/ability.js\").default | undefined} - Current ability for request scope. */\n  currentAbility() {\n    return this.getConfiguration().getCurrentAbility()\n  }\n\n  /** @returns {import(\"./http-server/client/request.js\").default} - The request.  */\n  request() { return this._request }\n\n  /** @returns {import(\"./http-server/client/response.js\").default} - The response.  */\n  response() { return this._response }\n}\n"]}
274
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"controller.js","sourceRoot":"","sources":["../../src/controller.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport ejs from \"ejs\"\nimport {incorporate} from \"incorporator\"\nimport * as inflection from \"inflection\"\nimport Logger from \"./logger.js\"\nimport Cookie from \"./http-server/cookie.js\"\nimport ParamsToObject from \"./http-server/client/params-to-object.js\"\nimport path from \"node:path\"\nimport restArgsError from \"./utils/rest-args-error.js\"\nimport querystring from \"querystring\"\n\nexport default class VelociousController {\n  /**\n   * @param {string} methodName - Method name.\n   * @returns {void} - No return value.\n   */\n  static beforeAction(methodName) {\n    if (!this._beforeActions) {\n      /** @type {Array<string>}  */\n      this._beforeActions = []\n    }\n\n    this._beforeActions.push(methodName)\n  }\n\n  /**\n   * @param {object} args - Options object.\n   * @param {string} args.action - Action.\n   * @param {import(\"./configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} args.controller - Controller.\n   * @param {object} args.params - Parameters object.\n   * @param {import(\"./http-server/client/request.js\").default} args.request - Request object.\n   * @param {import(\"./http-server/client/response.js\").default} args.response - Response object.\n   * @param {string} args.viewPath - View path.\n   */\n  constructor({action, configuration, controller, params, request, response, viewPath}) {\n    if (!action) throw new Error(\"No action given\")\n    if (!configuration) throw new Error(\"No configuration given\")\n    if (!controller) throw new Error(\"No controller given\")\n    if (!params) throw new Error(\"No params given\")\n    if (!request) throw new Error(\"No request given\")\n    if (!response) throw new Error(\"No response given\")\n    if (!viewPath) throw new Error(\"No viewPath given\")\n\n    this._action = action\n    this._controller = controller\n    this._configuration = configuration\n    this.logger = new Logger(this)\n    this._params = params\n    this._request = request\n    this._response = response\n    this.viewParams = {}\n    this._viewPath = viewPath\n  }\n\n  /** @returns {string} - The action.  */\n  getAction() { return this._action }\n\n  /** @returns {import(\"./configuration.js\").default} - The configuration.  */\n  getConfiguration() { return this._configuration }\n\n  /** @returns {Record<string, any>} - The params.  */\n  getParams() { return this._params }\n\n  /** @returns {import(\"./http-server/client/request.js\").default} - The request.  */\n  getRequest() { return this._request }\n\n  /**\n   * @param {string} name - Cookie name.\n   * @param {unknown} value - Cookie value.\n   * @param {object} [args] - Options object.\n   * @param {string} [args.domain] - Domain.\n   * @param {Date} [args.expires] - Expires date.\n   * @param {boolean} [args.httpOnly] - HttpOnly flag.\n   * @param {number} [args.maxAge] - Max-Age in seconds.\n   * @param {string} [args.path] - Path.\n   * @param {boolean} [args.secure] - Secure flag.\n   * @param {\"Lax\" | \"Strict\" | \"None\"} [args.sameSite] - SameSite value.\n   * @param {boolean} [args.encrypted] - Whether to encrypt the cookie value.\n   * @returns {Cookie} - Cookie instance.\n   */\n  setCookie(name, value, args = {}) {\n    const {encrypted = false, ...options} = args\n    const secret = encrypted ? this.getConfiguration().getCookieSecret() : undefined\n    const cookieValue = encrypted ? Cookie.encryptValue(value, secret) : String(value ?? \"\")\n    const cookie = new Cookie({name, value: cookieValue, options, encrypted})\n\n    this._response.addHeader(\"Set-Cookie\", cookie.toHeader())\n\n    return cookie\n  }\n\n  /** @returns {Cookie[]} - Cookies from the request. */\n  getCookies() {\n    if (!this._cookies) {\n      const secret = this.getConfiguration().getCookieSecret()\n      const headerValue = this._request.header(\"cookie\")\n\n      this._cookies = Cookie.parseHeader(headerValue, secret)\n    }\n\n    return this._cookies\n  }\n\n  /**\n   * @private\n   * @returns {typeof VelociousController} - The controller class.\n   */\n  _getControllerClass() {\n    const controllerClass = /** @type {typeof VelociousController} */ (this.constructor)\n\n    return controllerClass\n  }\n\n  async _runBeforeCallbacks() {\n    await this.logger.debug(\"_runBeforeCallbacks\")\n\n    let currentControllerClass = this._getControllerClass()\n\n    while (currentControllerClass) {\n      await this.logger.debug(`Running callbacks for ${currentControllerClass.name}`)\n\n      const beforeActions = currentControllerClass._beforeActions\n\n      if (beforeActions) {\n        for (const beforeActionName of beforeActions) {\n          const beforeAction = currentControllerClass.prototype[beforeActionName]\n\n          if (!beforeAction) throw new Error(`No such before action: ${beforeActionName}`)\n\n          const boundBeforeAction = beforeAction.bind(this)\n\n          await boundBeforeAction()\n        }\n      }\n\n      currentControllerClass = Object.getPrototypeOf(currentControllerClass)\n\n      if (!currentControllerClass?.name?.endsWith(\"Controller\")) break\n    }\n\n    await this.logger.debug(\"After runBeforeCallbacks\")\n  }\n\n  /** @returns {Record<string, any>} - The params.  */\n  params() {\n    // Merge query parameters so controllers can read them via params()\n    const mergedParams = {...this.queryParameters(), ...this._params}\n\n    if (!mergedParams.controller) mergedParams.controller = this._controller\n\n    return mergedParams\n  }\n\n  /** @returns {Record<string, any>} - The query parameters.  */\n  queryParameters() {\n    const query = this._request.path().split(\"?\")[1]\n\n    if (!query) return {}\n\n    try {\n      /** @type {Record<string, any>} */\n      const unparsedParams = querystring.parse(query)\n      const paramsToObject = new ParamsToObject(unparsedParams)\n\n      return paramsToObject.toObject()\n    } catch (error) {\n      const ensuredError = /** @type {Error & {velociousContext?: Record<string, unknown>}} */ (error)\n\n      ensuredError.velociousContext = {\n        ...(ensuredError.velociousContext || {}),\n        requestParsing: {\n          httpMethod: this._request.httpMethod(),\n          parameterKeys: Object.keys(querystring.parse(query)),\n          path: this._request.path(),\n          queryPreview: query.length > 300 ? `${query.slice(0, 300)}...` : query,\n          stage: \"query-parameters\"\n        }\n      }\n\n      throw ensuredError\n    }\n  }\n\n  /**\n   * @param {object} [args] - Options object.\n   * @param {object} [args.json] - Json.\n   * @param {number | string} [args.status] - Status.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async render({json, status, ...restArgs} = {}) {\n    restArgsError(restArgs)\n\n    if (json) {\n      return this.renderJsonArg(json)\n    }\n\n    if (status) {\n      this._response.setStatus(status)\n    }\n\n    return await this.renderView()\n  }\n\n  /** @param {object} json - JSON payload. */\n  renderJsonArg(json) {\n    const body = JSON.stringify(json)\n\n    this._response.setHeader(\"Content-Type\", \"application/json; charset=UTF-8\")\n    this._response.setBody(body)\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  renderView() {\n    return new Promise((resolve, reject) => {\n      const viewPath = `${this._viewPath}/${inflection.dasherize(inflection.underscore(this._action))}.ejs`\n      const actualViewParams = incorporate({controller: this}, this.viewParams)\n\n      ejs.renderFile(viewPath, actualViewParams, {}, (err, str) => {\n        if (err) {\n          if (err.code === \"ENOENT\") {\n            this.logger.warn(`Missing view file: ${viewPath}`)\n\n            if (this._response.getStatusCode() === 200) {\n              this._response.setStatus(\"internal-server-error\")\n            }\n\n            this._response.setHeader(\"Content-Type\", \"text/plain; charset=UTF-8\")\n            this._response.setBody(`Missing view file: ${viewPath}`)\n\n            resolve(null)\n          } else {\n            reject(err)\n          }\n        } else {\n          this._response.setHeader(\"Content-Type\", \"text/html; charset=UTF-8\")\n          this._response.setBody(str)\n\n          resolve(null)\n        }\n      })\n    })\n  }\n\n  /** @returns {void} - No return value.  */\n  renderText() {\n    throw new Error(\"renderText stub\")\n  }\n\n  /**\n   * Streams a file response from disk without loading the full file into controller memory.\n   * @param {string} filePath - File path.\n   * @param {object} [args] - Options object.\n   * @param {string} [args.contentType] - Content type.\n   * @param {number | string} [args.status] - Status.\n   * @returns {void} - No return value.\n   */\n  sendFile(filePath, args = {}) {\n    const {contentType, status, ...restArgs} = args\n\n    restArgsError(restArgs)\n\n    if (typeof filePath !== \"string\" || filePath.length < 1) {\n      throw new Error(`Expected file path to be a non-empty string, got: ${String(filePath)}`)\n    }\n\n    const detectedContentType = contentType || this.sendFileContentType(filePath)\n\n    if (detectedContentType) {\n      this._response.setHeader(\"Content-Type\", detectedContentType)\n    }\n\n    if (status) {\n      this._response.setStatus(status)\n    }\n\n    this._response.setFilePath(filePath)\n  }\n\n  /**\n   * @param {string} filePath - File path.\n   * @returns {string} - Content type value.\n   */\n  sendFileContentType(filePath) {\n    const extension = path.extname(filePath).toLowerCase()\n\n    if (extension === \".wasm\") return \"application/wasm\"\n    if (extension === \".js\") return \"text/javascript; charset=UTF-8\"\n    if (extension === \".json\") return \"application/json; charset=UTF-8\"\n    if (extension === \".css\") return \"text/css; charset=UTF-8\"\n    if (extension === \".html\") return \"text/html; charset=UTF-8\"\n    if (extension === \".txt\") return \"text/plain; charset=UTF-8\"\n    if (extension === \".svg\") return \"image/svg+xml\"\n    if (extension === \".png\") return \"image/png\"\n    if (extension === \".jpg\" || extension === \".jpeg\") return \"image/jpeg\"\n    if (extension === \".gif\") return \"image/gif\"\n\n    return \"application/octet-stream\"\n  }\n\n  /** @returns {import(\"./authorization/ability.js\").default | undefined} - Current ability for request scope. */\n  currentAbility() {\n    return this.getConfiguration().getCurrentAbility()\n  }\n\n  /** @returns {import(\"./http-server/client/request.js\").default} - The request.  */\n  request() { return this._request }\n\n  /** @returns {import(\"./http-server/client/response.js\").default} - The response.  */\n  response() { return this._response }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"test.d.ts","sourceRoot":"","sources":["../../../../../../src/environment-handlers/node/cli/commands/test.js"],"names":[],"mappings":"AAQA;IACE,yBAyFC;CACF;wBAjGuB,iCAAiC"}
1
+ {"version":3,"file":"test.d.ts","sourceRoot":"","sources":["../../../../../../src/environment-handlers/node/cli/commands/test.js"],"names":[],"mappings":"AAQA;IACE,yBA8FC;CACF;wBAtGuB,iCAAiC"}
@@ -63,6 +63,7 @@ export default class VelociousCliCommandsTest extends BaseCommand {
63
63
  process.exit(1);
64
64
  }
65
65
  if (testRunner.isFailed()) {
66
+ await testRunner.persistFailedTestConsoleOutputsToAssets();
66
67
  const failedTests = testRunner.getFailedTestDetails();
67
68
  if (failedTests.length > 0) {
68
69
  console.error(picocolors.red("\nFailed tests:"));
@@ -71,6 +72,9 @@ export default class VelociousCliCommandsTest extends BaseCommand {
71
72
  ? ` (${failed.filePath}:${failed.line})`
72
73
  : "";
73
74
  console.error(picocolors.red(`- ${failed.fullDescription}${location}`));
75
+ if (failed.consoleLogPath) {
76
+ console.error(picocolors.red(` Console log: ${failed.consoleLogPath}`));
77
+ }
74
78
  }
75
79
  }
76
80
  console.error(picocolors.red(`\nTest run failed with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`));
@@ -86,4 +90,4 @@ export default class VelociousCliCommandsTest extends BaseCommand {
86
90
  }
87
91
  }
88
92
  }
89
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.js","sourceRoot":"","sources":["../../../../../../src/environment-handlers/node/cli/commands/test.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport BaseCommand from \"../../../../cli/base-command.js\"\nimport picocolors from \"picocolors\"\nimport TestFilesFinder from \"../../../../testing/test-files-finder.js\"\nimport TestRunner from \"../../../../testing/test-runner.js\"\nimport {normalizeExamplePatterns, parseFilters} from \"../../../../testing/test-filter-parser.js\"\n\nexport default class VelociousCliCommandsTest extends BaseCommand {\n  async execute() {\n    this.getConfiguration().setEnvironment(\"test\")\n\n    let directory\n    const directories = []\n\n    if (process.env.VELOCIOUS_TEST_DIR) {\n      directory = process.env.VELOCIOUS_TEST_DIR\n      directories.push(process.env.VELOCIOUS_TEST_DIR)\n    } else {\n      directory = this.directory()\n      directories.push(`${this.directory()}/__tests__`)\n      directories.push(`${this.directory()}/tests`)\n      directories.push(`${this.directory()}/spec`)\n    }\n\n    const {includeTags, excludeTags, examplePatterns, filteredProcessArgs} = parseFilters(this.processArgs || [])\n    const testFilesFinder = new TestFilesFinder({directory, directories, processArgs: filteredProcessArgs})\n    const testFiles = await testFilesFinder.findTestFiles()\n    const testRunner = new TestRunner({\n      configuration: this.getConfiguration(),\n      excludeTags,\n      includeTags,\n      testFiles,\n      lineFilters: testFilesFinder.getLineFiltersByFile(),\n      examplePatterns: normalizeExamplePatterns(examplePatterns)\n    })\n    let signalHandled = false\n\n    const handleSignal = async (signal) => {\n      if (signalHandled) return\n      signalHandled = true\n      console.error(`\\nReceived ${signal}, running afterAll hooks before exit...`)\n\n      try {\n        await testRunner.runAfterAllsForActiveScopes()\n      } catch (error) {\n        console.error(\"Failed while running afterAll hooks:\", error)\n      } finally {\n        process.exit(130)\n      }\n    }\n\n    process.once(\"SIGINT\", () => { void handleSignal(\"SIGINT\") })\n    process.once(\"SIGTERM\", () => { void handleSignal(\"SIGTERM\") })\n\n    await testRunner.prepare()\n\n    if (testRunner.getTestsCount() === 0) {\n      throw new Error(`${testRunner.getTestsCount()} tests was found in ${testFiles.length} file(s)`)\n    }\n\n    await testRunner.run()\n\n    const executedTests = testRunner.getExecutedTestsCount()\n\n    const lineFilters = testRunner.getLineFilters()\n    const hasLineFilters = Object.keys(lineFilters).length > 0\n    const hasExampleFilters = examplePatterns.length > 0\n    const hasTagFilters = includeTags.length > 0 || excludeTags.length > 0\n\n    if ((hasTagFilters || hasLineFilters || hasExampleFilters) && executedTests === 0) {\n      console.error(picocolors.red(\"\\nNo tests matched the provided filters\"))\n      process.exit(1)\n    }\n\n    if (testRunner.isFailed()) {\n      const failedTests = testRunner.getFailedTestDetails()\n\n      if (failedTests.length > 0) {\n      console.error(picocolors.red(\"\\nFailed tests:\"))\n\n        for (const failed of failedTests) {\n          const location = failed.filePath && failed.line\n            ? ` (${failed.filePath}:${failed.line})`\n            : \"\"\n          console.error(picocolors.red(`- ${failed.fullDescription}${location}`))\n        }\n      }\n\n      console.error(picocolors.red(`\\nTest run failed with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`))\n      process.exit(1)\n    } else if (testRunner.areAnyTestsFocussed()) {\n      console.error(picocolors.red(`\\nFocussed run with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`))\n      process.exit(1)\n    } else {\n      console.log(picocolors.green(`\\nTest run succeeded with ${testRunner.getSuccessfulTests()} successful tests`))\n      process.exit(0)\n    }\n  }\n}\n"]}
93
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test.js","sourceRoot":"","sources":["../../../../../../src/environment-handlers/node/cli/commands/test.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport BaseCommand from \"../../../../cli/base-command.js\"\nimport picocolors from \"picocolors\"\nimport TestFilesFinder from \"../../../../testing/test-files-finder.js\"\nimport TestRunner from \"../../../../testing/test-runner.js\"\nimport {normalizeExamplePatterns, parseFilters} from \"../../../../testing/test-filter-parser.js\"\n\nexport default class VelociousCliCommandsTest extends BaseCommand {\n  async execute() {\n    this.getConfiguration().setEnvironment(\"test\")\n\n    let directory\n    const directories = []\n\n    if (process.env.VELOCIOUS_TEST_DIR) {\n      directory = process.env.VELOCIOUS_TEST_DIR\n      directories.push(process.env.VELOCIOUS_TEST_DIR)\n    } else {\n      directory = this.directory()\n      directories.push(`${this.directory()}/__tests__`)\n      directories.push(`${this.directory()}/tests`)\n      directories.push(`${this.directory()}/spec`)\n    }\n\n    const {includeTags, excludeTags, examplePatterns, filteredProcessArgs} = parseFilters(this.processArgs || [])\n    const testFilesFinder = new TestFilesFinder({directory, directories, processArgs: filteredProcessArgs})\n    const testFiles = await testFilesFinder.findTestFiles()\n    const testRunner = new TestRunner({\n      configuration: this.getConfiguration(),\n      excludeTags,\n      includeTags,\n      testFiles,\n      lineFilters: testFilesFinder.getLineFiltersByFile(),\n      examplePatterns: normalizeExamplePatterns(examplePatterns)\n    })\n    let signalHandled = false\n\n    const handleSignal = async (signal) => {\n      if (signalHandled) return\n      signalHandled = true\n      console.error(`\\nReceived ${signal}, running afterAll hooks before exit...`)\n\n      try {\n        await testRunner.runAfterAllsForActiveScopes()\n      } catch (error) {\n        console.error(\"Failed while running afterAll hooks:\", error)\n      } finally {\n        process.exit(130)\n      }\n    }\n\n    process.once(\"SIGINT\", () => { void handleSignal(\"SIGINT\") })\n    process.once(\"SIGTERM\", () => { void handleSignal(\"SIGTERM\") })\n\n    await testRunner.prepare()\n\n    if (testRunner.getTestsCount() === 0) {\n      throw new Error(`${testRunner.getTestsCount()} tests was found in ${testFiles.length} file(s)`)\n    }\n\n    await testRunner.run()\n\n    const executedTests = testRunner.getExecutedTestsCount()\n\n    const lineFilters = testRunner.getLineFilters()\n    const hasLineFilters = Object.keys(lineFilters).length > 0\n    const hasExampleFilters = examplePatterns.length > 0\n    const hasTagFilters = includeTags.length > 0 || excludeTags.length > 0\n\n    if ((hasTagFilters || hasLineFilters || hasExampleFilters) && executedTests === 0) {\n      console.error(picocolors.red(\"\\nNo tests matched the provided filters\"))\n      process.exit(1)\n    }\n\n    if (testRunner.isFailed()) {\n      await testRunner.persistFailedTestConsoleOutputsToAssets()\n      const failedTests = testRunner.getFailedTestDetails()\n\n      if (failedTests.length > 0) {\n      console.error(picocolors.red(\"\\nFailed tests:\"))\n\n        for (const failed of failedTests) {\n          const location = failed.filePath && failed.line\n            ? ` (${failed.filePath}:${failed.line})`\n            : \"\"\n          console.error(picocolors.red(`- ${failed.fullDescription}${location}`))\n\n          if (failed.consoleLogPath) {\n            console.error(picocolors.red(`  Console log: ${failed.consoleLogPath}`))\n          }\n        }\n      }\n\n      console.error(picocolors.red(`\\nTest run failed with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`))\n      process.exit(1)\n    } else if (testRunner.areAnyTestsFocussed()) {\n      console.error(picocolors.red(`\\nFocussed run with ${testRunner.getFailedTests()} failed tests and ${testRunner.getSuccessfulTests()} successfull`))\n      process.exit(1)\n    } else {\n      console.log(picocolors.green(`\\nTest run succeeded with ${testRunner.getSuccessfulTests()} successful tests`))\n      process.exit(0)\n    }\n  }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"AAuBA;IAIE;;;;;OAKG;IACH,2DAJG;QAAqB,WAAW,EAAxB,MAAM;QACyC,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QAC1B,aAAa,GAA3B,MAAM;KAChB,EAWA;IAnBD,mEAA2B;IAC3B,cAAiB;IAWf,eAA8B;IAC9B,oBAA8B;IAC9B,wDAAkC;IAClC,sBAAkC;IAElC,8BAA8B;IAC9B,gBADW,aAAa,EAAE,CACF;IAG1B;;;OAGG;IACH,iCAHW,MAAM,GACJ,IAAI,CAgBhB;IAED;;;OAGG;IACH,iCAHW,MAAM,GACJ,IAAI,CAgBhB;IAED;;;OAGG;IACH,wBAHW,KAAK,GACH,IAAI,CAehB;IAJC,wBAA+B;IAMjC,kCA6BC;IAED;;;OAGG;IACH,cAHW,MAAM,GACJ,IAAI,CAsDhB;IAED;;;OAGG;IACH,6BAHW,OAAO,cAAc,EAAE,OAAO,GAC5B,OAAO,CAOnB;IAED,0CAA0C;IAC1C,uBADc,IAAI,CA4DjB;IAfC,mCAME;IAWJ,wBAMC;IAED,kCA6BC;IAED;;;OAGG;IACH,4BAHW,aAAa,GACX,OAAO,CAAC,IAAI,CAAC,CAyEzB;IAED;;;OAGG;IACH,yBAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAmBzB;IAED;;;OAGG;IACH,+BAHW,OAAO,cAAc,EAAE,OAAO,GAAG,OAAO,wBAAwB,EAAE,OAAO,GACvE,OAAO,CAenB;CACF;mBA/ZkB,iBAAiB;0BAEV,qBAAqB;oBAD3B,cAAc;6BAEL,wBAAwB"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"AAmCA;IAIE;;;;;OAKG;IACH,2DAJG;QAAqB,WAAW,EAAxB,MAAM;QACyC,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QAC1B,aAAa,GAA3B,MAAM;KAChB,EAWA;IAnBD,mEAA2B;IAC3B,cAAiB;IAWf,eAA8B;IAC9B,oBAA8B;IAC9B,wDAAkC;IAClC,sBAAkC;IAElC,8BAA8B;IAC9B,gBADW,aAAa,EAAE,CACF;IAG1B;;;OAGG;IACH,iCAHW,MAAM,GACJ,IAAI,CAgBhB;IAED;;;OAGG;IACH,iCAHW,MAAM,GACJ,IAAI,CAgBhB;IAED;;;OAGG;IACH,wBAHW,KAAK,GACH,IAAI,CAehB;IAJC,wBAA+B;IAMjC,kCA6BC;IAED;;;OAGG;IACH,cAHW,MAAM,GACJ,IAAI,CAsDhB;IAED;;;OAGG;IACH,6BAHW,OAAO,cAAc,EAAE,OAAO,GAC5B,OAAO,CAOnB;IAED,0CAA0C;IAC1C,uBADc,IAAI,CA4DjB;IAfC,mCAME;IAWJ,wBAMC;IAED,kCA6BC;IAED;;;OAGG;IACH,4BAHW,aAAa,GACX,OAAO,CAAC,IAAI,CAAC,CAyEzB;IAED;;;OAGG;IACH,yBAHW,MAAM,GACJ,OAAO,CAAC,IAAI,CAAC,CAmBzB;IAED;;;OAGG;IACH,+BAHW,OAAO,cAAc,EAAE,OAAO,GAAG,OAAO,wBAAwB,EAAE,OAAO,GACvE,OAAO,CAenB;CACF;mBA3akB,iBAAiB;0BAEV,qBAAqB;oBAD3B,cAAc;6BAEL,wBAAwB"}
@@ -17,6 +17,17 @@ function summarizeRequestData(data) {
17
17
  const preview = data.toString("latin1", 0, Math.min(data.length, 160)).replaceAll("\r", "\\r").replaceAll("\n", "\\n");
18
18
  return { length: data.length, preview };
19
19
  }
20
+ /**
21
+ * @param {Error & {velociousContext?: Record<string, unknown>}} error - Error instance.
22
+ * @returns {Record<string, unknown>} - Safe bad-request details for logs.
23
+ */
24
+ function badRequestDetails(error) {
25
+ return {
26
+ errorClass: error.name,
27
+ message: error.message,
28
+ velociousContext: error.velociousContext
29
+ };
30
+ }
20
31
  export default class VeoliciousHttpServerClient {
21
32
  events = new EventEmitter();
22
33
  state = "initial";
@@ -77,7 +88,7 @@ export default class VeoliciousHttpServerClient {
77
88
  * @returns {void} - No return value.
78
89
  */
79
90
  handleBadRequest(error) {
80
- this.logger.warn(() => ["Failed to parse HTTP request", error]);
91
+ this.logger.warn(() => ["Failed to parse HTTP request", badRequestDetails(/** @type {Error & {velociousContext?: Record<string, unknown>}} */ (error))]);
81
92
  if (this.currentRequest && "getRequestParser" in this.currentRequest) {
82
93
  const httpRequest = /** @type {import("./request.js").default} */ (this.currentRequest);
83
94
  httpRequest.getRequestParser().destroy();
@@ -371,4 +382,4 @@ export default class VeoliciousHttpServerClient {
371
382
  return false;
372
383
  }
373
384
  }
374
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport crypto from \"crypto\"\nimport fs from \"node:fs/promises\"\nimport {createReadStream} from \"node:fs\"\nimport {digg} from \"diggerize\"\nimport EventEmitter from \"../../utils/event-emitter.js\"\nimport ensureError from \"../../utils/ensure-error.js\"\nimport Logger from \"../../logger.js\"\nimport Request from \"./request.js\"\nimport RequestRunner from \"./request-runner.js\"\nimport WebsocketSession from \"./websocket-session.js\"\n\n/**\n * @param {Buffer} data - Incoming request data.\n * @returns {{length: number, preview: string}} - Request data summary.\n */\nfunction summarizeRequestData(data) {\n  const preview = data.toString(\"latin1\", 0, Math.min(data.length, 160)).replaceAll(\"\\r\", \"\\\\r\").replaceAll(\"\\n\", \"\\\\n\")\n\n  return {length: data.length, preview}\n}\n\nexport default class VeoliciousHttpServerClient {\n  events = new EventEmitter()\n  state = \"initial\"\n\n  /**\n   * @param {object} args - Options object.\n   * @param {number} args.clientCount - Client count.\n   * @param {import(\"../../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} [args.remoteAddress] - Remote address.\n   */\n  constructor({clientCount, configuration, remoteAddress}) {\n    if (!configuration) throw new Error(\"No configuration given\")\n\n    this.logger = new Logger(this)\n    this.clientCount = clientCount\n    this.configuration = configuration\n    this.remoteAddress = remoteAddress\n\n    /** @type {RequestRunner[]} */\n    this.requestRunners = []\n  }\n\n  /**\n   * @param {string} message - Message text.\n   * @returns {void} - No return value.\n   */\n  _sendBadUpgradeResponse(message) {\n    const httpVersion = this.currentRequest?.httpVersion() || \"1.1\"\n    const body = `${message}\\n`\n    const headers = [\n      `HTTP/${httpVersion} 400 Bad Request`,\n      \"Connection: Close\",\n      \"Content-Type: text/plain; charset=UTF-8\",\n      `Content-Length: ${Buffer.byteLength(body, \"utf8\")}`,\n      \"\",\n      body\n    ].join(\"\\r\\n\")\n\n    this.events.emit(\"output\", headers)\n    this.events.emit(\"close\")\n  }\n\n  /**\n   * @param {string} message - Response message.\n   * @returns {void} - No return value.\n   */\n  _sendBadRequestResponse(message) {\n    const httpVersion = this.currentRequest?.httpVersion() || \"1.1\"\n    const body = `${message}\\n`\n    const headers = [\n      `HTTP/${httpVersion} 400 Bad Request`,\n      \"Connection: Close\",\n      \"Content-Type: text/plain; charset=UTF-8\",\n      `Content-Length: ${Buffer.byteLength(body, \"utf8\")}`,\n      \"\",\n      body\n    ].join(\"\\r\\n\")\n\n    this.events.emit(\"output\", headers)\n    this.events.emit(\"close\")\n  }\n\n  /**\n   * @param {Error} error - Error instance.\n   * @returns {void} - No return value.\n   */\n  handleBadRequest(error) {\n    this.logger.warn(() => [\"Failed to parse HTTP request\", error])\n\n    if (this.currentRequest && \"getRequestParser\" in this.currentRequest) {\n      const httpRequest = /** @type {import(\"./request.js\").default} */ (this.currentRequest)\n\n      httpRequest.getRequestParser().destroy()\n    }\n\n    this.currentRequest = undefined\n    this.state = \"initial\"\n\n    this._sendBadRequestResponse(\"Bad Request\")\n  }\n\n  executeCurrentRequest = () => {\n    this.logger.debug(\"executeCurrentRequest\")\n\n    if (!this.currentRequest) throw new Error(\"No current request\")\n    this.logger.debug(() => [\"executeCurrentRequest request\", {\n      clientCount: this.clientCount,\n      httpMethod: this.currentRequest.httpMethod(),\n      httpVersion: this.currentRequest.httpVersion(),\n      path: this.currentRequest.path(),\n      queueLength: this.requestRunners.length\n    }])\n\n    if (this._isWebsocketUpgrade(this.currentRequest)) {\n      this._upgradeToWebsocket()\n      return\n    }\n\n    // We are done parsing the given request and can theoretically start parsing a new one, before the current request is done - so reset the state.\n    this.state = \"initial\"\n\n    const requestRunner = new RequestRunner({\n      configuration: this.configuration,\n      request: this.currentRequest\n    })\n\n    this.requestRunners.push(requestRunner)\n\n    requestRunner.events.on(\"done\", this.requestDone)\n    requestRunner.run()\n  }\n\n  /**\n   * @param {Buffer} data - Data payload.\n   * @returns {void} - No return value.\n   */\n  onWrite(data) {\n    this.logger.debug(() => [\"onWrite start\", {\n      clientCount: this.clientCount,\n      state: this.state,\n      ...summarizeRequestData(data)\n    }])\n\n    if (this.websocketSession) {\n      this.websocketSession.onData(data)\n      return\n    }\n\n    try {\n      /** @type {Buffer | undefined} */\n      let remaining = data\n\n      while (remaining && remaining.length > 0) {\n        if (this.state == \"initial\") {\n          this.logger.debug(() => [\"onWrite creating request parser\", {clientCount: this.clientCount, remainingLength: remaining.length}])\n          this.currentRequest = new Request({client: this, configuration: this.configuration})\n          this.currentRequest.requestParser.events.on(\"done\", this.executeCurrentRequest)\n          this.state = \"requestStarted\"\n        } else if (this.state != \"requestStarted\") {\n          throw new Error(`Unknown state for client: ${this.state}`)\n        }\n\n        if (!this.currentRequest) throw new Error(\"No current request\")\n\n        remaining = this.currentRequest.feed(remaining)\n        this.logger.debug(() => [\"onWrite fed parser\", {\n          clientCount: this.clientCount,\n          hasRemaining: Boolean(remaining?.length),\n          remainingLength: remaining?.length || 0,\n          parserCompleted: this.currentRequest?.getRequestParser().hasCompleted\n        }])\n\n        if (remaining && remaining.length > 0) {\n          const requestParser = this.currentRequest.getRequestParser()\n\n          if (!requestParser.hasCompleted) {\n            this.logger.debug(() => [\"onWrite waiting for more data\", {clientCount: this.clientCount, remainingLength: remaining.length}])\n            break\n          }\n\n          this.state = \"initial\"\n          this.logger.debug(() => [\"onWrite parser completed with remaining bytes\", {clientCount: this.clientCount, remainingLength: remaining.length}])\n        }\n      }\n      this.logger.debug(() => [\"onWrite end\", {clientCount: this.clientCount, state: this.state, queueLength: this.requestRunners.length}])\n    } catch (error) {\n      this.handleBadRequest(ensureError(error))\n    }\n  }\n\n  /**\n   * @param {import(\"./request.js\").default} request - Request object.\n   * @returns {boolean} - Whether websocket upgrade.\n   */\n  _isWebsocketUpgrade(request) {\n    const upgradeHeader = request.header(\"upgrade\")?.toLowerCase()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()\n\n    return Boolean(upgradeHeader == \"websocket\" && connectionHeader?.includes(\"upgrade\"))\n  }\n\n  /** @returns {void} - No return value.  */\n  _upgradeToWebsocket() {\n    if (!this.currentRequest) throw new Error(\"No current request\")\n\n    const secWebsocketKey = this.currentRequest.header(\"sec-websocket-key\")\n\n    if (!secWebsocketKey) {\n      this._sendBadUpgradeResponse(\"Missing Sec-WebSocket-Key header\")\n      return\n    }\n\n    const websocketAcceptKey = crypto.createHash(\"sha1\")\n      .update(`${secWebsocketKey}258EAFA5-E914-47DA-95CA-C5AB0DC85B11`, \"binary\")\n      .digest(\"base64\")\n    const httpVersion = this.currentRequest.httpVersion() || \"1.1\"\n    const responseLines = [\n      `HTTP/${httpVersion} 101 Switching Protocols`,\n      \"Upgrade: websocket\",\n      \"Connection: Upgrade\",\n      `Sec-WebSocket-Accept: ${websocketAcceptKey}`,\n      \"\",\n      \"\"\n    ]\n    const response = responseLines.join(\"\\r\\n\")\n\n    const messageHandlerResolver = this.configuration.getWebsocketMessageHandlerResolver?.()\n    let messageHandler\n    let messageHandlerPromise\n\n    if (messageHandlerResolver) {\n      const resolvedHandler = messageHandlerResolver({\n        client: this,\n        configuration: this.configuration,\n        request: this.currentRequest\n      })\n\n      const resolvedThenable = /** @type {{then?: (...args: unknown[]) => unknown}} */ (resolvedHandler)\n\n      if (resolvedThenable?.then) {\n        messageHandlerPromise = /** @type {Promise<import(\"../../configuration-types.js\").WebsocketMessageHandler | void>} */ (resolvedHandler)\n      } else if (resolvedHandler) {\n        messageHandler = /** @type {import(\"../../configuration-types.js\").WebsocketMessageHandler} */ (resolvedHandler)\n      }\n    }\n\n    this.websocketSession = new WebsocketSession({\n      client: this,\n      configuration: this.configuration,\n      upgradeRequest: this.currentRequest,\n      messageHandler: messageHandler,\n      messageHandlerPromise: messageHandlerPromise\n    })\n    this.websocketSession.events.on(\"close\", () => {\n      this.websocketSession?.destroy()\n      this.websocketSession = undefined\n      this.events.emit(\"close\")\n    })\n    this.state = \"websocket\"\n    this.events.emit(\"output\", response)\n    void this.websocketSession.initializeChannel()\n  }\n\n  requestDone = () => {\n    this.logger.debug(() => [\"requestDone\", {clientCount: this.clientCount, queueLength: this.requestRunners.length}])\n    void this.sendDoneRequests().catch((error) => {\n      this.logger.warn(\"Failed while sending done requests\", error)\n      this.events.emit(\"close\")\n    })\n  }\n\n  async sendDoneRequests() {\n    while (true) {\n      const requestRunner = this.requestRunners[0]\n      const request = requestRunner?.getRequest()\n\n      if (requestRunner?.getState() == \"done\") {\n        const httpVersion = request.httpVersion()\n        const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n        const shouldCloseConnection = this.shouldCloseConnection(request)\n\n        this.requestRunners.shift()\n        this.logger.debug(() => [\"sendDoneRequests shifted queue\", {clientCount: this.clientCount, queueLength: this.requestRunners.length}])\n        try {\n          await this.sendResponse(requestRunner)\n        } catch (error) {\n          this.logger.error(() => [`Velocious client ${this.clientCount} failed while sending response`, error])\n          throw error\n        }\n        if (this.currentRequest === request && this.state === \"initial\") this.currentRequest = undefined\n        this.logger.debug(() => [\"sendDoneRequests\", {clientCount: this.clientCount, connectionHeader, httpVersion}])\n\n        if (shouldCloseConnection) {\n          this.logger.debug(() => [`Closing the connection because ${httpVersion} and connection header ${connectionHeader}`, {clientCount: this.clientCount}])\n          this.events.emit(\"close\")\n        }\n      } else {\n        break\n      }\n    }\n  }\n\n  /**\n   * @param {RequestRunner} requestRunner - Request runner.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async sendResponse(requestRunner) {\n    const response = digg(requestRunner, \"response\")\n    const request = requestRunner.getRequest()\n    const filePath = response.getFilePath()\n    const date = new Date()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n    const httpVersion = request.httpVersion()\n    const shouldCloseConnection = this.shouldCloseConnection(request)\n    const hasFilePath = typeof filePath === \"string\" && filePath.length > 0\n    const body = hasFilePath ? null : response.getBody()\n    const bodyIsString = typeof body === \"string\"\n    const bodyIsBinary = body instanceof Uint8Array\n\n    if (!hasFilePath && !bodyIsString && !bodyIsBinary) {\n      throw new Error(`Expected response body to be a string or Uint8Array, got ${typeof body}`)\n    }\n\n    this.logger.debug(\"sendResponse\", {clientCount: this.clientCount, connectionHeader, httpVersion})\n    this.logger.debug(() => [\"sendResponse payload\", {\n      clientCount: this.clientCount,\n      hasFilePath,\n      filePath,\n      bodyIsBinary,\n      bodyIsString\n    }])\n\n    if (shouldCloseConnection) {\n      response.setHeader(\"Connection\", \"Close\")\n    } else if (httpVersion == \"1.0\" && connectionHeader == \"keep-alive\") {\n      response.setHeader(\"Connection\", \"Keep-Alive\")\n    }\n\n    let contentLength\n\n    if (hasFilePath) {\n      const stats = await fs.stat(filePath)\n      contentLength = stats.size\n    } else {\n      contentLength = bodyIsString ? new TextEncoder().encode(body).length : body.byteLength\n    }\n\n    response.setHeader(\"Content-Length\", contentLength)\n    response.setHeader(\"Date\", date.toUTCString())\n    response.setHeader(\"Server\", \"Velocious\")\n\n    let headers = \"\"\n\n    headers += `HTTP/${request.httpVersion()} ${response.getStatusCode()} ${response.getStatusMessage()}\\r\\n`\n\n    for (const headerKey in response.headers) {\n      for (const headerValue of response.headers[headerKey]) {\n        headers += `${headerKey}: ${headerValue}\\r\\n`\n      }\n    }\n\n    headers += \"\\r\\n\"\n\n    this.events.emit(\"output\", headers)\n    this.logger.debug(() => [\"sendResponse headers emitted\", {clientCount: this.clientCount, headersLength: headers.length}])\n\n    if (hasFilePath) {\n      await this.sendFileOutput(filePath)\n    } else {\n      this.events.emit(\"output\", body)\n      this.logger.debug(() => [\"sendResponse body emitted\", {clientCount: this.clientCount, bodyLength: bodyIsString ? body.length : body.byteLength}])\n    }\n\n    if (\"getRequestParser\" in request) {\n      const httpRequest = /** @type {import(\"./request.js\").default} */ (request)\n      httpRequest.getRequestParser().destroy()\n    }\n  }\n\n  /**\n   * @param {string} filePath - File path.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async sendFileOutput(filePath) {\n    this.logger.debug(() => [\"sendFileOutput start\", {clientCount: this.clientCount, filePath}])\n    let totalBytes = 0\n    let chunkCount = 0\n\n    try {\n      for await (const chunk of createReadStream(filePath)) {\n        chunkCount += 1\n        totalBytes += chunk.length\n        this.logger.debug(() => [\"sendFileOutput chunk\", {clientCount: this.clientCount, chunkCount, chunkLength: chunk.length, totalBytes}])\n        this.events.emit(\"output\", chunk)\n      }\n      this.logger.debug(() => [\"sendFileOutput done\", {clientCount: this.clientCount, chunkCount, totalBytes}])\n    } catch (error) {\n      this.logger.error(() => [`Velocious client ${this.clientCount} failed while streaming file output: ${filePath}`, error])\n      throw error\n    }\n  }\n\n  /**\n   * @param {import(\"./request.js\").default | import(\"./websocket-request.js\").default} request - Request object.\n   * @returns {boolean} - Whether the connection should be closed.\n   */\n  shouldCloseConnection(request) {\n    const httpVersion = request.httpVersion()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n    const connectionTokens = connectionHeader\n      ? connectionHeader.split(\",\").map((token) => token.trim()).filter(Boolean)\n      : []\n\n    if (httpVersion == \"websocket\") return false\n    if (connectionTokens.includes(\"close\")) return true\n\n    if (httpVersion == \"1.0\" && connectionHeader != \"keep-alive\") return true\n\n    return false\n  }\n}\n"]}
385
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport crypto from \"crypto\"\nimport fs from \"node:fs/promises\"\nimport {createReadStream} from \"node:fs\"\nimport {digg} from \"diggerize\"\nimport EventEmitter from \"../../utils/event-emitter.js\"\nimport ensureError from \"../../utils/ensure-error.js\"\nimport Logger from \"../../logger.js\"\nimport Request from \"./request.js\"\nimport RequestRunner from \"./request-runner.js\"\nimport WebsocketSession from \"./websocket-session.js\"\n\n/**\n * @param {Buffer} data - Incoming request data.\n * @returns {{length: number, preview: string}} - Request data summary.\n */\nfunction summarizeRequestData(data) {\n  const preview = data.toString(\"latin1\", 0, Math.min(data.length, 160)).replaceAll(\"\\r\", \"\\\\r\").replaceAll(\"\\n\", \"\\\\n\")\n\n  return {length: data.length, preview}\n}\n\n/**\n * @param {Error & {velociousContext?: Record<string, unknown>}} error - Error instance.\n * @returns {Record<string, unknown>} - Safe bad-request details for logs.\n */\nfunction badRequestDetails(error) {\n  return {\n    errorClass: error.name,\n    message: error.message,\n    velociousContext: error.velociousContext\n  }\n}\n\nexport default class VeoliciousHttpServerClient {\n  events = new EventEmitter()\n  state = \"initial\"\n\n  /**\n   * @param {object} args - Options object.\n   * @param {number} args.clientCount - Client count.\n   * @param {import(\"../../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {string} [args.remoteAddress] - Remote address.\n   */\n  constructor({clientCount, configuration, remoteAddress}) {\n    if (!configuration) throw new Error(\"No configuration given\")\n\n    this.logger = new Logger(this)\n    this.clientCount = clientCount\n    this.configuration = configuration\n    this.remoteAddress = remoteAddress\n\n    /** @type {RequestRunner[]} */\n    this.requestRunners = []\n  }\n\n  /**\n   * @param {string} message - Message text.\n   * @returns {void} - No return value.\n   */\n  _sendBadUpgradeResponse(message) {\n    const httpVersion = this.currentRequest?.httpVersion() || \"1.1\"\n    const body = `${message}\\n`\n    const headers = [\n      `HTTP/${httpVersion} 400 Bad Request`,\n      \"Connection: Close\",\n      \"Content-Type: text/plain; charset=UTF-8\",\n      `Content-Length: ${Buffer.byteLength(body, \"utf8\")}`,\n      \"\",\n      body\n    ].join(\"\\r\\n\")\n\n    this.events.emit(\"output\", headers)\n    this.events.emit(\"close\")\n  }\n\n  /**\n   * @param {string} message - Response message.\n   * @returns {void} - No return value.\n   */\n  _sendBadRequestResponse(message) {\n    const httpVersion = this.currentRequest?.httpVersion() || \"1.1\"\n    const body = `${message}\\n`\n    const headers = [\n      `HTTP/${httpVersion} 400 Bad Request`,\n      \"Connection: Close\",\n      \"Content-Type: text/plain; charset=UTF-8\",\n      `Content-Length: ${Buffer.byteLength(body, \"utf8\")}`,\n      \"\",\n      body\n    ].join(\"\\r\\n\")\n\n    this.events.emit(\"output\", headers)\n    this.events.emit(\"close\")\n  }\n\n  /**\n   * @param {Error} error - Error instance.\n   * @returns {void} - No return value.\n   */\n  handleBadRequest(error) {\n    this.logger.warn(() => [\"Failed to parse HTTP request\", badRequestDetails(/** @type {Error & {velociousContext?: Record<string, unknown>}} */ (error))])\n\n    if (this.currentRequest && \"getRequestParser\" in this.currentRequest) {\n      const httpRequest = /** @type {import(\"./request.js\").default} */ (this.currentRequest)\n\n      httpRequest.getRequestParser().destroy()\n    }\n\n    this.currentRequest = undefined\n    this.state = \"initial\"\n\n    this._sendBadRequestResponse(\"Bad Request\")\n  }\n\n  executeCurrentRequest = () => {\n    this.logger.debug(\"executeCurrentRequest\")\n\n    if (!this.currentRequest) throw new Error(\"No current request\")\n    this.logger.debug(() => [\"executeCurrentRequest request\", {\n      clientCount: this.clientCount,\n      httpMethod: this.currentRequest.httpMethod(),\n      httpVersion: this.currentRequest.httpVersion(),\n      path: this.currentRequest.path(),\n      queueLength: this.requestRunners.length\n    }])\n\n    if (this._isWebsocketUpgrade(this.currentRequest)) {\n      this._upgradeToWebsocket()\n      return\n    }\n\n    // We are done parsing the given request and can theoretically start parsing a new one, before the current request is done - so reset the state.\n    this.state = \"initial\"\n\n    const requestRunner = new RequestRunner({\n      configuration: this.configuration,\n      request: this.currentRequest\n    })\n\n    this.requestRunners.push(requestRunner)\n\n    requestRunner.events.on(\"done\", this.requestDone)\n    requestRunner.run()\n  }\n\n  /**\n   * @param {Buffer} data - Data payload.\n   * @returns {void} - No return value.\n   */\n  onWrite(data) {\n    this.logger.debug(() => [\"onWrite start\", {\n      clientCount: this.clientCount,\n      state: this.state,\n      ...summarizeRequestData(data)\n    }])\n\n    if (this.websocketSession) {\n      this.websocketSession.onData(data)\n      return\n    }\n\n    try {\n      /** @type {Buffer | undefined} */\n      let remaining = data\n\n      while (remaining && remaining.length > 0) {\n        if (this.state == \"initial\") {\n          this.logger.debug(() => [\"onWrite creating request parser\", {clientCount: this.clientCount, remainingLength: remaining.length}])\n          this.currentRequest = new Request({client: this, configuration: this.configuration})\n          this.currentRequest.requestParser.events.on(\"done\", this.executeCurrentRequest)\n          this.state = \"requestStarted\"\n        } else if (this.state != \"requestStarted\") {\n          throw new Error(`Unknown state for client: ${this.state}`)\n        }\n\n        if (!this.currentRequest) throw new Error(\"No current request\")\n\n        remaining = this.currentRequest.feed(remaining)\n        this.logger.debug(() => [\"onWrite fed parser\", {\n          clientCount: this.clientCount,\n          hasRemaining: Boolean(remaining?.length),\n          remainingLength: remaining?.length || 0,\n          parserCompleted: this.currentRequest?.getRequestParser().hasCompleted\n        }])\n\n        if (remaining && remaining.length > 0) {\n          const requestParser = this.currentRequest.getRequestParser()\n\n          if (!requestParser.hasCompleted) {\n            this.logger.debug(() => [\"onWrite waiting for more data\", {clientCount: this.clientCount, remainingLength: remaining.length}])\n            break\n          }\n\n          this.state = \"initial\"\n          this.logger.debug(() => [\"onWrite parser completed with remaining bytes\", {clientCount: this.clientCount, remainingLength: remaining.length}])\n        }\n      }\n      this.logger.debug(() => [\"onWrite end\", {clientCount: this.clientCount, state: this.state, queueLength: this.requestRunners.length}])\n    } catch (error) {\n      this.handleBadRequest(ensureError(error))\n    }\n  }\n\n  /**\n   * @param {import(\"./request.js\").default} request - Request object.\n   * @returns {boolean} - Whether websocket upgrade.\n   */\n  _isWebsocketUpgrade(request) {\n    const upgradeHeader = request.header(\"upgrade\")?.toLowerCase()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()\n\n    return Boolean(upgradeHeader == \"websocket\" && connectionHeader?.includes(\"upgrade\"))\n  }\n\n  /** @returns {void} - No return value.  */\n  _upgradeToWebsocket() {\n    if (!this.currentRequest) throw new Error(\"No current request\")\n\n    const secWebsocketKey = this.currentRequest.header(\"sec-websocket-key\")\n\n    if (!secWebsocketKey) {\n      this._sendBadUpgradeResponse(\"Missing Sec-WebSocket-Key header\")\n      return\n    }\n\n    const websocketAcceptKey = crypto.createHash(\"sha1\")\n      .update(`${secWebsocketKey}258EAFA5-E914-47DA-95CA-C5AB0DC85B11`, \"binary\")\n      .digest(\"base64\")\n    const httpVersion = this.currentRequest.httpVersion() || \"1.1\"\n    const responseLines = [\n      `HTTP/${httpVersion} 101 Switching Protocols`,\n      \"Upgrade: websocket\",\n      \"Connection: Upgrade\",\n      `Sec-WebSocket-Accept: ${websocketAcceptKey}`,\n      \"\",\n      \"\"\n    ]\n    const response = responseLines.join(\"\\r\\n\")\n\n    const messageHandlerResolver = this.configuration.getWebsocketMessageHandlerResolver?.()\n    let messageHandler\n    let messageHandlerPromise\n\n    if (messageHandlerResolver) {\n      const resolvedHandler = messageHandlerResolver({\n        client: this,\n        configuration: this.configuration,\n        request: this.currentRequest\n      })\n\n      const resolvedThenable = /** @type {{then?: (...args: unknown[]) => unknown}} */ (resolvedHandler)\n\n      if (resolvedThenable?.then) {\n        messageHandlerPromise = /** @type {Promise<import(\"../../configuration-types.js\").WebsocketMessageHandler | void>} */ (resolvedHandler)\n      } else if (resolvedHandler) {\n        messageHandler = /** @type {import(\"../../configuration-types.js\").WebsocketMessageHandler} */ (resolvedHandler)\n      }\n    }\n\n    this.websocketSession = new WebsocketSession({\n      client: this,\n      configuration: this.configuration,\n      upgradeRequest: this.currentRequest,\n      messageHandler: messageHandler,\n      messageHandlerPromise: messageHandlerPromise\n    })\n    this.websocketSession.events.on(\"close\", () => {\n      this.websocketSession?.destroy()\n      this.websocketSession = undefined\n      this.events.emit(\"close\")\n    })\n    this.state = \"websocket\"\n    this.events.emit(\"output\", response)\n    void this.websocketSession.initializeChannel()\n  }\n\n  requestDone = () => {\n    this.logger.debug(() => [\"requestDone\", {clientCount: this.clientCount, queueLength: this.requestRunners.length}])\n    void this.sendDoneRequests().catch((error) => {\n      this.logger.warn(\"Failed while sending done requests\", error)\n      this.events.emit(\"close\")\n    })\n  }\n\n  async sendDoneRequests() {\n    while (true) {\n      const requestRunner = this.requestRunners[0]\n      const request = requestRunner?.getRequest()\n\n      if (requestRunner?.getState() == \"done\") {\n        const httpVersion = request.httpVersion()\n        const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n        const shouldCloseConnection = this.shouldCloseConnection(request)\n\n        this.requestRunners.shift()\n        this.logger.debug(() => [\"sendDoneRequests shifted queue\", {clientCount: this.clientCount, queueLength: this.requestRunners.length}])\n        try {\n          await this.sendResponse(requestRunner)\n        } catch (error) {\n          this.logger.error(() => [`Velocious client ${this.clientCount} failed while sending response`, error])\n          throw error\n        }\n        if (this.currentRequest === request && this.state === \"initial\") this.currentRequest = undefined\n        this.logger.debug(() => [\"sendDoneRequests\", {clientCount: this.clientCount, connectionHeader, httpVersion}])\n\n        if (shouldCloseConnection) {\n          this.logger.debug(() => [`Closing the connection because ${httpVersion} and connection header ${connectionHeader}`, {clientCount: this.clientCount}])\n          this.events.emit(\"close\")\n        }\n      } else {\n        break\n      }\n    }\n  }\n\n  /**\n   * @param {RequestRunner} requestRunner - Request runner.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async sendResponse(requestRunner) {\n    const response = digg(requestRunner, \"response\")\n    const request = requestRunner.getRequest()\n    const filePath = response.getFilePath()\n    const date = new Date()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n    const httpVersion = request.httpVersion()\n    const shouldCloseConnection = this.shouldCloseConnection(request)\n    const hasFilePath = typeof filePath === \"string\" && filePath.length > 0\n    const body = hasFilePath ? null : response.getBody()\n    const bodyIsString = typeof body === \"string\"\n    const bodyIsBinary = body instanceof Uint8Array\n\n    if (!hasFilePath && !bodyIsString && !bodyIsBinary) {\n      throw new Error(`Expected response body to be a string or Uint8Array, got ${typeof body}`)\n    }\n\n    this.logger.debug(\"sendResponse\", {clientCount: this.clientCount, connectionHeader, httpVersion})\n    this.logger.debug(() => [\"sendResponse payload\", {\n      clientCount: this.clientCount,\n      hasFilePath,\n      filePath,\n      bodyIsBinary,\n      bodyIsString\n    }])\n\n    if (shouldCloseConnection) {\n      response.setHeader(\"Connection\", \"Close\")\n    } else if (httpVersion == \"1.0\" && connectionHeader == \"keep-alive\") {\n      response.setHeader(\"Connection\", \"Keep-Alive\")\n    }\n\n    let contentLength\n\n    if (hasFilePath) {\n      const stats = await fs.stat(filePath)\n      contentLength = stats.size\n    } else {\n      contentLength = bodyIsString ? new TextEncoder().encode(body).length : body.byteLength\n    }\n\n    response.setHeader(\"Content-Length\", contentLength)\n    response.setHeader(\"Date\", date.toUTCString())\n    response.setHeader(\"Server\", \"Velocious\")\n\n    let headers = \"\"\n\n    headers += `HTTP/${request.httpVersion()} ${response.getStatusCode()} ${response.getStatusMessage()}\\r\\n`\n\n    for (const headerKey in response.headers) {\n      for (const headerValue of response.headers[headerKey]) {\n        headers += `${headerKey}: ${headerValue}\\r\\n`\n      }\n    }\n\n    headers += \"\\r\\n\"\n\n    this.events.emit(\"output\", headers)\n    this.logger.debug(() => [\"sendResponse headers emitted\", {clientCount: this.clientCount, headersLength: headers.length}])\n\n    if (hasFilePath) {\n      await this.sendFileOutput(filePath)\n    } else {\n      this.events.emit(\"output\", body)\n      this.logger.debug(() => [\"sendResponse body emitted\", {clientCount: this.clientCount, bodyLength: bodyIsString ? body.length : body.byteLength}])\n    }\n\n    if (\"getRequestParser\" in request) {\n      const httpRequest = /** @type {import(\"./request.js\").default} */ (request)\n      httpRequest.getRequestParser().destroy()\n    }\n  }\n\n  /**\n   * @param {string} filePath - File path.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async sendFileOutput(filePath) {\n    this.logger.debug(() => [\"sendFileOutput start\", {clientCount: this.clientCount, filePath}])\n    let totalBytes = 0\n    let chunkCount = 0\n\n    try {\n      for await (const chunk of createReadStream(filePath)) {\n        chunkCount += 1\n        totalBytes += chunk.length\n        this.logger.debug(() => [\"sendFileOutput chunk\", {clientCount: this.clientCount, chunkCount, chunkLength: chunk.length, totalBytes}])\n        this.events.emit(\"output\", chunk)\n      }\n      this.logger.debug(() => [\"sendFileOutput done\", {clientCount: this.clientCount, chunkCount, totalBytes}])\n    } catch (error) {\n      this.logger.error(() => [`Velocious client ${this.clientCount} failed while streaming file output: ${filePath}`, error])\n      throw error\n    }\n  }\n\n  /**\n   * @param {import(\"./request.js\").default | import(\"./websocket-request.js\").default} request - Request object.\n   * @returns {boolean} - Whether the connection should be closed.\n   */\n  shouldCloseConnection(request) {\n    const httpVersion = request.httpVersion()\n    const connectionHeader = request.header(\"connection\")?.toLowerCase()?.trim()\n    const connectionTokens = connectionHeader\n      ? connectionHeader.split(\",\").map((token) => token.trim()).filter(Boolean)\n      : []\n\n    if (httpVersion == \"websocket\") return false\n    if (connectionTokens.includes(\"close\")) return true\n\n    if (httpVersion == \"1.0\" && connectionHeader != \"keep-alive\") return true\n\n    return false\n  }\n}\n"]}
@@ -17,8 +17,9 @@ export default class ParamsToObject {
17
17
  * @param {any} value - Value to use.
18
18
  * @param {string} rest - Rest.
19
19
  * @param {Record<string, any> | any[]} result - Result.
20
+ * @param {string} [fullKey] - Original full key.
20
21
  * @returns {void} - No return value.
21
22
  */
22
- treatSecond(value: any, rest: string, result: Record<string, any> | any[]): void;
23
+ treatSecond(value: any, rest: string, result: Record<string, any> | any[], fullKey?: string): void;
23
24
  }
24
25
  //# sourceMappingURL=params-to-object.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"params-to-object.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/params-to-object.js"],"names":[],"mappings":"AAEA;IACE;;OAEG;IACH,oBAFW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAI7B;IADC,4BAAoB;IAGtB,oDAAoD;IACpD,YADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAYhC;IAED;;;;;OAKG;IACH,kBALW,MAAM,SACN,GAAG,UACH,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,GAAG,EAAE,GACzB,IAAI,CA0BhB;IAED;;;;;OAKG;IACH,mBALW,GAAG,QACH,MAAM,UACN,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,GAAG,EAAE,GACzB,IAAI,CAkChB;CACF"}
1
+ {"version":3,"file":"params-to-object.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/params-to-object.js"],"names":[],"mappings":"AAyBA;IACE;;OAEG;IACH,oBAFW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAI7B;IADC,4BAAoB;IAGtB,oDAAoD;IACpD,YADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAYhC;IAED;;;;;OAKG;IACH,kBALW,MAAM,SACN,GAAG,UACH,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,GAAG,EAAE,GACzB,IAAI,CA0BhB;IAED;;;;;;OAMG;IACH,mBANW,GAAG,QACH,MAAM,UACN,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,GAAG,EAAE,YAC3B,MAAM,GACJ,IAAI,CAkChB;CACF"}
@@ -1,4 +1,24 @@
1
1
  // @ts-check
2
+ /**
3
+ * @param {object} args - Args.
4
+ * @param {string} args.key - Parameter key.
5
+ * @param {string} args.rest - Remaining unmatched segment.
6
+ * @returns {Error} - Error with parser context attached.
7
+ */
8
+ function malformedNestedParamsKeyError(args) {
9
+ const { key, rest } = args;
10
+ const error = new Error(`Could not parse nested params key "${key}" at rest "${rest}"`);
11
+ /** @type {Error & {velociousContext?: Record<string, unknown>}} */
12
+ const typedError = error;
13
+ typedError.velociousContext = {
14
+ nestedParamsKey: {
15
+ key,
16
+ rest,
17
+ stage: "params-to-object"
18
+ }
19
+ };
20
+ return error;
21
+ }
2
22
  export default class ParamsToObject {
3
23
  /**
4
24
  * @param {Record<string, any>} object - Object.
@@ -40,7 +60,7 @@ export default class ParamsToObject {
40
60
  newResult = {};
41
61
  result[inputName] = newResult;
42
62
  }
43
- this.treatSecond(value, rest, newResult);
63
+ this.treatSecond(value, rest, newResult, key);
44
64
  }
45
65
  else {
46
66
  result[key] = value;
@@ -50,12 +70,13 @@ export default class ParamsToObject {
50
70
  * @param {any} value - Value to use.
51
71
  * @param {string} rest - Rest.
52
72
  * @param {Record<string, any> | any[]} result - Result.
73
+ * @param {string} [fullKey] - Original full key.
53
74
  * @returns {void} - No return value.
54
75
  */
55
- treatSecond(value, rest, result) {
76
+ treatSecond(value, rest, result, fullKey = rest) {
56
77
  const secondMatch = rest.match(/^\[(.*?)\]([\s\S]*)$/);
57
78
  if (!secondMatch)
58
- throw new Error(`Could not parse rest part: ${rest}`);
79
+ throw malformedNestedParamsKeyError({ key: fullKey, rest });
59
80
  const key = secondMatch[1];
60
81
  const newRest = secondMatch[2];
61
82
  /** @type {Array<any> | Record<string, any>} */
@@ -81,8 +102,8 @@ export default class ParamsToObject {
81
102
  newResult = {};
82
103
  result[key] = newResult;
83
104
  }
84
- this.treatSecond(value, newRest, newResult);
105
+ this.treatSecond(value, newRest, newResult, fullKey);
85
106
  }
86
107
  }
87
108
  }
88
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFyYW1zLXRvLW9iamVjdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9odHRwLXNlcnZlci9jbGllbnQvcGFyYW1zLXRvLW9iamVjdC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxZQUFZO0FBRVosTUFBTSxDQUFDLE9BQU8sT0FBTyxjQUFjO0lBQ2pDOztPQUVHO0lBQ0gsWUFBWSxNQUFNO1FBQ2hCLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFBO0lBQ3RCLENBQUM7SUFFRCxvREFBb0Q7SUFDcEQsUUFBUTtRQUNOLHNDQUFzQztRQUN0QyxNQUFNLE1BQU0sR0FBRyxFQUFFLENBQUE7UUFFakIsS0FBSSxNQUFNLEdBQUcsSUFBSSxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDN0IsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsQ0FBQTtZQUU5QixJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsRUFBRSxLQUFLLEVBQUUsTUFBTSxDQUFDLENBQUE7UUFDdkMsQ0FBQztRQUVELE9BQU8sTUFBTSxDQUFBO0lBQ2YsQ0FBQztJQUVEOzs7OztPQUtHO0lBQ0gsWUFBWSxDQUFDLEdBQUcsRUFBRSxLQUFLLEVBQUUsTUFBTTtRQUM3QixNQUFNLFVBQVUsR0FBRyxHQUFHLENBQUMsS0FBSyxDQUFDLHNCQUFzQixDQUFDLENBQUE7UUFFcEQsSUFBSSxVQUFVLEVBQUUsQ0FBQztZQUNmLE1BQU0sU0FBUyxHQUFHLFVBQVUsQ0FBQyxDQUFDLENBQUMsQ0FBQTtZQUMvQixNQUFNLElBQUksR0FBRyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUE7WUFFMUIsK0NBQStDO1lBQy9DLElBQUksU0FBUyxDQUFBO1lBRWIsSUFBSSxTQUFTLElBQUksTUFBTSxFQUFFLENBQUM7Z0JBQ3hCLFNBQVMsR0FBRyxNQUFNLENBQUMsU0FBUyxDQUFDLENBQUE7WUFDL0IsQ0FBQztpQkFBTSxJQUFJLElBQUksSUFBSSxJQUFJLEVBQUUsQ0FBQztnQkFDeEIsU0FBUyxHQUFHLEVBQUUsQ0FBQTtnQkFDZCxNQUFNLENBQUMsU0FBUyxDQUFDLEdBQUcsU0FBUyxDQUFBO1lBQy9CLENBQUM7aUJBQU0sQ0FBQztnQkFDTixTQUFTLEdBQUcsRUFBRSxDQUFBO2dCQUNkLE1BQU0sQ0FBQyxTQUFTLENBQUMsR0FBRyxTQUFTLENBQUE7WUFDL0IsQ0FBQztZQUVELElBQUksQ0FBQyxXQUFXLENBQUMsS0FBSyxFQUFFLElBQUksRUFBRSxTQUFTLENBQUMsQ0FBQTtRQUMxQyxDQUFDO2FBQU0sQ0FBQztZQUNOLE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBRyxLQUFLLENBQUE7UUFDckIsQ0FBQztJQUNILENBQUM7SUFFRDs7Ozs7T0FLRztJQUNILFdBQVcsQ0FBQyxLQUFLLEVBQUUsSUFBSSxFQUFFLE1BQU07UUFDN0IsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxzQkFBc0IsQ0FBQyxDQUFBO1FBRXRELElBQUksQ0FBQyxXQUFXO1lBQUUsTUFBTSxJQUFJLEtBQUssQ0FBQyw4QkFBOEIsSUFBSSxFQUFFLENBQUMsQ0FBQTtRQUV2RSxNQUFNLEdBQUcsR0FBRyxXQUFXLENBQUMsQ0FBQyxDQUFDLENBQUE7UUFDMUIsTUFBTSxPQUFPLEdBQUcsV0FBVyxDQUFDLENBQUMsQ0FBQyxDQUFBO1FBRTlCLCtDQUErQztRQUMvQyxJQUFJLFNBQVMsQ0FBQTtRQUViLElBQUksSUFBSSxJQUFJLElBQUksRUFBRSxDQUFDO1lBQ2pCLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUM7Z0JBQzNCLE1BQU0sSUFBSSxLQUFLLENBQUMsa0NBQWtDLElBQUksRUFBRSxDQUFDLENBQUE7WUFDM0QsQ0FBQztZQUVELE1BQU0sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUE7UUFDcEIsQ0FBQzthQUFNLElBQUksT0FBTyxJQUFJLEVBQUUsRUFBRSxDQUFDO1lBQ3pCLE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBRyxLQUFLLENBQUE7UUFDckIsQ0FBQzthQUFNLENBQUM7WUFDTixJQUFJLE9BQU8sTUFBTSxJQUFJLFFBQVEsSUFBSSxHQUFHLElBQUksTUFBTSxFQUFFLENBQUM7Z0JBQy9DLFNBQVMsR0FBRyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUE7WUFDekIsQ0FBQztpQkFBTSxJQUFJLE9BQU8sSUFBSSxJQUFJLEVBQUUsQ0FBQztnQkFDM0IsU0FBUyxHQUFHLEVBQUUsQ0FBQTtnQkFDZCxNQUFNLENBQUMsR0FBRyxDQUFDLEdBQUcsU0FBUyxDQUFBO1lBQ3pCLENBQUM7aUJBQU0sQ0FBQztnQkFDTixTQUFTLEdBQUcsRUFBRSxDQUFBO2dCQUNkLE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBRyxTQUFTLENBQUE7WUFDekIsQ0FBQztZQUVELElBQUksQ0FBQyxXQUFXLENBQUMsS0FBSyxFQUFFLE9BQU8sRUFBRSxTQUFTLENBQUMsQ0FBQTtRQUM3QyxDQUFDO0lBQ0gsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQHRzLWNoZWNrXG5cbmV4cG9ydCBkZWZhdWx0IGNsYXNzIFBhcmFtc1RvT2JqZWN0IHtcbiAgLyoqXG4gICAqIEBwYXJhbSB7UmVjb3JkPHN0cmluZywgYW55Pn0gb2JqZWN0IC0gT2JqZWN0LlxuICAgKi9cbiAgY29uc3RydWN0b3Iob2JqZWN0KSB7XG4gICAgdGhpcy5vYmplY3QgPSBvYmplY3RcbiAgfVxuXG4gIC8qKiBAcmV0dXJucyB7UmVjb3JkPHN0cmluZywgYW55Pn0gLSBUaGUgb2JqZWN0LiAgKi9cbiAgdG9PYmplY3QoKSB7XG4gICAgLyoqIEB0eXBlIHtSZWNvcmQ8c3RyaW5nLCB1bmtub3duPn0gKi9cbiAgICBjb25zdCByZXN1bHQgPSB7fVxuXG4gICAgZm9yKGNvbnN0IGtleSBpbiB0aGlzLm9iamVjdCkge1xuICAgICAgY29uc3QgdmFsdWUgPSB0aGlzLm9iamVjdFtrZXldXG5cbiAgICAgIHRoaXMudHJlYXRJbml0aWFsKGtleSwgdmFsdWUsIHJlc3VsdClcbiAgICB9XG5cbiAgICByZXR1cm4gcmVzdWx0XG4gIH1cblxuICAvKipcbiAgICogQHBhcmFtIHtzdHJpbmd9IGtleSAtIEtleS5cbiAgICogQHBhcmFtIHthbnl9IHZhbHVlIC0gVmFsdWUgdG8gdXNlLlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT4gfCBhbnlbXX0gcmVzdWx0IC0gUmVzdWx0LlxuICAgKiBAcmV0dXJucyB7dm9pZH0gLSBObyByZXR1cm4gdmFsdWUuXG4gICAqL1xuICB0cmVhdEluaXRpYWwoa2V5LCB2YWx1ZSwgcmVzdWx0KSB7XG4gICAgY29uc3QgZmlyc3RNYXRjaCA9IGtleS5tYXRjaCgvXiguKz8pKFxcWyhbXFxzXFxTXSskKSkvKVxuXG4gICAgaWYgKGZpcnN0TWF0Y2gpIHtcbiAgICAgIGNvbnN0IGlucHV0TmFtZSA9IGZpcnN0TWF0Y2hbMV1cbiAgICAgIGNvbnN0IHJlc3QgPSBmaXJzdE1hdGNoWzJdXG5cbiAgICAgIC8qKiBAdHlwZSB7QXJyYXk8YW55PiB8IFJlY29yZDxzdHJpbmcsIGFueT59ICovXG4gICAgICBsZXQgbmV3UmVzdWx0XG5cbiAgICAgIGlmIChpbnB1dE5hbWUgaW4gcmVzdWx0KSB7XG4gICAgICAgIG5ld1Jlc3VsdCA9IHJlc3VsdFtpbnB1dE5hbWVdXG4gICAgICB9IGVsc2UgaWYgKHJlc3QgPT0gXCJbXVwiKSB7XG4gICAgICAgIG5ld1Jlc3VsdCA9IFtdXG4gICAgICAgIHJlc3VsdFtpbnB1dE5hbWVdID0gbmV3UmVzdWx0XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBuZXdSZXN1bHQgPSB7fVxuICAgICAgICByZXN1bHRbaW5wdXROYW1lXSA9IG5ld1Jlc3VsdFxuICAgICAgfVxuXG4gICAgICB0aGlzLnRyZWF0U2Vjb25kKHZhbHVlLCByZXN0LCBuZXdSZXN1bHQpXG4gICAgfSBlbHNlIHtcbiAgICAgIHJlc3VsdFtrZXldID0gdmFsdWVcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogQHBhcmFtIHthbnl9IHZhbHVlIC0gVmFsdWUgdG8gdXNlLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gcmVzdCAtIFJlc3QuXG4gICAqIEBwYXJhbSB7UmVjb3JkPHN0cmluZywgYW55PiB8IGFueVtdfSByZXN1bHQgLSBSZXN1bHQuXG4gICAqIEByZXR1cm5zIHt2b2lkfSAtIE5vIHJldHVybiB2YWx1ZS5cbiAgICovXG4gIHRyZWF0U2Vjb25kKHZhbHVlLCByZXN0LCByZXN1bHQpIHtcbiAgICBjb25zdCBzZWNvbmRNYXRjaCA9IHJlc3QubWF0Y2goL15cXFsoLio/KVxcXShbXFxzXFxTXSopJC8pXG5cbiAgICBpZiAoIXNlY29uZE1hdGNoKSB0aHJvdyBuZXcgRXJyb3IoYENvdWxkIG5vdCBwYXJzZSByZXN0IHBhcnQ6ICR7cmVzdH1gKVxuXG4gICAgY29uc3Qga2V5ID0gc2Vjb25kTWF0Y2hbMV1cbiAgICBjb25zdCBuZXdSZXN0ID0gc2Vjb25kTWF0Y2hbMl1cblxuICAgIC8qKiBAdHlwZSB7QXJyYXk8YW55PiB8IFJlY29yZDxzdHJpbmcsIGFueT59ICovXG4gICAgbGV0IG5ld1Jlc3VsdFxuXG4gICAgaWYgKHJlc3QgPT0gXCJbXVwiKSB7XG4gICAgICBpZiAoIUFycmF5LmlzQXJyYXkocmVzdWx0KSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEV4cGVjdGVkIGFycmF5IHJlc3VsdCBmb3IgcmVzdCAke3Jlc3R9YClcbiAgICAgIH1cblxuICAgICAgcmVzdWx0LnB1c2godmFsdWUpXG4gICAgfSBlbHNlIGlmIChuZXdSZXN0ID09IFwiXCIpIHtcbiAgICAgIHJlc3VsdFtrZXldID0gdmFsdWVcbiAgICB9IGVsc2Uge1xuICAgICAgaWYgKHR5cGVvZiByZXN1bHQgPT0gXCJvYmplY3RcIiAmJiBrZXkgaW4gcmVzdWx0KSB7XG4gICAgICAgIG5ld1Jlc3VsdCA9IHJlc3VsdFtrZXldXG4gICAgICB9IGVsc2UgaWYgKG5ld1Jlc3QgPT0gXCJbXVwiKSB7XG4gICAgICAgIG5ld1Jlc3VsdCA9IFtdXG4gICAgICAgIHJlc3VsdFtrZXldID0gbmV3UmVzdWx0XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBuZXdSZXN1bHQgPSB7fVxuICAgICAgICByZXN1bHRba2V5XSA9IG5ld1Jlc3VsdFxuICAgICAgfVxuXG4gICAgICB0aGlzLnRyZWF0U2Vjb25kKHZhbHVlLCBuZXdSZXN0LCBuZXdSZXN1bHQpXG4gICAgfVxuICB9XG59XG5cbiJdfQ==
109
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGFyYW1zLXRvLW9iamVjdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9odHRwLXNlcnZlci9jbGllbnQvcGFyYW1zLXRvLW9iamVjdC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxZQUFZO0FBRVo7Ozs7O0dBS0c7QUFDSCxTQUFTLDZCQUE2QixDQUFDLElBQUk7SUFDekMsTUFBTSxFQUFDLEdBQUcsRUFBRSxJQUFJLEVBQUMsR0FBRyxJQUFJLENBQUE7SUFDeEIsTUFBTSxLQUFLLEdBQUcsSUFBSSxLQUFLLENBQUMsc0NBQXNDLEdBQUcsY0FBYyxJQUFJLEdBQUcsQ0FBQyxDQUFBO0lBQ3ZGLG1FQUFtRTtJQUNuRSxNQUFNLFVBQVUsR0FBRyxLQUFLLENBQUE7SUFFeEIsVUFBVSxDQUFDLGdCQUFnQixHQUFHO1FBQzVCLGVBQWUsRUFBRTtZQUNmLEdBQUc7WUFDSCxJQUFJO1lBQ0osS0FBSyxFQUFFLGtCQUFrQjtTQUMxQjtLQUNGLENBQUE7SUFFRCxPQUFPLEtBQUssQ0FBQTtBQUNkLENBQUM7QUFFRCxNQUFNLENBQUMsT0FBTyxPQUFPLGNBQWM7SUFDakM7O09BRUc7SUFDSCxZQUFZLE1BQU07UUFDaEIsSUFBSSxDQUFDLE1BQU0sR0FBRyxNQUFNLENBQUE7SUFDdEIsQ0FBQztJQUVELG9EQUFvRDtJQUNwRCxRQUFRO1FBQ04sc0NBQXNDO1FBQ3RDLE1BQU0sTUFBTSxHQUFHLEVBQUUsQ0FBQTtRQUVqQixLQUFJLE1BQU0sR0FBRyxJQUFJLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUM3QixNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxDQUFBO1lBRTlCLElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxFQUFFLEtBQUssRUFBRSxNQUFNLENBQUMsQ0FBQTtRQUN2QyxDQUFDO1FBRUQsT0FBTyxNQUFNLENBQUE7SUFDZixDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSCxZ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bSB7b2JqZWN0fSBhcmdzIC0gQXJncy5cbiAqIEBwYXJhbSB7c3RyaW5nfSBhcmdzLmtleSAtIFBhcmFtZXRlciBrZXkuXG4gKiBAcGFyYW0ge3N0cmluZ30gYXJncy5yZXN0IC0gUmVtYWluaW5nIHVubWF0Y2hlZCBzZWdtZW50LlxuICogQHJldHVybnMge0Vycm9yfSAtIEVycm9yIHdpdGggcGFyc2VyIGNvbnRleHQgYXR0YWNoZWQuXG4gKi9cbmZ1bmN0aW9uIG1hbGZvcm1lZE5lc3RlZFBhcmFtc0tleUVycm9yKGFyZ3MpIHtcbiAgY29uc3Qge2tleSwgcmVzdH0gPSBhcmdzXG4gIGNvbnN0IGVycm9yID0gbmV3IEVycm9yKGBDb3VsZCBub3QgcGFyc2UgbmVzdGVkIHBhcmFtcyBrZXkgXCIke2tleX1cIiBhdCByZXN0IFwiJHtyZXN0fVwiYClcbiAgLyoqIEB0eXBlIHtFcnJvciAmIHt2ZWxvY2lvdXNDb250ZXh0PzogUmVjb3JkPHN0cmluZywgdW5rbm93bj59fSAqL1xuICBjb25zdCB0eXBlZEVycm9yID0gZXJyb3JcblxuICB0eXBlZEVycm9yLnZlbG9jaW91c0NvbnRleHQgPSB7XG4gICAgbmVzdGVkUGFyYW1zS2V5OiB7XG4gICAgICBrZXksXG4gICAgICByZXN0LFxuICAgICAgc3RhZ2U6IFwicGFyYW1zLXRvLW9iamVjdFwiXG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIGVycm9yXG59XG5cbmV4cG9ydCBkZWZhdWx0IGNsYXNzIFBhcmFtc1RvT2JqZWN0IHtcbiAgLyoqXG4gICAqIEBwYXJhbSB7UmVjb3JkPHN0cmluZywgYW55Pn0gb2JqZWN0IC0gT2JqZWN0LlxuICAgKi9cbiAgY29uc3RydWN0b3Iob2JqZWN0KSB7XG4gICAgdGhpcy5vYmplY3QgPSBvYmplY3RcbiAgfVxuXG4gIC8qKiBAcmV0dXJucyB7UmVjb3JkPHN0cmluZywgYW55Pn0gLSBUaGUgb2JqZWN0LiAgKi9cbiAgdG9PYmplY3QoKSB7XG4gICAgLyoqIEB0eXBlIHtSZWNvcmQ8c3RyaW5nLCB1bmtub3duPn0gKi9cbiAgICBjb25zdCByZXN1bHQgPSB7fVxuXG4gICAgZm9yKGNvbnN0IGtleSBpbiB0aGlzLm9iamVjdCkge1xuICAgICAgY29uc3QgdmFsdWUgPSB0aGlzLm9iamVjdFtrZXldXG5cbiAgICAgIHRoaXMudHJlYXRJbml0aWFsKGtleSwgdmFsdWUsIHJlc3VsdClcbiAgICB9XG5cbiAgICByZXR1cm4gcmVzdWx0XG4gIH1cblxuICAvKipcbiAgICogQHBhcmFtIHtzdHJpbmd9IGtleSAtIEtleS5cbiAgICogQHBhcmFtIHthbnl9IHZhbHVlIC0gVmFsdWUgdG8gdXNlLlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT4gfCBhbnlbXX0gcmVzdWx0IC0gUmVzdWx0LlxuICAgKiBAcmV0dXJucyB7dm9pZH0gLSBObyByZXR1cm4gdmFsdWUuXG4gICAqL1xuICB0cmVhdEluaXRpYWwoa2V5LCB2YWx1ZSwgcmVzdWx0KSB7XG4gICAgY29uc3QgZmlyc3RNYXRjaCA9IGtleS5tYXRjaCgvXiguKz8pKFxcWyhbXFxzXFxTXSskKSkvKVxuXG4gICAgaWYgKGZpcnN0TWF0Y2gpIHtcbiAgICAgIGNvbnN0IGlucHV0TmFtZSA9IGZpcnN0TWF0Y2hbMV1cbiAgICAgIGNvbnN0IHJlc3QgPSBmaXJzdE1hdGNoWzJdXG5cbiAgICAgIC8qKiBAdHlwZSB7QXJyYXk8YW55PiB8IFJlY29yZDxzdHJpbmcsIGFueT59ICovXG4gICAgICBsZXQgbmV3UmVzdWx0XG5cbiAgICAgIGlmIChpbnB1dE5hbWUgaW4gcmVzdWx0KSB7XG4gICAgICAgIG5ld1Jlc3VsdCA9IHJlc3VsdFtpbnB1dE5hbWVdXG4gICAgICB9IGVsc2UgaWYgKHJlc3QgPT0gXCJbXVwiKSB7XG4gICAgICAgIG5ld1Jlc3VsdCA9IFtdXG4gICAgICAgIHJlc3VsdFtpbnB1dE5hbWVdID0gbmV3UmVzdWx0XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBuZXdSZXN1bHQgPSB7fVxuICAgICAgICByZXN1bHRbaW5wdXROYW1lXSA9IG5ld1Jlc3VsdFxuICAgICAgfVxuXG4gICAgICB0aGlzLnRyZWF0U2Vjb25kKHZhbHVlLCByZXN0LCBuZXdSZXN1bHQsIGtleSlcbiAgICB9IGVsc2Uge1xuICAgICAgcmVzdWx0W2tleV0gPSB2YWx1ZVxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0ge2FueX0gdmFsdWUgLSBWYWx1ZSB0byB1c2UuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSByZXN0IC0gUmVzdC5cbiAgICogQHBhcmFtIHtSZWNvcmQ8c3RyaW5nLCBhbnk+IHwgYW55W119IHJlc3VsdCAtIFJlc3VsdC5cbiAgICogQHBhcmFtIHtzdHJpbmd9IFtmdWxsS2V5XSAtIE9yaWdpbmFsIGZ1bGwga2V5LlxuICAgKiBAcmV0dXJucyB7dm9pZH0gLSBObyByZXR1cm4gdmFsdWUuXG4gICAqL1xuICB0cmVhdFNlY29uZCh2YWx1ZSwgcmVzdCwgcmVzdWx0LCBmdWxsS2V5ID0gcmVzdCkge1xuICAgIGNvbnN0IHNlY29uZE1hdGNoID0gcmVzdC5tYXRjaCgvXlxcWyguKj8pXFxdKFtcXHNcXFNdKikkLylcblxuICAgIGlmICghc2Vjb25kTWF0Y2gpIHRocm93IG1hbGZvcm1lZE5lc3RlZFBhcmFtc0tleUVycm9yKHtrZXk6IGZ1bGxLZXksIHJlc3R9KVxuXG4gICAgY29uc3Qga2V5ID0gc2Vjb25kTWF0Y2hbMV1cbiAgICBjb25zdCBuZXdSZXN0ID0gc2Vjb25kTWF0Y2hbMl1cblxuICAgIC8qKiBAdHlwZSB7QXJyYXk8YW55PiB8IFJlY29yZDxzdHJpbmcsIGFueT59ICovXG4gICAgbGV0IG5ld1Jlc3VsdFxuXG4gICAgaWYgKHJlc3QgPT0gXCJbXVwiKSB7XG4gICAgICBpZiAoIUFycmF5LmlzQXJyYXkocmVzdWx0KSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEV4cGVjdGVkIGFycmF5IHJlc3VsdCBmb3IgcmVzdCAke3Jlc3R9YClcbiAgICAgIH1cblxuICAgICAgcmVzdWx0LnB1c2godmFsdWUpXG4gICAgfSBlbHNlIGlmIChuZXdSZXN0ID09IFwiXCIpIHtcbiAgICAgIHJlc3VsdFtrZXldID0gdmFsdWVcbiAgICB9IGVsc2Uge1xuICAgICAgaWYgKHR5cGVvZiByZXN1bHQgPT0gXCJvYmplY3RcIiAmJiBrZXkgaW4gcmVzdWx0KSB7XG4gICAgICAgIG5ld1Jlc3VsdCA9IHJlc3VsdFtrZXldXG4gICAgICB9IGVsc2UgaWYgKG5ld1Jlc3QgPT0gXCJbXVwiKSB7XG4gICAgICAgIG5ld1Jlc3VsdCA9IFtdXG4gICAgICAgIHJlc3VsdFtrZXldID0gbmV3UmVzdWx0XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBuZXdSZXN1bHQgPSB7fVxuICAgICAgICByZXN1bHRba2V5XSA9IG5ld1Jlc3VsdFxuICAgICAgfVxuXG4gICAgICB0aGlzLnRyZWF0U2Vjb25kKHZhbHVlLCBuZXdSZXN0LCBuZXdSZXN1bHQsIGZ1bGxLZXkpXG4gICAgfVxuICB9XG59XG4iXX0=
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../../src/http-server/client/request-buffer/index.js"],"names":[],"mappings":"AAUA;IAkBE;;;OAGG;IACH,+BAFG;QAA0D,aAAa,EAA/D,OAAO,2BAA2B,EAAE,OAAO;KACrD,EAIA;IAxBD,mBAAc;IAEd,uBAAuB;IACvB,MADW,MAAM,EAAE,CACV;IAET,mEAA2B;IAE3B,qCAAqC;IACrC,eADW,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CACf;IAElB,4BAA0B;IAE1B,mBAAiB;IACjB,WAAW;IACX,qBAAmB;IACnB,cAAgB;IAOd,2DAAkC;IAClC,eAA8C;IAGhD,gBAEC;IAED;;;OAGG;IACH,WAHW,MAAM,GACJ,MAAM,GAAG,SAAS,CAsG9B;IArES,0BAA6B;IAS7B,8BAA8B;IA8DxC;;;OAGG;IACH,gBAHW,MAAM,GACJ,MAAM,CAQlB;IAED;;OAEG;IACH,kBAFa,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAalC;IAED;;OAEG;IACH,oBAFa,IAAI,CAWhB;IAJC,2BAA6B;IAM/B,gCAEC;IAED;;OAEG;IACH,mBAFa,IAAI,CAKhB;IAED;;;OAGG;IACH,YAHW,MAAM,GACJ,IAAI,CAoChB;IAED;;;OAGG;IACH,yBAHW,MAAM,GACJ,MAAM,GAAG,SAAS,CAU9B;IAED;;OAEG;IACH,kBAFW,MAAM,QAQhB;IADwC,sBAAgD;IAGzF;;;OAGG;IACH,kBAHW,MAAM,GACJ,IAAI,CAgDhB;IArBO,iBAAwB;IACxB,qBAA4C;IAC5C,yBAAoD;IACpD,wBAAiD;IAQjD,uBAAuB;IACvB,eADW,MAAM,EAAE,CACI;IAW/B;;;OAGG;IACH,sBAHW,MAAM,GACJ,IAAI,CAchB;IALC,mBAA0B;IAC1B,oBAA2B;IAC3B,aAAoB;IAKtB,wBASC;IAPG,cAAmE;IASvE;;;OAGG;IACH,+BAHW,MAAM,GACJ,OAAO,CAInB;IAED;;OAEG;IACH,qBAFa,OAAO,CAMnB;IAED;;OAEG;IACH,yBAFa,IAAI,CAOhB;IAJC,wBAA0B;IAC1B,yBAAiC;IAKnC;;;OAGG;IACH,yBAHW,MAAM,GACJ,IAAI,CAuBhB;IAED;;OAEG;IACH,qBAFa,IAAI,CAQhB;IAED;;;OAGG;IACH,mBAHW,MAAM,GACJ,IAAI,CAKhB;IAED,4BAaC;IAED,mCAMC;IAED,mCASC;CACF;mBAlckB,aAAa;mBAEb,oBAAoB;yBAHd,qBAAqB"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../../src/http-server/client/request-buffer/index.js"],"names":[],"mappings":"AAsBA;IAkBE;;;OAGG;IACH,+BAFG;QAA0D,aAAa,EAA/D,OAAO,2BAA2B,EAAE,OAAO;KACrD,EAIA;IAxBD,mBAAc;IAEd,uBAAuB;IACvB,MADW,MAAM,EAAE,CACV;IAET,mEAA2B;IAE3B,qCAAqC;IACrC,eADW,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CACf;IAElB,4BAA0B;IAE1B,mBAAiB;IACjB,WAAW;IACX,qBAAmB;IACnB,cAAgB;IAOd,2DAAkC;IAClC,eAA8C;IAGhD,gBAEC;IAED;;;OAGG;IACH,WAHW,MAAM,GACJ,MAAM,GAAG,SAAS,CAsG9B;IArES,0BAA6B;IAS7B,8BAA8B;IA8DxC;;;OAGG;IACH,gBAHW,MAAM,GACJ,MAAM,CAQlB;IAED;;OAEG;IACH,kBAFa,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAalC;IAED;;OAEG;IACH,oBAFa,IAAI,CAWhB;IAJC,2BAA6B;IAM/B,gCAEC;IAED;;OAEG;IACH,mBAFa,IAAI,CAKhB;IAED;;;OAGG;IACH,YAHW,MAAM,GACJ,IAAI,CAoChB;IAED;;;OAGG;IACH,yBAHW,MAAM,GACJ,MAAM,GAAG,SAAS,CAU9B;IAED;;OAEG;IACH,kBAFW,MAAM,QAQhB;IADwC,sBAAgD;IAGzF;;;OAGG;IACH,kBAHW,MAAM,GACJ,IAAI,CAgDhB;IArBO,iBAAwB;IACxB,qBAA4C;IAC5C,yBAAoD;IACpD,wBAAiD;IAQjD,uBAAuB;IACvB,eADW,MAAM,EAAE,CACI;IAW/B;;;OAGG;IACH,sBAHW,MAAM,GACJ,IAAI,CAchB;IALC,mBAA0B;IAC1B,oBAA2B;IAC3B,aAAoB;IAKtB,wBASC;IAPG,cAAmE;IASvE;;;OAGG;IACH,+BAHW,MAAM,GACJ,OAAO,CAInB;IAED;;OAEG;IACH,qBAFa,OAAO,CAMnB;IAED;;OAEG;IACH,yBAFa,IAAI,CAOhB;IAJC,wBAA0B;IAC1B,yBAAiC;IAKnC;;;OAGG;IACH,yBAHW,MAAM,GACJ,IAAI,CAuBhB;IAED;;OAEG;IACH,qBAFa,IAAI,CAQhB;IAED;;;OAGG;IACH,mBAHW,MAAM,GACJ,IAAI,CAKhB;IAED,4BAaC;IAED,mCAMC;IAED,mCA2BC;CACF;mBAhekB,aAAa;mBAEb,oBAAoB;yBAHd,qBAAqB"}