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.
- package/README.md +103 -0
- package/build/src/configuration.d.ts +8 -0
- package/build/src/configuration.d.ts.map +1 -1
- package/build/src/configuration.js +13 -1
- package/build/src/database/drivers/base-table.d.ts +5 -0
- package/build/src/database/drivers/base-table.d.ts.map +1 -1
- package/build/src/database/drivers/base-table.js +11 -1
- package/build/src/database/drivers/base.d.ts +9 -0
- package/build/src/database/drivers/base.d.ts.map +1 -1
- package/build/src/database/drivers/base.js +12 -1
- package/build/src/database/drivers/mssql/index.d.ts.map +1 -1
- package/build/src/database/drivers/mssql/index.js +7 -3
- package/build/src/database/drivers/mysql/index.d.ts.map +1 -1
- package/build/src/database/drivers/mysql/index.js +2 -1
- package/build/src/database/drivers/pgsql/index.d.ts.map +1 -1
- package/build/src/database/drivers/pgsql/index.js +2 -1
- package/build/src/database/drivers/sqlite/base.d.ts.map +1 -1
- package/build/src/database/drivers/sqlite/base.js +3 -2
- package/build/src/database/migration/index.d.ts.map +1 -1
- package/build/src/database/migration/index.js +11 -2
- package/build/src/database/record/index.d.ts.map +1 -1
- package/build/src/database/record/index.js +15 -2
- package/build/src/http-client/response.d.ts +4 -0
- package/build/src/http-client/response.d.ts.map +1 -1
- package/build/src/http-client/response.js +19 -17
- package/build/src/http-client/websocket-client.d.ts +105 -0
- package/build/src/http-client/websocket-client.d.ts.map +1 -0
- package/build/src/http-client/websocket-client.js +237 -0
- package/build/src/http-server/client/index.d.ts +14 -0
- package/build/src/http-server/client/index.d.ts.map +1 -1
- package/build/src/http-server/client/index.js +72 -1
- package/build/src/http-server/client/params-to-object.d.ts +7 -7
- package/build/src/http-server/client/params-to-object.d.ts.map +1 -1
- package/build/src/http-server/client/params-to-object.js +4 -4
- package/build/src/http-server/client/request-buffer/form-data-part.d.ts +21 -2
- package/build/src/http-server/client/request-buffer/form-data-part.d.ts.map +1 -1
- package/build/src/http-server/client/request-buffer/form-data-part.js +62 -4
- package/build/src/http-server/client/request-parser.js +2 -2
- package/build/src/http-server/client/request-runner.d.ts +4 -4
- package/build/src/http-server/client/request-runner.d.ts.map +1 -1
- package/build/src/http-server/client/request-runner.js +2 -2
- package/build/src/http-server/client/uploaded-file/memory-uploaded-file.d.ts +21 -0
- package/build/src/http-server/client/uploaded-file/memory-uploaded-file.d.ts.map +1 -0
- package/build/src/http-server/client/uploaded-file/memory-uploaded-file.js +26 -0
- package/build/src/http-server/client/uploaded-file/temporary-uploaded-file.d.ts +21 -0
- package/build/src/http-server/client/uploaded-file/temporary-uploaded-file.d.ts.map +1 -0
- package/build/src/http-server/client/uploaded-file/temporary-uploaded-file.js +26 -0
- package/build/src/http-server/client/uploaded-file/uploaded-file.d.ts +29 -0
- package/build/src/http-server/client/uploaded-file/uploaded-file.d.ts.map +1 -0
- package/build/src/http-server/client/uploaded-file/uploaded-file.js +34 -0
- package/build/src/http-server/client/websocket-request.d.ts +40 -0
- package/build/src/http-server/client/websocket-request.d.ts.map +1 -0
- package/build/src/http-server/client/websocket-request.js +88 -0
- package/build/src/http-server/client/websocket-session.d.ts +73 -0
- package/build/src/http-server/client/websocket-session.d.ts.map +1 -0
- package/build/src/http-server/client/websocket-session.js +231 -0
- package/build/src/http-server/server-client.d.ts.map +1 -1
- package/build/src/http-server/server-client.js +6 -3
- package/build/src/http-server/websocket-events-host.d.ts +22 -0
- package/build/src/http-server/websocket-events-host.d.ts.map +1 -0
- package/build/src/http-server/websocket-events-host.js +29 -0
- package/build/src/http-server/websocket-events.d.ts +20 -0
- package/build/src/http-server/websocket-events.d.ts.map +1 -0
- package/build/src/http-server/websocket-events.js +23 -0
- package/build/src/http-server/worker-handler/index.d.ts +19 -4
- package/build/src/http-server/worker-handler/index.d.ts.map +1 -1
- package/build/src/http-server/worker-handler/index.js +24 -3
- package/build/src/http-server/worker-handler/worker-thread.d.ts +18 -2
- package/build/src/http-server/worker-handler/worker-thread.d.ts.map +1 -1
- package/build/src/http-server/worker-handler/worker-thread.js +23 -2
- package/build/src/routes/resolver.d.ts +5 -0
- package/build/src/routes/resolver.d.ts.map +1 -1
- package/build/src/routes/resolver.js +28 -2
- 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,
|
|
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
|
|
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
|
-
|
|
79
|
+
const contentLengthNumber = this._contentLengthNumber();
|
|
80
|
+
if (!contentLengthNumber) {
|
|
93
81
|
this.completeResponse();
|
|
94
82
|
break;
|
|
95
83
|
}
|
|
96
|
-
else
|
|
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":"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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 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":"
|
|
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"}
|