@xelis/sdk 0.5.2 → 0.5.3
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/lib/websocket.d.ts +4 -5
- package/lib/websocket.js +23 -21
- package/package.json +1 -1
- package/react/daemon.d.ts +5 -5
- package/react/daemon.js +46 -26
package/lib/websocket.d.ts
CHANGED
|
@@ -1,4 +1,3 @@
|
|
|
1
|
-
/// <reference types="node" />
|
|
2
1
|
import { MessageEvent } from 'ws';
|
|
3
2
|
import WebSocket from 'isomorphic-ws';
|
|
4
3
|
import { RPCResponse } from './types';
|
|
@@ -6,14 +5,14 @@ export declare class WS {
|
|
|
6
5
|
endpoint: string;
|
|
7
6
|
socket?: WebSocket;
|
|
8
7
|
timeout: number;
|
|
9
|
-
connected: boolean;
|
|
10
8
|
unsubscribeSuspense: number;
|
|
9
|
+
reconnectOnConnectionLoss: boolean;
|
|
10
|
+
maxConnectionTries: number;
|
|
11
|
+
connectionTries: number;
|
|
11
12
|
private events;
|
|
12
13
|
constructor();
|
|
13
14
|
connect(endpoint: string): Promise<unknown>;
|
|
14
|
-
|
|
15
|
-
onClose(cb: (event: WebSocket.CloseEvent) => void): void;
|
|
16
|
-
onError(cb: (err: WebSocket.ErrorEvent) => void): void;
|
|
15
|
+
tryReconnect(): void;
|
|
17
16
|
private clearEvent;
|
|
18
17
|
closeAllListens(event: string): Promise<void>;
|
|
19
18
|
listenEvent<T>(event: string, onData: (msgEvent: MessageEvent, data?: T, err?: Error) => void): Promise<() => Promise<void>>;
|
package/lib/websocket.js
CHANGED
|
@@ -10,48 +10,48 @@ function createRequestMethod(method, params) {
|
|
|
10
10
|
}
|
|
11
11
|
export class WS {
|
|
12
12
|
constructor() {
|
|
13
|
+
this.connectionTries = 0;
|
|
13
14
|
this.endpoint = "";
|
|
14
15
|
this.timeout = 3000;
|
|
15
|
-
this.connected = false;
|
|
16
16
|
this.events = {};
|
|
17
17
|
this.unsubscribeSuspense = 1000;
|
|
18
|
+
this.maxConnectionTries = 3;
|
|
19
|
+
this.reconnectOnConnectionLoss = true;
|
|
18
20
|
}
|
|
19
21
|
connect(endpoint) {
|
|
20
|
-
|
|
22
|
+
// force disconnect if already connected
|
|
23
|
+
if (this.socket && this.socket.readyState === WebSocket.OPEN) {
|
|
21
24
|
this.socket.close();
|
|
22
25
|
}
|
|
26
|
+
this.connectionTries = 0;
|
|
23
27
|
return new Promise((resolve, reject) => {
|
|
24
28
|
this.socket = new WebSocket(endpoint);
|
|
25
29
|
this.endpoint = endpoint;
|
|
26
30
|
this.socket.addEventListener(`open`, (event) => {
|
|
27
|
-
this.connected = true;
|
|
28
31
|
resolve(event);
|
|
29
32
|
});
|
|
30
|
-
this.socket.addEventListener(`close`, () => {
|
|
31
|
-
this.
|
|
32
|
-
|
|
33
|
+
this.socket.addEventListener(`close`, (event) => {
|
|
34
|
+
if (this.reconnectOnConnectionLoss && !event.wasClean) {
|
|
35
|
+
this.tryReconnect();
|
|
36
|
+
reject(new Error(`Reconnecting.`));
|
|
37
|
+
}
|
|
38
|
+
else {
|
|
39
|
+
reject(event);
|
|
40
|
+
}
|
|
33
41
|
});
|
|
34
42
|
this.socket.addEventListener(`error`, (err) => {
|
|
35
|
-
this.connected = false;
|
|
36
43
|
reject(err);
|
|
37
44
|
});
|
|
38
45
|
});
|
|
39
46
|
}
|
|
40
|
-
|
|
41
|
-
this.
|
|
42
|
-
|
|
43
|
-
onClose(cb) {
|
|
44
|
-
if (!this.socket)
|
|
47
|
+
tryReconnect() {
|
|
48
|
+
this.connectionTries++;
|
|
49
|
+
if (this.connectionTries > this.maxConnectionTries) {
|
|
45
50
|
return;
|
|
51
|
+
}
|
|
52
|
+
this.socket = new WebSocket(this.endpoint);
|
|
46
53
|
this.socket.addEventListener(`close`, (event) => {
|
|
47
|
-
|
|
48
|
-
});
|
|
49
|
-
}
|
|
50
|
-
onError(cb) {
|
|
51
|
-
if (!this.socket)
|
|
52
|
-
return;
|
|
53
|
-
this.socket.addEventListener(`error`, (err) => {
|
|
54
|
-
cb(err);
|
|
54
|
+
this.tryReconnect();
|
|
55
55
|
});
|
|
56
56
|
}
|
|
57
57
|
clearEvent(event) {
|
|
@@ -157,7 +157,9 @@ export class WS {
|
|
|
157
157
|
this.socket && this.socket.removeEventListener(`message`, onMessage);
|
|
158
158
|
reject(new Error(`timeout`));
|
|
159
159
|
}, this.timeout);
|
|
160
|
-
this.socket && this.socket.
|
|
160
|
+
if (this.socket && this.socket.readyState === WebSocket.OPEN) {
|
|
161
|
+
this.socket.send(data);
|
|
162
|
+
}
|
|
161
163
|
});
|
|
162
164
|
}
|
|
163
165
|
dataCall(method, params) {
|
package/package.json
CHANGED
package/react/daemon.d.ts
CHANGED
|
@@ -2,11 +2,11 @@ import React, { PropsWithChildren } from 'react';
|
|
|
2
2
|
import { MessageEvent } from 'ws';
|
|
3
3
|
import DaemonWS from '../daemon/websocket';
|
|
4
4
|
import { RPCEvent } from '../daemon/types';
|
|
5
|
+
export declare const INITIATING = -1;
|
|
5
6
|
interface NodeSocket {
|
|
6
7
|
daemon: DaemonWS;
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
connected: boolean;
|
|
8
|
+
err?: Error;
|
|
9
|
+
readyState: number;
|
|
10
10
|
}
|
|
11
11
|
interface NodeSocketProviderProps {
|
|
12
12
|
endpoint: string;
|
|
@@ -14,9 +14,9 @@ interface NodeSocketProviderProps {
|
|
|
14
14
|
export declare const NodeSocketProvider: (props: PropsWithChildren<NodeSocketProviderProps>) => React.JSX.Element;
|
|
15
15
|
interface NodeSocketSubscribeProps<T> {
|
|
16
16
|
event: RPCEvent;
|
|
17
|
-
|
|
17
|
+
onLoad: () => void;
|
|
18
18
|
onData: (msgEvent: MessageEvent, data?: T, err?: Error | undefined) => void;
|
|
19
19
|
}
|
|
20
|
-
export declare const useNodeSocketSubscribe: ({ event,
|
|
20
|
+
export declare const useNodeSocketSubscribe: ({ event, onLoad, onData }: NodeSocketSubscribeProps<any>, dependencies: React.DependencyList) => void;
|
|
21
21
|
export declare const useNodeSocket: () => NodeSocket;
|
|
22
22
|
export default useNodeSocket;
|
package/react/daemon.js
CHANGED
|
@@ -2,49 +2,69 @@ import React, { createContext, useContext, useEffect, useState } from 'react';
|
|
|
2
2
|
import to from 'await-to-js';
|
|
3
3
|
import DaemonWS from '../daemon/websocket';
|
|
4
4
|
const daemon = new DaemonWS();
|
|
5
|
+
export const INITIATING = -1;
|
|
5
6
|
const Context = createContext({
|
|
6
|
-
|
|
7
|
-
err: null,
|
|
8
|
-
loading: false,
|
|
7
|
+
err: undefined,
|
|
9
8
|
daemon,
|
|
9
|
+
readyState: INITIATING
|
|
10
10
|
});
|
|
11
11
|
export const NodeSocketProvider = (props) => {
|
|
12
12
|
const { children, endpoint } = props;
|
|
13
|
-
const [
|
|
14
|
-
const [connected, setConnected] = useState(false);
|
|
13
|
+
const [readyState, setReadyState] = useState(INITIATING);
|
|
15
14
|
const [err, setErr] = useState();
|
|
16
15
|
useEffect(() => {
|
|
17
|
-
const
|
|
18
|
-
|
|
19
|
-
const [err,
|
|
16
|
+
const connect = async () => {
|
|
17
|
+
setErr(undefined);
|
|
18
|
+
const [err, _] = await to(daemon.connect(endpoint));
|
|
20
19
|
if (err)
|
|
21
20
|
setErr(err);
|
|
22
|
-
else
|
|
23
|
-
setConnected(true);
|
|
24
|
-
setLoading(false);
|
|
25
|
-
daemon.onError((err) => {
|
|
26
|
-
setErr(err);
|
|
27
|
-
setConnected(false);
|
|
28
|
-
});
|
|
29
|
-
daemon.onClose(() => {
|
|
30
|
-
setConnected(false);
|
|
31
|
-
});
|
|
32
21
|
};
|
|
33
|
-
|
|
22
|
+
connect();
|
|
34
23
|
}, [endpoint]);
|
|
35
|
-
|
|
24
|
+
useEffect(() => {
|
|
25
|
+
if (!daemon.socket)
|
|
26
|
+
return;
|
|
27
|
+
setReadyState(daemon.socket.readyState);
|
|
28
|
+
const onOpen = () => {
|
|
29
|
+
if (!daemon.socket)
|
|
30
|
+
return;
|
|
31
|
+
setReadyState(daemon.socket.readyState);
|
|
32
|
+
setErr(undefined);
|
|
33
|
+
};
|
|
34
|
+
const onClose = (event) => {
|
|
35
|
+
if (!daemon.socket)
|
|
36
|
+
return;
|
|
37
|
+
setReadyState(daemon.socket.readyState);
|
|
38
|
+
setErr(new Error(event.reason));
|
|
39
|
+
};
|
|
40
|
+
const onError = (err) => {
|
|
41
|
+
if (!daemon.socket)
|
|
42
|
+
return;
|
|
43
|
+
setReadyState(daemon.socket.readyState);
|
|
44
|
+
setErr(new Error(err.message));
|
|
45
|
+
};
|
|
46
|
+
daemon.socket.addEventListener(`open`, onOpen);
|
|
47
|
+
daemon.socket.addEventListener(`close`, onClose);
|
|
48
|
+
daemon.socket.addEventListener(`error`, onError);
|
|
49
|
+
return () => {
|
|
50
|
+
if (!daemon.socket)
|
|
51
|
+
return;
|
|
52
|
+
daemon.socket.removeEventListener(`open`, onOpen);
|
|
53
|
+
daemon.socket.removeEventListener(`close`, onClose);
|
|
54
|
+
daemon.socket.removeEventListener(`error`, onError);
|
|
55
|
+
};
|
|
56
|
+
}, [daemon.socket]);
|
|
57
|
+
return React.createElement(Context.Provider, { value: { daemon, err, readyState } }, children);
|
|
36
58
|
};
|
|
37
|
-
export const useNodeSocketSubscribe = ({ event,
|
|
59
|
+
export const useNodeSocketSubscribe = ({ event, onLoad, onData }, dependencies) => {
|
|
38
60
|
const nodeSocket = useNodeSocket();
|
|
39
61
|
useEffect(() => {
|
|
40
|
-
if (
|
|
62
|
+
if (nodeSocket.readyState !== WebSocket.OPEN)
|
|
41
63
|
return;
|
|
42
|
-
if (typeof
|
|
43
|
-
|
|
64
|
+
if (typeof onLoad === `function`)
|
|
65
|
+
onLoad();
|
|
44
66
|
let closeEvent;
|
|
45
67
|
const listen = async () => {
|
|
46
|
-
if (!nodeSocket.daemon)
|
|
47
|
-
return;
|
|
48
68
|
closeEvent = await nodeSocket.daemon.listenEvent(event, onData);
|
|
49
69
|
};
|
|
50
70
|
listen();
|