@polkadot-api/ws-provider 0.5.0 → 0.6.1

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 (44) hide show
  1. package/dist/index/esm/default-provider.mjs +153 -0
  2. package/dist/index/esm/default-provider.mjs.map +1 -0
  3. package/dist/index/esm/follow-enhancer.mjs +90 -0
  4. package/dist/index/esm/follow-enhancer.mjs.map +1 -0
  5. package/dist/index/esm/index.mjs +3 -0
  6. package/dist/index/esm/index.mjs.map +1 -0
  7. package/dist/{node/esm/types.mjs → index/esm/types-common.mjs} +1 -1
  8. package/dist/index/esm/types-common.mjs.map +1 -0
  9. package/dist/index/index.d.ts +47 -0
  10. package/dist/index/index.js +250 -0
  11. package/dist/index/index.js.map +1 -0
  12. package/dist/node/esm/default-provider.mjs +153 -0
  13. package/dist/node/esm/default-provider.mjs.map +1 -0
  14. package/dist/node/esm/follow-enhancer.mjs +8 -0
  15. package/dist/node/esm/follow-enhancer.mjs.map +1 -1
  16. package/dist/node/esm/legacy-provider.mjs +42 -0
  17. package/dist/node/esm/legacy-provider.mjs.map +1 -0
  18. package/dist/node/esm/node.mjs +3 -3
  19. package/dist/node/esm/node.mjs.map +1 -1
  20. package/dist/{web/esm/types.mjs → node/esm/types-common.mjs} +1 -1
  21. package/dist/node/esm/types-common.mjs.map +1 -0
  22. package/dist/node/node.d.ts +5 -0
  23. package/dist/node/node.js +163 -126
  24. package/dist/node/node.js.map +1 -1
  25. package/dist/web/esm/default-provider.mjs +153 -0
  26. package/dist/web/esm/default-provider.mjs.map +1 -0
  27. package/dist/web/esm/follow-enhancer.mjs +8 -0
  28. package/dist/web/esm/follow-enhancer.mjs.map +1 -1
  29. package/dist/web/esm/legacy-provider.mjs +42 -0
  30. package/dist/web/esm/legacy-provider.mjs.map +1 -0
  31. package/dist/web/esm/types-common.mjs +10 -0
  32. package/dist/web/esm/types-common.mjs.map +1 -0
  33. package/dist/web/esm/web.mjs +3 -3
  34. package/dist/web/esm/web.mjs.map +1 -1
  35. package/dist/web/web.d.ts +5 -0
  36. package/dist/web/web.js +163 -126
  37. package/dist/web/web.js.map +1 -1
  38. package/package.json +17 -6
  39. package/dist/node/esm/types.mjs.map +0 -1
  40. package/dist/node/esm/ws-provider.mjs +0 -162
  41. package/dist/node/esm/ws-provider.mjs.map +0 -1
  42. package/dist/web/esm/types.mjs.map +0 -1
  43. package/dist/web/esm/ws-provider.mjs +0 -162
  44. package/dist/web/esm/ws-provider.mjs.map +0 -1
