Files
tether-ts/src/utils/websocket.ts
T
2026-04-21 22:26:47 -05:00

76 lines
2.2 KiB
TypeScript

export class WebSocketHandler {
private ws: WebSocket | null = null;
private url: string = '';
private subscribedQueries = new Map<string, (data: any) => void>();
private onOpen: () => void = () => {};
private onClose: () => void = () => {};
private reconnectAttempts: number = 0;
private maxReconnectAttempts: number = 5;
private reconnectInterval: number = 1000;
private sendQueue: string[] = [];
startConnection = (url: string) => {
this.url = url;
this.ws = new WebSocket(url);
const ws = this.ws;
ws.onopen = () => {
console.log('Connected to Tether');
this.onOpen();
if (this.sendQueue.length > 0) {
this.sendQueue.forEach(message => this.ws?.send(message));
this.sendQueue = [];
}
this.reconnectAttempts = 0;
};
ws.onmessage = (event: MessageEvent) => {
const data = JSON.parse(String(event.data)) as {
type: string;
query?: string;
data?: unknown;
error?: string;
};
if (data.type === 'query') {
this.subscribedQueries.forEach((callback, query) => {
if (data.query === query) {
callback(data.data);
}
});
} else if (data.type === 'error') {
console.error(data.error);
}
};
ws.onclose = () => {
console.log('Disconnected from Tether');
this.attemptReconnect();
};
};
attemptReconnect = () => {
this.ws?.close();
this.reconnectAttempts++;
if (this.reconnectAttempts > this.maxReconnectAttempts) {
console.error('Max reconnect attempts reached');
return;
}
setTimeout(() => {
this.startConnection(this.url);
}, this.reconnectInterval);
};
close = () => {
this.ws?.close();
this.ws = null;
};
send = (message: string) => {
if (this.ws?.readyState !== WebSocket.OPEN) {
this.sendQueue.push(message);
return;
}
this.ws?.send(message);
};
}