@nktkas/hyperliquid 0.13.1 → 0.14.0
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/LICENSE +20 -20
- package/README.md +334 -107
- package/esm/deps/jsr.io/@std/bytes/1.0.5/_types.d.ts +9 -0
- package/esm/deps/jsr.io/@std/bytes/1.0.5/_types.d.ts.map +1 -0
- package/esm/deps/jsr.io/@std/bytes/1.0.5/_types.js +2 -0
- package/esm/deps/jsr.io/@std/bytes/{1.0.4 → 1.0.5}/concat.d.ts +3 -1
- package/esm/deps/jsr.io/@std/bytes/1.0.5/concat.d.ts.map +1 -0
- package/esm/deps/jsr.io/@std/bytes/{1.0.4 → 1.0.5}/concat.js +1 -1
- package/esm/deps/jsr.io/@std/encoding/1.0.7/_types.d.ts +9 -0
- package/esm/deps/jsr.io/@std/encoding/1.0.7/_types.d.ts.map +1 -0
- package/esm/deps/jsr.io/@std/encoding/1.0.7/_types.js +2 -0
- package/esm/deps/jsr.io/@std/encoding/{1.0.6 → 1.0.7}/_validate_binary_like.d.ts.map +1 -1
- package/esm/deps/jsr.io/@std/encoding/{1.0.6 → 1.0.7}/_validate_binary_like.js +1 -1
- package/{script/deps/jsr.io/@std/encoding/1.0.6 → esm/deps/jsr.io/@std/encoding/1.0.7}/hex.d.ts +3 -1
- package/esm/deps/jsr.io/@std/encoding/1.0.7/hex.d.ts.map +1 -0
- package/esm/deps/jsr.io/@std/encoding/{1.0.6 → 1.0.7}/hex.js +1 -1
- package/esm/deps/jsr.io/@std/msgpack/1.0.3/_types.d.ts +9 -0
- package/esm/deps/jsr.io/@std/msgpack/1.0.3/_types.d.ts.map +1 -0
- package/esm/deps/jsr.io/@std/msgpack/1.0.3/_types.js +2 -0
- package/esm/deps/jsr.io/@std/msgpack/{1.0.2 → 1.0.3}/encode.d.ts +3 -1
- package/esm/deps/jsr.io/@std/msgpack/1.0.3/encode.d.ts.map +1 -0
- package/esm/deps/jsr.io/@std/msgpack/{1.0.2 → 1.0.3}/encode.js +2 -2
- package/esm/mod.d.ts +4 -12
- package/esm/mod.d.ts.map +1 -1
- package/esm/mod.js +2 -2
- package/{script/src/transports → esm/src}/base.d.ts +10 -13
- package/esm/src/base.d.ts.map +1 -0
- package/esm/src/base.js +14 -0
- package/esm/src/clients/event.d.ts +59 -18
- package/esm/src/clients/event.d.ts.map +1 -1
- package/esm/src/clients/event.js +74 -18
- package/esm/src/clients/public.d.ts +348 -53
- package/esm/src/clients/public.d.ts.map +1 -1
- package/esm/src/clients/public.js +349 -50
- package/esm/src/clients/wallet.d.ts +345 -73
- package/esm/src/clients/wallet.d.ts.map +1 -1
- package/esm/src/clients/wallet.js +979 -229
- package/esm/src/signing.d.ts +135 -0
- package/esm/src/signing.d.ts.map +1 -0
- package/esm/src/signing.js +188 -0
- package/esm/src/transports/http/http_transport.d.ts +1 -1
- package/esm/src/transports/http/http_transport.d.ts.map +1 -1
- package/esm/src/transports/http/http_transport.js +1 -1
- package/esm/src/transports/websocket/_hyperliquid_event_target.d.ts +64 -0
- package/esm/src/transports/websocket/_hyperliquid_event_target.d.ts.map +1 -0
- package/esm/src/transports/websocket/_hyperliquid_event_target.js +52 -0
- package/esm/src/transports/websocket/{reconnecting_websocket.d.ts → _reconnecting_websocket.d.ts} +26 -26
- package/esm/src/transports/websocket/_reconnecting_websocket.d.ts.map +1 -0
- package/esm/src/transports/websocket/{reconnecting_websocket.js → _reconnecting_websocket.js} +91 -76
- package/{script/src/transports/websocket/websocket_request_dispatcher.d.ts → esm/src/transports/websocket/_websocket_request_dispatcher.d.ts} +3 -7
- package/esm/src/transports/websocket/_websocket_request_dispatcher.d.ts.map +1 -0
- package/esm/src/transports/websocket/{websocket_request_dispatcher.js → _websocket_request_dispatcher.js} +6 -10
- package/esm/src/transports/websocket/websocket_transport.d.ts +15 -21
- package/esm/src/transports/websocket/websocket_transport.d.ts.map +1 -1
- package/esm/src/transports/websocket/websocket_transport.js +55 -52
- package/esm/src/types/exchange/requests.d.ts +298 -79
- package/esm/src/types/exchange/requests.d.ts.map +1 -1
- package/esm/src/types/exchange/responses.d.ts +38 -38
- package/esm/src/types/exchange/responses.d.ts.map +1 -1
- package/esm/src/types/explorer/requests.d.ts +23 -10
- package/esm/src/types/explorer/requests.d.ts.map +1 -1
- package/esm/src/types/explorer/responses.d.ts +45 -3
- package/esm/src/types/explorer/responses.d.ts.map +1 -1
- package/esm/src/types/info/accounts.d.ts +279 -189
- package/esm/src/types/info/accounts.d.ts.map +1 -1
- package/esm/src/types/info/assets.d.ts +191 -185
- package/esm/src/types/info/assets.d.ts.map +1 -1
- package/esm/src/types/info/delegations.d.ts +117 -0
- package/esm/src/types/info/delegations.d.ts.map +1 -0
- package/esm/src/types/info/orders.d.ts +94 -94
- package/esm/src/types/info/orders.d.ts.map +1 -1
- package/esm/src/types/info/requests.d.ts +192 -64
- package/esm/src/types/info/requests.d.ts.map +1 -1
- package/esm/src/types/info/vaults.d.ts +42 -85
- package/esm/src/types/info/vaults.d.ts.map +1 -1
- package/esm/src/types/mod.d.ts +14 -0
- package/esm/src/types/mod.d.ts.map +1 -0
- package/esm/src/types/subscriptions/requests.d.ts +11 -1
- package/esm/src/types/subscriptions/requests.d.ts.map +1 -1
- package/{script/src/types/subscriptions/common.d.ts → esm/src/types/subscriptions/responses.d.ts} +18 -3
- package/esm/src/types/subscriptions/responses.d.ts.map +1 -0
- package/package.json +20 -4
- package/script/deps/jsr.io/@derzade/typescript-event-target/1.1.1/mod.js +13 -3
- package/script/deps/jsr.io/@derzade/typescript-event-target/1.1.1/src/TypedEventTarget.js +23 -13
- package/script/deps/jsr.io/@noble/hashes/1.7.1/src/_assert.js +55 -45
- package/script/deps/jsr.io/@noble/hashes/1.7.1/src/_u64.js +97 -87
- package/script/deps/jsr.io/@noble/hashes/1.7.1/src/crypto.js +14 -4
- package/script/deps/jsr.io/@noble/hashes/1.7.1/src/sha3.js +288 -278
- package/script/deps/jsr.io/@noble/hashes/1.7.1/src/utils.js +238 -228
- package/script/deps/jsr.io/@std/bytes/1.0.5/_types.d.ts +9 -0
- package/script/deps/jsr.io/@std/bytes/1.0.5/_types.d.ts.map +1 -0
- package/script/deps/jsr.io/@std/bytes/1.0.5/_types.js +13 -0
- package/script/deps/jsr.io/@std/bytes/{1.0.4 → 1.0.5}/concat.d.ts +3 -1
- package/script/deps/jsr.io/@std/bytes/1.0.5/concat.d.ts.map +1 -0
- package/script/deps/jsr.io/@std/bytes/1.0.5/concat.js +45 -0
- package/script/deps/jsr.io/@std/encoding/1.0.7/_types.d.ts +9 -0
- package/script/deps/jsr.io/@std/encoding/1.0.7/_types.d.ts.map +1 -0
- package/script/deps/jsr.io/@std/encoding/1.0.7/_types.js +13 -0
- package/script/deps/jsr.io/@std/encoding/{1.0.6 → 1.0.7}/_validate_binary_like.d.ts.map +1 -1
- package/script/deps/jsr.io/@std/encoding/1.0.7/_validate_binary_like.js +39 -0
- package/{esm/deps/jsr.io/@std/encoding/1.0.6 → script/deps/jsr.io/@std/encoding/1.0.7}/hex.d.ts +3 -1
- package/script/deps/jsr.io/@std/encoding/1.0.7/hex.d.ts.map +1 -0
- package/script/deps/jsr.io/@std/encoding/1.0.7/hex.js +123 -0
- package/script/deps/jsr.io/@std/msgpack/1.0.3/_types.d.ts +9 -0
- package/script/deps/jsr.io/@std/msgpack/1.0.3/_types.d.ts.map +1 -0
- package/script/deps/jsr.io/@std/msgpack/1.0.3/_types.js +13 -0
- package/script/deps/jsr.io/@std/msgpack/{1.0.2 → 1.0.3}/encode.d.ts +3 -1
- package/script/deps/jsr.io/@std/msgpack/1.0.3/encode.d.ts.map +1 -0
- package/script/deps/jsr.io/@std/msgpack/1.0.3/encode.js +250 -0
- package/script/mod.d.ts +4 -12
- package/script/mod.d.ts.map +1 -1
- package/script/mod.js +21 -13
- package/{esm/src/transports → script/src}/base.d.ts +10 -13
- package/script/src/base.d.ts.map +1 -0
- package/script/src/base.js +29 -0
- package/script/src/clients/event.d.ts +59 -18
- package/script/src/clients/event.d.ts.map +1 -1
- package/script/src/clients/event.js +551 -485
- package/script/src/clients/public.d.ts +348 -53
- package/script/src/clients/public.d.ts.map +1 -1
- package/script/src/clients/public.js +1017 -708
- package/script/src/clients/wallet.d.ts +345 -73
- package/script/src/clients/wallet.d.ts.map +1 -1
- package/script/src/clients/wallet.js +1737 -977
- package/script/src/signing.d.ts +135 -0
- package/script/src/signing.d.ts.map +1 -0
- package/script/src/signing.js +203 -0
- package/script/src/transports/http/http_transport.d.ts +1 -1
- package/script/src/transports/http/http_transport.d.ts.map +1 -1
- package/script/src/transports/http/http_transport.js +174 -164
- package/script/src/transports/websocket/_hyperliquid_event_target.d.ts +64 -0
- package/script/src/transports/websocket/_hyperliquid_event_target.d.ts.map +1 -0
- package/script/src/transports/websocket/_hyperliquid_event_target.js +66 -0
- package/script/src/transports/websocket/{reconnecting_websocket.d.ts → _reconnecting_websocket.d.ts} +26 -26
- package/script/src/transports/websocket/_reconnecting_websocket.d.ts.map +1 -0
- package/script/src/transports/websocket/_reconnecting_websocket.js +400 -0
- package/{esm/src/transports/websocket/websocket_request_dispatcher.d.ts → script/src/transports/websocket/_websocket_request_dispatcher.d.ts} +3 -7
- package/script/src/transports/websocket/_websocket_request_dispatcher.d.ts.map +1 -0
- package/script/src/transports/websocket/_websocket_request_dispatcher.js +212 -0
- package/script/src/transports/websocket/websocket_transport.d.ts +15 -21
- package/script/src/transports/websocket/websocket_transport.d.ts.map +1 -1
- package/script/src/transports/websocket/websocket_transport.js +235 -222
- package/script/src/types/exchange/requests.d.ts +298 -79
- package/script/src/types/exchange/requests.d.ts.map +1 -1
- package/script/src/types/exchange/requests.js +12 -2
- package/script/src/types/exchange/responses.d.ts +38 -38
- package/script/src/types/exchange/responses.d.ts.map +1 -1
- package/script/src/types/exchange/responses.js +12 -2
- package/script/src/types/explorer/requests.d.ts +23 -10
- package/script/src/types/explorer/requests.d.ts.map +1 -1
- package/script/src/types/explorer/requests.js +12 -2
- package/script/src/types/explorer/responses.d.ts +45 -3
- package/script/src/types/explorer/responses.d.ts.map +1 -1
- package/script/src/types/explorer/responses.js +12 -2
- package/script/src/types/info/accounts.d.ts +279 -189
- package/script/src/types/info/accounts.d.ts.map +1 -1
- package/script/src/types/info/accounts.js +12 -2
- package/script/src/types/info/assets.d.ts +191 -185
- package/script/src/types/info/assets.d.ts.map +1 -1
- package/script/src/types/info/assets.js +12 -2
- package/script/src/types/info/delegations.d.ts +117 -0
- package/script/src/types/info/delegations.d.ts.map +1 -0
- package/script/src/types/info/delegations.js +12 -0
- package/script/src/types/info/orders.d.ts +94 -94
- package/script/src/types/info/orders.d.ts.map +1 -1
- package/script/src/types/info/orders.js +12 -2
- package/script/src/types/info/requests.d.ts +192 -64
- package/script/src/types/info/requests.d.ts.map +1 -1
- package/script/src/types/info/requests.js +12 -2
- package/script/src/types/info/vaults.d.ts +42 -85
- package/script/src/types/info/vaults.d.ts.map +1 -1
- package/script/src/types/info/vaults.js +12 -2
- package/script/src/types/mod.d.ts +14 -0
- package/script/src/types/mod.d.ts.map +1 -0
- package/script/src/types/mod.js +12 -0
- package/script/src/types/subscriptions/requests.d.ts +11 -1
- package/script/src/types/subscriptions/requests.d.ts.map +1 -1
- package/script/src/types/subscriptions/requests.js +12 -2
- package/{esm/src/types/subscriptions/common.d.ts → script/src/types/subscriptions/responses.d.ts} +18 -3
- package/script/src/types/subscriptions/responses.d.ts.map +1 -0
- package/script/src/types/subscriptions/responses.js +12 -0
- package/esm/deps/jsr.io/@std/bytes/1.0.4/concat.d.ts.map +0 -1
- package/esm/deps/jsr.io/@std/encoding/1.0.6/hex.d.ts.map +0 -1
- package/esm/deps/jsr.io/@std/msgpack/1.0.2/encode.d.ts.map +0 -1
- package/esm/src/transports/base.d.ts.map +0 -1
- package/esm/src/transports/base.js +0 -14
- package/esm/src/transports/websocket/hyperliquid_event_target.d.ts +0 -66
- package/esm/src/transports/websocket/hyperliquid_event_target.d.ts.map +0 -1
- package/esm/src/transports/websocket/hyperliquid_event_target.js +0 -33
- package/esm/src/transports/websocket/reconnecting_websocket.d.ts.map +0 -1
- package/esm/src/transports/websocket/websocket_request_dispatcher.d.ts.map +0 -1
- package/esm/src/types/common.d.ts +0 -3
- package/esm/src/types/common.d.ts.map +0 -1
- package/esm/src/types/exchange/common.d.ts +0 -36
- package/esm/src/types/exchange/common.d.ts.map +0 -1
- package/esm/src/types/explorer/common.d.ts +0 -37
- package/esm/src/types/explorer/common.d.ts.map +0 -1
- package/esm/src/types/subscriptions/common.d.ts.map +0 -1
- package/esm/src/types/subscriptions/common.js +0 -1
- package/esm/src/utils/key_sort.d.ts +0 -21
- package/esm/src/utils/key_sort.d.ts.map +0 -1
- package/esm/src/utils/key_sort.js +0 -124
- package/esm/src/utils/signing.d.ts +0 -109
- package/esm/src/utils/signing.d.ts.map +0 -1
- package/esm/src/utils/signing.js +0 -164
- package/script/deps/jsr.io/@std/bytes/1.0.4/concat.d.ts.map +0 -1
- package/script/deps/jsr.io/@std/bytes/1.0.4/concat.js +0 -35
- package/script/deps/jsr.io/@std/encoding/1.0.6/_validate_binary_like.js +0 -29
- package/script/deps/jsr.io/@std/encoding/1.0.6/hex.d.ts.map +0 -1
- package/script/deps/jsr.io/@std/encoding/1.0.6/hex.js +0 -113
- package/script/deps/jsr.io/@std/msgpack/1.0.2/encode.d.ts.map +0 -1
- package/script/deps/jsr.io/@std/msgpack/1.0.2/encode.js +0 -240
- package/script/src/transports/base.d.ts.map +0 -1
- package/script/src/transports/base.js +0 -18
- package/script/src/transports/websocket/hyperliquid_event_target.d.ts +0 -66
- package/script/src/transports/websocket/hyperliquid_event_target.d.ts.map +0 -1
- package/script/src/transports/websocket/hyperliquid_event_target.js +0 -37
- package/script/src/transports/websocket/reconnecting_websocket.d.ts.map +0 -1
- package/script/src/transports/websocket/reconnecting_websocket.js +0 -374
- package/script/src/transports/websocket/websocket_request_dispatcher.d.ts.map +0 -1
- package/script/src/transports/websocket/websocket_request_dispatcher.js +0 -206
- package/script/src/types/common.d.ts +0 -3
- package/script/src/types/common.d.ts.map +0 -1
- package/script/src/types/common.js +0 -2
- package/script/src/types/exchange/common.d.ts +0 -36
- package/script/src/types/exchange/common.d.ts.map +0 -1
- package/script/src/types/exchange/common.js +0 -2
- package/script/src/types/explorer/common.d.ts +0 -37
- package/script/src/types/explorer/common.d.ts.map +0 -1
- package/script/src/types/explorer/common.js +0 -2
- package/script/src/types/subscriptions/common.d.ts.map +0 -1
- package/script/src/types/subscriptions/common.js +0 -2
- package/script/src/utils/key_sort.d.ts +0 -21
- package/script/src/utils/key_sort.d.ts.map +0 -1
- package/script/src/utils/key_sort.js +0 -127
- package/script/src/utils/signing.d.ts +0 -109
- package/script/src/utils/signing.d.ts.map +0 -1
- package/script/src/utils/signing.js +0 -172
- /package/esm/deps/jsr.io/@std/encoding/{1.0.6 → 1.0.7}/_validate_binary_like.d.ts +0 -0
- /package/esm/src/types/{common.js → info/delegations.js} +0 -0
- /package/esm/src/types/{exchange/common.js → mod.js} +0 -0
- /package/esm/src/types/{explorer/common.js → subscriptions/responses.js} +0 -0
- /package/script/deps/jsr.io/@std/encoding/{1.0.6 → 1.0.7}/_validate_binary_like.d.ts +0 -0
|
@@ -0,0 +1,400 @@
|
|
|
1
|
+
// deno-lint-ignore-file no-explicit-any
|
|
2
|
+
(function (factory) {
|
|
3
|
+
if (typeof module === "object" && typeof module.exports === "object") {
|
|
4
|
+
var v = factory(require, exports);
|
|
5
|
+
if (v !== undefined) module.exports = v;
|
|
6
|
+
}
|
|
7
|
+
else if (typeof define === "function" && define.amd) {
|
|
8
|
+
define(["require", "exports", "../../base.js"], factory);
|
|
9
|
+
}
|
|
10
|
+
})(function (require, exports) {
|
|
11
|
+
"use strict";
|
|
12
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
13
|
+
exports.ReconnectingWebSocket = exports.ReconnectingWebSocketError = void 0;
|
|
14
|
+
const base_js_1 = require("../../base.js");
|
|
15
|
+
/** Simple FIFO (First In, First Out) buffer implementation. */
|
|
16
|
+
class FIFOMessageBuffer {
|
|
17
|
+
constructor() {
|
|
18
|
+
Object.defineProperty(this, "messages", {
|
|
19
|
+
enumerable: true,
|
|
20
|
+
configurable: true,
|
|
21
|
+
writable: true,
|
|
22
|
+
value: []
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
push(data) {
|
|
26
|
+
this.messages.push(data);
|
|
27
|
+
}
|
|
28
|
+
shift() {
|
|
29
|
+
return this.messages.shift();
|
|
30
|
+
}
|
|
31
|
+
clear() {
|
|
32
|
+
this.messages = [];
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
/** Error thrown when reconnection problems occur. */
|
|
36
|
+
class ReconnectingWebSocketError extends base_js_1.TransportError {
|
|
37
|
+
constructor(code, originalError) {
|
|
38
|
+
super(`Error when reconnecting WebSocket: ${code}`);
|
|
39
|
+
Object.defineProperty(this, "code", {
|
|
40
|
+
enumerable: true,
|
|
41
|
+
configurable: true,
|
|
42
|
+
writable: true,
|
|
43
|
+
value: code
|
|
44
|
+
});
|
|
45
|
+
Object.defineProperty(this, "originalError", {
|
|
46
|
+
enumerable: true,
|
|
47
|
+
configurable: true,
|
|
48
|
+
writable: true,
|
|
49
|
+
value: originalError
|
|
50
|
+
});
|
|
51
|
+
this.name = "ReconnectingWebSocketError";
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
exports.ReconnectingWebSocketError = ReconnectingWebSocketError;
|
|
55
|
+
/**
|
|
56
|
+
* A WebSocket that automatically reconnects when disconnected.
|
|
57
|
+
* Fully compatible with standard WebSocket API.
|
|
58
|
+
*/
|
|
59
|
+
class ReconnectingWebSocket {
|
|
60
|
+
/**
|
|
61
|
+
* Creates a new reconnecting WebSocket.
|
|
62
|
+
* @param url - The WebSocket URL to connect to.
|
|
63
|
+
* @param protocols - The WebSocket protocols to use.
|
|
64
|
+
* @param options - The configuration options.
|
|
65
|
+
*/
|
|
66
|
+
constructor(url, protocols, options) {
|
|
67
|
+
/** Controller for handling connection termination */
|
|
68
|
+
Object.defineProperty(this, "_terminationController", {
|
|
69
|
+
enumerable: true,
|
|
70
|
+
configurable: true,
|
|
71
|
+
writable: true,
|
|
72
|
+
value: new AbortController()
|
|
73
|
+
});
|
|
74
|
+
/** WebSocket protocols */
|
|
75
|
+
Object.defineProperty(this, "_protocols", {
|
|
76
|
+
enumerable: true,
|
|
77
|
+
configurable: true,
|
|
78
|
+
writable: true,
|
|
79
|
+
value: void 0
|
|
80
|
+
});
|
|
81
|
+
/** The underlying WebSocket instance */
|
|
82
|
+
Object.defineProperty(this, "_socket", {
|
|
83
|
+
enumerable: true,
|
|
84
|
+
configurable: true,
|
|
85
|
+
writable: true,
|
|
86
|
+
value: void 0
|
|
87
|
+
});
|
|
88
|
+
/** Current number of reconnection attempts */
|
|
89
|
+
Object.defineProperty(this, "_reconnectCount", {
|
|
90
|
+
enumerable: true,
|
|
91
|
+
configurable: true,
|
|
92
|
+
writable: true,
|
|
93
|
+
value: 0
|
|
94
|
+
});
|
|
95
|
+
/** Array of registered event listeners */
|
|
96
|
+
Object.defineProperty(this, "_eventListeners", {
|
|
97
|
+
enumerable: true,
|
|
98
|
+
configurable: true,
|
|
99
|
+
writable: true,
|
|
100
|
+
value: []
|
|
101
|
+
});
|
|
102
|
+
/** Configuration options */
|
|
103
|
+
Object.defineProperty(this, "reconnectOptions", {
|
|
104
|
+
enumerable: true,
|
|
105
|
+
configurable: true,
|
|
106
|
+
writable: true,
|
|
107
|
+
value: void 0
|
|
108
|
+
});
|
|
109
|
+
/** An AbortSignal that is triggered when the connection is permanently closed */
|
|
110
|
+
Object.defineProperty(this, "terminationSignal", {
|
|
111
|
+
enumerable: true,
|
|
112
|
+
configurable: true,
|
|
113
|
+
writable: true,
|
|
114
|
+
value: this._terminationController.signal
|
|
115
|
+
});
|
|
116
|
+
Object.defineProperty(this, "CLOSED", {
|
|
117
|
+
enumerable: true,
|
|
118
|
+
configurable: true,
|
|
119
|
+
writable: true,
|
|
120
|
+
value: WebSocket.CLOSED
|
|
121
|
+
});
|
|
122
|
+
Object.defineProperty(this, "CLOSING", {
|
|
123
|
+
enumerable: true,
|
|
124
|
+
configurable: true,
|
|
125
|
+
writable: true,
|
|
126
|
+
value: WebSocket.CLOSING
|
|
127
|
+
});
|
|
128
|
+
Object.defineProperty(this, "CONNECTING", {
|
|
129
|
+
enumerable: true,
|
|
130
|
+
configurable: true,
|
|
131
|
+
writable: true,
|
|
132
|
+
value: WebSocket.CONNECTING
|
|
133
|
+
});
|
|
134
|
+
Object.defineProperty(this, "OPEN", {
|
|
135
|
+
enumerable: true,
|
|
136
|
+
configurable: true,
|
|
137
|
+
writable: true,
|
|
138
|
+
value: WebSocket.OPEN
|
|
139
|
+
});
|
|
140
|
+
this.reconnectOptions = {
|
|
141
|
+
maxRetries: options?.maxRetries ?? 3,
|
|
142
|
+
connectionTimeout: options?.connectionTimeout === undefined ? 10_000 : options.connectionTimeout,
|
|
143
|
+
connectionDelay: options?.connectionDelay ?? ((attempt) => Math.min(~~(1 << attempt) * 150, 10_000)),
|
|
144
|
+
shouldReconnect: options?.shouldReconnect ?? (() => true),
|
|
145
|
+
messageBuffer: options?.messageBuffer ?? new FIFOMessageBuffer(),
|
|
146
|
+
};
|
|
147
|
+
this._protocols = protocols;
|
|
148
|
+
this._socket = this.reconnectOptions.connectionTimeout
|
|
149
|
+
? this._connectWithTimeout(url, this._protocols, this.reconnectOptions.connectionTimeout)
|
|
150
|
+
: new WebSocket(url, this._protocols);
|
|
151
|
+
this._initEventListeners();
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Creates a WebSocket connection with timeout.
|
|
155
|
+
* @param url - The WebSocket URL to connect to.
|
|
156
|
+
* @param protocols - The WebSocket protocols to use.
|
|
157
|
+
* @param timeout - The connection timeout in ms.
|
|
158
|
+
* @returns A new WebSocket instance.
|
|
159
|
+
*/
|
|
160
|
+
_connectWithTimeout(url, protocols, timeout) {
|
|
161
|
+
const socket = new WebSocket(url, protocols);
|
|
162
|
+
const timeoutId = setTimeout(() => {
|
|
163
|
+
socket.removeEventListener("open", openHandler);
|
|
164
|
+
socket.removeEventListener("close", closeHandler);
|
|
165
|
+
socket.close(3008, "Timeout"); // https://www.iana.org/assignments/websocket/websocket.xml#close-code-number
|
|
166
|
+
}, timeout);
|
|
167
|
+
const openHandler = () => {
|
|
168
|
+
socket.removeEventListener("close", closeHandler);
|
|
169
|
+
clearTimeout(timeoutId);
|
|
170
|
+
};
|
|
171
|
+
const closeHandler = () => {
|
|
172
|
+
socket.removeEventListener("open", openHandler);
|
|
173
|
+
clearTimeout(timeoutId);
|
|
174
|
+
};
|
|
175
|
+
socket.addEventListener("open", openHandler, { once: true });
|
|
176
|
+
socket.addEventListener("close", closeHandler, { once: true });
|
|
177
|
+
return socket;
|
|
178
|
+
}
|
|
179
|
+
/** Initializes the internal event listeners for the WebSocket. */
|
|
180
|
+
_initEventListeners() {
|
|
181
|
+
this._socket.addEventListener("open", () => {
|
|
182
|
+
// Reset reconnection count
|
|
183
|
+
this._reconnectCount = 0;
|
|
184
|
+
// Send all buffered messages
|
|
185
|
+
let message;
|
|
186
|
+
while ((message = this.reconnectOptions.messageBuffer.shift()) !== undefined) {
|
|
187
|
+
this._socket.send(message);
|
|
188
|
+
}
|
|
189
|
+
}, { once: true });
|
|
190
|
+
this._socket.addEventListener("close", async (event) => {
|
|
191
|
+
try {
|
|
192
|
+
// If the termination signal is already aborted, do not attempt to reconnect
|
|
193
|
+
if (this.terminationSignal.aborted)
|
|
194
|
+
return;
|
|
195
|
+
// Check if reconnection should be attempted
|
|
196
|
+
if (++this._reconnectCount > this.reconnectOptions.maxRetries) {
|
|
197
|
+
this._cleanup(new ReconnectingWebSocketError("RECONNECTION_LIMIT_REACHED"));
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
200
|
+
const userDecision = await this.reconnectOptions.shouldReconnect(event);
|
|
201
|
+
if (this.terminationSignal.aborted)
|
|
202
|
+
return; // Check again after the await
|
|
203
|
+
if (!userDecision) {
|
|
204
|
+
this._cleanup(new ReconnectingWebSocketError("RECONNECTION_STOPPED_BY_USER"));
|
|
205
|
+
return;
|
|
206
|
+
}
|
|
207
|
+
// Delay before reconnecting
|
|
208
|
+
const delay = typeof this.reconnectOptions.connectionDelay === "number"
|
|
209
|
+
? this.reconnectOptions.connectionDelay
|
|
210
|
+
: await this.reconnectOptions.connectionDelay(this._reconnectCount);
|
|
211
|
+
if (this.terminationSignal.aborted)
|
|
212
|
+
return; // Check again after the await
|
|
213
|
+
await sleep(delay, this.terminationSignal);
|
|
214
|
+
// Reconnect the socket
|
|
215
|
+
this._socket = this.reconnectOptions.connectionTimeout
|
|
216
|
+
? this._connectWithTimeout(this.url, this._protocols, this.reconnectOptions.connectionTimeout)
|
|
217
|
+
: new WebSocket(this.url, this._protocols);
|
|
218
|
+
// Reconnect all listeners
|
|
219
|
+
this._initEventListeners();
|
|
220
|
+
this._eventListeners.forEach(({ type, listenerProxy, options }) => {
|
|
221
|
+
this._socket.addEventListener(type, listenerProxy, options);
|
|
222
|
+
});
|
|
223
|
+
}
|
|
224
|
+
catch (error) {
|
|
225
|
+
this._cleanup(new ReconnectingWebSocketError("UNKNOWN_ERROR", error));
|
|
226
|
+
}
|
|
227
|
+
}, { once: true });
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Clean up internal resources.
|
|
231
|
+
* @param reason - The reason for cleanup.
|
|
232
|
+
*/
|
|
233
|
+
_cleanup(reason) {
|
|
234
|
+
this._terminationController.abort(reason);
|
|
235
|
+
this.reconnectOptions.messageBuffer.clear();
|
|
236
|
+
this._eventListeners = [];
|
|
237
|
+
}
|
|
238
|
+
/**
|
|
239
|
+
* Check if two event listeners are the same (just like EventTarget).
|
|
240
|
+
* @param a - First event listener configuration.
|
|
241
|
+
* @param b - Second event listener configuration.
|
|
242
|
+
* @returns True if the listeners match.
|
|
243
|
+
*/
|
|
244
|
+
_isListenerMatch(a, b) {
|
|
245
|
+
const aCapture = Boolean(typeof a.options === "object" ? a.options.capture : a.options);
|
|
246
|
+
const bCapture = Boolean(typeof b.options === "object" ? b.options.capture : b.options);
|
|
247
|
+
return a.type === b.type && a.listener === b.listener && aCapture === bCapture;
|
|
248
|
+
}
|
|
249
|
+
// WebSocket property implementations
|
|
250
|
+
get url() {
|
|
251
|
+
return this._socket.url;
|
|
252
|
+
}
|
|
253
|
+
get readyState() {
|
|
254
|
+
return this._socket.readyState;
|
|
255
|
+
}
|
|
256
|
+
get bufferedAmount() {
|
|
257
|
+
return this._socket.bufferedAmount;
|
|
258
|
+
}
|
|
259
|
+
get extensions() {
|
|
260
|
+
return this._socket.extensions;
|
|
261
|
+
}
|
|
262
|
+
get protocol() {
|
|
263
|
+
return this._socket.protocol;
|
|
264
|
+
}
|
|
265
|
+
get binaryType() {
|
|
266
|
+
return this._socket.binaryType;
|
|
267
|
+
}
|
|
268
|
+
set binaryType(value) {
|
|
269
|
+
this._socket.binaryType = value;
|
|
270
|
+
}
|
|
271
|
+
get onclose() {
|
|
272
|
+
return this._socket.onclose;
|
|
273
|
+
}
|
|
274
|
+
set onclose(value) {
|
|
275
|
+
this._socket.onclose = value;
|
|
276
|
+
}
|
|
277
|
+
get onerror() {
|
|
278
|
+
return this._socket.onerror;
|
|
279
|
+
}
|
|
280
|
+
set onerror(value) {
|
|
281
|
+
this._socket.onerror = value;
|
|
282
|
+
}
|
|
283
|
+
get onmessage() {
|
|
284
|
+
return this._socket.onmessage;
|
|
285
|
+
}
|
|
286
|
+
set onmessage(value) {
|
|
287
|
+
this._socket.onmessage = value;
|
|
288
|
+
}
|
|
289
|
+
get onopen() {
|
|
290
|
+
return this._socket.onopen;
|
|
291
|
+
}
|
|
292
|
+
set onopen(value) {
|
|
293
|
+
this._socket.onopen = value;
|
|
294
|
+
}
|
|
295
|
+
/**
|
|
296
|
+
* @param permanently - If true, the connection will be permanently closed. Default is true.
|
|
297
|
+
*/
|
|
298
|
+
close(code, reason, permanently = true) {
|
|
299
|
+
if (permanently)
|
|
300
|
+
this._cleanup(new ReconnectingWebSocketError("USER_INITIATED_CLOSE"));
|
|
301
|
+
this._socket.close(code, reason);
|
|
302
|
+
}
|
|
303
|
+
/**
|
|
304
|
+
* @note If the connection is not open, the data will be buffered and sent when the connection is established.
|
|
305
|
+
*/
|
|
306
|
+
send(data) {
|
|
307
|
+
if (this._socket.readyState !== WebSocket.OPEN && !this.terminationSignal.aborted) {
|
|
308
|
+
this.reconnectOptions.messageBuffer.push(data);
|
|
309
|
+
}
|
|
310
|
+
else {
|
|
311
|
+
this._socket.send(data);
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
addEventListener(type, listener, options) {
|
|
315
|
+
// Check if the listener is already added
|
|
316
|
+
const exists = this._eventListeners.some((e) => this._isListenerMatch(e, { type, listener, options }));
|
|
317
|
+
if (exists)
|
|
318
|
+
return;
|
|
319
|
+
// Wrap the original listener to follow the once option when reconnecting
|
|
320
|
+
const listenerProxy = (event) => {
|
|
321
|
+
try {
|
|
322
|
+
if (typeof listener === "function") {
|
|
323
|
+
listener.call(this, event);
|
|
324
|
+
}
|
|
325
|
+
else {
|
|
326
|
+
listener.handleEvent(event);
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
finally {
|
|
330
|
+
if (typeof options === "object" && options.once === true) {
|
|
331
|
+
const index = this._eventListeners.findIndex((e) => this._isListenerMatch(e, { type, listener, options }));
|
|
332
|
+
if (index !== -1)
|
|
333
|
+
this._eventListeners.splice(index, 1);
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
};
|
|
337
|
+
this._eventListeners.push({ type, listener, options, listenerProxy });
|
|
338
|
+
this._socket.addEventListener(type, listenerProxy, options);
|
|
339
|
+
}
|
|
340
|
+
removeEventListener(type, listener, options) {
|
|
341
|
+
const index = this._eventListeners.findIndex((e) => this._isListenerMatch(e, { type, listener, options }));
|
|
342
|
+
if (index !== -1) {
|
|
343
|
+
const { listenerProxy } = this._eventListeners[index];
|
|
344
|
+
this._socket.removeEventListener(type, listenerProxy, options);
|
|
345
|
+
this._eventListeners.splice(index, 1);
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
dispatchEvent(event) {
|
|
349
|
+
return this._socket.dispatchEvent(event);
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
exports.ReconnectingWebSocket = ReconnectingWebSocket;
|
|
353
|
+
Object.defineProperty(ReconnectingWebSocket, "CLOSED", {
|
|
354
|
+
enumerable: true,
|
|
355
|
+
configurable: true,
|
|
356
|
+
writable: true,
|
|
357
|
+
value: WebSocket.CLOSED
|
|
358
|
+
});
|
|
359
|
+
Object.defineProperty(ReconnectingWebSocket, "CLOSING", {
|
|
360
|
+
enumerable: true,
|
|
361
|
+
configurable: true,
|
|
362
|
+
writable: true,
|
|
363
|
+
value: WebSocket.CLOSING
|
|
364
|
+
});
|
|
365
|
+
Object.defineProperty(ReconnectingWebSocket, "CONNECTING", {
|
|
366
|
+
enumerable: true,
|
|
367
|
+
configurable: true,
|
|
368
|
+
writable: true,
|
|
369
|
+
value: WebSocket.CONNECTING
|
|
370
|
+
});
|
|
371
|
+
Object.defineProperty(ReconnectingWebSocket, "OPEN", {
|
|
372
|
+
enumerable: true,
|
|
373
|
+
configurable: true,
|
|
374
|
+
writable: true,
|
|
375
|
+
value: WebSocket.OPEN
|
|
376
|
+
});
|
|
377
|
+
/**
|
|
378
|
+
* Returns a promise that resolves after the specified number of ms,
|
|
379
|
+
* or rejects as soon as the given signal is aborted.
|
|
380
|
+
* @param ms - The number of ms to sleep.
|
|
381
|
+
* @param signal - An optional abort signal.
|
|
382
|
+
* @returns A promise that resolves after the specified delay.
|
|
383
|
+
*/
|
|
384
|
+
function sleep(ms, signal) {
|
|
385
|
+
if (signal?.aborted)
|
|
386
|
+
return Promise.reject(signal.reason);
|
|
387
|
+
return new Promise((resolve, reject) => {
|
|
388
|
+
const onAbort = () => {
|
|
389
|
+
clearTimeout(timer);
|
|
390
|
+
reject(signal?.reason);
|
|
391
|
+
};
|
|
392
|
+
const onTimeout = () => {
|
|
393
|
+
signal?.removeEventListener("abort", onAbort);
|
|
394
|
+
resolve();
|
|
395
|
+
};
|
|
396
|
+
const timer = setTimeout(onTimeout, ms);
|
|
397
|
+
signal?.addEventListener("abort", onAbort, { once: true });
|
|
398
|
+
});
|
|
399
|
+
}
|
|
400
|
+
});
|
|
@@ -1,15 +1,11 @@
|
|
|
1
|
-
import { TransportError } from "
|
|
2
|
-
import type { HyperliquidEventTarget } from "./
|
|
1
|
+
import { TransportError } from "../../base.js";
|
|
2
|
+
import type { HyperliquidEventTarget } from "./_hyperliquid_event_target.js";
|
|
3
3
|
/**
|
|
4
4
|
* Error thrown when a WebSocket request fails:
|
|
5
5
|
* - When the WebSocket connection is closed
|
|
6
6
|
* - When the server responds with an error message
|
|
7
7
|
*/
|
|
8
8
|
export declare class WebSocketRequestError extends TransportError {
|
|
9
|
-
/**
|
|
10
|
-
* Creates a new WebSocket request error.
|
|
11
|
-
* @param message - The error message.
|
|
12
|
-
*/
|
|
13
9
|
constructor(message: string);
|
|
14
10
|
}
|
|
15
11
|
/**
|
|
@@ -60,4 +56,4 @@ export declare class WebSocketRequestDispatcher {
|
|
|
60
56
|
*/
|
|
61
57
|
static requestToId(value: unknown): string;
|
|
62
58
|
}
|
|
63
|
-
//# sourceMappingURL=
|
|
59
|
+
//# sourceMappingURL=_websocket_request_dispatcher.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"_websocket_request_dispatcher.d.ts","sourceRoot":"","sources":["../../../../src/src/transports/websocket/_websocket_request_dispatcher.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AAC/C,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,gCAAgC,CAAC;AAE7E;;;;GAIG;AACH,qBAAa,qBAAsB,SAAQ,cAAc;gBACzC,OAAO,EAAE,MAAM;CAI9B;AAED;;;GAGG;AACH,qBAAa,0BAA0B;IAevB,SAAS,CAAC,MAAM,EAAE,SAAS;IAdvC,2BAA2B;IAC3B,SAAS,CAAC,MAAM,EAAE,MAAM,CAAK;IAE7B,oDAAoD;IACpD,SAAS,CAAC,OAAO,EAAE,GAAG,CAClB,MAAM,GAAG,MAAM,EACf;QAAE,OAAO,EAAE,CAAC,KAAK,EAAE,OAAO,KAAK,IAAI,CAAC;QAAC,MAAM,EAAE,CAAC,MAAM,EAAE,OAAO,KAAK,IAAI,CAAA;KAAE,CAC3E,CAAa;IAEd;;;;OAIG;gBACmB,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,sBAAsB;IA2DzE,+BAA+B;IAC/B,SAAS,KAAK,MAAM,IAAI,MAAM,CAE7B;IAED;;;;;;OAMG;IACG,OAAO,CACT,MAAM,EAAE,MAAM,GAAG,WAAW,GAAG,aAAa,EAC5C,OAAO,EAAE,OAAO,EAChB,MAAM,CAAC,EAAE,WAAW,GACrB,OAAO,CAAC,OAAO,CAAC;IAkCnB;;;;OAIG;IACH,SAAS,CAAC,OAAO,CAAC,EAAE,EAAE,MAAM,GAAG,MAAM,EAAE,KAAK,EAAE,OAAO,GAAG,IAAI;IAK5D;;;;OAIG;IACH,SAAS,CAAC,MAAM,CAAC,EAAE,EAAE,MAAM,GAAG,MAAM,EAAE,MAAM,EAAE,OAAO,GAAG,IAAI;IAK5D;;;;OAIG;IACH,MAAM,CAAC,WAAW,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM;CAK7C"}
|
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
(function (factory) {
|
|
2
|
+
if (typeof module === "object" && typeof module.exports === "object") {
|
|
3
|
+
var v = factory(require, exports);
|
|
4
|
+
if (v !== undefined) module.exports = v;
|
|
5
|
+
}
|
|
6
|
+
else if (typeof define === "function" && define.amd) {
|
|
7
|
+
define(["require", "exports", "../../base.js"], factory);
|
|
8
|
+
}
|
|
9
|
+
})(function (require, exports) {
|
|
10
|
+
"use strict";
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.WebSocketRequestDispatcher = exports.WebSocketRequestError = void 0;
|
|
13
|
+
const base_js_1 = require("../../base.js");
|
|
14
|
+
/**
|
|
15
|
+
* Error thrown when a WebSocket request fails:
|
|
16
|
+
* - When the WebSocket connection is closed
|
|
17
|
+
* - When the server responds with an error message
|
|
18
|
+
*/
|
|
19
|
+
class WebSocketRequestError extends base_js_1.TransportError {
|
|
20
|
+
constructor(message) {
|
|
21
|
+
super(message);
|
|
22
|
+
this.name = "WebSocketRequestError";
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
exports.WebSocketRequestError = WebSocketRequestError;
|
|
26
|
+
/**
|
|
27
|
+
* Manages WebSocket requests to the Hyperliquid API.
|
|
28
|
+
* Handles request creation, sending, and mapping responses to their corresponding requests.
|
|
29
|
+
*/
|
|
30
|
+
class WebSocketRequestDispatcher {
|
|
31
|
+
/**
|
|
32
|
+
* Creates a new WebSocket request dispatcher.
|
|
33
|
+
* @param socket - WebSocket connection instance for sending requests to the Hyperliquid WebSocket API
|
|
34
|
+
* @param hlEvents - Used to recognize Hyperliquid responses and match them with sent requests
|
|
35
|
+
*/
|
|
36
|
+
constructor(socket, hlEvents) {
|
|
37
|
+
Object.defineProperty(this, "socket", {
|
|
38
|
+
enumerable: true,
|
|
39
|
+
configurable: true,
|
|
40
|
+
writable: true,
|
|
41
|
+
value: socket
|
|
42
|
+
});
|
|
43
|
+
/** Last used request ID */
|
|
44
|
+
Object.defineProperty(this, "lastId", {
|
|
45
|
+
enumerable: true,
|
|
46
|
+
configurable: true,
|
|
47
|
+
writable: true,
|
|
48
|
+
value: 0
|
|
49
|
+
});
|
|
50
|
+
/** Map of pending requests waiting for responses */
|
|
51
|
+
Object.defineProperty(this, "pending", {
|
|
52
|
+
enumerable: true,
|
|
53
|
+
configurable: true,
|
|
54
|
+
writable: true,
|
|
55
|
+
value: new Map()
|
|
56
|
+
});
|
|
57
|
+
// Monitor responses and match the pending request
|
|
58
|
+
hlEvents.addEventListener("subscriptionResponse", (event) => {
|
|
59
|
+
// Use a stringified request as an id
|
|
60
|
+
const id = WebSocketRequestDispatcher.requestToId(event.detail.subscription);
|
|
61
|
+
this.resolve(id, event.detail);
|
|
62
|
+
});
|
|
63
|
+
hlEvents.addEventListener("post", (event) => {
|
|
64
|
+
const data = event.detail.response.type === "info"
|
|
65
|
+
? event.detail.response.payload.data
|
|
66
|
+
: event.detail.response.payload;
|
|
67
|
+
this.resolve(event.detail.id, data);
|
|
68
|
+
});
|
|
69
|
+
hlEvents.addEventListener("error", (event) => {
|
|
70
|
+
try {
|
|
71
|
+
// Error event doesn't have an id, use original request to match
|
|
72
|
+
const request = event.detail.match(/{.*}/)?.[0];
|
|
73
|
+
if (request) {
|
|
74
|
+
const parsedRequest = JSON.parse(request);
|
|
75
|
+
if ("id" in parsedRequest && typeof parsedRequest.id === "number") {
|
|
76
|
+
// If a post request was sent, it is possible to get the id from the request
|
|
77
|
+
this.reject(parsedRequest.id, new WebSocketRequestError(`Cannot complete WebSocket request: ${event.detail}`));
|
|
78
|
+
}
|
|
79
|
+
else if ("subscription" in parsedRequest &&
|
|
80
|
+
typeof parsedRequest.subscription === "object" &&
|
|
81
|
+
parsedRequest.subscription !== null) {
|
|
82
|
+
// If a subscription/unsubscribe request was sent, use the request as an id
|
|
83
|
+
const id = WebSocketRequestDispatcher.requestToId(parsedRequest.subscription);
|
|
84
|
+
this.reject(id, new WebSocketRequestError(`Cannot complete WebSocket request: ${event.detail}`));
|
|
85
|
+
}
|
|
86
|
+
else {
|
|
87
|
+
// If the request is not recognized, use the parsed request as an id
|
|
88
|
+
const id = WebSocketRequestDispatcher.requestToId(parsedRequest);
|
|
89
|
+
this.reject(id, new WebSocketRequestError(`Cannot complete WebSocket request: ${event.detail}`));
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
catch {
|
|
94
|
+
// Ignore JSON parsing errors
|
|
95
|
+
}
|
|
96
|
+
});
|
|
97
|
+
// Throws all pending requests if the connection is dropped
|
|
98
|
+
socket.addEventListener("close", () => {
|
|
99
|
+
this.pending.forEach(({ reject }) => {
|
|
100
|
+
reject(new WebSocketRequestError("Cannot complete WebSocket request: connection is closed"));
|
|
101
|
+
});
|
|
102
|
+
this.pending.clear();
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
/** Gets the next request ID */
|
|
106
|
+
get nextId() {
|
|
107
|
+
return ++this.lastId;
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Sends a request to the Hyperliquid API.
|
|
111
|
+
* @param method - The method of websocket request.
|
|
112
|
+
* @param payload - The payload to send with the request.
|
|
113
|
+
* @param signal - An optional abort signal.
|
|
114
|
+
* @returns A promise that resolves with the parsed JSON response body.
|
|
115
|
+
*/
|
|
116
|
+
async request(method, payload, signal) {
|
|
117
|
+
signal?.throwIfAborted();
|
|
118
|
+
// Create a request object
|
|
119
|
+
let id;
|
|
120
|
+
let request;
|
|
121
|
+
if (method === "post") {
|
|
122
|
+
id = this.nextId;
|
|
123
|
+
request = { method, id, request: payload };
|
|
124
|
+
}
|
|
125
|
+
else {
|
|
126
|
+
id = WebSocketRequestDispatcher.requestToId(payload);
|
|
127
|
+
request = { method, subscription: payload };
|
|
128
|
+
}
|
|
129
|
+
// Send the request
|
|
130
|
+
this.socket.send(JSON.stringify(request));
|
|
131
|
+
// Wait for a response
|
|
132
|
+
let onAbort;
|
|
133
|
+
return await new Promise((resolve, reject) => {
|
|
134
|
+
// Add an abort listener
|
|
135
|
+
onAbort = () => reject(signal?.reason);
|
|
136
|
+
signal?.addEventListener("abort", onAbort, { once: true });
|
|
137
|
+
// Add the promise to the pending list
|
|
138
|
+
this.pending.set(id, { resolve, reject });
|
|
139
|
+
}).finally(() => {
|
|
140
|
+
// Remove the abort listener when the promise is settled
|
|
141
|
+
signal?.removeEventListener("abort", onAbort);
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Resolves a pending request.
|
|
146
|
+
* @param id - A request ID or a stringified request.
|
|
147
|
+
* @param value - A resolution value.
|
|
148
|
+
*/
|
|
149
|
+
resolve(id, value) {
|
|
150
|
+
this.pending.get(id)?.resolve(value);
|
|
151
|
+
this.pending.delete(id);
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Rejects a pending request.
|
|
155
|
+
* @param id - A request ID or a stringified request.
|
|
156
|
+
* @param reason - A rejection reason.
|
|
157
|
+
*/
|
|
158
|
+
reject(id, reason) {
|
|
159
|
+
this.pending.get(id)?.reject(reason);
|
|
160
|
+
this.pending.delete(id);
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Normalizes a request object to an ID.
|
|
164
|
+
* @param value - A request object.
|
|
165
|
+
* @returns A stringified request.
|
|
166
|
+
*/
|
|
167
|
+
static requestToId(value) {
|
|
168
|
+
const lowerHex = deepLowerHex(value);
|
|
169
|
+
const sorted = deepSortKeys(lowerHex);
|
|
170
|
+
return JSON.stringify(sorted);
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
exports.WebSocketRequestDispatcher = WebSocketRequestDispatcher;
|
|
174
|
+
/**
|
|
175
|
+
* Convert all hexadecimal strings to lowercase in an object/array.
|
|
176
|
+
* @param obj - The object/array to convert hexadecimal strings to lowercase.
|
|
177
|
+
* @returns A new object/array with hexadecimal strings converted to lowercase.
|
|
178
|
+
*/
|
|
179
|
+
function deepLowerHex(obj) {
|
|
180
|
+
if (Array.isArray(obj)) {
|
|
181
|
+
return obj.map(deepLowerHex);
|
|
182
|
+
}
|
|
183
|
+
else if (obj && typeof obj === "object") {
|
|
184
|
+
return Object.entries(obj).reduce((acc, [key, val]) => ({
|
|
185
|
+
...acc,
|
|
186
|
+
[key]: deepLowerHex(val),
|
|
187
|
+
}), {});
|
|
188
|
+
}
|
|
189
|
+
else if (typeof obj === "string" && /^0x[0-9A-Fa-f]+$/.test(obj)) {
|
|
190
|
+
return obj.toLowerCase();
|
|
191
|
+
}
|
|
192
|
+
else {
|
|
193
|
+
return obj;
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
/**
|
|
197
|
+
* Deeply sort the keys of an object.
|
|
198
|
+
* @param obj - An object to sort the keys of.
|
|
199
|
+
* @returns A new object with sorted keys.
|
|
200
|
+
*/
|
|
201
|
+
function deepSortKeys(obj) {
|
|
202
|
+
if (obj === null || typeof obj !== "object") {
|
|
203
|
+
return obj;
|
|
204
|
+
}
|
|
205
|
+
if (Array.isArray(obj)) {
|
|
206
|
+
return obj.map(deepSortKeys);
|
|
207
|
+
}
|
|
208
|
+
return Object.fromEntries(Object.entries(obj)
|
|
209
|
+
.sort(([a], [b]) => a.localeCompare(b))
|
|
210
|
+
.map(([k, v]) => [k, deepSortKeys(v)]));
|
|
211
|
+
}
|
|
212
|
+
});
|