@@ -0,0 +1,153 @@
1
+ import { getSyncProvider } from '@polkadot-api/json-rpc-provider-proxy';
2
+ import { WsEvent } from './types-common.mjs';
3
+ import { followEnhancer } from './follow-enhancer.mjs';
4
+
5
+ const timeoutError = {
6
+ type: WsEvent.ERROR,
7
+ event: { type: "timeout" }
8
+ };
9
+ const noop = () => {
10
+ };
11
+ const defaultConfig = {
12
+ onStatusChanged: noop,
13
+ innerEnhancer: (x) => x,
14
+ timeout: 5e3
15
+ };
16
+ const mapEndpoints = (endpoints) => endpoints.map((x) => typeof x === "string" ? [x] : [x.uri, x.protocol]);
17
+ const getWsProvider = (endpoints, config) => {
18
+ const { onStatusChanged, innerEnhancer, timeout } = {
19
+ ...defaultConfig,
20
+ ...config
21
+ };
22
+ const actualEndpoints = mapEndpoints(
23
+ Array.isArray(endpoints) ? endpoints : [endpoints]
24
+ );
25
+ const WebsocketClass = config?.websocketClass ?? globalThis.WebSocket;
26
+ if (!WebsocketClass) throw new Error("Missing WebSocket class");
27
+ let idx = 0;
28
+ let status;
29
+ let switchTo = null;
30
+ let disconnect = noop;
31
+ let outerCleanup = noop;
32
+ const result = followEnhancer(
33
+ getSyncProvider(async () => {
34
+ const [uri, protocols] = switchTo || actualEndpoints[idx++ % endpoints.length];
35
+ switchTo = null;
36
+ const socket = new WebsocketClass(uri, protocols);
37
+ const forceSocketClose = () => {
38
+ try {
39
+ socket.addEventListener("error", noop, { once: true });
40
+ socket.close();
41
+ } catch {
42
+ }
43
+ };
44
+ onStatusChanged(
45
+ status = {
46
+ type: WsEvent.CONNECTING,
47
+ uri,
48
+ protocols
49
+ }
50
+ );
51
+ await new Promise((resolve, reject) => {
52
+ const onOpen = () => {
53
+ initialCleanup();
54
+ resolve();
55
+ };
56
+ const onError = (e) => {
57
+ initialCleanup();
58
+ if (e == null) forceSocketClose();
59
+ console.error(
60
+ `Unable to connect to ${uri}${protocols ? ", protocols: " + protocols : ""}`
61
+ );
62
+ onStatusChanged(
63
+ status = {
64
+ type: e ? WsEvent.ERROR : WsEvent.CLOSE,
65
+ event: e
66
+ }
67
+ );
68
+ setTimeout(reject, e ? 300 : 0, e);
69
+ };
70
+ const timeoutToken = timeout !== Infinity ? setTimeout(() => {
71
+ initialCleanup();
72
+ forceSocketClose();
73
+ onStatusChanged(status = timeoutError);
74
+ reject(timeoutError.event);
75
+ }, timeout) : void 0;
76
+ const initialCleanup = () => {
77
+ clearTimeout(timeoutToken);
78
+ socket.removeEventListener("error", onError);
79
+ socket.removeEventListener("open", onOpen);
80
+ };
81
+ socket.addEventListener("open", onOpen);
82
+ socket.addEventListener("error", onError);
83
+ disconnect = () => {
84
+ onError(null);
85
+ };
86
+ });
87
+ onStatusChanged(
88
+ status = {
89
+ type: WsEvent.CONNECTED,
90
+ uri,
91
+ protocols
92
+ }
93
+ );
94
+ let _onInnerMessage;
95
+ const inner = innerEnhancer((onInnerMessage) => {
96
+ _onInnerMessage = onInnerMessage;
97
+ return {
98
+ send: (m) => {
99
+ socket.send(m);
100
+ },
101
+ disconnect: () => {
102
+ disconnect();
103
+ }
104
+ };
105
+ });
106
+ return (onMessage, onHalt) => {
107
+ const connection = inner(onMessage);
108
+ const _onMessage = (e) => {
109
+ if (typeof e.data === "string") _onInnerMessage(e.data);
110
+ };
111
+ const innerHalt = (reason) => (e) => {
112
+ console.warn(`WS halt (${reason})`);
113
+ onStatusChanged(
114
+ status = {
115
+ type: reason,
116
+ event: e
117
+ }
118
+ );
119
+ onHalt();
120
+ };
121
+ const onError = innerHalt(WsEvent.ERROR);
122
+ const onClose = innerHalt(WsEvent.CLOSE);
123
+ socket.addEventListener("message", _onMessage);
124
+ socket.addEventListener("error", onError);
125
+ socket.addEventListener("close", onClose);
126
+ disconnect = (withHalt) => {
127
+ outerCleanup();
128
+ disconnect = noop;
129
+ socket.removeEventListener("message", _onMessage);
130
+ socket.removeEventListener("error", onError);
131
+ socket.removeEventListener("close", onClose);
132
+ forceSocketClose();
133
+ if (withHalt) onClose({});
134
+ };
135
+ return connection;
136
+ };
137
+ }),
138
+ () => {
139
+ switchFn();
140
+ }
141
+ );
142
+ outerCleanup = result.cleanup;
143
+ delete result.cleanup;
144
+ const switchFn = (...args) => {
145
+ if (status.type === WsEvent.CLOSE) return;
146
+ if (args.length) switchTo = args;
147
+ if (status.type !== WsEvent.ERROR) disconnect(true);
148
+ };
149
+ return Object.assign(result, { switch: switchFn, getStatus: () => status });
150
+ };
151
+
152
+ export { defaultConfig, getWsProvider, mapEndpoints, noop };
153
+ //# sourceMappingURL=default-provider.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"default-provider.mjs","sources":["../../../src/default-provider.ts"],"sourcesContent":["import { getSyncProvider } from \"@polkadot-api/json-rpc-provider-proxy\"\nimport { StatusChange, WsJsonRpcProvider, WsEvent } from \"./types-common\"\nimport { followEnhancer } from \"./follow-enhancer\"\nimport { JsonRpcProvider } from \"@polkadot-api/json-rpc-provider\"\nimport { WebSocketClass } from \"./types-new\"\n\nconst timeoutError: StatusChange = {\n type: WsEvent.ERROR,\n event: { type: \"timeout\" },\n}\n\nexport const noop = () => {}\n\nexport const defaultConfig: {\n onStatusChanged: (status: StatusChange) => void\n innerEnhancer: (input: JsonRpcProvider) => JsonRpcProvider\n timeout: number\n} = {\n onStatusChanged: noop,\n innerEnhancer: (x: JsonRpcProvider) => x,\n timeout: 5_000,\n}\n\nexport const mapEndpoints = (\n endpoints: Array<string | { uri: string; protocol: string | string[] }>,\n): Array<[string, string | string[]] | [string]> =>\n endpoints.map((x) => (typeof x === \"string\" ? [x] : [x.uri, x.protocol]))\n\nexport const getWsProvider = (\n endpoints:\n | string\n | Array<string | { uri: string; protocol: string | string[] }>,\n config?: Partial<{\n onStatusChanged: (status: StatusChange) => void\n innerEnhancer: (input: JsonRpcProvider) => JsonRpcProvider\n timeout: number\n websocketClass: WebSocketClass\n }>,\n): WsJsonRpcProvider => {\n const { onStatusChanged, innerEnhancer, timeout } = {\n ...defaultConfig,\n ...config,\n }\n const actualEndpoints = mapEndpoints(\n Array.isArray(endpoints) ? endpoints : [endpoints],\n )\n const WebsocketClass = config?.websocketClass ?? globalThis.WebSocket\n if (!WebsocketClass) throw new Error(\"Missing WebSocket class\")\n\n let idx = 0\n let status: StatusChange\n let switchTo: [string] | [string, string | string[]] | null = null\n let disconnect: (withHalt?: boolean) => void = noop\n\n let outerCleanup: () => void = noop\n const result = followEnhancer(\n getSyncProvider(async () => {\n const [uri, protocols] =\n switchTo || actualEndpoints[idx++ % endpoints.length]\n switchTo = null\n const socket = new WebsocketClass(uri, protocols)\n const forceSocketClose = () => {\n try {\n socket.addEventListener(\"error\", noop, { once: true })\n socket.close()\n } catch {}\n }\n onStatusChanged(\n (status = {\n type: WsEvent.CONNECTING,\n uri,\n protocols,\n }),\n )\n\n await new Promise<void>((resolve, reject) => {\n const onOpen = () => {\n initialCleanup()\n resolve()\n }\n\n const onError = (e: Event | null) => {\n initialCleanup()\n if (e == null) forceSocketClose()\n console.error(\n `Unable to connect to ${uri}${\n protocols ? \", protocols: \" + protocols : \"\"\n }`,\n )\n onStatusChanged(\n (status = {\n type: e ? WsEvent.ERROR : WsEvent.CLOSE,\n event: e,\n }),\n )\n setTimeout(reject, e ? 300 : 0, e)\n }\n\n const timeoutToken =\n timeout !== Infinity\n ? setTimeout(() => {\n initialCleanup()\n forceSocketClose()\n onStatusChanged((status = timeoutError))\n reject(timeoutError.event)\n }, timeout)\n : undefined\n\n const initialCleanup = () => {\n clearTimeout(timeoutToken)\n socket.removeEventListener(\"error\", onError)\n socket.removeEventListener(\"open\", onOpen)\n }\n socket.addEventListener(\"open\", onOpen)\n socket.addEventListener(\"error\", onError)\n disconnect = () => {\n onError(null)\n }\n })\n\n onStatusChanged(\n (status = {\n type: WsEvent.CONNECTED,\n uri,\n protocols,\n }),\n )\n\n let _onInnerMessage: (msg: string) => void\n const inner = innerEnhancer((onInnerMessage) => {\n _onInnerMessage = onInnerMessage\n return {\n send: (m) => {\n socket.send(m)\n },\n disconnect: () => {\n disconnect()\n },\n }\n })\n\n return (onMessage, onHalt) => {\n const connection = inner(onMessage)\n const _onMessage = (e: MessageEvent) => {\n if (typeof e.data === \"string\") _onInnerMessage(e.data)\n }\n const innerHalt =\n (reason: WsEvent.CLOSE | WsEvent.ERROR) => (e: any) => {\n console.warn(`WS halt (${reason})`)\n onStatusChanged(\n (status = {\n type: reason,\n event: e,\n }),\n )\n onHalt()\n }\n const onError = innerHalt(WsEvent.ERROR)\n const onClose = innerHalt(WsEvent.CLOSE)\n\n socket.addEventListener(\"message\", _onMessage)\n socket.addEventListener(\"error\", onError)\n socket.addEventListener(\"close\", onClose)\n disconnect = (withHalt) => {\n outerCleanup()\n disconnect = noop\n socket.removeEventListener(\"message\", _onMessage)\n socket.removeEventListener(\"error\", onError)\n socket.removeEventListener(\"close\", onClose)\n forceSocketClose()\n if (withHalt) onClose({})\n }\n\n return connection\n }\n }),\n () => {\n switchFn()\n },\n )\n outerCleanup = result.cleanup\n delete (result as any).cleanup\n\n const switchFn: WsJsonRpcProvider[\"switch\"] = (...args) => {\n if (status.type === WsEvent.CLOSE) return\n if (args.length) switchTo = args as any\n if (status.type !== WsEvent.ERROR) disconnect(true)\n }\n\n return Object.assign(result, { switch: switchFn, getStatus: () => status })\n}\n"],"names":[],"mappings":"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"}
@@ -0,0 +1,90 @@
1
+ const methods = {};
2
+ ["v1", "unstable"].forEach((version) => {
3
+ methods[`chainHead_${version}_follow`] = "follow";
4
+ methods[`chainHead_${version}_unfollow`] = "unfollow";
5
+ });
6
+ const followEnhancer = (base, forceDisconnect) => {
7
+ const prematureStops = /* @__PURE__ */ new Set();
8
+ const preOpId = /* @__PURE__ */ new Map();
9
+ const onGoing = /* @__PURE__ */ new Set();
10
+ let methodsRequestId;
11
+ const result = (onMsg) => {
12
+ const { send, disconnect } = base((fromProvider) => {
13
+ let nStops = {
14
+ latest: Date.now(),
15
+ count: 0
16
+ };
17
+ const parsed = JSON.parse(fromProvider);
18
+ if ("id" in parsed) {
19
+ const { id, result: result2 } = parsed;
20
+ if (id === methodsRequestId) {
21
+ methodsRequestId = void 0;
22
+ if (result2 && !result2.methods.some((x) => {
23
+ const [group, , name] = x.split("_");
24
+ return group === "chainHead" && name === "follow";
25
+ })) {
26
+ onMsg(fromProvider);
27
+ forceDisconnect();
28
+ return;
29
+ }
30
+ }
31
+ const msg = preOpId.get(id);
32
+ if (msg) {
33
+ preOpId.delete(id);
34
+ if (prematureStops.has(result2)) {
35
+ prematureStops.delete(result2);
36
+ return;
37
+ }
38
+ onGoing.add(result2);
39
+ const currentSize = onGoing.size + preOpId.size;
40
+ if (currentSize > 2)
41
+ console.warn(
42
+ `Too many chainHead follow subscriptions (${currentSize})`
43
+ );
44
+ else if (parsed.error) {
45
+ console.warn(`chainHead follow failed on the ${currentSize} sub`);
46
+ forceDisconnect();
47
+ preOpId.set(id, msg);
48
+ send(msg);
49
+ return;
50
+ }
51
+ }
52
+ } else {
53
+ const { subscription, result: result2 } = parsed.params;
54
+ if (result2?.event === "stop") {
55
+ const diff = Date.now() - nStops.latest;
56
+ nStops.latest += diff;
57
+ nStops.count = diff < 1e3 ? nStops.count + 1 : 1;
58
+ if (onGoing.has(subscription)) onGoing.delete(subscription);
59
+ else prematureStops.add(subscription);
60
+ }
61
+ }
62
+ onMsg(fromProvider);
63
+ if (nStops.count > 2) forceDisconnect();
64
+ });
65
+ return {
66
+ send(toProvider) {
67
+ const parsed = JSON.parse(toProvider);
68
+ if (parsed.method === "rpc_methods") methodsRequestId = parsed.id;
69
+ const method = methods[parsed.method];
70
+ if (method === "follow") {
71
+ preOpId.set(parsed.id, toProvider);
72
+ } else if (method === "unfollow") {
73
+ onGoing.delete(parsed.params[0]);
74
+ }
75
+ send(toProvider);
76
+ },
77
+ disconnect
78
+ };
79
+ };
80
+ return Object.assign(result, {
81
+ cleanup: () => {
82
+ prematureStops.clear();
83
+ preOpId.clear();
84
+ onGoing.clear();
85
+ }
86
+ });
87
+ };
88
+
89
+ export { followEnhancer };
90
+ //# sourceMappingURL=follow-enhancer.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"follow-enhancer.mjs","sources":["../../../src/follow-enhancer.ts"],"sourcesContent":["import { JsonRpcProvider } from \"@polkadot-api/json-rpc-provider\"\n\nconst methods: Record<string, \"follow\" | \"unfollow\"> = {}\n;[\"v1\", \"unstable\"].forEach((version) => {\n methods[`chainHead_${version}_follow`] = \"follow\"\n methods[`chainHead_${version}_unfollow`] = \"unfollow\"\n})\n\nexport const followEnhancer: (\n input: JsonRpcProvider,\n forceDisconnect: () => void,\n) => JsonRpcProvider & {\n cleanup: () => void\n} = (base, forceDisconnect) => {\n const prematureStops = new Set<string>()\n const preOpId = new Map<string, string>()\n const onGoing = new Set<string>()\n let methodsRequestId: string | undefined\n\n const result: JsonRpcProvider = (onMsg) => {\n const { send, disconnect } = base((fromProvider) => {\n let nStops: { latest: number; count: number } = {\n latest: Date.now(),\n count: 0,\n }\n const parsed = JSON.parse(fromProvider)\n // it's a response\n if (\"id\" in parsed) {\n const { id, result } = parsed\n if (id === methodsRequestId) {\n methodsRequestId = undefined\n if (\n result &&\n !(result.methods as string[]).some((x) => {\n const [group, , name] = x.split(\"_\")\n return group === \"chainHead\" && name === \"follow\"\n })\n ) {\n onMsg(fromProvider)\n forceDisconnect()\n return\n }\n }\n\n const msg = preOpId.get(id)\n if (msg) {\n preOpId.delete(id)\n if (prematureStops.has(result)) {\n prematureStops.delete(result)\n return\n }\n\n onGoing.add(result)\n const currentSize = onGoing.size + preOpId.size\n if (currentSize > 2)\n console.warn(\n `Too many chainHead follow subscriptions (${currentSize})`,\n )\n else if (parsed.error) {\n console.warn(`chainHead follow failed on the ${currentSize} sub`)\n forceDisconnect()\n preOpId.set(id, msg)\n send(msg)\n return\n }\n }\n } else {\n // it's a notifiaction\n const { subscription, result } = (parsed as any).params\n if (result?.event === \"stop\") {\n const diff = Date.now() - nStops.latest\n nStops.latest += diff\n nStops.count = diff < 1000 ? nStops.count + 1 : 1\n\n if (onGoing.has(subscription)) onGoing.delete(subscription)\n else prematureStops.add(subscription)\n }\n }\n\n onMsg(fromProvider)\n if (nStops.count > 2) forceDisconnect()\n })\n\n return {\n send(toProvider) {\n const parsed = JSON.parse(toProvider)\n if (parsed.method === \"rpc_methods\") methodsRequestId = parsed.id\n\n const method = methods[parsed.method]\n if (method === \"follow\") {\n preOpId.set(parsed.id, toProvider)\n } else if (method === \"unfollow\") {\n onGoing.delete(parsed.params[0])\n }\n send(toProvider)\n },\n disconnect,\n }\n }\n\n return Object.assign(result, {\n cleanup: () => {\n prematureStops.clear()\n preOpId.clear()\n onGoing.clear()\n },\n })\n}\n"],"names":["result"],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ export { WsEvent } from './types-common.mjs';
2
+ export { getWsProvider } from './default-provider.mjs';
3
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.mjs","sources":[],"sourcesContent":[],"names":[],"mappings":";"}
@@ -7,4 +7,4 @@ var WsEvent = /* @__PURE__ */ ((WsEvent2) => {
7
7
  })(WsEvent || {});
