2021-04-08 14:12:03 -04:00
|
|
|
/**
|
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one
|
|
|
|
* or more contributor license agreements. See the NOTICE file
|
|
|
|
* distributed with this work for additional information
|
|
|
|
* regarding copyright ownership. The ASF licenses this file
|
|
|
|
* to you under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance
|
|
|
|
* with the License. You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing,
|
|
|
|
* software distributed under the License is distributed on an
|
|
|
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
|
|
* KIND, either express or implied. See the License for the
|
|
|
|
* specific language governing permissions and limitations
|
|
|
|
* under the License.
|
|
|
|
*/
|
|
|
|
const jwt = require('jsonwebtoken');
|
|
|
|
const config = require('../config.test.json');
|
|
|
|
|
|
|
|
import { describe, expect, test, beforeEach, afterEach } from '@jest/globals';
|
|
|
|
import * as http from 'http';
|
|
|
|
import * as net from 'net';
|
2021-11-03 06:17:36 -04:00
|
|
|
import { WebSocket } from 'ws';
|
2021-04-08 14:12:03 -04:00
|
|
|
|
|
|
|
// NOTE: these mock variables needs to start with "mock" due to
|
|
|
|
// calls to `jest.mock` being hoisted to the top of the file.
|
|
|
|
// https://jestjs.io/docs/es6-class-mocks#calling-jestmock-with-the-module-factory-parameter
|
|
|
|
const mockRedisXrange = jest.fn();
|
|
|
|
|
|
|
|
jest.mock('ws');
|
|
|
|
jest.mock('ioredis', () => {
|
|
|
|
return jest.fn().mockImplementation(() => {
|
|
|
|
return { xrange: mockRedisXrange };
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
const wsMock = WebSocket as jest.Mocked<typeof WebSocket>;
|
|
|
|
const channelId = 'bc9e040c-7b4a-4817-99b9-292832d97ec7';
|
|
|
|
const streamReturnValue: server.StreamResult[] = [
|
|
|
|
[
|
|
|
|
'1615426152415-0',
|
|
|
|
[
|
|
|
|
'data',
|
|
|
|
`{"channel_id": "${channelId}", "job_id": "c9b99965-8f1e-4ce5-aa43-d6fc94d6a510", "user_id": "1", "status": "done", "errors": [], "result_url": "/superset/explore_json/data/ejr-37281682b1282cdb8f25e0de0339b386"}`,
|
|
|
|
],
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'1615426152516-0',
|
|
|
|
[
|
|
|
|
'data',
|
|
|
|
`{"channel_id": "${channelId}", "job_id": "f1e5bb1f-f2f1-4f21-9b2f-c9b91dcc9b59", "user_id": "1", "status": "done", "errors": [], "result_url": "/api/v1/chart/data/qc-64e8452dc9907dd77746cb75a19202de"}`,
|
|
|
|
],
|
|
|
|
],
|
|
|
|
];
|
|
|
|
|
|
|
|
import * as server from '../src/index';
|
2021-04-26 17:05:10 -04:00
|
|
|
import { statsd } from '../src/index';
|
2021-04-08 14:12:03 -04:00
|
|
|
|
|
|
|
describe('server', () => {
|
2021-04-26 17:05:10 -04:00
|
|
|
let statsdIncrementMock: jest.SpyInstance;
|
|
|
|
|
2021-04-08 14:12:03 -04:00
|
|
|
beforeEach(() => {
|
|
|
|
mockRedisXrange.mockClear();
|
|
|
|
server.resetState();
|
2021-04-26 17:05:10 -04:00
|
|
|
statsdIncrementMock = jest.spyOn(statsd, 'increment').mockReturnValue();
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
statsdIncrementMock.mockRestore();
|
2021-04-08 14:12:03 -04:00
|
|
|
});
|
|
|
|
|
2021-04-15 12:42:01 -04:00
|
|
|
describe('HTTP requests', () => {
|
|
|
|
test('services health checks', () => {
|
|
|
|
const endMock = jest.fn();
|
|
|
|
const writeHeadMock = jest.fn();
|
|
|
|
|
|
|
|
const request = {
|
|
|
|
url: '/health',
|
|
|
|
method: 'GET',
|
|
|
|
headers: {
|
|
|
|
host: 'example.com',
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const response = {
|
|
|
|
writeHead: writeHeadMock,
|
|
|
|
end: endMock,
|
|
|
|
};
|
|
|
|
|
|
|
|
server.httpRequest(request as any, response as any);
|
|
|
|
|
|
|
|
expect(writeHeadMock).toBeCalledTimes(1);
|
|
|
|
expect(writeHeadMock).toHaveBeenLastCalledWith(200);
|
|
|
|
|
|
|
|
expect(endMock).toBeCalledTimes(1);
|
|
|
|
expect(endMock).toHaveBeenLastCalledWith('OK');
|
|
|
|
});
|
|
|
|
|
2023-01-25 18:35:08 -05:00
|
|
|
test('responds with a 404 when not found', () => {
|
2021-04-15 12:42:01 -04:00
|
|
|
const endMock = jest.fn();
|
|
|
|
const writeHeadMock = jest.fn();
|
|
|
|
|
|
|
|
const request = {
|
|
|
|
url: '/unsupported',
|
|
|
|
method: 'GET',
|
|
|
|
headers: {
|
|
|
|
host: 'example.com',
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const response = {
|
|
|
|
writeHead: writeHeadMock,
|
|
|
|
end: endMock,
|
|
|
|
};
|
|
|
|
|
|
|
|
server.httpRequest(request as any, response as any);
|
|
|
|
|
|
|
|
expect(writeHeadMock).toBeCalledTimes(1);
|
|
|
|
expect(writeHeadMock).toHaveBeenLastCalledWith(404);
|
|
|
|
|
|
|
|
expect(endMock).toBeCalledTimes(1);
|
|
|
|
expect(endMock).toHaveBeenLastCalledWith('Not Found');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-04-08 14:12:03 -04:00
|
|
|
describe('incrementId', () => {
|
|
|
|
test('it increments a valid Redis stream ID', () => {
|
|
|
|
expect(server.incrementId('1607477697866-0')).toEqual('1607477697866-1');
|
|
|
|
});
|
|
|
|
|
|
|
|
test('it handles an invalid Redis stream ID', () => {
|
|
|
|
expect(server.incrementId('foo')).toEqual('foo');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('redisUrlFromConfig', () => {
|
|
|
|
test('it builds a valid Redis URL from defaults', () => {
|
|
|
|
expect(
|
|
|
|
server.redisUrlFromConfig({
|
|
|
|
port: 6379,
|
|
|
|
host: '127.0.0.1',
|
|
|
|
password: '',
|
|
|
|
db: 0,
|
|
|
|
ssl: false,
|
|
|
|
}),
|
|
|
|
).toEqual('redis://127.0.0.1:6379/0');
|
|
|
|
});
|
|
|
|
test('it builds a valid Redis URL with a password', () => {
|
|
|
|
expect(
|
|
|
|
server.redisUrlFromConfig({
|
|
|
|
port: 6380,
|
|
|
|
host: 'redis.local',
|
|
|
|
password: 'foo',
|
|
|
|
db: 1,
|
|
|
|
ssl: false,
|
|
|
|
}),
|
|
|
|
).toEqual('redis://:foo@redis.local:6380/1');
|
|
|
|
});
|
|
|
|
test('it builds a valid Redis URL with SSL', () => {
|
|
|
|
expect(
|
|
|
|
server.redisUrlFromConfig({
|
|
|
|
port: 6379,
|
|
|
|
host: '127.0.0.1',
|
|
|
|
password: '',
|
|
|
|
db: 0,
|
|
|
|
ssl: true,
|
|
|
|
}),
|
|
|
|
).toEqual('rediss://127.0.0.1:6379/0');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('processStreamResults', () => {
|
|
|
|
test('sends data to channel', async () => {
|
|
|
|
const ws = new wsMock('localhost');
|
|
|
|
const sendMock = jest.spyOn(ws, 'send');
|
|
|
|
const socketInstance = { ws: ws, channel: channelId, pongTs: Date.now() };
|
2021-04-26 17:05:10 -04:00
|
|
|
|
|
|
|
expect(statsdIncrementMock).toBeCalledTimes(0);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.trackClient(channelId, socketInstance);
|
2021-04-26 17:05:10 -04:00
|
|
|
expect(statsdIncrementMock).toBeCalledTimes(1);
|
|
|
|
expect(statsdIncrementMock).toHaveBeenNthCalledWith(
|
|
|
|
1,
|
|
|
|
'ws_connected_client',
|
|
|
|
);
|
2021-04-08 14:12:03 -04:00
|
|
|
|
|
|
|
server.processStreamResults(streamReturnValue);
|
2021-04-26 17:05:10 -04:00
|
|
|
expect(statsdIncrementMock).toBeCalledTimes(1);
|
2021-04-08 14:12:03 -04:00
|
|
|
|
|
|
|
const message1 = `{"id":"1615426152415-0","channel_id":"${channelId}","job_id":"c9b99965-8f1e-4ce5-aa43-d6fc94d6a510","user_id":"1","status":"done","errors":[],"result_url":"/superset/explore_json/data/ejr-37281682b1282cdb8f25e0de0339b386"}`;
|
|
|
|
const message2 = `{"id":"1615426152516-0","channel_id":"${channelId}","job_id":"f1e5bb1f-f2f1-4f21-9b2f-c9b91dcc9b59","user_id":"1","status":"done","errors":[],"result_url":"/api/v1/chart/data/qc-64e8452dc9907dd77746cb75a19202de"}`;
|
|
|
|
expect(sendMock).toHaveBeenCalledWith(message1);
|
|
|
|
expect(sendMock).toHaveBeenCalledWith(message2);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('channel not present', async () => {
|
|
|
|
const ws = new wsMock('localhost');
|
|
|
|
const sendMock = jest.spyOn(ws, 'send');
|
|
|
|
|
2021-04-26 17:05:10 -04:00
|
|
|
expect(statsdIncrementMock).toBeCalledTimes(0);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.processStreamResults(streamReturnValue);
|
2021-04-26 17:05:10 -04:00
|
|
|
expect(statsdIncrementMock).toBeCalledTimes(0);
|
2021-04-08 14:12:03 -04:00
|
|
|
|
|
|
|
expect(sendMock).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('error sending data to client', async () => {
|
|
|
|
const ws = new wsMock('localhost');
|
|
|
|
const sendMock = jest.spyOn(ws, 'send').mockImplementation(() => {
|
|
|
|
throw new Error();
|
|
|
|
});
|
|
|
|
const cleanChannelMock = jest.spyOn(server, 'cleanChannel');
|
|
|
|
const socketInstance = { ws: ws, channel: channelId, pongTs: Date.now() };
|
2021-04-26 17:05:10 -04:00
|
|
|
|
|
|
|
expect(statsdIncrementMock).toBeCalledTimes(0);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.trackClient(channelId, socketInstance);
|
2021-04-26 17:05:10 -04:00
|
|
|
expect(statsdIncrementMock).toBeCalledTimes(1);
|
|
|
|
expect(statsdIncrementMock).toHaveBeenNthCalledWith(
|
|
|
|
1,
|
|
|
|
'ws_connected_client',
|
|
|
|
);
|
2021-04-08 14:12:03 -04:00
|
|
|
|
|
|
|
server.processStreamResults(streamReturnValue);
|
2021-04-26 17:05:10 -04:00
|
|
|
expect(statsdIncrementMock).toBeCalledTimes(2);
|
|
|
|
expect(statsdIncrementMock).toHaveBeenNthCalledWith(
|
|
|
|
2,
|
|
|
|
'ws_client_send_error',
|
|
|
|
);
|
2021-04-08 14:12:03 -04:00
|
|
|
|
|
|
|
expect(sendMock).toHaveBeenCalled();
|
|
|
|
expect(cleanChannelMock).toHaveBeenCalledWith(channelId);
|
|
|
|
|
|
|
|
cleanChannelMock.mockRestore();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('fetchRangeFromStream', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
mockRedisXrange.mockClear();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('success with results', async () => {
|
|
|
|
mockRedisXrange.mockResolvedValueOnce(streamReturnValue);
|
|
|
|
const cb = jest.fn();
|
|
|
|
await server.fetchRangeFromStream({
|
|
|
|
sessionId: '123',
|
|
|
|
startId: '-',
|
|
|
|
endId: '+',
|
|
|
|
listener: cb,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(mockRedisXrange).toHaveBeenCalledWith(
|
|
|
|
'test-async-events-123',
|
|
|
|
'-',
|
|
|
|
'+',
|
|
|
|
);
|
|
|
|
expect(cb).toHaveBeenCalledWith(streamReturnValue);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('success no results', async () => {
|
|
|
|
const cb = jest.fn();
|
|
|
|
await server.fetchRangeFromStream({
|
|
|
|
sessionId: '123',
|
|
|
|
startId: '-',
|
|
|
|
endId: '+',
|
|
|
|
listener: cb,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(mockRedisXrange).toHaveBeenCalledWith(
|
|
|
|
'test-async-events-123',
|
|
|
|
'-',
|
|
|
|
'+',
|
|
|
|
);
|
|
|
|
expect(cb).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('error', async () => {
|
|
|
|
const cb = jest.fn();
|
|
|
|
mockRedisXrange.mockRejectedValueOnce(new Error());
|
|
|
|
await server.fetchRangeFromStream({
|
|
|
|
sessionId: '123',
|
|
|
|
startId: '-',
|
|
|
|
endId: '+',
|
|
|
|
listener: cb,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(mockRedisXrange).toHaveBeenCalledWith(
|
|
|
|
'test-async-events-123',
|
|
|
|
'-',
|
|
|
|
'+',
|
|
|
|
);
|
|
|
|
expect(cb).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('wsConnection', () => {
|
|
|
|
let ws: WebSocket;
|
|
|
|
let wsEventMock: jest.SpyInstance;
|
|
|
|
let trackClientSpy: jest.SpyInstance;
|
|
|
|
let fetchRangeFromStreamSpy: jest.SpyInstance;
|
|
|
|
let dateNowSpy: jest.SpyInstance;
|
|
|
|
let socketInstanceExpected: server.SocketInstance;
|
|
|
|
|
|
|
|
const getRequest = (token: string, url: string): http.IncomingMessage => {
|
|
|
|
const request = new http.IncomingMessage(new net.Socket());
|
|
|
|
request.method = 'GET';
|
|
|
|
request.headers = { cookie: `${config.jwtCookieName}=${token}` };
|
|
|
|
request.url = url;
|
|
|
|
return request;
|
|
|
|
};
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
ws = new wsMock('localhost');
|
|
|
|
wsEventMock = jest.spyOn(ws, 'on');
|
|
|
|
trackClientSpy = jest.spyOn(server, 'trackClient');
|
|
|
|
fetchRangeFromStreamSpy = jest.spyOn(server, 'fetchRangeFromStream');
|
|
|
|
dateNowSpy = jest
|
|
|
|
.spyOn(global.Date, 'now')
|
|
|
|
.mockImplementation(() =>
|
|
|
|
new Date('2021-03-10T11:01:58.135Z').valueOf(),
|
|
|
|
);
|
|
|
|
socketInstanceExpected = {
|
|
|
|
ws,
|
|
|
|
channel: channelId,
|
|
|
|
pongTs: 1615374118135,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
wsEventMock.mockRestore();
|
|
|
|
trackClientSpy.mockRestore();
|
|
|
|
fetchRangeFromStreamSpy.mockRestore();
|
|
|
|
dateNowSpy.mockRestore();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('invalid JWT', async () => {
|
|
|
|
const invalidToken = jwt.sign({ channel: channelId }, 'invalid secret');
|
|
|
|
const request = getRequest(invalidToken, 'http://localhost');
|
|
|
|
|
|
|
|
expect(() => {
|
|
|
|
server.wsConnection(ws, request);
|
|
|
|
}).toThrow();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('valid JWT, no lastId', async () => {
|
|
|
|
const validToken = jwt.sign({ channel: channelId }, config.jwtSecret);
|
|
|
|
const request = getRequest(validToken, 'http://localhost');
|
|
|
|
|
|
|
|
server.wsConnection(ws, request);
|
|
|
|
|
|
|
|
expect(trackClientSpy).toHaveBeenCalledWith(
|
|
|
|
channelId,
|
|
|
|
socketInstanceExpected,
|
|
|
|
);
|
|
|
|
expect(fetchRangeFromStreamSpy).not.toHaveBeenCalled();
|
|
|
|
expect(wsEventMock).toHaveBeenCalledWith('pong', expect.any(Function));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('valid JWT, with lastId', async () => {
|
|
|
|
const validToken = jwt.sign({ channel: channelId }, config.jwtSecret);
|
|
|
|
const lastId = '1615426152415-0';
|
|
|
|
const request = getRequest(
|
|
|
|
validToken,
|
|
|
|
`http://localhost?last_id=${lastId}`,
|
|
|
|
);
|
|
|
|
|
|
|
|
server.wsConnection(ws, request);
|
|
|
|
|
|
|
|
expect(trackClientSpy).toHaveBeenCalledWith(
|
|
|
|
channelId,
|
|
|
|
socketInstanceExpected,
|
|
|
|
);
|
|
|
|
expect(fetchRangeFromStreamSpy).toHaveBeenCalledWith({
|
|
|
|
sessionId: channelId,
|
|
|
|
startId: '1615426152415-1',
|
|
|
|
endId: '+',
|
|
|
|
listener: server.processStreamResults,
|
|
|
|
});
|
|
|
|
expect(wsEventMock).toHaveBeenCalledWith('pong', expect.any(Function));
|
|
|
|
});
|
|
|
|
|
|
|
|
test('valid JWT, with lastId and lastFirehoseId', async () => {
|
|
|
|
const validToken = jwt.sign({ channel: channelId }, config.jwtSecret);
|
|
|
|
const lastId = '1615426152415-0';
|
|
|
|
const lastFirehoseId = '1715426152415-0';
|
|
|
|
const request = getRequest(
|
|
|
|
validToken,
|
|
|
|
`http://localhost?last_id=${lastId}`,
|
|
|
|
);
|
|
|
|
|
|
|
|
server.setLastFirehoseId(lastFirehoseId);
|
|
|
|
server.wsConnection(ws, request);
|
|
|
|
|
|
|
|
expect(trackClientSpy).toHaveBeenCalledWith(
|
|
|
|
channelId,
|
|
|
|
socketInstanceExpected,
|
|
|
|
);
|
|
|
|
expect(fetchRangeFromStreamSpy).toHaveBeenCalledWith({
|
|
|
|
sessionId: channelId,
|
|
|
|
startId: '1615426152415-1',
|
|
|
|
endId: lastFirehoseId,
|
|
|
|
listener: server.processStreamResults,
|
|
|
|
});
|
|
|
|
expect(wsEventMock).toHaveBeenCalledWith('pong', expect.any(Function));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('httpUpgrade', () => {
|
|
|
|
let socket: net.Socket;
|
|
|
|
let socketDestroySpy: jest.SpyInstance;
|
|
|
|
let wssUpgradeSpy: jest.SpyInstance;
|
|
|
|
|
|
|
|
const getRequest = (token: string, url: string): http.IncomingMessage => {
|
|
|
|
const request = new http.IncomingMessage(new net.Socket());
|
|
|
|
request.method = 'GET';
|
|
|
|
request.headers = { cookie: `${config.jwtCookieName}=${token}` };
|
|
|
|
request.url = url;
|
|
|
|
return request;
|
|
|
|
};
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
socket = new net.Socket();
|
|
|
|
socketDestroySpy = jest.spyOn(socket, 'destroy');
|
|
|
|
wssUpgradeSpy = jest.spyOn(server.wss, 'handleUpgrade');
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
wssUpgradeSpy.mockRestore();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('invalid JWT', async () => {
|
|
|
|
const invalidToken = jwt.sign({ channel: channelId }, 'invalid secret');
|
|
|
|
const request = getRequest(invalidToken, 'http://localhost');
|
|
|
|
|
|
|
|
server.httpUpgrade(request, socket, Buffer.alloc(5));
|
|
|
|
expect(socketDestroySpy).toHaveBeenCalled();
|
|
|
|
expect(wssUpgradeSpy).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('valid JWT, no channel', async () => {
|
|
|
|
const validToken = jwt.sign({ foo: 'bar' }, config.jwtSecret);
|
|
|
|
const request = getRequest(validToken, 'http://localhost');
|
|
|
|
|
|
|
|
server.httpUpgrade(request, socket, Buffer.alloc(5));
|
|
|
|
|
|
|
|
expect(socketDestroySpy).toHaveBeenCalled();
|
|
|
|
expect(wssUpgradeSpy).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('valid upgrade', async () => {
|
|
|
|
const validToken = jwt.sign({ channel: channelId }, config.jwtSecret);
|
|
|
|
const request = getRequest(validToken, 'http://localhost');
|
|
|
|
|
|
|
|
server.httpUpgrade(request, socket, Buffer.alloc(5));
|
|
|
|
|
|
|
|
expect(socketDestroySpy).not.toHaveBeenCalled();
|
|
|
|
expect(wssUpgradeSpy).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-11-03 06:17:36 -04:00
|
|
|
const setReadyState = (ws: WebSocket, value: typeof ws.readyState) => {
|
|
|
|
// workaround for not being able to do
|
|
|
|
// spyOn(instance,'readyState','get').and.returnValue(value);
|
|
|
|
// See for details: https://github.com/facebook/jest/issues/9675
|
|
|
|
Object.defineProperty(ws, 'readyState', {
|
|
|
|
configurable: true,
|
|
|
|
get() {
|
|
|
|
return value;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2021-04-08 14:12:03 -04:00
|
|
|
describe('checkSockets', () => {
|
|
|
|
let ws: WebSocket;
|
|
|
|
let pingSpy: jest.SpyInstance;
|
|
|
|
let terminateSpy: jest.SpyInstance;
|
|
|
|
let socketInstance: server.SocketInstance;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
ws = new wsMock('localhost');
|
|
|
|
pingSpy = jest.spyOn(ws, 'ping');
|
|
|
|
terminateSpy = jest.spyOn(ws, 'terminate');
|
|
|
|
socketInstance = { ws: ws, channel: channelId, pongTs: Date.now() };
|
|
|
|
});
|
|
|
|
|
|
|
|
test('active sockets', () => {
|
2021-11-03 06:17:36 -04:00
|
|
|
setReadyState(ws, WebSocket.OPEN);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.trackClient(channelId, socketInstance);
|
|
|
|
|
|
|
|
server.checkSockets();
|
|
|
|
|
|
|
|
expect(pingSpy).toHaveBeenCalled();
|
|
|
|
expect(terminateSpy).not.toHaveBeenCalled();
|
|
|
|
expect(Object.keys(server.sockets).length).toBe(1);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('stale sockets', () => {
|
2021-11-03 06:17:36 -04:00
|
|
|
setReadyState(ws, WebSocket.OPEN);
|
2021-04-08 14:12:03 -04:00
|
|
|
socketInstance.pongTs = Date.now() - 60000;
|
|
|
|
server.trackClient(channelId, socketInstance);
|
|
|
|
|
|
|
|
server.checkSockets();
|
|
|
|
|
|
|
|
expect(pingSpy).not.toHaveBeenCalled();
|
|
|
|
expect(terminateSpy).toHaveBeenCalled();
|
|
|
|
expect(Object.keys(server.sockets).length).toBe(0);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('closed sockets', () => {
|
2021-11-03 06:17:36 -04:00
|
|
|
setReadyState(ws, WebSocket.CLOSED);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.trackClient(channelId, socketInstance);
|
|
|
|
|
|
|
|
server.checkSockets();
|
|
|
|
|
|
|
|
expect(pingSpy).not.toHaveBeenCalled();
|
|
|
|
expect(terminateSpy).not.toHaveBeenCalled();
|
|
|
|
expect(Object.keys(server.sockets).length).toBe(0);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('no sockets', () => {
|
|
|
|
// don't error
|
|
|
|
server.checkSockets();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('cleanChannel', () => {
|
|
|
|
let ws: WebSocket;
|
|
|
|
let socketInstance: server.SocketInstance;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
ws = new wsMock('localhost');
|
|
|
|
socketInstance = { ws: ws, channel: channelId, pongTs: Date.now() };
|
|
|
|
});
|
|
|
|
|
|
|
|
test('active sockets', () => {
|
2021-11-03 06:17:36 -04:00
|
|
|
setReadyState(ws, WebSocket.OPEN);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.trackClient(channelId, socketInstance);
|
|
|
|
|
|
|
|
server.cleanChannel(channelId);
|
|
|
|
|
|
|
|
expect(server.channels[channelId].sockets.length).toBe(1);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('closing sockets', () => {
|
2021-11-03 06:17:36 -04:00
|
|
|
setReadyState(ws, WebSocket.CLOSING);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.trackClient(channelId, socketInstance);
|
|
|
|
|
|
|
|
server.cleanChannel(channelId);
|
|
|
|
|
|
|
|
expect(server.channels[channelId]).toBeUndefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('multiple sockets', () => {
|
2021-11-03 06:17:36 -04:00
|
|
|
setReadyState(ws, WebSocket.OPEN);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.trackClient(channelId, socketInstance);
|
|
|
|
|
|
|
|
const ws2 = new wsMock('localhost');
|
2021-11-03 06:17:36 -04:00
|
|
|
setReadyState(ws2, WebSocket.OPEN);
|
2021-04-08 14:12:03 -04:00
|
|
|
const socketInstance2 = {
|
|
|
|
ws: ws2,
|
|
|
|
channel: channelId,
|
|
|
|
pongTs: Date.now(),
|
|
|
|
};
|
|
|
|
server.trackClient(channelId, socketInstance2);
|
|
|
|
|
|
|
|
server.cleanChannel(channelId);
|
|
|
|
|
|
|
|
expect(server.channels[channelId].sockets.length).toBe(2);
|
|
|
|
|
2021-11-03 06:17:36 -04:00
|
|
|
setReadyState(ws2, WebSocket.CLOSED);
|
2021-04-08 14:12:03 -04:00
|
|
|
server.cleanChannel(channelId);
|
|
|
|
|
|
|
|
expect(server.channels[channelId].sockets.length).toBe(1);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('invalid channel', () => {
|
|
|
|
// don't error
|
|
|
|
server.cleanChannel(channelId);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|