velocious 1.0.131 → 1.0.132

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.
Files changed (74) hide show
  1. package/README.md +103 -0
  2. package/build/src/configuration.d.ts +8 -0
  3. package/build/src/configuration.d.ts.map +1 -1
  4. package/build/src/configuration.js +13 -1
  5. package/build/src/database/drivers/base-table.d.ts +5 -0
  6. package/build/src/database/drivers/base-table.d.ts.map +1 -1
  7. package/build/src/database/drivers/base-table.js +11 -1
  8. package/build/src/database/drivers/base.d.ts +9 -0
  9. package/build/src/database/drivers/base.d.ts.map +1 -1
  10. package/build/src/database/drivers/base.js +12 -1
  11. package/build/src/database/drivers/mssql/index.d.ts.map +1 -1
  12. package/build/src/database/drivers/mssql/index.js +7 -3
  13. package/build/src/database/drivers/mysql/index.d.ts.map +1 -1
  14. package/build/src/database/drivers/mysql/index.js +2 -1
  15. package/build/src/database/drivers/pgsql/index.d.ts.map +1 -1
  16. package/build/src/database/drivers/pgsql/index.js +2 -1
  17. package/build/src/database/drivers/sqlite/base.d.ts.map +1 -1
  18. package/build/src/database/drivers/sqlite/base.js +3 -2
  19. package/build/src/database/migration/index.d.ts.map +1 -1
  20. package/build/src/database/migration/index.js +11 -2
  21. package/build/src/database/record/index.d.ts.map +1 -1
  22. package/build/src/database/record/index.js +15 -2
  23. package/build/src/http-client/response.d.ts +4 -0
  24. package/build/src/http-client/response.d.ts.map +1 -1
  25. package/build/src/http-client/response.js +19 -17
  26. package/build/src/http-client/websocket-client.d.ts +105 -0
  27. package/build/src/http-client/websocket-client.d.ts.map +1 -0
  28. package/build/src/http-client/websocket-client.js +237 -0
  29. package/build/src/http-server/client/index.d.ts +14 -0
  30. package/build/src/http-server/client/index.d.ts.map +1 -1
  31. package/build/src/http-server/client/index.js +72 -1
  32. package/build/src/http-server/client/params-to-object.d.ts +7 -7
  33. package/build/src/http-server/client/params-to-object.d.ts.map +1 -1
  34. package/build/src/http-server/client/params-to-object.js +4 -4
  35. package/build/src/http-server/client/request-buffer/form-data-part.d.ts +21 -2
  36. package/build/src/http-server/client/request-buffer/form-data-part.d.ts.map +1 -1
  37. package/build/src/http-server/client/request-buffer/form-data-part.js +62 -4
  38. package/build/src/http-server/client/request-parser.js +2 -2
  39. package/build/src/http-server/client/request-runner.d.ts +4 -4
  40. package/build/src/http-server/client/request-runner.d.ts.map +1 -1
  41. package/build/src/http-server/client/request-runner.js +2 -2
  42. package/build/src/http-server/client/uploaded-file/memory-uploaded-file.d.ts +21 -0
  43. package/build/src/http-server/client/uploaded-file/memory-uploaded-file.d.ts.map +1 -0
  44. package/build/src/http-server/client/uploaded-file/memory-uploaded-file.js +26 -0
  45. package/build/src/http-server/client/uploaded-file/temporary-uploaded-file.d.ts +21 -0
  46. package/build/src/http-server/client/uploaded-file/temporary-uploaded-file.d.ts.map +1 -0
  47. package/build/src/http-server/client/uploaded-file/temporary-uploaded-file.js +26 -0
  48. package/build/src/http-server/client/uploaded-file/uploaded-file.d.ts +29 -0
  49. package/build/src/http-server/client/uploaded-file/uploaded-file.d.ts.map +1 -0
  50. package/build/src/http-server/client/uploaded-file/uploaded-file.js +34 -0
  51. package/build/src/http-server/client/websocket-request.d.ts +40 -0
  52. package/build/src/http-server/client/websocket-request.d.ts.map +1 -0
  53. package/build/src/http-server/client/websocket-request.js +88 -0
  54. package/build/src/http-server/client/websocket-session.d.ts +73 -0
  55. package/build/src/http-server/client/websocket-session.d.ts.map +1 -0
  56. package/build/src/http-server/client/websocket-session.js +231 -0
  57. package/build/src/http-server/server-client.d.ts.map +1 -1
  58. package/build/src/http-server/server-client.js +6 -3
  59. package/build/src/http-server/websocket-events-host.d.ts +22 -0
  60. package/build/src/http-server/websocket-events-host.d.ts.map +1 -0
  61. package/build/src/http-server/websocket-events-host.js +29 -0
  62. package/build/src/http-server/websocket-events.d.ts +20 -0
  63. package/build/src/http-server/websocket-events.d.ts.map +1 -0
  64. package/build/src/http-server/websocket-events.js +23 -0
  65. package/build/src/http-server/worker-handler/index.d.ts +19 -4
  66. package/build/src/http-server/worker-handler/index.d.ts.map +1 -1
  67. package/build/src/http-server/worker-handler/index.js +24 -3
  68. package/build/src/http-server/worker-handler/worker-thread.d.ts +18 -2
  69. package/build/src/http-server/worker-handler/worker-thread.d.ts.map +1 -1
  70. package/build/src/http-server/worker-handler/worker-thread.js +23 -2
  71. package/build/src/routes/resolver.d.ts +5 -0
  72. package/build/src/routes/resolver.d.ts.map +1 -1
  73. package/build/src/routes/resolver.js +28 -2
  74. package/package.json +1 -1
@@ -26,6 +26,10 @@ export default class Response {
26
26
  tryToParse(): void;
27
27
  statusLine: string;
28
28
  completeResponse(): void;
29
+ /**
30
+ * @returns {number}
31
+ */
32
+ _contentLengthNumber(): number;
29
33
  }
30
34
  import Header from "./header.js";
