fix: fix reconnection logic
This commit is contained in:
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "tap-sdk-js",
|
||||
"version": "1.1.3",
|
||||
"version": "1.2.3",
|
||||
"description": "",
|
||||
"scripts": {
|
||||
"build": "tsc",
|
||||
|
||||
@@ -12,6 +12,7 @@ export type WSEvent = {
|
||||
export type WSState = {
|
||||
socket: WebSocket | null;
|
||||
connected: boolean;
|
||||
connecting: boolean;
|
||||
retries: number;
|
||||
emitter: EventEmitter<WSEvent>;
|
||||
};
|
||||
@@ -28,49 +29,87 @@ export function createWebSocketManager(config: WSConfig) {
|
||||
const state: WSState = {
|
||||
socket: null,
|
||||
connected: false,
|
||||
connecting: false,
|
||||
retries: 0,
|
||||
emitter: new EventEmitter(),
|
||||
};
|
||||
|
||||
function connect() {
|
||||
return new Promise<void>((resolve) => {
|
||||
state.connecting = true;
|
||||
return new Promise<void>((resolve, reject) => {
|
||||
const socket = new WebSocket(config.url);
|
||||
socket.on("open", () => {
|
||||
|
||||
const onOpen = () => {
|
||||
socket.off("open", onOpen);
|
||||
socket.off("error", onError);
|
||||
socket.off("close", onClose);
|
||||
|
||||
state.socket = socket;
|
||||
state.connected = true;
|
||||
state.connecting = false;
|
||||
state.retries = 0;
|
||||
state.emitter.emit("open");
|
||||
resolve();
|
||||
});
|
||||
|
||||
// Re-attach persistent listeners
|
||||
socket.on("message", (msg) => {
|
||||
state.emitter.emit("message", msg.toString());
|
||||
});
|
||||
|
||||
socket.on("error", (err) => {
|
||||
if (state.connected) state.emitter.emit("error", err);
|
||||
else state.emitter.emit("connectionError", err);
|
||||
state.emitter.emit("error", err);
|
||||
});
|
||||
|
||||
socket.on("close", () => {
|
||||
state.emitter.emit("close");
|
||||
state.connected = false;
|
||||
state.emitter.emit("close");
|
||||
retry();
|
||||
});
|
||||
};
|
||||
|
||||
const onError = (err: Error) => {
|
||||
socket.off("open", onOpen);
|
||||
socket.off("error", onError);
|
||||
socket.off("close", onClose);
|
||||
state.connecting = false;
|
||||
state.emitter.emit("connectionError", err);
|
||||
reject(err);
|
||||
};
|
||||
|
||||
const onClose = () => {
|
||||
socket.off("open", onOpen);
|
||||
socket.off("error", onError);
|
||||
socket.off("close", onClose);
|
||||
state.connecting = false;
|
||||
const err = new Error("Connection closed before open");
|
||||
state.emitter.emit("connectionError", err);
|
||||
reject(err);
|
||||
};
|
||||
|
||||
socket.on("open", onOpen);
|
||||
socket.on("error", onError);
|
||||
socket.on("close", onClose);
|
||||
});
|
||||
}
|
||||
|
||||
function retry() {
|
||||
if (state.connected || state.connecting) return;
|
||||
state.retries++;
|
||||
setTimeout(
|
||||
() => {
|
||||
connect().catch(() => retry());
|
||||
state.retries++;
|
||||
start();
|
||||
},
|
||||
backoffDelay(state.retries, config.backoffBaseMs),
|
||||
backoffDelay(state.retries - 1, config.backoffBaseMs),
|
||||
);
|
||||
}
|
||||
|
||||
async function start() {
|
||||
await connect().catch(() => retry());
|
||||
if (state.connected || state.connecting) return;
|
||||
try {
|
||||
await connect();
|
||||
} catch (e) {
|
||||
retry();
|
||||
}
|
||||
}
|
||||
|
||||
function send(message: string) {
|
||||
|
||||
@@ -141,6 +141,7 @@ export function createClient(userConfig: ZakoTapOptions): ZakoTapClientHandle {
|
||||
socket.events.on("connectionError", (err) =>
|
||||
emitter.emit("connectionError", err),
|
||||
);
|
||||
socket.events.on("close", () => emitter.emit("close"));
|
||||
|
||||
socket.events.on("message", (content) => {
|
||||
const data = JSON.parse(content);
|
||||
|
||||
@@ -26,6 +26,11 @@ export type TapEvents = {
|
||||
* Event called on initial connection error.
|
||||
*/
|
||||
connectionError: (e: Error) => any;
|
||||
|
||||
/**
|
||||
* Event called on connection close.
|
||||
*/
|
||||
close: () => any;
|
||||
};
|
||||
|
||||
export const createEmitter = () => new EventEmitter<TapEvents>();
|
||||
|
||||
Reference in New Issue
Block a user