mirror of
https://github.com/minescope/mineping.git
synced 2025-06-18 01:16:17 +03:00
test: implement tests using vitest framework
This commit is contained in:
parent
ef2bebe755
commit
435e59739c
1499
package-lock.json
generated
Normal file
1499
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
23
package.json
23
package.json
@ -1,22 +1,29 @@
|
||||
{
|
||||
"name": "@minescope/mineping",
|
||||
"version": "1.6.1",
|
||||
"version": "1.7.0-beta.0",
|
||||
"description": "Ping both Minecraft Bedrock and Java servers.",
|
||||
"main": "index.js",
|
||||
"type": "module",
|
||||
"types": "types/index.d.ts",
|
||||
"keywords": [],
|
||||
"author": {
|
||||
"name": "Timofey (xzeldon)",
|
||||
"email": "contact@zeldon.ru",
|
||||
"url": "https://zeldon.ru"
|
||||
"scripts": {
|
||||
"test": "vitest run",
|
||||
"test:watch": "vitest"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git://github.com/minescope/mineping.git"
|
||||
},
|
||||
"type": "module",
|
||||
"license": "MIT",
|
||||
"keywords": [],
|
||||
"author": {
|
||||
"name": "Timofey Gelazoniya",
|
||||
"email": "timofey@z4n.me",
|
||||
"url": "https://zeldon.ru"
|
||||
},
|
||||
"engines": {
|
||||
"node": ">=14"
|
||||
},
|
||||
"license": "MIT"
|
||||
"devDependencies": {
|
||||
"vitest": "^3.2.3"
|
||||
}
|
||||
}
|
||||
|
117
test/bedrock.test.js
Normal file
117
test/bedrock.test.js
Normal file
@ -0,0 +1,117 @@
|
||||
import dgram from "node:dgram";
|
||||
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
||||
import { pingBedrock } from "../lib/bedrock.js";
|
||||
|
||||
vi.mock("node:dgram");
|
||||
|
||||
describe("bedrock.js", () => {
|
||||
let mockSocket;
|
||||
|
||||
beforeEach(() => {
|
||||
// A store for event handlers, closed over by the mockSocket.
|
||||
const handlers = {};
|
||||
|
||||
// Create a stateful mock socket to simulate EventEmitter.
|
||||
mockSocket = {
|
||||
send: vi.fn(),
|
||||
close: vi.fn(),
|
||||
on: vi.fn((event, handler) => {
|
||||
handlers[event] = handler;
|
||||
}),
|
||||
emit: vi.fn((event, ...args) => {
|
||||
if (handlers[event]) {
|
||||
handlers[event](...args);
|
||||
}
|
||||
}),
|
||||
};
|
||||
|
||||
dgram.createSocket = vi.fn().mockReturnValue(mockSocket);
|
||||
vi.useFakeTimers();
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
vi.restoreAllMocks();
|
||||
vi.useRealTimers();
|
||||
});
|
||||
|
||||
it("should ping a server and parse MOTD", async () => {
|
||||
const host = "play.example.com";
|
||||
const options = { port: 25565, timeout: 10000 };
|
||||
const pingPromise = pingBedrock(host, options);
|
||||
|
||||
const motd =
|
||||
"MCPE;§l§bOasys§fPE;0;1337;1096;1999;-37530542056358113;oasys-pe.ru;Adventure;1";
|
||||
const mockPongPacket = createMockPongPacket(motd);
|
||||
|
||||
mockSocket.emit("message", mockPongPacket);
|
||||
|
||||
const result = await pingPromise;
|
||||
|
||||
expect(dgram.createSocket).toHaveBeenCalledWith("udp4");
|
||||
expect(mockSocket.send).toHaveBeenCalledWith(
|
||||
expect.any(Buffer),
|
||||
0,
|
||||
33,
|
||||
options.port,
|
||||
host
|
||||
);
|
||||
expect(mockSocket.close).toHaveBeenCalled();
|
||||
expect(result).toEqual({
|
||||
edition: "MCPE",
|
||||
name: "§l§bOasys§fPE",
|
||||
version: { protocolVersion: 0, minecraftVersion: "1337" },
|
||||
players: { online: 1096, max: 1999 },
|
||||
serverId: "-37530542056358113",
|
||||
mapName: "oasys-pe.ru",
|
||||
gameMode: "Adventure",
|
||||
});
|
||||
});
|
||||
|
||||
describe("errors", () => {
|
||||
it("should throw an error if host is not provided", () => {
|
||||
expect(() => pingBedrock(null)).toThrow("Host argument is not provided");
|
||||
});
|
||||
|
||||
it("should reject on socket timeout", async () => {
|
||||
const pingPromise = pingBedrock("play.example.com", { timeout: 1000 });
|
||||
|
||||
vi.advanceTimersByTime(1000);
|
||||
|
||||
await expect(pingPromise).rejects.toThrow("Socket timeout");
|
||||
expect(mockSocket.close).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it("should reject on a generic socket error", async () => {
|
||||
const pingPromise = pingBedrock("play.example.com");
|
||||
|
||||
// Simulate a DNS or network error by emitting it.
|
||||
mockSocket.emit("error", new Error("EHOSTUNREACH"));
|
||||
|
||||
await expect(pingPromise).rejects.toThrow("EHOSTUNREACH");
|
||||
});
|
||||
|
||||
it("should only reject once, even if multiple errors occur", async () => {
|
||||
const pingPromise = pingBedrock("play.example.com");
|
||||
|
||||
// Fire a socket error first.
|
||||
mockSocket.emit("error", new Error("First error"));
|
||||
|
||||
// Then, try to trigger another error by sending a bad message.
|
||||
mockSocket.emit("message", Buffer.alloc(0));
|
||||
|
||||
await expect(pingPromise).rejects.toThrow("First error");
|
||||
expect(mockSocket.close).toHaveBeenCalledTimes(1);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
function createMockPongPacket(motd) {
|
||||
const motdBuffer = Buffer.from(motd, "utf-8");
|
||||
const packet = Buffer.alloc(35 + motdBuffer.length);
|
||||
packet.writeUInt8(0x1c, 0);
|
||||
packet.writeBigInt64LE(BigInt(Date.now()), 1);
|
||||
Buffer.from("00ffff00fefefefefdfdfdfd12345678", "hex").copy(packet, 17);
|
||||
packet.writeUInt16BE(motdBuffer.length, 33);
|
||||
motdBuffer.copy(packet, 35);
|
||||
return packet;
|
||||
}
|
112
test/java.test.js
Normal file
112
test/java.test.js
Normal file
@ -0,0 +1,112 @@
|
||||
import net from "node:net";
|
||||
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
||||
import { pingJava } from "../lib/java.js";
|
||||
import varint from "../lib/varint.js";
|
||||
|
||||
vi.mock("node:net");
|
||||
|
||||
describe("pingJava", () => {
|
||||
let mockSocket;
|
||||
|
||||
beforeEach(() => {
|
||||
const mockHandlers = {};
|
||||
mockSocket = {
|
||||
write: vi.fn(),
|
||||
destroy: vi.fn(),
|
||||
setNoDelay: vi.fn(),
|
||||
on: vi.fn((event, handler) => (mockHandlers[event] = handler)),
|
||||
emit: vi.fn((event, ...args) => mockHandlers[event]?.(...args)),
|
||||
};
|
||||
net.createConnection = vi.fn().mockReturnValue(mockSocket);
|
||||
vi.useFakeTimers();
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
vi.restoreAllMocks();
|
||||
});
|
||||
|
||||
it("should ping a server and handle a chunked response", async () => {
|
||||
const host = "mc.hypixel.net";
|
||||
const options = {
|
||||
port: 25565,
|
||||
timeout: 5000,
|
||||
protocolVersion: 765,
|
||||
virtualHost: "mc.hypixel.net",
|
||||
};
|
||||
|
||||
const pingPromise = pingJava(host, options);
|
||||
|
||||
mockSocket.emit("connect");
|
||||
|
||||
expect(net.createConnection).toHaveBeenCalledWith({
|
||||
host,
|
||||
port: options.port,
|
||||
});
|
||||
expect(mockSocket.setNoDelay).toHaveBeenCalledWith(true);
|
||||
expect(mockSocket.write).toHaveBeenCalledTimes(2);
|
||||
|
||||
const mockResponse = {
|
||||
version: { name: "1.21", protocol: 765 },
|
||||
players: { max: 20, online: 5, sample: [] },
|
||||
description: "A Minecraft Server",
|
||||
favicon: "...",
|
||||
};
|
||||
const fullPacket = createMockJavaResponse(mockResponse);
|
||||
const chunk1 = fullPacket.subarray(0, 10);
|
||||
const chunk2 = fullPacket.subarray(10);
|
||||
|
||||
mockSocket.emit("data", chunk1);
|
||||
mockSocket.emit("data", chunk2);
|
||||
|
||||
const result = await pingPromise;
|
||||
expect(result).toEqual(mockResponse);
|
||||
expect(mockSocket.destroy).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
describe("errors", () => {
|
||||
it("should throw an error if host is not provided", () => {
|
||||
expect(() => pingJava(null)).toThrow("Host argument is not provided");
|
||||
});
|
||||
|
||||
it("should reject on socket timeout before data is received", async () => {
|
||||
const pingPromise = pingJava("localhost", { timeout: 1000 });
|
||||
mockSocket.emit("connect");
|
||||
|
||||
// Advance time to trigger the timeout
|
||||
vi.advanceTimersByTime(1000);
|
||||
|
||||
await expect(pingPromise).rejects.toThrow("Socket timeout");
|
||||
expect(mockSocket.destroy).toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it("should reject on connection error", async () => {
|
||||
const pingPromise = pingJava("localhost");
|
||||
|
||||
// Simulate a connection refusal
|
||||
mockSocket.emit("error", new Error("ECONNREFUSED"));
|
||||
|
||||
await expect(pingPromise).rejects.toThrow("ECONNREFUSED");
|
||||
});
|
||||
|
||||
it("should only reject once, even if multiple errors occur", async () => {
|
||||
const pingPromise = pingJava("localhost");
|
||||
|
||||
// Fire two errors back-to-back
|
||||
mockSocket.emit("error", new Error("First error"));
|
||||
mockSocket.emit("error", new Error("Second error"));
|
||||
|
||||
await expect(pingPromise).rejects.toThrow("First error");
|
||||
expect(mockSocket.destroy).toHaveBeenCalledTimes(1);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
function createMockJavaResponse(response) {
|
||||
const jsonString = JSON.stringify(response);
|
||||
const jsonBuffer = Buffer.from(jsonString, "utf8");
|
||||
const responseLength = varint.encodeInt(jsonBuffer.length);
|
||||
const packetId = varint.encodeInt(0);
|
||||
const packetData = Buffer.concat([packetId, responseLength, jsonBuffer]);
|
||||
const packetLength = varint.encodeInt(packetData.length);
|
||||
return Buffer.concat([packetLength, packetData]);
|
||||
}
|
76
test/varint.test.js
Normal file
76
test/varint.test.js
Normal file
@ -0,0 +1,76 @@
|
||||
import { describe, it, expect } from "vitest";
|
||||
import varint from "../lib/varint.js";
|
||||
|
||||
describe("varint.js", () => {
|
||||
it("should encode and decode integers symmetrically (round-trip)", () => {
|
||||
const testValues = [
|
||||
0,
|
||||
1,
|
||||
127, // Max 1-byte
|
||||
128, // Min 2-byte
|
||||
255,
|
||||
16383, // Max 2-byte
|
||||
16384, // Min 3-byte
|
||||
2147483647, // Max signed 32-bit int
|
||||
-1, // Critical edge case (encodes as max unsigned int)
|
||||
];
|
||||
|
||||
testValues.forEach((value) => {
|
||||
const encoded = varint.encodeInt(value);
|
||||
const decoded = varint.decodeInt(encoded, 0);
|
||||
expect(decoded, `Value ${value} failed round-trip`).toBe(value);
|
||||
});
|
||||
});
|
||||
|
||||
it("should decode an integer from a non-zero offset", () => {
|
||||
// [255 (invalid varint), 128 (valid varint), 127 (valid varint)]
|
||||
const buffer = Buffer.from([0xff, 0x80, 0x01, 0x7f]);
|
||||
expect(varint.decodeInt(buffer, 1)).toBe(128);
|
||||
});
|
||||
|
||||
it("should throw an error for a malformed varint that is too long", () => {
|
||||
const invalidBuffer = Buffer.from([0x80, 0x80, 0x80, 0x80, 0x80, 0x80]);
|
||||
expect(() => varint.decodeInt(invalidBuffer, 0)).toThrow(
|
||||
"VarInt is too big"
|
||||
);
|
||||
});
|
||||
|
||||
it("should correctly predict the encoded length of a varint", () => {
|
||||
const boundaries = [0, 127, 128, 16383, 16384, 2097151, 2097152];
|
||||
boundaries.forEach((value) => {
|
||||
const predictedLength = varint.decodeLength(value);
|
||||
const actualLength = varint.encodeInt(value).length;
|
||||
expect(predictedLength).toBe(actualLength);
|
||||
});
|
||||
});
|
||||
|
||||
it("should encode 16-bit unsigned shorts in big-endian format", () => {
|
||||
expect(varint.encodeUShort(0)).toEqual(Buffer.from([0x00, 0x00]));
|
||||
expect(varint.encodeUShort(256)).toEqual(Buffer.from([0x01, 0x00]));
|
||||
expect(varint.encodeUShort(65535)).toEqual(Buffer.from([0xff, 0xff]));
|
||||
});
|
||||
|
||||
it("should correctly assemble and parse a Minecraft handshake packet", () => {
|
||||
const protocolVersion = -1;
|
||||
const virtualHost = "mc.example.com";
|
||||
const port = 25565;
|
||||
|
||||
const payload = Buffer.concat([
|
||||
varint.encodeInt(0),
|
||||
varint.encodeInt(protocolVersion),
|
||||
varint.encodeInt(virtualHost.length),
|
||||
varint.encodeString(virtualHost),
|
||||
varint.encodeUShort(port),
|
||||
varint.encodeInt(1),
|
||||
]);
|
||||
|
||||
const finalPacket = varint.concat([payload]);
|
||||
|
||||
const decodedPacketLength = varint.decodeInt(finalPacket, 0);
|
||||
expect(decodedPacketLength).toBe(payload.length);
|
||||
|
||||
const lengthOfPacketLength = varint.decodeLength(decodedPacketLength);
|
||||
const decodedPayload = finalPacket.subarray(lengthOfPacketLength);
|
||||
expect(decodedPayload).toEqual(payload);
|
||||
});
|
||||
});
|
Loading…
x
Reference in New Issue
Block a user