add basic ws connectivity tests
This commit is contained in:
@@ -74,7 +74,7 @@ export type WsKey = typeof WS_KEY_MAP[keyof typeof WS_KEY_MAP];
|
|||||||
export interface WSClientConfigurableOptions {
|
export interface WSClientConfigurableOptions {
|
||||||
key?: string;
|
key?: string;
|
||||||
secret?: string;
|
secret?: string;
|
||||||
livenet?: boolean;
|
testnet?: boolean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The API group this client should connect to.
|
* The API group this client should connect to.
|
||||||
@@ -94,7 +94,7 @@ export interface WSClientConfigurableOptions {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface WebsocketClientOptions extends WSClientConfigurableOptions {
|
export interface WebsocketClientOptions extends WSClientConfigurableOptions {
|
||||||
livenet: boolean;
|
testnet?: boolean;
|
||||||
market: APIMarket;
|
market: APIMarket;
|
||||||
pongTimeout: number;
|
pongTimeout: number;
|
||||||
pingInterval: number;
|
pingInterval: number;
|
||||||
|
|||||||
@@ -37,16 +37,36 @@ function neverGuard(x: never, msg: string): Error {
|
|||||||
|
|
||||||
const loggerCategory = { category: 'bybit-ws' };
|
const loggerCategory = { category: 'bybit-ws' };
|
||||||
|
|
||||||
|
export type WsClientEvent =
|
||||||
|
| 'open'
|
||||||
|
| 'update'
|
||||||
|
| 'close'
|
||||||
|
| 'error'
|
||||||
|
| 'reconnect'
|
||||||
|
| 'reconnected'
|
||||||
|
| 'response';
|
||||||
|
|
||||||
|
interface WebsocketClientEvents {
|
||||||
|
open: (evt: { wsKey: WsKey; event: any }) => void;
|
||||||
|
reconnect: (evt: { wsKey: WsKey; event: any }) => void;
|
||||||
|
reconnected: (evt: { wsKey: WsKey; event: any }) => void;
|
||||||
|
close: (evt: { wsKey: WsKey; event: any }) => void;
|
||||||
|
response: (response: any) => void;
|
||||||
|
update: (response: any) => void;
|
||||||
|
error: (response: any) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Type safety for on and emit handlers: https://stackoverflow.com/a/61609010/880837
|
||||||
export declare interface WebsocketClient {
|
export declare interface WebsocketClient {
|
||||||
on(
|
on<U extends keyof WebsocketClientEvents>(
|
||||||
event: 'open' | 'reconnected',
|
event: U,
|
||||||
listener: ({ wsKey: WsKey, event: any }) => void
|
listener: WebsocketClientEvents[U]
|
||||||
): this;
|
): this;
|
||||||
on(
|
|
||||||
event: 'response' | 'update' | 'error',
|
emit<U extends keyof WebsocketClientEvents>(
|
||||||
listener: (response: any) => void
|
event: U,
|
||||||
): this;
|
...args: Parameters<WebsocketClientEvents[U]>
|
||||||
on(event: 'reconnect' | 'close', listener: ({ wsKey: WsKey }) => void): this;
|
): boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class WebsocketClient extends EventEmitter {
|
export class WebsocketClient extends EventEmitter {
|
||||||
@@ -65,7 +85,7 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
this.wsStore = new WsStore(this.logger);
|
this.wsStore = new WsStore(this.logger);
|
||||||
|
|
||||||
this.options = {
|
this.options = {
|
||||||
livenet: false,
|
testnet: false,
|
||||||
pongTimeout: 1000,
|
pongTimeout: 1000,
|
||||||
pingInterval: 10000,
|
pingInterval: 10000,
|
||||||
reconnectTimeout: 500,
|
reconnectTimeout: 500,
|
||||||
@@ -89,7 +109,7 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
this.restClient = new InverseClient(
|
this.restClient = new InverseClient(
|
||||||
undefined,
|
undefined,
|
||||||
undefined,
|
undefined,
|
||||||
this.isLivenet(),
|
!this.isTestnet(),
|
||||||
this.options.restOptions,
|
this.options.restOptions,
|
||||||
this.options.requestOptions
|
this.options.requestOptions
|
||||||
);
|
);
|
||||||
@@ -99,7 +119,7 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
this.restClient = new LinearClient(
|
this.restClient = new LinearClient(
|
||||||
undefined,
|
undefined,
|
||||||
undefined,
|
undefined,
|
||||||
this.isLivenet(),
|
!this.isTestnet(),
|
||||||
this.options.restOptions,
|
this.options.restOptions,
|
||||||
this.options.requestOptions
|
this.options.requestOptions
|
||||||
);
|
);
|
||||||
@@ -109,7 +129,7 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
this.restClient = new SpotClient(
|
this.restClient = new SpotClient(
|
||||||
undefined,
|
undefined,
|
||||||
undefined,
|
undefined,
|
||||||
this.isLivenet(),
|
!this.isTestnet(),
|
||||||
this.options.restOptions,
|
this.options.restOptions,
|
||||||
this.options.requestOptions
|
this.options.requestOptions
|
||||||
);
|
);
|
||||||
@@ -134,8 +154,8 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public isLivenet(): boolean {
|
public isTestnet(): boolean {
|
||||||
return this.options.livenet === true;
|
return this.options.testnet === true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public isLinear(): boolean {
|
public isLinear(): boolean {
|
||||||
@@ -216,6 +236,13 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
this.getWs(wsKey)?.close();
|
this.getWs(wsKey)?.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public closeAll() {
|
||||||
|
const keys = this.wsStore.getKeys();
|
||||||
|
keys.forEach((key) => {
|
||||||
|
this.close(key);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Request connection of all dependent (public & private) websockets, instead of waiting for automatic connection by library
|
* Request connection of all dependent (public & private) websockets, instead of waiting for automatic connection by library
|
||||||
*/
|
*/
|
||||||
@@ -528,9 +555,8 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
this.logger.info('Websocket connected', {
|
this.logger.info('Websocket connected', {
|
||||||
...loggerCategory,
|
...loggerCategory,
|
||||||
wsKey,
|
wsKey,
|
||||||
livenet: this.isLivenet(),
|
testnet: this.isTestnet(),
|
||||||
linear: this.isLinear(),
|
market: this.options.market,
|
||||||
spot: this.isSpot(),
|
|
||||||
});
|
});
|
||||||
this.emit('open', { wsKey, event });
|
this.emit('open', { wsKey, event });
|
||||||
} else if (
|
} else if (
|
||||||
@@ -560,7 +586,12 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
|
|
||||||
const msg = JSON.parse((event && event.data) || event);
|
const msg = JSON.parse((event && event.data) || event);
|
||||||
if (msg['success'] || msg?.pong) {
|
if (msg['success'] || msg?.pong) {
|
||||||
return this.onWsMessageResponse(msg, wsKey);
|
if (isWsPong(msg)) {
|
||||||
|
this.logger.silly('Received pong', { ...loggerCategory, wsKey });
|
||||||
|
} else {
|
||||||
|
this.emit('response', msg);
|
||||||
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (msg.topic) {
|
if (msg.topic) {
|
||||||
@@ -602,18 +633,10 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
this.wsStore.getConnectionState(wsKey) !== WsConnectionStateEnum.CLOSING
|
this.wsStore.getConnectionState(wsKey) !== WsConnectionStateEnum.CLOSING
|
||||||
) {
|
) {
|
||||||
this.reconnectWithDelay(wsKey, this.options.reconnectTimeout!);
|
this.reconnectWithDelay(wsKey, this.options.reconnectTimeout!);
|
||||||
this.emit('reconnect', { wsKey });
|
this.emit('reconnect', { wsKey, event });
|
||||||
} else {
|
} else {
|
||||||
this.setWsState(wsKey, WsConnectionStateEnum.INITIAL);
|
this.setWsState(wsKey, WsConnectionStateEnum.INITIAL);
|
||||||
this.emit('close', { wsKey });
|
this.emit('close', { wsKey, event });
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private onWsMessageResponse(response: any, wsKey: WsKey) {
|
|
||||||
if (isWsPong(response)) {
|
|
||||||
this.logger.silly('Received pong', { ...loggerCategory, wsKey });
|
|
||||||
} else {
|
|
||||||
this.emit('response', response);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -630,7 +653,7 @@ export class WebsocketClient extends EventEmitter {
|
|||||||
return this.options.wsUrl;
|
return this.options.wsUrl;
|
||||||
}
|
}
|
||||||
|
|
||||||
const networkKey = this.isLivenet() ? 'livenet' : 'testnet';
|
const networkKey = this.isTestnet() ? 'testnet' : 'livenet';
|
||||||
|
|
||||||
switch (wsKey) {
|
switch (wsKey) {
|
||||||
case WS_KEY_MAP.linearPublic: {
|
case WS_KEY_MAP.linearPublic: {
|
||||||
|
|||||||
75
test/inverse/private.ws.test.ts
Normal file
75
test/inverse/private.ws.test.ts
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
import {
|
||||||
|
WebsocketClient,
|
||||||
|
WSClientConfigurableOptions,
|
||||||
|
WS_KEY_MAP,
|
||||||
|
} from '../../src';
|
||||||
|
import {
|
||||||
|
logAllEvents,
|
||||||
|
promiseSleep,
|
||||||
|
silentLogger,
|
||||||
|
waitForSocketEvent,
|
||||||
|
WS_OPEN_EVENT_PARTIAL,
|
||||||
|
} from '../ws.util';
|
||||||
|
|
||||||
|
describe('Private Inverse Perps Websocket Client', () => {
|
||||||
|
let wsClient: WebsocketClient;
|
||||||
|
|
||||||
|
const API_KEY = process.env.API_KEY_COM;
|
||||||
|
const API_SECRET = process.env.API_SECRET_COM;
|
||||||
|
|
||||||
|
it('should have api credentials to test with', () => {
|
||||||
|
expect(API_KEY).toStrictEqual(expect.any(String));
|
||||||
|
expect(API_SECRET).toStrictEqual(expect.any(String));
|
||||||
|
});
|
||||||
|
|
||||||
|
const wsClientOptions: WSClientConfigurableOptions = {
|
||||||
|
market: 'inverse',
|
||||||
|
key: API_KEY,
|
||||||
|
secret: API_SECRET,
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
wsClient = new WebsocketClient(wsClientOptions, silentLogger);
|
||||||
|
wsClient.connectPrivate();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
// await promiseSleep(2000);
|
||||||
|
wsClient.closeAll();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should open a ws connection', async () => {
|
||||||
|
const wsOpenPromise = waitForSocketEvent(wsClient, 'open');
|
||||||
|
|
||||||
|
expect(wsOpenPromise).resolves.toMatchObject({
|
||||||
|
event: WS_OPEN_EVENT_PARTIAL,
|
||||||
|
wsKey: WS_KEY_MAP.inverse,
|
||||||
|
});
|
||||||
|
|
||||||
|
await Promise.all([wsOpenPromise]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should subscribe to private wallet events', async () => {
|
||||||
|
const wsResponsePromise = waitForSocketEvent(wsClient, 'response');
|
||||||
|
// const wsUpdatePromise = waitForSocketEvent(wsClient, 'update');
|
||||||
|
|
||||||
|
const wsTopic = 'wallet';
|
||||||
|
expect(wsResponsePromise).resolves.toMatchObject({
|
||||||
|
request: {
|
||||||
|
args: [wsTopic],
|
||||||
|
op: 'subscribe',
|
||||||
|
},
|
||||||
|
success: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
// No easy way to trigger a private event (other than executing trades)
|
||||||
|
// expect(wsUpdatePromise).resolves.toMatchObject({
|
||||||
|
// topic: wsTopic,
|
||||||
|
// data: expect.any(Array),
|
||||||
|
// });
|
||||||
|
|
||||||
|
wsClient.subscribe(wsTopic);
|
||||||
|
|
||||||
|
await Promise.all([wsResponsePromise]);
|
||||||
|
});
|
||||||
|
});
|
||||||
75
test/inverse/public.ws.test.ts
Normal file
75
test/inverse/public.ws.test.ts
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
import {
|
||||||
|
LinearClient,
|
||||||
|
WebsocketClient,
|
||||||
|
WSClientConfigurableOptions,
|
||||||
|
WS_KEY_MAP,
|
||||||
|
} from '../../src';
|
||||||
|
import {
|
||||||
|
promiseSleep,
|
||||||
|
silentLogger,
|
||||||
|
waitForSocketEvent,
|
||||||
|
WS_OPEN_EVENT_PARTIAL,
|
||||||
|
} from '../ws.util';
|
||||||
|
|
||||||
|
describe('Public Inverse Perps Websocket Client', () => {
|
||||||
|
let wsClient: WebsocketClient;
|
||||||
|
|
||||||
|
const wsClientOptions: WSClientConfigurableOptions = {
|
||||||
|
market: 'inverse',
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
wsClient = new WebsocketClient(wsClientOptions, silentLogger);
|
||||||
|
wsClient.connectPublic();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
wsClient.closeAll();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should open a private ws connection', async () => {
|
||||||
|
const wsOpenPromise = waitForSocketEvent(wsClient, 'open');
|
||||||
|
|
||||||
|
expect(wsOpenPromise).resolves.toMatchObject({
|
||||||
|
event: WS_OPEN_EVENT_PARTIAL,
|
||||||
|
wsKey: WS_KEY_MAP.inverse,
|
||||||
|
});
|
||||||
|
|
||||||
|
await Promise.all([wsOpenPromise]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should subscribe to public orderBookL2_25 events', async () => {
|
||||||
|
const wsResponsePromise = waitForSocketEvent(wsClient, 'response');
|
||||||
|
const wsUpdatePromise = waitForSocketEvent(wsClient, 'update');
|
||||||
|
|
||||||
|
const wsTopic = 'orderBookL2_25.BTCUSD';
|
||||||
|
expect(wsResponsePromise).resolves.toMatchObject({
|
||||||
|
request: {
|
||||||
|
args: [wsTopic],
|
||||||
|
op: 'subscribe',
|
||||||
|
},
|
||||||
|
success: true,
|
||||||
|
});
|
||||||
|
expect(wsUpdatePromise).resolves.toMatchObject({
|
||||||
|
topic: wsTopic,
|
||||||
|
data: expect.any(Array),
|
||||||
|
});
|
||||||
|
|
||||||
|
wsClient.subscribe(wsTopic);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await wsResponsePromise;
|
||||||
|
} catch (e) {
|
||||||
|
console.error(
|
||||||
|
`Wait for "${wsTopic}" subscription response exception: `,
|
||||||
|
e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await wsUpdatePromise;
|
||||||
|
} catch (e) {
|
||||||
|
console.error(`Wait for "${wsTopic}" event exception: `, e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
73
test/linear/private.ws.test.ts
Normal file
73
test/linear/private.ws.test.ts
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
import {
|
||||||
|
WebsocketClient,
|
||||||
|
WSClientConfigurableOptions,
|
||||||
|
WS_KEY_MAP,
|
||||||
|
} from '../../src';
|
||||||
|
import {
|
||||||
|
promiseSleep,
|
||||||
|
silentLogger,
|
||||||
|
waitForSocketEvent,
|
||||||
|
WS_OPEN_EVENT_PARTIAL,
|
||||||
|
} from '../ws.util';
|
||||||
|
|
||||||
|
describe('Private Linear Websocket Client', () => {
|
||||||
|
let wsClient: WebsocketClient;
|
||||||
|
|
||||||
|
const API_KEY = process.env.API_KEY_COM;
|
||||||
|
const API_SECRET = process.env.API_SECRET_COM;
|
||||||
|
|
||||||
|
it('should have api credentials to test with', () => {
|
||||||
|
expect(API_KEY).toStrictEqual(expect.any(String));
|
||||||
|
expect(API_SECRET).toStrictEqual(expect.any(String));
|
||||||
|
});
|
||||||
|
|
||||||
|
const wsClientOptions: WSClientConfigurableOptions = {
|
||||||
|
market: 'linear',
|
||||||
|
key: API_KEY,
|
||||||
|
secret: API_SECRET,
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
wsClient = new WebsocketClient(wsClientOptions, silentLogger);
|
||||||
|
wsClient.connectPrivate();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
wsClient.closeAll();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should open a ws connection', async () => {
|
||||||
|
const wsOpenPromise = waitForSocketEvent(wsClient, 'open');
|
||||||
|
|
||||||
|
expect(wsOpenPromise).resolves.toMatchObject({
|
||||||
|
event: WS_OPEN_EVENT_PARTIAL,
|
||||||
|
wsKey: WS_KEY_MAP.linearPrivate,
|
||||||
|
});
|
||||||
|
|
||||||
|
await Promise.all([wsOpenPromise]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should subscribe to private wallet events', async () => {
|
||||||
|
const wsResponsePromise = waitForSocketEvent(wsClient, 'response');
|
||||||
|
// const wsUpdatePromise = waitForSocketEvent(wsClient, 'update');
|
||||||
|
|
||||||
|
const wsTopic = 'wallet';
|
||||||
|
expect(wsResponsePromise).resolves.toMatchObject({
|
||||||
|
request: {
|
||||||
|
args: [wsTopic],
|
||||||
|
op: 'subscribe',
|
||||||
|
},
|
||||||
|
success: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
// No easy way to trigger a private event (other than executing trades)
|
||||||
|
// expect(wsUpdatePromise).resolves.toMatchObject({
|
||||||
|
// topic: wsTopic,
|
||||||
|
// data: expect.any(Array),
|
||||||
|
// });
|
||||||
|
|
||||||
|
wsClient.subscribe(wsTopic);
|
||||||
|
|
||||||
|
await Promise.all([wsResponsePromise]);
|
||||||
|
});
|
||||||
|
});
|
||||||
76
test/linear/public.ws.test.ts
Normal file
76
test/linear/public.ws.test.ts
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
import {
|
||||||
|
LinearClient,
|
||||||
|
WebsocketClient,
|
||||||
|
WSClientConfigurableOptions,
|
||||||
|
WS_KEY_MAP,
|
||||||
|
} from '../../src';
|
||||||
|
import {
|
||||||
|
silentLogger,
|
||||||
|
waitForSocketEvent,
|
||||||
|
WS_OPEN_EVENT_PARTIAL,
|
||||||
|
} from '../ws.util';
|
||||||
|
|
||||||
|
describe('Public Linear Websocket Client', () => {
|
||||||
|
let wsClient: WebsocketClient;
|
||||||
|
|
||||||
|
const wsClientOptions: WSClientConfigurableOptions = {
|
||||||
|
market: 'linear',
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
wsClient = new WebsocketClient(wsClientOptions, silentLogger);
|
||||||
|
wsClient.connectPublic();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
wsClient.closeAll();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should open a private ws connection', async () => {
|
||||||
|
const wsOpenPromise = waitForSocketEvent(wsClient, 'open');
|
||||||
|
|
||||||
|
expect(wsOpenPromise).resolves.toMatchObject({
|
||||||
|
event: WS_OPEN_EVENT_PARTIAL,
|
||||||
|
wsKey: WS_KEY_MAP.linearPublic,
|
||||||
|
});
|
||||||
|
|
||||||
|
await Promise.all([wsOpenPromise]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should subscribe to public orderBookL2_25 events', async () => {
|
||||||
|
const wsResponsePromise = waitForSocketEvent(wsClient, 'response');
|
||||||
|
const wsUpdatePromise = waitForSocketEvent(wsClient, 'update');
|
||||||
|
|
||||||
|
const wsTopic = 'orderBookL2_25.BTCUSDT';
|
||||||
|
expect(wsResponsePromise).resolves.toMatchObject({
|
||||||
|
request: {
|
||||||
|
args: [wsTopic],
|
||||||
|
op: 'subscribe',
|
||||||
|
},
|
||||||
|
success: true,
|
||||||
|
});
|
||||||
|
expect(wsUpdatePromise).resolves.toMatchObject({
|
||||||
|
topic: wsTopic,
|
||||||
|
data: {
|
||||||
|
order_book: expect.any(Array),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
wsClient.subscribe(wsTopic);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await wsResponsePromise;
|
||||||
|
} catch (e) {
|
||||||
|
console.error(
|
||||||
|
`Wait for "${wsTopic}" subscription response exception: `,
|
||||||
|
e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await wsUpdatePromise;
|
||||||
|
} catch (e) {
|
||||||
|
console.error(`Wait for "${wsTopic}" event exception: `, e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
82
test/ws.util.ts
Normal file
82
test/ws.util.ts
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
import { WebsocketClient, WsClientEvent } from '../src';
|
||||||
|
|
||||||
|
export const silentLogger = {
|
||||||
|
silly: () => {},
|
||||||
|
debug: () => {},
|
||||||
|
notice: () => {},
|
||||||
|
info: () => {},
|
||||||
|
warning: () => {},
|
||||||
|
error: () => {},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WS_OPEN_EVENT_PARTIAL = {
|
||||||
|
type: 'open',
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Resolves a promise if an event is seen before a timeout (defaults to 2.5 seconds) */
|
||||||
|
export function waitForSocketEvent(
|
||||||
|
wsClient: WebsocketClient,
|
||||||
|
event: WsClientEvent,
|
||||||
|
timeoutMs: number = 4.5 * 1000
|
||||||
|
) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const timeout = setTimeout(() => {
|
||||||
|
reject(
|
||||||
|
`Failed to receive "${event}" event before timeout. Check that these are correct: topic, api keys (if private), signature process (if private)`
|
||||||
|
);
|
||||||
|
}, timeoutMs);
|
||||||
|
|
||||||
|
let resolvedOnce = false;
|
||||||
|
|
||||||
|
wsClient.on(event, (event) => {
|
||||||
|
clearTimeout(timeout);
|
||||||
|
resolve(event);
|
||||||
|
resolvedOnce = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
wsClient.on('error', (event) => {
|
||||||
|
clearTimeout(timeout);
|
||||||
|
if (!resolvedOnce) {
|
||||||
|
reject(event);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// if (event !== 'close') {
|
||||||
|
// wsClient.on('close', (event) => {
|
||||||
|
// clearTimeout(timeout);
|
||||||
|
|
||||||
|
// if (!resolvedOnce) {
|
||||||
|
// reject(event);
|
||||||
|
// }
|
||||||
|
// });
|
||||||
|
// }
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function logAllEvents(wsClient: WebsocketClient) {
|
||||||
|
wsClient.on('update', (data) => {
|
||||||
|
console.log('wsUpdate: ', JSON.stringify(data, null, 2));
|
||||||
|
});
|
||||||
|
|
||||||
|
wsClient.on('open', (data) => {
|
||||||
|
console.log('wsOpen: ', data.wsKey);
|
||||||
|
});
|
||||||
|
wsClient.on('response', (data) => {
|
||||||
|
console.log('wsResponse ', JSON.stringify(data, null, 2));
|
||||||
|
});
|
||||||
|
wsClient.on('reconnect', ({ wsKey }) => {
|
||||||
|
console.log('wsReconnecting ', wsKey);
|
||||||
|
});
|
||||||
|
wsClient.on('reconnected', (data) => {
|
||||||
|
console.log('wsReconnected ', data?.wsKey);
|
||||||
|
});
|
||||||
|
wsClient.on('close', (data) => {
|
||||||
|
// console.log('wsClose: ', data);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function promiseSleep(ms: number) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
setTimeout(resolve, ms);
|
||||||
|
});
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user