mirror of
https://github.com/discordjs/discord.js.git
synced 2026-03-09 16:13:31 +01:00
* chore: vitest config * feat: vitest * fix: do not actually create ws * chore: config * chore: lockfile * chore: revert downgrade, up node * chore: package - 'git add -A' * chore: delete mock-socket * chore: delete mock-socket * fix: lockfile --------- Co-authored-by: almeidx <github@almeidx.dev> Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
155 lines
4.8 KiB
TypeScript
155 lines
4.8 KiB
TypeScript
// @ts-nocheck
|
|
import { Buffer } from 'node:buffer';
|
|
import { createSocket as _createSocket } from 'node:dgram';
|
|
import { EventEmitter } from 'node:events';
|
|
import { describe, test, expect, vitest, beforeEach, afterEach } from 'vitest';
|
|
import { VoiceUDPSocket } from '../src/networking/VoiceUDPSocket';
|
|
|
|
vitest.mock('node:dgram');
|
|
vitest.useFakeTimers();
|
|
|
|
const createSocket = _createSocket as unknown as vitest.Mock<typeof _createSocket>;
|
|
|
|
beforeEach(() => {
|
|
createSocket.mockReset();
|
|
});
|
|
|
|
class FakeSocket extends EventEmitter {
|
|
public send(buffer: Buffer, port: number, address: string) {}
|
|
|
|
public close() {
|
|
this.emit('close');
|
|
}
|
|
}
|
|
|
|
// ip = 91.90.123.93, port = 54148
|
|
const VALID_RESPONSE = Buffer.from([
|
|
0x0, 0x2, 0x0, 0x46, 0x0, 0x4, 0xeb, 0x23, 0x39, 0x31, 0x2e, 0x39, 0x30, 0x2e, 0x31, 0x32, 0x33, 0x2e, 0x39, 0x33,
|
|
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xd3, 0x84,
|
|
]);
|
|
|
|
async function wait() {
|
|
return new Promise((resolve) => {
|
|
setImmediate(resolve);
|
|
vitest.advanceTimersToNextTimer();
|
|
});
|
|
}
|
|
|
|
describe('VoiceUDPSocket#performIPDiscovery', () => {
|
|
let socket: VoiceUDPSocket;
|
|
|
|
afterEach(() => {
|
|
socket.destroy();
|
|
});
|
|
|
|
/*
|
|
Ensures that the UDP socket sends data and parses the response correctly
|
|
*/
|
|
test('Resolves and cleans up with a successful flow', async () => {
|
|
const fake = new FakeSocket();
|
|
fake.send = vitest.fn().mockImplementation((buffer: Buffer, port: number, address: string) => {
|
|
fake.emit('message', VALID_RESPONSE);
|
|
});
|
|
createSocket.mockImplementation((type) => fake as any);
|
|
socket = new VoiceUDPSocket({ ip: '1.2.3.4', port: 25_565 });
|
|
|
|
expect(createSocket).toHaveBeenCalledWith('udp4');
|
|
expect(fake.listenerCount('message')).toEqual(1);
|
|
await expect(socket.performIPDiscovery(1_234)).resolves.toEqual({
|
|
ip: '91.90.123.93',
|
|
port: 54_148,
|
|
});
|
|
// Ensure clean up occurs
|
|
expect(fake.listenerCount('message')).toEqual(1);
|
|
});
|
|
|
|
/*
|
|
In the case where an unrelated message is received before the IP discovery buffer,
|
|
the UDP socket should wait indefinitely until the correct buffer arrives.
|
|
*/
|
|
test('Waits for a valid response in an unexpected flow', async () => {
|
|
const fake = new FakeSocket();
|
|
const fakeResponse = Buffer.from([1, 2, 3, 4, 5]);
|
|
fake.send = vitest.fn().mockImplementation(async (buffer: Buffer, port: number, address: string) => {
|
|
fake.emit('message', fakeResponse);
|
|
await wait();
|
|
fake.emit('message', VALID_RESPONSE);
|
|
});
|
|
createSocket.mockImplementation(() => fake as any);
|
|
socket = new VoiceUDPSocket({ ip: '1.2.3.4', port: 25_565 });
|
|
|
|
expect(createSocket).toHaveBeenCalledWith('udp4');
|
|
expect(fake.listenerCount('message')).toEqual(1);
|
|
await expect(socket.performIPDiscovery(1_234)).resolves.toEqual({
|
|
ip: '91.90.123.93',
|
|
port: 54_148,
|
|
});
|
|
// Ensure clean up occurs
|
|
expect(fake.listenerCount('message')).toEqual(1);
|
|
});
|
|
|
|
test('Rejects if socket closes before IP discovery can be completed', async () => {
|
|
const fake = new FakeSocket();
|
|
fake.send = vitest.fn().mockImplementation(async (buffer: Buffer, port: number, address: string) => {
|
|
await wait();
|
|
fake.close();
|
|
});
|
|
createSocket.mockImplementation(() => fake as any);
|
|
socket = new VoiceUDPSocket({ ip: '1.2.3.4', port: 25_565 });
|
|
|
|
expect(createSocket).toHaveBeenCalledWith('udp4');
|
|
await expect(socket.performIPDiscovery(1_234)).rejects.toThrowError();
|
|
});
|
|
|
|
test('Stays alive when messages are echoed back', async () => {
|
|
const fake = new FakeSocket();
|
|
fake.send = vitest.fn().mockImplementation(async (buffer: Buffer) => {
|
|
await wait();
|
|
fake.emit('message', buffer);
|
|
});
|
|
createSocket.mockImplementation(() => fake as any);
|
|
socket = new VoiceUDPSocket({ ip: '1.2.3.4', port: 25_565 });
|
|
|
|
let closed = false;
|
|
socket.on('close', () => (closed = true));
|
|
|
|
for (let index = 0; index < 30; index++) {
|
|
vitest.advanceTimersToNextTimer();
|
|
await wait();
|
|
}
|
|
|
|
expect(closed).toEqual(false);
|
|
});
|
|
|
|
test('Recovers from intermittent responses', async () => {
|
|
const fake = new FakeSocket();
|
|
const fakeSend = vitest.fn();
|
|
fake.send = fakeSend;
|
|
createSocket.mockImplementation(() => fake as any);
|
|
socket = new VoiceUDPSocket({ ip: '1.2.3.4', port: 25_565 });
|
|
|
|
let closed = false;
|
|
|
|
socket.on('close', () => (closed = true));
|
|
|
|
for (let index = 0; index < 10; index++) {
|
|
vitest.advanceTimersToNextTimer();
|
|
await wait();
|
|
}
|
|
|
|
fakeSend.mockImplementation(async (buffer: Buffer) => {
|
|
await wait();
|
|
fake.emit('message', buffer);
|
|
});
|
|
expect(closed).toEqual(false);
|
|
for (let index = 0; index < 30; index++) {
|
|
vitest.advanceTimersToNextTimer();
|
|
await wait();
|
|
}
|
|
|
|
expect(closed).toEqual(false);
|
|
});
|
|
});
|