add basic ws connectivity tests

This commit is contained in:
tiagosiebler
2022-09-15 14:11:17 +01:00
parent 3f5039ef8b
commit 1b422a1beb
7 changed files with 435 additions and 31 deletions

View File

@@ -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;

View File

@@ -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: {

View 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]);
});
});

View 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);
}
});
});

View 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]);
});
});

View 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
View 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);
});
}