@xelis/sdk 0.5.2 → 0.5.4

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.
@@ -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
- close(code?: number | undefined, data?: string | Buffer | undefined): void;
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,52 @@ 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
- if (this.socket && this.socket.readyState !== WebSocket.CLOSED) {
22
+ // force disconnect if already connected
23
+ if (this.socket && this.socket.readyState === WebSocket.OPEN) {
21
24
  this.socket.close();
22
25
  }
26
+ this.events = {};
27
+ this.connectionTries = 0;
23
28
  return new Promise((resolve, reject) => {
24
29
  this.socket = new WebSocket(endpoint);
25
30
  this.endpoint = endpoint;
26
31
  this.socket.addEventListener(`open`, (event) => {
27
- this.connected = true;
28
32
  resolve(event);
29
33
  });
30
- this.socket.addEventListener(`close`, () => {
31
- this.connected = false;
32
- reject();
34
+ this.socket.addEventListener(`close`, (event) => {
35
+ if (this.reconnectOnConnectionLoss && !event.wasClean) {
36
+ this.tryReconnect();
37
+ reject(new Error(`Unhandled close. Reconnecting...`));
38
+ }
39
+ else {
40
+ reject(event);
41
+ }
33
42
  });
34
43
  this.socket.addEventListener(`error`, (err) => {
35
- this.connected = false;
36
44
  reject(err);
37
45
  });
38
46
  });
39
47
  }
40
- close(code, data) {
41
- this.socket && this.socket.close(code, data);
42
- }
43
- onClose(cb) {
44
- if (!this.socket)
48
+ tryReconnect() {
49
+ this.connectionTries++;
50
+ if (this.connectionTries > this.maxConnectionTries) {
45
51
  return;
46
- this.socket.addEventListener(`close`, (event) => {
47
- cb(event);
52
+ }
53
+ this.socket = new WebSocket(this.endpoint);
54
+ this.socket.addEventListener(`open`, () => {
55
+ this.connectionTries = 0;
48
56
  });
49
- }
50
- onError(cb) {
51
- if (!this.socket)
52
- return;
53
- this.socket.addEventListener(`error`, (err) => {
54
- cb(err);
57
+ this.socket.addEventListener(`close`, (event) => {
58
+ this.tryReconnect();
55
59
  });
56
60
  }
57
61
  clearEvent(event) {
@@ -101,7 +105,7 @@ export class WS {
101
105
  this.events[event].listeners.push(onMessage);
102
106
  }
103
107
  else {
104
- // important if multiple listenEvent are called without await atleast we store listener before getting id
108
+ // important if multiple listenEvent are called without await at least we store listener before getting id
105
109
  this.events[event] = { listeners: [onMessage] };
106
110
  const [err, res] = await to(this.call(`subscribe`, { notify: event }));
107
111
  if (err) {
@@ -121,12 +125,19 @@ export class WS {
121
125
  break;
122
126
  }
123
127
  }
128
+ // no more listener so we unsubscribe from daemon websocket if socket still open
124
129
  if (listeners.length === 0) {
125
- this.events[event].unsubscribeTimeoutId = setTimeout(async () => {
126
- // no more listener so we unsubscribe from daemon websocket
127
- this.call(`unsubscribe`, { notify: event });
130
+ if (this.socket && this.socket.readyState === WebSocket.OPEN) {
131
+ // we use a grace period to unsubscribe (mostly because of react useEffect and avoid unecessary subscribe)
132
+ this.events[event].unsubscribeTimeoutId = setTimeout(async () => {
133
+ this.call(`unsubscribe`, { notify: event });
134
+ Reflect.deleteProperty(this.events, event);
135
+ }, this.unsubscribeSuspense);
136
+ }
137
+ else {
138
+ // socket is closed so we don't send unsubscribe and no grace period delete right away
128
139
  Reflect.deleteProperty(this.events, event);
129
- }, this.unsubscribeSuspense);
140
+ }
130
141
  }
131
142
  }
132
143
  this.socket && this.socket.removeEventListener(`message`, onMessage);
@@ -157,7 +168,9 @@ export class WS {
157
168
  this.socket && this.socket.removeEventListener(`message`, onMessage);
158
169
  reject(new Error(`timeout`));
159
170
  }, this.timeout);
160
- this.socket && this.socket.send(data);
171
+ if (this.socket && this.socket.readyState === WebSocket.OPEN) {
172
+ this.socket.send(data);
173
+ }
161
174
  });
162
175
  }
163
176
  dataCall(method, params) {
package/package.json CHANGED
@@ -1,5 +1,5 @@
1
1
  {
2
- "version": "0.5.2",
2
+ "version": "0.5.4",
3
3
  "name": "@xelis/sdk",
4
4
  "description": "Xelis software development kit for JS",
5
5
  "repository": {
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
- loading: boolean;
8
- err: any;
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
- onConnected: () => void;
17
+ onLoad: () => void;
18
18
  onData: (msgEvent: MessageEvent, data?: T, err?: Error | undefined) => void;
19
19
  }
20
- export declare const useNodeSocketSubscribe: ({ event, onConnected, onData }: NodeSocketSubscribeProps<any>, dependencies: React.DependencyList) => void;
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
- connected: false,
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 [loading, setLoading] = useState(true);
14
- const [connected, setConnected] = useState(false);
13
+ const [readyState, setReadyState] = useState(INITIATING);
15
14
  const [err, setErr] = useState();
16
15
  useEffect(() => {
17
- const load = async () => {
18
- setLoading(true);
19
- const [err, res] = await to(daemon.connect(endpoint));
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
- load();
22
+ connect();
34
23
  }, [endpoint]);
35
- return React.createElement(Context.Provider, { value: { daemon, loading, err, connected } }, children);
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, onConnected, onData }, dependencies) => {
59
+ export const useNodeSocketSubscribe = ({ event, onLoad, onData }, dependencies) => {
38
60
  const nodeSocket = useNodeSocket();
39
61
  useEffect(() => {
40
- if (!nodeSocket.connected)
62
+ if (nodeSocket.readyState !== WebSocket.OPEN)
41
63
  return;
42
- if (typeof onConnected === `function`)
43
- onConnected();
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();