8
8
 
9
9
  export { WsEvent };
10
- //# sourceMappingURL=types.mjs.map
10
+ //# sourceMappingURL=types-common.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types-common.mjs","sources":["../../../src/types-common.ts"],"sourcesContent":["import { JsonRpcProvider } from \"@polkadot-api/json-rpc-provider\"\n\nexport enum WsEvent {\n CONNECTING,\n CONNECTED,\n ERROR,\n CLOSE,\n}\n\nexport type WsConnecting = {\n type: WsEvent.CONNECTING\n uri: string\n protocols?: string | string[]\n}\nexport type WsConnected = {\n type: WsEvent.CONNECTED\n uri: string\n protocols?: string | string[]\n}\nexport type WsError = {\n type: WsEvent.ERROR\n event: any\n}\nexport type WsClose = {\n type: WsEvent.CLOSE\n event: any\n}\nexport type StatusChange = WsConnecting | WsConnected | WsError | WsClose\nexport type WsJsonRpcProvider = JsonRpcProvider & {\n switch: (uri?: string, protocol?: string[]) => void\n getStatus: () => StatusChange\n}\nexport { type JsonRpcProvider }\n"],"names":["WsEvent"],"mappings":"AAEO,IAAK,OAAA,qBAAAA,QAAAA,KAAL;AACL,EAAAA,QAAAA,CAAAA,QAAAA,CAAA,YAAA,CAAA,GAAA,CAAA,CAAA,GAAA,YAAA;AACA,EAAAA,QAAAA,CAAAA,QAAAA,CAAA,WAAA,CAAA,GAAA,CAAA,CAAA,GAAA,WAAA;AACA,EAAAA,QAAAA,CAAAA,QAAAA,CAAA,OAAA,CAAA,GAAA,CAAA,CAAA,GAAA,OAAA;AACA,EAAAA,QAAAA,CAAAA,QAAAA,CAAA,OAAA,CAAA,GAAA,CAAA,CAAA,GAAA,OAAA;AAJU,EAAA,OAAAA,QAAAA;AAAA,CAAA,EAAA,OAAA,IAAA,EAAA;;;;"}
@@ -0,0 +1,47 @@
1
+ import { JsonRpcProvider } from '@polkadot-api/json-rpc-provider';
2
+ export { JsonRpcProvider } from '@polkadot-api/json-rpc-provider';
3
+
4
+ declare enum WsEvent {
5
+ CONNECTING = 0,
6
+ CONNECTED = 1,
7
+ ERROR = 2,
8
+ CLOSE = 3
9
+ }
10
+ type WsConnecting = {
11
+ type: WsEvent.CONNECTING;
12
+ uri: string;
13
+ protocols?: string | string[];
14
+ };
15
+ type WsConnected = {
16
+ type: WsEvent.CONNECTED;
17
+ uri: string;
18
+ protocols?: string | string[];
19
+ };
20
+ type WsError = {
21
+ type: WsEvent.ERROR;
22
+ event: any;
23
+ };
24
+ type WsClose = {
25
+ type: WsEvent.CLOSE;
26
+ event: any;
27
+ };
28
+ type StatusChange = WsConnecting | WsConnected | WsError | WsClose;
29
+ type WsJsonRpcProvider = JsonRpcProvider & {
30
+ switch: (uri?: string, protocol?: string[]) => void;
31
+ getStatus: () => StatusChange;
32
+ };
33
+
34
+ type WebSocketClass = typeof WebSocket;
35
+
36
+ declare const getWsProvider: (endpoints: string | Array<string | {
37
+ uri: string;
38
+ protocol: string | string[];
39
+ }>, config?: Partial<{
40
+ onStatusChanged: (status: StatusChange) => void;
41
+ innerEnhancer: (input: JsonRpcProvider) => JsonRpcProvider;
42
+ timeout: number;
43
+ websocketClass: WebSocketClass;
44
+ }>) => WsJsonRpcProvider;
45
+
46
+ export { WsEvent, getWsProvider };
47
+ export type { StatusChange, WebSocketClass, WsClose, WsConnected, WsConnecting, WsError, WsJsonRpcProvider };
@@ -0,0 +1,250 @@
1
+ 'use strict';
2
+
3
+ var jsonRpcProviderProxy = require('@polkadot-api/json-rpc-provider-proxy');
4
+
5
+ var WsEvent = /* @__PURE__ */ ((WsEvent2) => {
6
+ WsEvent2[WsEvent2["CONNECTING"] = 0] = "CONNECTING";
7
+ WsEvent2[WsEvent2["CONNECTED"] = 1] = "CONNECTED";
8
+ WsEvent2[WsEvent2["ERROR"] = 2] = "ERROR";
9
+ WsEvent2[WsEvent2["CLOSE"] = 3] = "CLOSE";
10
+ return WsEvent2;
11
+ })(WsEvent || {});
12
+
13
+ const methods = {};
14
+ ["v1", "unstable"].forEach((version) => {
15
+ methods[`chainHead_${version}_follow`] = "follow";
16
+ methods[`chainHead_${version}_unfollow`] = "unfollow";
17
+ });
18
+ const followEnhancer = (base, forceDisconnect) => {
19
+ const prematureStops = /* @__PURE__ */ new Set();
20
+ const preOpId = /* @__PURE__ */ new Map();
21
+ const onGoing = /* @__PURE__ */ new Set();
22
+ let methodsRequestId;
23
+ const result = (onMsg) => {
24
+ const { send, disconnect } = base((fromProvider) => {
25
+ let nStops = {
26
+ latest: Date.now(),
27
+ count: 0
28
+ };
29
+ const parsed = JSON.parse(fromProvider);
30
+ if ("id" in parsed) {
31
+ const { id, result: result2 } = parsed;
32
+ if (id === methodsRequestId) {
33
+ methodsRequestId = void 0;
34
+ if (result2 && !result2.methods.some((x) => {
35
+ const [group, , name] = x.split("_");
36
+ return group === "chainHead" && name === "follow";
37
+ })) {
38
+ onMsg(fromProvider);
39
+ forceDisconnect();
40
+ return;
41
+ }
42
+ }
43
+ const msg = preOpId.get(id);
44
+ if (msg) {
45
+ preOpId.delete(id);
46
+ if (prematureStops.has(result2)) {
47
+ prematureStops.delete(result2);
48
+ return;
49
+ }
50
+ onGoing.add(result2);
51
+ const currentSize = onGoing.size + preOpId.size;
52
+ if (currentSize > 2)
53
+ console.warn(
54
+ `Too many chainHead follow subscriptions (${currentSize})`
55
+ );
56
+ else if (parsed.error) {
57
+ console.warn(`chainHead follow failed on the ${currentSize} sub`);
58
+ forceDisconnect();
59
+ preOpId.set(id, msg);
60
+ send(msg);
61
+ return;
62
+ }
63
+ }
64
+ } else {
65
+ const { subscription, result: result2 } = parsed.params;
66
+ if (result2?.event === "stop") {
67
+ const diff = Date.now() - nStops.latest;
68
+ nStops.latest += diff;
69
+ nStops.count = diff < 1e3 ? nStops.count + 1 : 1;
70
+ if (onGoing.has(subscription)) onGoing.delete(subscription);
71
+ else prematureStops.add(subscription);
72
+ }
73
+ }
74
+ onMsg(fromProvider);
75
+ if (nStops.count > 2) forceDisconnect();
76
+ });
77
+ return {
78
+ send(toProvider) {
79
+ const parsed = JSON.parse(toProvider);
80
+ if (parsed.method === "rpc_methods") methodsRequestId = parsed.id;
81
+ const method = methods[parsed.method];
82
+ if (method === "follow") {
83
+ preOpId.set(parsed.id, toProvider);
84
+ } else if (method === "unfollow") {
85
+ onGoing.delete(parsed.params[0]);
86
+ }
87
+ send(toProvider);
88
+ },
89
+ disconnect
90
+ };
91
+ };
92
+ return Object.assign(result, {
93
+ cleanup: () => {
94
+ prematureStops.clear();
95
+ preOpId.clear();
96
+ onGoing.clear();
97
+ }
98
+ });
99
+ };
100
+
101
+ const timeoutError = {
102
+ type: WsEvent.ERROR,
103
+ event: { type: "timeout" }
104
+ };
105
+ const noop = () => {
106
+ };
107
+ const defaultConfig = {
108
+ onStatusChanged: noop,
109
+ innerEnhancer: (x) => x,
110
+ timeout: 5e3
111
+ };
112
+ const mapEndpoints = (endpoints) => endpoints.map((x) => typeof x === "string" ? [x] : [x.uri, x.protocol]);
113
+ const getWsProvider = (endpoints, config) => {
114
+ const { onStatusChanged, innerEnhancer, timeout } = {
115
+ ...defaultConfig,
116
+ ...config
117
+ };
118
+ const actualEndpoints = mapEndpoints(
119
+ Array.isArray(endpoints) ? endpoints : [endpoints]
120
+ );
121
+ const WebsocketClass = config?.websocketClass ?? globalThis.WebSocket;
122
+ if (!WebsocketClass) throw new Error("Missing WebSocket class");
123
+ let idx = 0;
124
+ let status;
125
+ let switchTo = null;
126
+ let disconnect = noop;
127
+ let outerCleanup = noop;
128
+ const result = followEnhancer(
129
+ jsonRpcProviderProxy.getSyncProvider(async () => {
130
+ const [uri, protocols] = switchTo || actualEndpoints[idx++ % endpoints.length];
131
+ switchTo = null;
132
+ const socket = new WebsocketClass(uri, protocols);
133
+ const forceSocketClose = () => {
134
+ try {
135
+ socket.addEventListener("error", noop, { once: true });
136
+ socket.close();
137
+ } catch {
138
+ }
139
+ };
140
+ onStatusChanged(
141
+ status = {
142
+ type: WsEvent.CONNECTING,
143
+ uri,
144
+ protocols
145
+ }
146
+ );
147
+ await new Promise((resolve, reject) => {
148
+ const onOpen = () => {
149
+ initialCleanup();
150
+ resolve();
151
+ };
152
+ const onError = (e) => {
153
+ initialCleanup();
154
+ if (e == null) forceSocketClose();
155
+ console.error(
156
+ `Unable to connect to ${uri}${protocols ? ", protocols: " + protocols : ""}`
157
+ );
158
+ onStatusChanged(
159
+ status = {
160
+ type: e ? WsEvent.ERROR : WsEvent.CLOSE,
161
+ event: e
162
+ }
163
+ );
164
+ setTimeout(reject, e ? 300 : 0, e);
165
+ };
166
+ const timeoutToken = timeout !== Infinity ? setTimeout(() => {
167
+ initialCleanup();
168
+ forceSocketClose();
169
+ onStatusChanged(status = timeoutError);
170
+ reject(timeoutError.event);
171
+ }, timeout) : void 0;
172
+ const initialCleanup = () => {
173
+ clearTimeout(timeoutToken);
174
+ socket.removeEventListener("error", onError);
175
+ socket.removeEventListener("open", onOpen);
176
+ };
177
+ socket.addEventListener("open", onOpen);
178
+ socket.addEventListener("error", onError);
179
+ disconnect = () => {
180
+ onError(null);
181
+ };
182
+ });
183
+ onStatusChanged(
184
+ status = {
185
+ type: WsEvent.CONNECTED,
186
+ uri,
187
+ protocols
188
+ }
189
+ );
190
+ let _onInnerMessage;
191
+ const inner = innerEnhancer((onInnerMessage) => {
192
+ _onInnerMessage = onInnerMessage;
193
+ return {
194
+ send: (m) => {
195
+ socket.send(m);
196
+ },
197
+ disconnect: () => {
198
+ disconnect();
199
+ }
200
+ };
201
+ });
202
+ return (onMessage, onHalt) => {
203
+ const connection = inner(onMessage);
204
+ const _onMessage = (e) => {
205
+ if (typeof e.data === "string") _onInnerMessage(e.data);
206
+ };
207
+ const innerHalt = (reason) => (e) => {
208
+ console.warn(`WS halt (${reason})`);
209
+ onStatusChanged(
210
+ status = {
211
+ type: reason,
212
+ event: e
213
+ }
214
+ );
215
+ onHalt();
216
+ };
217
+ const onError = innerHalt(WsEvent.ERROR);
218
+ const onClose = innerHalt(WsEvent.CLOSE);
219
+ socket.addEventListener("message", _onMessage);
220
+ socket.addEventListener("error", onError);
221
+ socket.addEventListener("close", onClose);
222
+ disconnect = (withHalt) => {
223
+ outerCleanup();
224
+ disconnect = noop;
225
+ socket.removeEventListener("message", _onMessage);
226
+ socket.removeEventListener("error", onError);
227
+ socket.removeEventListener("close", onClose);
228
+ forceSocketClose();
229
+ if (withHalt) onClose({});
230
+ };
231
+ return connection;
232
+ };
233
+ }),
234
+ () => {
235
+ switchFn();
236
+ }
237
+ );
238
+ outerCleanup = result.cleanup;
239
+ delete result.cleanup;
240
+ const switchFn = (...args) => {
241
+ if (status.type === WsEvent.CLOSE) return;
242
+ if (args.length) switchTo = args;
243
+ if (status.type !== WsEvent.ERROR) disconnect(true);
244
+ };
245
+ return Object.assign(result, { switch: switchFn, getStatus: () => status });
246
+ };
247
+
248
+ exports.WsEvent = WsEvent;
249
+ exports.getWsProvider = getWsProvider;
250
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sources":["../../src/types-common.ts","../../src/follow-enhancer.ts","../../src/default-provider.ts"],"sourcesContent":["import { JsonRpcProvider } from \"@polkadot-api/json-rpc-provider\"\n\nexport enum WsEvent {\n CONNECTING,\n CONNECTED,\n ERROR,\n CLOSE,\n}\n\nexport type WsConnecting = {\n type: WsEvent.CONNECTING\n uri: string\n protocols?: string | string[]\n}\nexport type WsConnected = {\n type: WsEvent.CONNECTED\n uri: string\n protocols?: string | string[]\n}\nexport type WsError = {\n type: WsEvent.ERROR\n event: any\n}\nexport type WsClose = {\n type: WsEvent.CLOSE\n event: any\n}\nexport type StatusChange = WsConnecting | WsConnected | WsError | WsClose\nexport type WsJsonRpcProvider = JsonRpcProvider & {\n switch: (uri?: string, protocol?: string[]) => void\n getStatus: () => StatusChange\n}\nexport { type JsonRpcProvider }\n","import { JsonRpcProvider } from \"@polkadot-api/json-rpc-provider\"\n\nconst methods: Record<string, \"follow\" | \"unfollow\"> = {}\n;[\"v1\", \"unstable\"].forEach((version) => {\n methods[`chainHead_${version}_follow`] = \"follow\"\n methods[`chainHead_${version}_unfollow`] = \"unfollow\"\n})\n\nexport const followEnhancer: (\n input: JsonRpcProvider,\n forceDisconnect: () => void,\n) => JsonRpcProvider & {\n cleanup: () => void\n} = (base, forceDisconnect) => {\n const prematureStops = new Set<string>()\n const preOpId = new Map<string, string>()\n const onGoing = new Set<string>()\n let methodsRequestId: string | undefined\n\n const result: JsonRpcProvider = (onMsg) => {\n const { send, disconnect } = base((fromProvider) => {\n let nStops: { latest: number; count: number } = {\n latest: Date.now(),\n count: 0,\n }\n const parsed = JSON.parse(fromProvider)\n // it's a response\n if (\"id\" in parsed) {\n const { id, result } = parsed\n if (id === methodsRequestId) {\n methodsRequestId = undefined\n if (\n result &&\n !(result.methods as string[]).some((x) => {\n const [group, , name] = x.split(\"_\")\n return group === \"chainHead\" && name === \"follow\"\n })\n ) {\n onMsg(fromProvider)\n forceDisconnect()\n return\n }\n }\n\n const msg = preOpId.get(id)\n if (msg) {\n preOpId.delete(id)\n if (prematureStops.has(result)) {\n prematureStops.delete(result)\n return\n }\n\n onGoing.add(result)\n const currentSize = onGoing.size + preOpId.size\n if (currentSize > 2)\n console.warn(\n `Too many chainHead follow subscriptions (${currentSize})`,\n )\n else if (parsed.error) {\n console.warn(`chainHead follow failed on the ${currentSize} sub`)\n forceDisconnect()\n preOpId.set(id, msg)\n send(msg)\n return\n }\n }\n } else {\n // it's a notifiaction\n const { subscription, result } = (parsed as any).params\n if (result?.event === \"stop\") {\n const diff = Date.now() - nStops.latest\n nStops.latest += diff\n nStops.count = diff < 1000 ? nStops.count + 1 : 1\n\n if (onGoing.has(subscription)) onGoing.delete(subscription)\n else prematureStops.add(subscription)\n }\n }\n\n onMsg(fromProvider)\n if (nStops.count > 2) forceDisconnect()\n })\n\n return {\n send(toProvider) {\n const parsed = JSON.parse(toProvider)\n if (parsed.method === \"rpc_methods\") methodsRequestId = parsed.id\n\n const method = methods[parsed.method]\n if (method === \"follow\") {\n preOpId.set(parsed.id, toProvider)\n } else if (method === \"unfollow\") {\n onGoing.delete(parsed.params[0])\n }\n send(toProvider)\n },\n disconnect,\n }\n }\n\n return Object.assign(result, {\n cleanup: () => {\n prematureStops.clear()\n preOpId.clear()\n onGoing.clear()\n },\n })\n}\n","import { getSyncProvider } from \"@polkadot-api/json-rpc-provider-proxy\"\nimport { StatusChange, WsJsonRpcProvider, WsEvent } from \"./types-common\"\nimport { followEnhancer } from \"./follow-enhancer\"\nimport { JsonRpcProvider } from \"@polkadot-api/json-rpc-provider\"\nimport { WebSocketClass } from \"./types-new\"\n\nconst timeoutError: StatusChange = {\n type: WsEvent.ERROR,\n event: { type: \"timeout\" },\n}\n\nexport const noop = () => {}\n\nexport const defaultConfig: {\n onStatusChanged: (status: StatusChange) => void\n innerEnhancer: (input: JsonRpcProvider) => JsonRpcProvider\n timeout: number\n} = {\n onStatusChanged: noop,\n innerEnhancer: (x: JsonRpcProvider) => x,\n timeout: 5_000,\n}\n\nexport const mapEndpoints = (\n endpoints: Array<string | { uri: string; protocol: string | string[] }>,\n): Array<[string, string | string[]] | [string]> =>\n endpoints.map((x) => (typeof x === \"string\" ? [x] : [x.uri, x.protocol]))\n\nexport const getWsProvider = (\n endpoints:\n | string\n | Array<string | { uri: string; protocol: string | string[] }>,\n config?: Partial<{\n onStatusChanged: (status: StatusChange) => void\n innerEnhancer: (input: JsonRpcProvider) => JsonRpcProvider\n timeout: number\n websocketClass: WebSocketClass\n }>,\n): WsJsonRpcProvider => {\n const { onStatusChanged, innerEnhancer, timeout } = {\n ...defaultConfig,\n ...config,\n }\n const actualEndpoints = mapEndpoints(\n Array.isArray(endpoints) ? endpoints : [endpoints],\n )\n const WebsocketClass = config?.websocketClass ?? globalThis.WebSocket\n if (!WebsocketClass) throw new Error(\"Missing WebSocket class\")\n\n let idx = 0\n let status: StatusChange\n let switchTo: [string] | [string, string | string[]] | null = null\n let disconnect: (withHalt?: boolean) => void = noop\n\n let outerCleanup: () => void = noop\n const result = followEnhancer(\n getSyncProvider(async () => {\n const [uri, protocols] =\n switchTo || actualEndpoints[idx++ % endpoints.length]\n switchTo = null\n const socket = new WebsocketClass(uri, protocols)\n const forceSocketClose = () => {\n try {\n socket.addEventListener(\"error\", noop, { once: true })\n socket.close()\n } catch {}\n }\n onStatusChanged(\n (status = {\n type: WsEvent.CONNECTING,\n uri,\n protocols,\n }),\n )\n\n await new Promise<void>((resolve, reject) => {\n const onOpen = () => {\n initialCleanup()\n resolve()\n }\n\n const onError = (e: Event | null) => {\n initialCleanup()\n if (e == null) forceSocketClose()\n console.error(\n `Unable to connect to ${uri}${\n protocols ? \", protocols: \" + protocols : \"\"\n }`,\n )\n onStatusChanged(\n (status = {\n type: e ? WsEvent.ERROR : WsEvent.CLOSE,\n event: e,\n }),\n )\n setTimeout(reject, e ? 300 : 0, e)\n }\n\n const timeoutToken =\n timeout !== Infinity\n ? setTimeout(() => {\n initialCleanup()\n forceSocketClose()\n onStatusChanged((status = timeoutError))\n reject(timeoutError.event)\n }, timeout)\n : undefined\n\n const initialCleanup = () => {\n clearTimeout(timeoutToken)\n socket.removeEventListener(\"error\", onError)\n socket.removeEventListener(\"open\", onOpen)\n }\n socket.addEventListener(\"open\", onOpen)\n socket.addEventListener(\"error\", onError)\n disconnect = () => {\n onError(null)\n }\n })\n\n onStatusChanged(\n (status = {\n type: WsEvent.CONNECTED,\n uri,\n protocols,\n }),\n )\n\n let _onInnerMessage: (msg: string) => void\n const inner = innerEnhancer((onInnerMessage) => {\n _onInnerMessage = onInnerMessage\n return {\n send: (m) => {\n socket.send(m)\n },\n disconnect: () => {\n disconnect()\n },\n }\n })\n\n return (onMessage, onHalt) => {\n const connection = inner(onMessage)\n const _onMessage = (e: MessageEvent) => {\n if (typeof e.data === \"string\") _onInnerMessage(e.data)\n }\n const innerHalt =\n (reason: WsEvent.CLOSE | WsEvent.ERROR) => (e: any) => {\n console.warn(`WS halt (${reason})`)\n onStatusChanged(\n (status = {\n type: reason,\n event: e,\n }),\n )\n onHalt()\n }\n const onError = innerHalt(WsEvent.ERROR)\n const onClose = innerHalt(WsEvent.CLOSE)\n\n socket.addEventListener(\"message\", _onMessage)\n socket.addEventListener(\"error\", onError)\n socket.addEventListener(\"close\", onClose)\n disconnect = (withHalt) => {\n outerCleanup()\n disconnect = noop\n socket.removeEventListener(\"message\", _onMessage)\n socket.removeEventListener(\"error\", onError)\n socket.removeEventListener(\"close\", onClose)\n forceSocketClose()\n if (withHalt) onClose({})\n }\n\n return connection\n }\n }),\n () => {\n switchFn()\n },\n )\n outerCleanup = result.cleanup\n delete (result as any).cleanup\n\n const switchFn: WsJsonRpcProvider[\"switch\"] = (...args) => {\n if (status.type === WsEvent.CLOSE) return\n if (args.length) switchTo = args as any\n if (status.type !== WsEvent.ERROR) disconnect(true)\n }\n\n return Object.assign(result, { switch: switchFn, getStatus: () => status })\n}\n"],"names":["WsEvent","result","getSyncProvider"],"mappings":"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"}