31
35
  //# sourceMappingURL=response.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"response.d.ts","sourceRoot":"","sources":["../../../src/http-client/response.js"],"names":[],"mappings":"AAIA;IACE;;;;OAIG;IACH,oCAHG;QAAqB,MAAM,EAAnB,MAAM;QACkB,UAAU,EAAlC,MAAa,IAAI;KAC3B,EAaA;IATC,uBAAuB;IACvB,SADW,MAAM,EAAE,CACF;IAEjB,eAAyC;IACzC,kBATsB,IAAI,CASE;IAC5B,cAA0B;IAE1B,qBAAqB;IACrB,UADW,MAAM,CACc;IAGjC,2BAA2B;IAC3B,WADY,MAAM,QAIjB;IAED;;;OAGG;IACH,gBAHW,MAAM,GACJ,MAAM,CAclB;IAED,YAaC;IAED,mBAkEC;IAzBS,mBAAsB;IA2BhC,yBAGC;CACF;mBArIkB,aAAa"}
1
+ {"version":3,"file":"response.d.ts","sourceRoot":"","sources":["../../../src/http-client/response.js"],"names":[],"mappings":"AAIA;IACE;;;;OAIG;IACH,oCAHG;QAAqB,MAAM,EAAnB,MAAM;QACkB,UAAU,EAAlC,MAAa,IAAI;KAC3B,EAaA;IATC,uBAAuB;IACvB,SADW,MAAM,EAAE,CACF;IAEjB,eAAyC;IACzC,kBATsB,IAAI,CASE;IAC5B,cAA0B;IAE1B,qBAAqB;IACrB,UADW,MAAM,CACc;IAGjC,2BAA2B;IAC3B,WADY,MAAM,QAIjB;IAED;;;OAGG;IACH,gBAHW,MAAM,GACJ,MAAM,CAclB;IAED,YAaC;IAED,mBAsDC;IA3BS,mBAAsB;IA6BhC,yBAGC;IAED;;OAEG;IACH,wBAFa,MAAM,CAalB;CACF;mBAzIkB,aAAa"}
@@ -50,20 +50,7 @@ export default class Response {
50
50
  tryToParse() {
51
51
  while (true) {
52
52
  if (this.state == "body") {
53
- const contentLengthValue = this.getHeader("Content-Length")?.getValue();
54
- if (contentLengthValue === undefined) {
55
- throw new Error("No content length given");
56
- }
57
- let contentLengthNumber;
58
- if (typeof contentLengthValue == "number") {
59
- contentLengthNumber = contentLengthValue;
60
- }
61
- else if (contentLengthValue == "string") {
62
- contentLengthNumber = parseInt(contentLengthValue);
63
- }
64
- else {
65
- throw new Error(`Content-Length is not a number: ${contentLengthValue}`);
66
- }
53
+ const contentLengthNumber = this._contentLengthNumber();
67
54
  if (this.response.byteLength >= contentLengthNumber) {
68
55
  this.completeResponse();
69
56
  break;
@@ -89,11 +76,12 @@ export default class Response {
89
76
  }
90
77
  else if (this.state == "headers") {
91
78
  if (line == "") {
92
- if (this.method == "GET" || this.method == "HEAD") {
79
+ const contentLengthNumber = this._contentLengthNumber();
80
+ if (!contentLengthNumber) {
93
81
  this.completeResponse();
94
82
  break;
95
83
  }
96
- else if (this.method == "POST") {
84
+ else {
97
85
  this.state = "body";
98
86
  }
99
87
  }
@@ -116,5 +104,19 @@ export default class Response {
116
104
  this.state = "done";
117
105
  this.onComplete();
118
106
  }
107
+ /**
108
+ * @returns {number}
109
+ */
110
+ _contentLengthNumber() {
111
+ const header = this.headers.find((currentHeader) => currentHeader.getName().toLowerCase() == "content-length");
112
+ if (!header)
113
+ return 0;
114
+ const contentLengthValue = header.getValue();
115
+ if (typeof contentLengthValue === "number")
116
+ return contentLengthValue;
117
+ if (typeof contentLengthValue === "string")
118
+ return parseInt(contentLengthValue);
119
+ throw new Error(`Content-Length is not a number: ${contentLengthValue}`);
120
+ }
119
121
  }
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"response.js","sourceRoot":"","sources":["../../../src/http-client/response.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Header from \"./header.js\"\n\nexport default class Response {\n  /**\n   * @param {object} args\n   * @param {string} args.method\n   * @param {function() : void} args.onComplete\n   */\n  constructor({method = \"GET\", onComplete}) {\n    if (!method) throw new Error(`Invalid method given: ${method}`)\n\n    /** @type {Header[]} */\n    this.headers = []\n\n    this.method = method.toUpperCase().trim()\n    this.onComplete = onComplete\n    this.state = \"status-line\"\n\n    /** @type {Buffer} */\n    this.response = Buffer.alloc(0);\n  }\n\n  /** @param {Buffer} data */\n  feed(data) {\n    this.response = Buffer.concat([this.response, data])\n    this.tryToParse()\n  }\n\n  /**\n   * @param {string} name\n   * @returns {Header}\n   */\n  getHeader(name) {\n    const compareName = name.toLowerCase().trim()\n\n    for (const header of this.headers) {\n      const headerCompareName = header.getName().toLowerCase().trim()\n\n      if (compareName == headerCompareName) {\n        return header\n      }\n    }\n\n    throw new Error(`Header ${name} not found`)\n  }\n\n  json() {\n    const contentTypeHeader = this.getHeader(\"Content-Type\")?.getValue()\n\n    if (typeof contentTypeHeader != \"string\") throw new Error(`Content-Type wasn't a string: ${contentTypeHeader}`)\n\n    if (!contentTypeHeader.toLowerCase().trim().startsWith(\"application/json\")) {\n      throw new Error(`Content-Type is not JSON: ${contentTypeHeader}`)\n    }\n\n    const body = this.response.toString()\n    const json = JSON.parse(body)\n\n    return json\n  }\n\n  tryToParse() {\n    while (true) {\n      if (this.state == \"body\") {\n        const contentLengthValue = this.getHeader(\"Content-Length\")?.getValue()\n\n        if (contentLengthValue === undefined) {\n          throw new Error(\"No content length given\")\n        }\n\n        let contentLengthNumber\n\n        if (typeof contentLengthValue == \"number\") {\n          contentLengthNumber = contentLengthValue\n        } else if (contentLengthValue == \"string\") {\n          contentLengthNumber = parseInt(contentLengthValue)\n        } else {\n          throw new Error(`Content-Length is not a number: ${contentLengthValue}`)\n        }\n\n        if (this.response.byteLength >= contentLengthNumber) {\n          this.completeResponse()\n          break\n        }\n      } else {\n        const response = this.response.toString()\n        let lineEndIndex = response.indexOf(\"\\r\\n\")\n        let lineEndLength = 2\n\n        if (lineEndIndex === -1) {\n          lineEndIndex = response.indexOf(\"\\n\")\n          lineEndLength = 1\n        }\n\n        if (lineEndIndex === -1) {\n          break // We need to get fed more to continue reading\n        } else {\n          const line = response.substring(0, lineEndIndex)\n\n          this.response = this.response.slice(lineEndIndex + lineEndLength)\n\n          if (this.state == \"status-line\") {\n            this.statusLine = line\n            this.state = \"headers\"\n          } else if (this.state == \"headers\") {\n            if (line == \"\") {\n              if (this.method == \"GET\" || this.method == \"HEAD\") {\n                this.completeResponse()\n                break\n              } else if (this.method == \"POST\") {\n                this.state = \"body\"\n              }\n            } else {\n              const headerMatch = line.match(/^(.+?):\\s*(.+)$/)\n\n              if (!headerMatch) throw new Error(`Invalid header: ${line}`)\n\n              const header = new Header(headerMatch[1], headerMatch[2])\n\n              this.headers.push(header)\n            }\n          } else {\n            throw new Error(`Unexpected state: ${this.state}`)\n          }\n        }\n      }\n    }\n  }\n\n  completeResponse() {\n    this.state = \"done\"\n    this.onComplete()\n  }\n}\n"]}
122
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"response.js","sourceRoot":"","sources":["../../../src/http-client/response.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,OAAO,MAAM,MAAM,aAAa,CAAA;AAEhC,MAAM,CAAC,OAAO,OAAO,QAAQ;IAC3B;;;;OAIG;IACH,YAAY,EAAC,MAAM,GAAG,KAAK,EAAE,UAAU,EAAC;QACtC,IAAI,CAAC,MAAM;YAAE,MAAM,IAAI,KAAK,CAAC,yBAAyB,MAAM,EAAE,CAAC,CAAA;QAE/D,uBAAuB;QACvB,IAAI,CAAC,OAAO,GAAG,EAAE,CAAA;QAEjB,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAA;QACzC,IAAI,CAAC,UAAU,GAAG,UAAU,CAAA;QAC5B,IAAI,CAAC,KAAK,GAAG,aAAa,CAAA;QAE1B,qBAAqB;QACrB,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAClC,CAAC;IAED,2BAA2B;IAC3B,IAAI,CAAC,IAAI;QACP,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC,CAAA;QACpD,IAAI,CAAC,UAAU,EAAE,CAAA;IACnB,CAAC;IAED;;;OAGG;IACH,SAAS,CAAC,IAAI;QACZ,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAA;QAE7C,KAAK,MAAM,MAAM,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YAClC,MAAM,iBAAiB,GAAG,MAAM,CAAC,OAAO,EAAE,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAA;YAE/D,IAAI,WAAW,IAAI,iBAAiB,EAAE,CAAC;gBACrC,OAAO,MAAM,CAAA;YACf,CAAC;QACH,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,UAAU,IAAI,YAAY,CAAC,CAAA;IAC7C,CAAC;IAED,IAAI;QACF,MAAM,iBAAiB,GAAG,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,EAAE,QAAQ,EAAE,CAAA;QAEpE,IAAI,OAAO,iBAAiB,IAAI,QAAQ;YAAE,MAAM,IAAI,KAAK,CAAC,iCAAiC,iBAAiB,EAAE,CAAC,CAAA;QAE/G,IAAI,CAAC,iBAAiB,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC,UAAU,CAAC,kBAAkB,CAAC,EAAE,CAAC;YAC3E,MAAM,IAAI,KAAK,CAAC,6BAA6B,iBAAiB,EAAE,CAAC,CAAA;QACnE,CAAC;QAED,MAAM,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAA;QACrC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;QAE7B,OAAO,IAAI,CAAA;IACb,CAAC;IAED,UAAU;QACR,OAAO,IAAI,EAAE,CAAC;YACZ,IAAI,IAAI,CAAC,KAAK,IAAI,MAAM,EAAE,CAAC;gBACzB,MAAM,mBAAmB,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAA;gBAEvD,IAAI,IAAI,CAAC,QAAQ,CAAC,UAAU,IAAI,mBAAmB,EAAE,CAAC;oBACpD,IAAI,CAAC,gBAAgB,EAAE,CAAA;oBACvB,MAAK;gBACP,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAA;gBACzC,IAAI,YAAY,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAA;gBAC3C,IAAI,aAAa,GAAG,CAAC,CAAA;gBAErB,IAAI,YAAY,KAAK,CAAC,CAAC,EAAE,CAAC;oBACxB,YAAY,GAAG,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAA;oBACrC,aAAa,GAAG,CAAC,CAAA;gBACnB,CAAC;gBAED,IAAI,YAAY,KAAK,CAAC,CAAC,EAAE,CAAC;oBACxB,MAAK,CAAC,8CAA8C;gBACtD,CAAC;qBAAM,CAAC;oBACN,MAAM,IAAI,GAAG,QAAQ,CAAC,SAAS,CAAC,CAAC,EAAE,YAAY,CAAC,CAAA;oBAEhD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,YAAY,GAAG,aAAa,CAAC,CAAA;oBAEjE,IAAI,IAAI,CAAC,KAAK,IAAI,aAAa,EAAE,CAAC;wBAChC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAA;wBACtB,IAAI,CAAC,KAAK,GAAG,SAAS,CAAA;oBACxB,CAAC;yBAAM,IAAI,IAAI,CAAC,KAAK,IAAI,SAAS,EAAE,CAAC;wBACnC,IAAI,IAAI,IAAI,EAAE,EAAE,CAAC;4BACf,MAAM,mBAAmB,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAA;4BAEvD,IAAI,CAAC,mBAAmB,EAAE,CAAC;gCACzB,IAAI,CAAC,gBAAgB,EAAE,CAAA;gCACvB,MAAK;4BACP,CAAC;iCAAM,CAAC;gCACN,IAAI,CAAC,KAAK,GAAG,MAAM,CAAA;4BACrB,CAAC;wBACH,CAAC;6BAAM,CAAC;4BACN,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAA;4BAEjD,IAAI,CAAC,WAAW;gCAAE,MAAM,IAAI,KAAK,CAAC,mBAAmB,IAAI,EAAE,CAAC,CAAA;4BAE5D,MAAM,MAAM,GAAG,IAAI,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC,CAAA;4BAEzD,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;wBAC3B,CAAC;oBACH,CAAC;yBAAM,CAAC;wBACN,MAAM,IAAI,KAAK,CAAC,qBAAqB,IAAI,CAAC,KAAK,EAAE,CAAC,CAAA;oBACpD,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAED,gBAAgB;QACd,IAAI,CAAC,KAAK,GAAG,MAAM,CAAA;QACnB,IAAI,CAAC,UAAU,EAAE,CAAA;IACnB,CAAC;IAED;;OAEG;IACH,oBAAoB;QAClB,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,aAAa,EAAE,EAAE,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC,WAAW,EAAE,IAAI,gBAAgB,CAAC,CAAA;QAE9G,IAAI,CAAC,MAAM;YAAE,OAAO,CAAC,CAAA;QAErB,MAAM,kBAAkB,GAAG,MAAM,CAAC,QAAQ,EAAE,CAAA;QAE5C,IAAI,OAAO,kBAAkB,KAAK,QAAQ;YAAE,OAAO,kBAAkB,CAAA;QACrE,IAAI,OAAO,kBAAkB,KAAK,QAAQ;YAAE,OAAO,QAAQ,CAAC,kBAAkB,CAAC,CAAA;QAE/E,MAAM,IAAI,KAAK,CAAC,mCAAmC,kBAAkB,EAAE,CAAC,CAAA;IAC1E,CAAC;CACF","sourcesContent":["// @ts-check\n\nimport Header from \"./header.js\"\n\nexport default class Response {\n  /**\n   * @param {object} args\n   * @param {string} args.method\n   * @param {function() : void} args.onComplete\n   */\n  constructor({method = \"GET\", onComplete}) {\n    if (!method) throw new Error(`Invalid method given: ${method}`)\n\n    /** @type {Header[]} */\n    this.headers = []\n\n    this.method = method.toUpperCase().trim()\n    this.onComplete = onComplete\n    this.state = \"status-line\"\n\n    /** @type {Buffer} */\n    this.response = Buffer.alloc(0);\n  }\n\n  /** @param {Buffer} data */\n  feed(data) {\n    this.response = Buffer.concat([this.response, data])\n    this.tryToParse()\n  }\n\n  /**\n   * @param {string} name\n   * @returns {Header}\n   */\n  getHeader(name) {\n    const compareName = name.toLowerCase().trim()\n\n    for (const header of this.headers) {\n      const headerCompareName = header.getName().toLowerCase().trim()\n\n      if (compareName == headerCompareName) {\n        return header\n      }\n    }\n\n    throw new Error(`Header ${name} not found`)\n  }\n\n  json() {\n    const contentTypeHeader = this.getHeader(\"Content-Type\")?.getValue()\n\n    if (typeof contentTypeHeader != \"string\") throw new Error(`Content-Type wasn't a string: ${contentTypeHeader}`)\n\n    if (!contentTypeHeader.toLowerCase().trim().startsWith(\"application/json\")) {\n      throw new Error(`Content-Type is not JSON: ${contentTypeHeader}`)\n    }\n\n    const body = this.response.toString()\n    const json = JSON.parse(body)\n\n    return json\n  }\n\n  tryToParse() {\n    while (true) {\n      if (this.state == \"body\") {\n        const contentLengthNumber = this._contentLengthNumber()\n\n        if (this.response.byteLength >= contentLengthNumber) {\n          this.completeResponse()\n          break\n        }\n      } else {\n        const response = this.response.toString()\n        let lineEndIndex = response.indexOf(\"\\r\\n\")\n        let lineEndLength = 2\n\n        if (lineEndIndex === -1) {\n          lineEndIndex = response.indexOf(\"\\n\")\n          lineEndLength = 1\n        }\n\n        if (lineEndIndex === -1) {\n          break // We need to get fed more to continue reading\n        } else {\n          const line = response.substring(0, lineEndIndex)\n\n          this.response = this.response.slice(lineEndIndex + lineEndLength)\n\n          if (this.state == \"status-line\") {\n            this.statusLine = line\n            this.state = \"headers\"\n          } else if (this.state == \"headers\") {\n            if (line == \"\") {\n              const contentLengthNumber = this._contentLengthNumber()\n\n              if (!contentLengthNumber) {\n                this.completeResponse()\n                break\n              } else {\n                this.state = \"body\"\n              }\n            } else {\n              const headerMatch = line.match(/^(.+?):\\s*(.+)$/)\n\n              if (!headerMatch) throw new Error(`Invalid header: ${line}`)\n\n              const header = new Header(headerMatch[1], headerMatch[2])\n\n              this.headers.push(header)\n            }\n          } else {\n            throw new Error(`Unexpected state: ${this.state}`)\n          }\n        }\n      }\n    }\n  }\n\n  completeResponse() {\n    this.state = \"done\"\n    this.onComplete()\n  }\n\n  /**\n   * @returns {number}\n   */\n  _contentLengthNumber() {\n    const header = this.headers.find((currentHeader) => currentHeader.getName().toLowerCase() == \"content-length\")\n\n    if (!header) return 0\n\n    const contentLengthValue = header.getValue()\n\n    if (typeof contentLengthValue === \"number\") return contentLengthValue\n    if (typeof contentLengthValue === \"string\") return parseInt(contentLengthValue)\n\n    throw new Error(`Content-Length is not a number: ${contentLengthValue}`)\n  }\n}\n"]}
@@ -0,0 +1,105 @@
1
+ /**
2
+ * A small websocket client that mirrors simple HTTP-style calls and channel subscriptions.
3
+ */
4
+ export default class VelociousWebsocketClient {
5
+ /**
6
+ * @param {object} [args]
7
+ * @param {boolean} [args.debug]
8
+ * @param {string} [args.url] Full websocket URL (default: ws://127.0.0.1:3006/websocket)
9
+ */
10
+ constructor({ debug, url }?: {
11
+ debug?: boolean;
12
+ url?: string;
13
+ });
14
+ debug: boolean;
15
+ pendingRequests: Map<any, any>;
16
+ subscribedChannels: Set<any>;
17
+ url: string;
18
+ listeners: Map<any, any>;
19
+ nextID: number;
20
+ /**
21
+ * Ensure a websocket connection is open.
22
+ * @returns {Promise<void>}
23
+ */
24
+ connect(): Promise<void>;
25
+ connectPromise: Promise<any>;
26
+ socket: WebSocket;
27
+ /**
28
+ * Close the websocket and clear pending state.
29
+ * @returns {Promise<void>}
30
+ */
31
+ close(): Promise<void>;
32
+ /**
33
+ * Perform a POST request over the websocket.
34
+ * @param {string} path
35
+ * @param {any} [body]
36
+ * @param {{headers?: Record<string, string>}} [options]
37
+ * @returns {Promise<VelociousWebsocketResponse>}
38
+ */
39
+ post(path: string, body?: any, options?: {
40
+ headers?: Record<string, string>;
41
+ }): Promise<VelociousWebsocketResponse>;
42
+ /**
43
+ * Perform a GET request over the websocket.
44
+ * @param {string} path
45
+ * @param {{headers?: Record<string, string>}} [options]
46
+ * @returns {Promise<VelociousWebsocketResponse>}
47
+ */
48
+ get(path: string, options?: {
49
+ headers?: Record<string, string>;
50
+ }): Promise<VelociousWebsocketResponse>;
51
+ /**
52
+ * Subscribe to a channel for server-sent events.
53
+ * @param {string} channel
54
+ * @param {(payload: any) => void} callback
55
+ * @returns {() => void} unsubscribe function
56
+ */
57
+ on(channel: string, callback: (payload: any) => void): () => void;
58
+ /**
59
+ * @private
60
+ * @param {string} method
61
+ * @param {string} path
62
+ * @param {object} [options]
63
+ * @param {any} [options.body]
64
+ * @param {Record<string, string>} [options.headers]
65
+ * @returns {Promise<VelociousWebsocketResponse>}
66
+ */
67
+ private request;
68
+ /**
69
+ * @private
70
+ * @param {MessageEvent<any>} event
71
+ */
72
+ private onMessage;
73
+ /**
74
+ * Reject all pending requests when the socket closes unexpectedly.
75
+ * @private
76
+ */
77
+ private onClose;
78
+ /**
79
+ * @private
80
+ * @param {Record<string, any>} payload
81
+ */
82
+ private _sendMessage;
83
+ /**
84
+ * @private
85
+ * @param {...any} args
86
+ * @returns {void}
87
+ */
88
+ private _debug;
89
+ }
90
+ declare class VelociousWebsocketResponse {
91
+ /**
92
+ * @param {object} message
93
+ */
94
+ constructor(message: object);
95
+ body: any;
96
+ headers: any;
97
+ id: any;
98
+ statusCode: any;
99
+ statusMessage: any;
100
+ type: any;
101
+ /** @returns {any} */
102
+ json(): any;
103
+ }
104
+ export {};
105
+ //# sourceMappingURL=websocket-client.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"websocket-client.d.ts","sourceRoot":"","sources":["../../../src/http-client/websocket-client.js"],"names":[],"mappings":"AAEA;;GAEG;AACH;IACE;;;;OAIG;IACH,6BAHG;QAAuB,KAAK,GAApB,OAAO;QACO,GAAG,GAAjB,MAAM;KAChB,EAUA;IANC,eAAkB;IAClB,+BAAgC;IAChC,6BAAmC;IACnC,YAAiD;IACjD,yBAA0B;IAC1B,eAAe;IAGjB;;;OAGG;IACH,WAFa,OAAO,CAAC,IAAI,CAAC,CA+BzB;IAzBC,6BAsBE;IArBA,kBAAqC;IA0BzC;;;OAGG;IACH,SAFa,OAAO,CAAC,IAAI,CAAC,CAYzB;IAED;;;;;;OAMG;IACH,WALW,MAAM,SACN,GAAG,YACH;QAAC,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;KAAC,GAChC,OAAO,CAAC,0BAA0B,CAAC,CAI/C;IAED;;;;;OAKG;IACH,UAJW,MAAM,YACN;QAAC,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;KAAC,GAChC,OAAO,CAAC,0BAA0B,CAAC,CAI/C;IAED;;;;;OAKG;IACH,YAJW,MAAM,YACN,CAAC,OAAO,EAAE,GAAG,KAAK,IAAI,GACpB,MAAM,IAAI,CAyBtB;IAED;;;;;;;;OAQG;IACH,gBAiBC;IAED;;;OAGG;IACH,kBA6CC;IAED;;;OAGG;IACH,gBAOC;IAED;;;OAGG;IACH,qBASC;IAED;;;;OAIG;IACH,eAIC;CACF;AAED;IACE;;OAEG;IACH,qBAFW,MAAM,EAShB;IANC,UAAwB;IACxB,aAAoC;IACpC,QAAoB;IACpB,gBAA2C;IAC3C,mBAAkD;IAClD,UAAwB;IAG1B,qBAAqB;IACrB,QADc,GAAG,CAOhB;CACF"}
@@ -0,0 +1,237 @@
1
+ // @ts-check
2
+ /**
3
+ * A small websocket client that mirrors simple HTTP-style calls and channel subscriptions.
4
+ */
5
+ export default class VelociousWebsocketClient {
6
+ /**
7
+ * @param {object} [args]
8
+ * @param {boolean} [args.debug]
9
+ * @param {string} [args.url] Full websocket URL (default: ws://127.0.0.1:3006/websocket)
10
+ */
11
+ constructor({ debug = false, url } = {}) {
12
+ if (!globalThis.WebSocket)
13
+ throw new Error("WebSocket global is not available");
14
+ this.debug = debug;
15
+ this.pendingRequests = new Map();
16
+ this.subscribedChannels = new Set();
17
+ this.url = url || "ws://127.0.0.1:3006/websocket";
18
+ this.listeners = new Map();
19
+ this.nextID = 1;
20
+ }
21
+ /**
22
+ * Ensure a websocket connection is open.
23
+ * @returns {Promise<void>}
24
+ */
25
+ async connect() {
26
+ if (this.socket && this.socket.readyState === this.socket.OPEN)
27
+ return;
28
+ if (this.connectPromise)
29
+ return this.connectPromise;
30
+ this.connectPromise = new Promise((resolve, reject) => {
31
+ this.socket = new WebSocket(this.url);
32
+ const cleanup = () => {
33
+ this.socket?.removeEventListener("open", onOpen);
34
+ this.socket?.removeEventListener("error", onError);
35
+ };
36
+ const onOpen = () => {
37
+ cleanup();
38
+ resolve();
39
+ };
40
+ const onError = (event) => {
41
+ cleanup();
42
+ const error = event?.error || new Error("Websocket connection error");
43
+ reject(error);
44
+ };
45
+ this.socket.addEventListener("open", onOpen);
46
+ this.socket.addEventListener("error", onError);
47
+ this.socket.addEventListener("message", this.onMessage);
48
+ this.socket.addEventListener("close", this.onClose);
49
+ });
50
+ return this.connectPromise;
51
+ }
52
+ /**
53
+ * Close the websocket and clear pending state.
54
+ * @returns {Promise<void>}
55
+ */
56
+ async close() {
57
+ if (!this.socket)
58
+ return;
59
+ await new Promise((resolve) => {
60
+ this.socket?.addEventListener("close", () => resolve());
61
+ this.socket?.close();
62
+ });
63
+ this.socket = undefined;
64
+ this.connectPromise = undefined;
65
+ }
66
+ /**
67
+ * Perform a POST request over the websocket.
68
+ * @param {string} path
69
+ * @param {any} [body]
70
+ * @param {{headers?: Record<string, string>}} [options]
71
+ * @returns {Promise<VelociousWebsocketResponse>}
72
+ */
73
+ async post(path, body, options = {}) {
74
+ return await this.request("POST", path, { ...options, body });
75
+ }
76
+ /**
77
+ * Perform a GET request over the websocket.
78
+ * @param {string} path
79
+ * @param {{headers?: Record<string, string>}} [options]
80
+ * @returns {Promise<VelociousWebsocketResponse>}
81
+ */
82
+ async get(path, options = {}) {
83
+ return await this.request("GET", path, options);
84
+ }
85
+ /**
86
+ * Subscribe to a channel for server-sent events.
87
+ * @param {string} channel
88
+ * @param {(payload: any) => void} callback
89
+ * @returns {() => void} unsubscribe function
90
+ */
91
+ on(channel, callback) {
92
+ if (!this.listeners.has(channel)) {
93
+ this.listeners.set(channel, new Set());
94
+ this.subscribedChannels.add(channel);
95
+ void this.connect().then(() => {
96
+ this._sendMessage({ channel, type: "subscribe" });
97
+ }).catch((error) => this._debug("Subscribe failed", error));
98
+ }
99
+ const channelListeners = this.listeners.get(channel);
100
+ if (!channelListeners)
101
+ throw new Error("Listeners map not initialized");
102
+ channelListeners.add(callback);
103
+ return () => {
104
+ channelListeners.delete(callback);
105
+ if (channelListeners.size === 0) {
106
+ this.listeners.delete(channel);
107
+ }
108
+ };
109
+ }
110
+ /**
111
+ * @private
112
+ * @param {string} method
113
+ * @param {string} path
114
+ * @param {object} [options]
115
+ * @param {any} [options.body]
116
+ * @param {Record<string, string>} [options.headers]
117
+ * @returns {Promise<VelociousWebsocketResponse>}
118
+ */
119
+ async request(method, path, { body, headers } = {}) {
120
+ await this.connect();
121
+ const id = `ws-${this.nextID++}`;
122
+ const payload = {
123
+ body,
124
+ headers,
125
+ id,
126
+ method,
127
+ path,
128
+ type: "request"
129
+ };
130
+ return await new Promise((resolve, reject) => {
131
+ this.pendingRequests.set(id, { resolve, reject });
132
+ this._sendMessage(payload);
133
+ });
134
+ }
135
+ /**
136
+ * @private
137
+ * @param {MessageEvent<any>} event
138
+ */
139
+ onMessage = (event) => {
140
+ const raw = typeof event.data === "string" ? event.data : event.data?.toString?.();
141
+ if (!raw)
142
+ return;
143
+ let message;
144
+ try {
145
+ message = JSON.parse(raw);
146
+ }
147
+ catch (error) {
148
+ this._debug("Failed to parse websocket message", error);
149
+ return;
150
+ }
151
+ const { type } = message;
152
+ if (type === "response") {
153
+ const { id } = message;
154
+ const pending = id ? this.pendingRequests.get(id) : undefined;
155
+ if (pending) {
156
+ this.pendingRequests.delete(id);
157
+ pending.resolve(new VelociousWebsocketResponse(message));
158
+ }
159
+ else {
160
+ this._debug(`No pending request for response id ${id}`);
161
+ }
162
+ }
163
+ else if (type === "event") {
164
+ const { channel, payload } = message;
165
+ const callbacks = this.listeners.get(channel);
166
+ callbacks?.forEach((callback) => {
167
+ try {
168
+ callback(payload);
169
+ }
170
+ catch (error) {
171
+ this._debug("Listener error", error);
172
+ }
173
+ });
174
+ }
175
+ else if (type === "error" && message.id) {
176
+ const pending = this.pendingRequests.get(message.id);
177
+ if (pending) {
178
+ this.pendingRequests.delete(message.id);
179
+ pending.reject(new Error(message.error || "Unknown websocket error"));
180
+ }
181
+ }
182
+ };
183
+ /**
184
+ * Reject all pending requests when the socket closes unexpectedly.
185
+ * @private
186
+ */
187
+ onClose = () => {
188
+ for (const [id, { reject }] of this.pendingRequests.entries()) {
189
+ reject(new Error(`Websocket closed before response for ${id}`));
190
+ }
191
+ this.pendingRequests.clear();
192
+ this.connectPromise = undefined;
193
+ };
194
+ /**
195
+ * @private
196
+ * @param {Record<string, any>} payload
197
+ */
198
+ _sendMessage(payload) {
199
+ if (!this.socket || this.socket.readyState !== this.socket.OPEN) {
200
+ throw new Error("Websocket is not open");
201
+ }
202
+ const json = JSON.stringify(payload);
203
+ this._debug("Sending", json);
204
+ this.socket.send(json);
205
+ }
206
+ /**
207
+ * @private
208
+ * @param {...any} args
209
+ * @returns {void}
210
+ */
211
+ _debug(...args) {
212
+ if (!this.debug)
213
+ return;
214
+ console.debug("[VelociousWebsocketClient]", ...args);
215
+ }
216
+ }
217
+ class VelociousWebsocketResponse {
218
+ /**
219
+ * @param {object} message
220
+ */
221
+ constructor(message) {
222
+ this.body = message.body;
223
+ this.headers = message.headers || {};
224
+ this.id = message.id;
225
+ this.statusCode = message.statusCode || 200;
226
+ this.statusMessage = message.statusMessage || "OK";
227
+ this.type = message.type;
228
+ }
229
+ /** @returns {any} */
230
+ json() {
231
+ if (typeof this.body !== "string") {
232
+ throw new Error("Response body is not a string");
233
+ }
234
+ return JSON.parse(this.body);
235
+ }
236
+ }
237
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-client.js","sourceRoot":"","sources":["../../../src/http-client/websocket-client.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\n/**\n * A small websocket client that mirrors simple HTTP-style calls and channel subscriptions.\n */\nexport default class VelociousWebsocketClient {\n  /**\n   * @param {object} [args]\n   * @param {boolean} [args.debug]\n   * @param {string} [args.url] Full websocket URL (default: ws://127.0.0.1:3006/websocket)\n   */\n  constructor({debug = false, url} = {}) {\n    if (!globalThis.WebSocket) throw new Error(\"WebSocket global is not available\")\n\n    this.debug = debug\n    this.pendingRequests = new Map()\n    this.subscribedChannels = new Set()\n    this.url = url || \"ws://127.0.0.1:3006/websocket\"\n    this.listeners = new Map()\n    this.nextID = 1\n  }\n\n  /**\n   * Ensure a websocket connection is open.\n   * @returns {Promise<void>}\n   */\n  async connect() {\n    if (this.socket && this.socket.readyState === this.socket.OPEN) return\n    if (this.connectPromise) return this.connectPromise\n\n    this.connectPromise = new Promise((resolve, reject) => {\n      this.socket = new WebSocket(this.url)\n\n      const cleanup = () => {\n        this.socket?.removeEventListener(\"open\", onOpen)\n        this.socket?.removeEventListener(\"error\", onError)\n      }\n\n      const onOpen = () => {\n        cleanup()\n        resolve()\n      }\n      const onError = (event) => {\n        cleanup()\n        const error = event?.error || new Error(\"Websocket connection error\")\n        reject(error)\n      }\n\n      this.socket.addEventListener(\"open\", onOpen)\n      this.socket.addEventListener(\"error\", onError)\n      this.socket.addEventListener(\"message\", this.onMessage)\n      this.socket.addEventListener(\"close\", this.onClose)\n    })\n\n    return this.connectPromise\n  }\n\n  /**\n   * Close the websocket and clear pending state.\n   * @returns {Promise<void>}\n   */\n  async close() {\n    if (!this.socket) return\n\n    await new Promise((resolve) => {\n      this.socket?.addEventListener(\"close\", () => resolve())\n      this.socket?.close()\n    })\n\n    this.socket = undefined\n    this.connectPromise = undefined\n  }\n\n  /**\n   * Perform a POST request over the websocket.\n   * @param {string} path\n   * @param {any} [body]\n   * @param {{headers?: Record<string, string>}} [options]\n   * @returns {Promise<VelociousWebsocketResponse>}\n   */\n  async post(path, body, options = {}) {\n    return await this.request(\"POST\", path, {...options, body})\n  }\n\n  /**\n   * Perform a GET request over the websocket.\n   * @param {string} path\n   * @param {{headers?: Record<string, string>}} [options]\n   * @returns {Promise<VelociousWebsocketResponse>}\n   */\n  async get(path, options = {}) {\n    return await this.request(\"GET\", path, options)\n  }\n\n  /**\n   * Subscribe to a channel for server-sent events.\n   * @param {string} channel\n   * @param {(payload: any) => void} callback\n   * @returns {() => void} unsubscribe function\n   */\n  on(channel, callback) {\n    if (!this.listeners.has(channel)) {\n      this.listeners.set(channel, new Set())\n      this.subscribedChannels.add(channel)\n\n      void this.connect().then(() => {\n        this._sendMessage({channel, type: \"subscribe\"})\n      }).catch((error) => this._debug(\"Subscribe failed\", error))\n    }\n\n    const channelListeners = this.listeners.get(channel)\n\n    if (!channelListeners) throw new Error(\"Listeners map not initialized\")\n\n    channelListeners.add(callback)\n\n    return () => {\n      channelListeners.delete(callback)\n\n      if (channelListeners.size === 0) {\n        this.listeners.delete(channel)\n      }\n    }\n  }\n\n  /**\n   * @private\n   * @param {string} method\n   * @param {string} path\n   * @param {object} [options]\n   * @param {any} [options.body]\n   * @param {Record<string, string>} [options.headers]\n   * @returns {Promise<VelociousWebsocketResponse>}\n   */\n  async request(method, path, {body, headers} = {}) {\n    await this.connect()\n\n    const id = `ws-${this.nextID++}`\n    const payload = {\n      body,\n      headers,\n      id,\n      method,\n      path,\n      type: \"request\"\n    }\n\n    return await new Promise((resolve, reject) => {\n      this.pendingRequests.set(id, {resolve, reject})\n      this._sendMessage(payload)\n    })\n  }\n\n  /**\n   * @private\n   * @param {MessageEvent<any>} event\n   */\n  onMessage = (event) => {\n    const raw = typeof event.data === \"string\" ? event.data : event.data?.toString?.()\n\n    if (!raw) return\n\n    let message\n\n    try {\n      message = JSON.parse(raw)\n    } catch (error) {\n      this._debug(\"Failed to parse websocket message\", error)\n      return\n    }\n\n    const {type} = message\n\n    if (type === \"response\") {\n      const {id} = message\n      const pending = id ? this.pendingRequests.get(id) : undefined\n\n      if (pending) {\n        this.pendingRequests.delete(id)\n        pending.resolve(new VelociousWebsocketResponse(message))\n      } else {\n        this._debug(`No pending request for response id ${id}`)\n      }\n    } else if (type === \"event\") {\n      const {channel, payload} = message\n      const callbacks = this.listeners.get(channel)\n\n      callbacks?.forEach((callback) => {\n        try {\n          callback(payload)\n        } catch (error) {\n          this._debug(\"Listener error\", error)\n        }\n      })\n    } else if (type === \"error\" && message.id) {\n      const pending = this.pendingRequests.get(message.id)\n\n      if (pending) {\n        this.pendingRequests.delete(message.id)\n        pending.reject(new Error(message.error || \"Unknown websocket error\"))\n      }\n    }\n  }\n\n  /**\n   * Reject all pending requests when the socket closes unexpectedly.\n   * @private\n   */\n  onClose = () => {\n    for (const [id, {reject}] of this.pendingRequests.entries()) {\n      reject(new Error(`Websocket closed before response for ${id}`))\n    }\n\n    this.pendingRequests.clear()\n    this.connectPromise = undefined\n  }\n\n  /**\n   * @private\n   * @param {Record<string, any>} payload\n   */\n  _sendMessage(payload) {\n    if (!this.socket || this.socket.readyState !== this.socket.OPEN) {\n      throw new Error(\"Websocket is not open\")\n    }\n\n    const json = JSON.stringify(payload)\n\n    this._debug(\"Sending\", json)\n    this.socket.send(json)\n  }\n\n  /**\n   * @private\n   * @param  {...any} args\n   * @returns {void}\n   */\n  _debug(...args) {\n    if (!this.debug) return\n\n    console.debug(\"[VelociousWebsocketClient]\", ...args)\n  }\n}\n\nclass VelociousWebsocketResponse {\n  /**\n   * @param {object} message\n   */\n  constructor(message) {\n    this.body = message.body\n    this.headers = message.headers || {}\n    this.id = message.id\n    this.statusCode = message.statusCode || 200\n    this.statusMessage = message.statusMessage || \"OK\"\n    this.type = message.type\n  }\n\n  /** @returns {any} */\n  json() {\n    if (typeof this.body !== \"string\") {\n      throw new Error(\"Response body is not a string\")\n    }\n\n    return JSON.parse(this.body)\n  }\n}\n"]}
@@ -18,6 +18,11 @@ export default class VeoliciousHttpServerClient {
18
18
  remoteAddress: string;
19
19
  /** @type {RequestRunner[]} */
20
20
  requestRunners: RequestRunner[];
21
+ /**
22
+ * @param {string} message
23
+ * @returns {void}
24
+ */
25
+ _sendBadUpgradeResponse(message: string): void;
21
26
  executeCurrentRequest: () => void;
22
27
  /**
23
28
  * @param {Buffer} data
@@ -25,6 +30,14 @@ export default class VeoliciousHttpServerClient {
25
30
  */
26
31
  onWrite(data: Buffer): void;
27
32
  currentRequest: Request;
33
+ /**
34
+ * @param {import("./request.js").default} request
35
+ * @returns {boolean}
36
+ */
37
+ _isWebsocketUpgrade(request: import("./request.js").default): boolean;
38
+ /** @returns {void} */
39
+ _upgradeToWebsocket(): void;
40
+ websocketSession: WebsocketSession;
28
41
  requestDone: () => void;
29
42
  sendDoneRequests(): void;
30
43
  /**
@@ -37,4 +50,5 @@ import { EventEmitter } from "events";
37
50
  import { Logger } from "../../logger.js";
38
51
  import RequestRunner from "./request-runner.js";
39
52
  import Request from "./request.js";
53
+ import WebsocketSession from "./websocket-session.js";
40
54
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"AAQA;IAIE;;;;;OAKG;IACH,2DAJG;QAAqB,WAAW,EAAxB,MAAM;QACyC,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QAC1B,aAAa,GAA3B,MAAM;KAAsB,EAYtC;IAnBD,8BAA2B;IAC3B,cAAiB;IAWf,eAA8B;IAC9B,oBAA8B;IAC9B,wDAAkC;IAClC,sBAAkC;IAElC,8BAA8B;IAC9B,gBADW,aAAa,EAAE,CACF;IAG1B,kCAiBC;IAED;;;OAGG;IACH,cAHW,MAAM,GACJ,IAAI,CAehB;IAXG,wBAAoF;IAaxF,wBAEC;IAED,yBAqBC;IAED;;;OAGG;IACH,4BAHW,aAAa,GACX,IAAI,CAwChB;CACF;6BAvI0B,QAAQ;uBACd,iBAAiB;0BAEZ,qBAAqB;oBAD3B,cAAc"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/http-server/client/index.js"],"names":[],"mappings":"AAUA;IAIE;;;;;OAKG;IACH,2DAJG;QAAqB,WAAW,EAAxB,MAAM;QACyC,aAAa,EAA5D,OAAO,wBAAwB,EAAE,OAAO;QAC1B,aAAa,GAA3B,MAAM;KAAsB,EAYtC;IAnBD,8BAA2B;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,kCAsBC;IAED;;;OAGG;IACH,cAHW,MAAM,GACJ,IAAI,CAoBhB;IAXG,wBAAoF;IAaxF;;;OAGG;IACH,6BAHW,OAAO,cAAc,EAAE,OAAO,GAC5B,OAAO,CAOnB;IAED,sBAAsB;IACtB,uBADc,IAAI,CAoCjB;IAXC,mCAGE;IAUJ,wBAEC;IAED,yBAqBC;IAED;;;OAGG;IACH,4BAHW,aAAa,GACX,IAAI,CAwChB;CACF;6BAvN0B,QAAQ;uBACd,iBAAiB;0BAEZ,qBAAqB;oBAD3B,cAAc;6BAEL,wBAAwB"}