├── .gitignore
├── LICENSE
├── README.md
├── SECURITY.md
├── examples
├── 01-basic-ping.js
├── 02-error-handling.js
├── 03-server-dashboard.js
└── 04-cli.js
├── index.js
├── lib
├── bedrock.js
├── java.js
└── varint.js
├── package-lock.json
├── package.json
├── test
├── bedrock.test.js
├── java.test.js
└── varint.test.js
├── tsconfig.json
└── types
├── index.d.ts
└── lib
├── bedrock.d.ts
├── java.d.ts
└── varint.d.ts
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2022 Timofey Gelazoniya
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # mineping
2 |
3 | [](https://www.npmjs.com/package/@minescope/mineping)
4 |
5 | A simple and efficient JavaScript library for pinging Minecraft servers. It supports both Java and Bedrock editions through a async/await-friendly API.
6 |
7 | `@minescope/mineping` automatically resolves SRV records for Java servers and parses rich status data, including MOTD, player counts, version info, and server icons. Comes with full TypeScript support.
8 |
9 | *Mirror on my [
Git](https://git.zeldon.ru/zeldon/mineping)*
10 |
11 | ## Features
12 |
13 | - **Dual Protocol Support:** Ping both Java and Bedrock servers with a consistent API.
14 | - **SRV Record Resolution:** Automatically resolves SRV records for Java Edition servers, so you don't have to worry about custom ports.
15 | - **Rich Data:** Parses the full response from servers, including player samples, favicons, gamemodes, and more.
16 | - **Lightweight:** Has only **one** runtime dependency — [debug](https://www.npmjs.com/package/debug) (used for tracing).
17 |
18 | ## Requirements
19 |
20 | > **[Node.js](https://nodejs.org/) 14 or newer is required**
21 |
22 | ## Install
23 |
24 | To install `mineping`, simply run the following command:
25 |
26 | ```bash
27 | npm i @minescope/mineping
28 | ```
29 |
30 | ## API & Usage Examples
31 |
32 | The library exports two main functions: `pingJava` and `pingBedrock`. Both are asynchronous and return a `Promise`.
33 |
34 | ### 1. Basic Server Ping
35 |
36 | Java:
37 | ```js
38 | import { pingJava } from "@minescope/mineping";
39 |
40 | const data = await pingJava("0.0.0.0");
41 | console.log(data)
42 | ```
43 | ```js
44 | {
45 | version: { name: '1.21.5', protocol: 770 },
46 | enforcesSecureChat: true,
47 | description: '§1Welcome to §2My Minecraft Server!',
48 | players: { max: 20, online: 0 }
49 | }
50 | ```
51 |
52 | Bedrock:
53 | ```js
54 | import { pingBedrock } from "@minescope/mineping";
55 |
56 | const data = await pingBedrock("0.0.0.0");
57 | console.log(data)
58 | ```
59 | ```js
60 | {
61 | edition: 'MCPE',
62 | name: 'Dedicated Server',
63 | levelName: 'Bedrock level',
64 | gamemode: 'Survival',
65 | version: { protocol: 800, minecraft: '1.21.84' },
66 | players: { online: 0, max: 10 },
67 | port: { v4: 19132, v6: 19133 },
68 | guid: 12143264093420916401n,
69 | isNintendoLimited: false,
70 | isEditorModeEnabled: false
71 | }
72 | ```
73 |
74 | ## Loading and configuration the module
75 |
76 | ### CommonJS
77 |
78 | `mineping` is an ESM-only module — you are not able to import it with `require()`.
79 | If you cannot switch to ESM, you can use the async `import()` function from CommonJS to load `mineping` asynchronously:
80 |
81 | ```js
82 | const pingJava = (...args) =>
83 | import("@minescope/mineping").then((module) => module.pingJava(...args));
84 | const pingBedrock = (...args) =>
85 | import("@minescope/mineping").then((module) => module.pingBedrock(...args));
86 | ```
87 |
88 | ## Debugging
89 |
90 | `mineping` uses the [`debug`](https://www.npmjs.com/package/debug) library to provide detailed tracing information, which can be useful for diagnosing connection issues or understanding the library's internal workings.
91 |
92 | To enable debug logs, set the `DEBUG` environment variable when running your script. The library uses two namespaces:
93 |
94 | - `mineping:java` for the Java Edition pinger.
95 | - `mineping:bedrock` for the Bedrock Edition pinger.
96 |
97 | ### Examples
98 |
99 | **Enable all `mineping` debug logs:**
100 |
101 | You can use a wildcard (`*`) to enable all logs from this library.
102 |
103 | ```bash
104 | DEBUG=mineping:* node your-script.js
105 | ```
106 |
107 |
108 | Click to see output for DEBUG="mineping:*" node examples/01-basic-ping.js
109 |
110 | ```bash
111 | DEBUG="mineping:*" node examples/01-basic-ping.js
112 | mineping:java pinging Java server hypixel.net with options: {} +0ms
113 | mineping:java attempting SRV lookup for _minecraft._tcp.hypixel.net with 5000ms timeout +2ms
114 | mineping:java SRV lookup successful, new target: mc.hypixel.net:25565 +2ms
115 | mineping:java creating TCP connection to mc.hypixel.net:25565 +0ms
116 | mineping:java socket connected to mc.hypixel.net:25565, sending packets... +182ms
117 | mineping:java received 1440 bytes of data, total buffer size is now 1440 bytes +130ms
118 | mineping:java packet incomplete, waiting for more data +0ms
119 | mineping:java received 12960 bytes of data, total buffer size is now 14400 bytes +1ms
120 | mineping:java packet incomplete, waiting for more data +0ms
121 | mineping:java received 1601 bytes of data, total buffer size is now 16001 bytes +129ms
122 | mineping:java received raw JSON response +0ms
123 | mineping:java successfully parsed full response +0ms
124 | mineping:java cleaning up resources for mc.hypixel.net:25565 +0ms
125 | --- Java Server ---
126 | {
127 | version: { name: 'Requires MC 1.8 / 1.21', protocol: 47 },
128 | players: { max: 200000, online: 28654, sample: [] },
129 | description: ' §aHypixel Network §c[1.8-1.21]\n' +
130 | ' §6§lSB 0.23 §2§lFORAGING §8§l- §e§lSUMMER EVENT',
131 | favicon: ''... 5738 more characters
132 | }
133 |
134 | ====================
135 |
136 | mineping:bedrock pinging Bedrock server geo.hivebedrock.network:19132 with 5000ms timeout +0ms
137 | mineping:bedrock sending Unconnected Ping packet to geo.hivebedrock.network:19132 +1ms
138 | mineping:bedrock packet: +0ms
139 | mineping:bedrock received 124 bytes from geo.hivebedrock.network:19132 +104ms
140 | mineping:bedrock received raw MOTD string: MCPE;BEDWARS + BUILD BATTLE;121;1.0;13074;100001;-4669279440237021648;Hive Games;Survival +0ms
141 | mineping:bedrock cleaning up resources for geo.hivebedrock.network:19132 +0ms
142 | --- Bedrock Server ---
143 | {
144 | edition: 'MCPE',
145 | name: 'BEDWARS + BUILD BATTLE',
146 | levelName: 'Hive Games',
147 | gamemode: 'Survival',
148 | version: { protocol: 121, minecraft: '1.0' },
149 | players: { online: 13074, max: 100001 },
150 | port: { v4: undefined, v6: undefined },
151 | guid: -4669279440237021648n,
152 | isNintendoLimited: undefined,
153 | isEditorModeEnabled: undefined
154 | }
155 | ```
156 |
157 |
158 | **_PowerShell_ uses different syntax to set environment variables:**
159 |
160 | ```powershell
161 | $env:DEBUG="mineping:*";node your-script.js
162 | ```
163 |
164 | ## Acknowledgements
165 |
166 | Special thanks to the following projects for inspiration and protocol details:
167 |
168 | - [mcping](https://github.com/Scetch/mcping) crate for Rust
169 | - [mcping-js](https://github.com/Cryptkeeper/mcping-js) library for querying Minecraft Java Edition servers
170 | - The amazing community at [minecraft.wiki](https://minecraft.wiki/) for documenting the protocols.
171 |
--------------------------------------------------------------------------------
/SECURITY.md:
--------------------------------------------------------------------------------
1 | # Reporting a Vulnerability
2 |
3 | If you believe you have found a security vulnerability, let me know by sending email to timofey@z4n.me I will investigate that and do my best to quickly fix the problem.
4 |
5 | Please don't open an issue to or discuss this security vulnerability in a public place. Thanks for understanding!
6 |
--------------------------------------------------------------------------------
/examples/01-basic-ping.js:
--------------------------------------------------------------------------------
1 | import { pingJava, pingBedrock } from "../index.js";
2 |
3 | try {
4 | const javaData = await pingJava("hypixel.net");
5 | console.log("--- Java Server ---");
6 | console.log(javaData);
7 | } catch (error) {
8 | console.error("Could not ping Java server:", error);
9 | }
10 |
11 | console.log("\n" + "=".repeat(20) + "\n");
12 |
13 | try {
14 | const motd = await pingBedrock("geo.hivebedrock.network");
15 | console.log("--- Bedrock Server ---");
16 | console.log(motd);
17 | } catch (error) {
18 | console.error("Could not ping Bedrock server:", error);
19 | }
20 |
--------------------------------------------------------------------------------
/examples/02-error-handling.js:
--------------------------------------------------------------------------------
1 | import { pingJava } from "../index.js";
2 |
3 | const offlineServer = "this.server.does.not.exist";
4 | const port = 12345;
5 |
6 | console.log(`Pinging an offline server: ${offlineServer}:${port}`);
7 |
8 | try {
9 | // We set a short timeout to fail faster.
10 | const data = await pingJava(offlineServer, { port, timeout: 500 });
11 | console.log("Success!?", data);
12 | } catch (error) {
13 | console.error("Caught expected error:", error.message);
14 | }
15 |
--------------------------------------------------------------------------------
/examples/03-server-dashboard.js:
--------------------------------------------------------------------------------
1 | import { pingJava, pingBedrock } from "../index.js";
2 |
3 | const servers = [
4 | { type: "Java", host: "mc.hypixel.net" },
5 | { type: "Java", host: "play.cubecraft.net" },
6 | { type: "Java", host: "an-offline-java-server.com" },
7 | { type: "Bedrock", host: "geo.hivebedrock.network" },
8 | { type: "Bedrock", host: "buzz.insanitycraft.net" },
9 | { type: "Bedrock", host: "an.offline.bedrock.server" },
10 | ];
11 |
12 | console.log("Pinging all servers...");
13 |
14 | // Create an array of ping promises
15 | const pingPromises = servers.map((server) => {
16 | if (server.type === "Java") {
17 | return pingJava(server.host, { timeout: 3000 });
18 | } else {
19 | return pingBedrock(server.host, { timeout: 3000 });
20 | }
21 | });
22 |
23 | // Wait for all pings to complete (or fail)
24 | const results = await Promise.allSettled(pingPromises);
25 |
26 | // Process and display results
27 | const displayData = results.map((result, index) => {
28 | const server = servers[index];
29 | if (result.status === "fulfilled") {
30 | const data = result.value;
31 | return {
32 | Server: `${server.type} - ${server.host}`,
33 | Status: "✅ Online",
34 | Players: `${data.players.online} / ${data.players.max}`,
35 | Version: data.version.name ?? data.version.minecraft,
36 | };
37 | } else {
38 | return {
39 | Server: `${server.type} - ${server.host}`,
40 | Status: "❌ Offline",
41 | Players: "N/A",
42 | Version: `Error: ${result.reason.message.slice(0, 30)}...`,
43 | };
44 | }
45 | });
46 |
47 | console.table(displayData);
48 |
--------------------------------------------------------------------------------
/examples/04-cli.js:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env node
2 |
3 | /**
4 | * Usage examples:
5 | * - Java (with custom timeout): node cli.js -j --host="mc.hypixel.net" --timeout 1000
6 | * - Bedrock: node cli.js -b --host="play.timecrack.net"
7 | */
8 |
9 | import { pingBedrock, pingJava } from "../index.js";
10 |
11 | const DEFAULT_TIMEOUT = 5000;
12 | const JAVA_DEFAULT_PORT = 25565;
13 | const BEDROCK_DEFAULT_PORT = 19132;
14 |
15 | try {
16 | const args = parseArgs(process.argv.slice(2));
17 |
18 | if (shouldShowHelp(args)) {
19 | printHelp();
20 | process.exit(0);
21 | }
22 |
23 | validateArgs(args);
24 |
25 | const port = Number(args.port) || getDefaultPort(args);
26 | const timeout = Number(args.timeout) || DEFAULT_TIMEOUT;
27 |
28 | if (args.j) {
29 | await pingJavaServer(args.host, port, timeout);
30 | } else if (args.b) {
31 | await pingBedrockServer(args.host, port, timeout);
32 | }
33 | } catch (err) {
34 | console.error(`ERROR: ${err.message}`);
35 | process.exit(1);
36 | }
37 |
38 | function parseArgs(rawArgs) {
39 | const args = {};
40 |
41 | for (let i = 0; i < rawArgs.length; i++) {
42 | const arg = rawArgs[i];
43 |
44 | if (arg.startsWith("--")) {
45 | // Handle --key=value and --key value formats
46 | const [key, value] = arg.slice(2).split("=");
47 | args[key] = value ?? rawArgs[++i] ?? true;
48 | } else if (arg.startsWith("-")) {
49 | // Handle short flags (-j, -b, -h)
50 | const flags = arg.slice(1).split("");
51 | flags.forEach((flag) => {
52 | args[flag] = true;
53 | });
54 | }
55 | }
56 |
57 | return args;
58 | }
59 |
60 | function validateArgs(args) {
61 | if (args.j && args.b) {
62 | printInterestingFacts();
63 | process.exit(0);
64 | }
65 |
66 | if (!args.host) {
67 | throw new Error("The host argument not found! Use -h or --help.");
68 | }
69 |
70 | if (!args.j && !args.b) {
71 | throw new Error("Must specify either -j or -b flag. Use -h or --help.");
72 | }
73 |
74 | if (args.port && (isNaN(args.port) || args.port < 1 || args.port > 65535)) {
75 | throw new Error("Port must be a number between 1 and 65535");
76 | }
77 |
78 | if (args.timeout && (isNaN(args.timeout) || args.timeout < 0)) {
79 | throw new Error("Timeout must be a positive number");
80 | }
81 | }
82 |
83 | function shouldShowHelp(args) {
84 | return args.help || args.h || Object.keys(args).length === 0;
85 | }
86 |
87 | function printHelp() {
88 | console.log(`node cli.js [..]
89 | A simple to use, efficient, and full-featured Minecraft server info parser!
90 |
91 | USAGE:
92 | node cli.js [OPTIONS] --host --port --timeout
93 |
94 | OPTIONS:
95 | -j Use for Minecraft Java Edition
96 | -b Use for Minecraft Bedrock Edition
97 | -h, --help Show this help message
98 |
99 | --host The server address (required)
100 | --port The server port (default: ${JAVA_DEFAULT_PORT} for Java, ${BEDROCK_DEFAULT_PORT} for Bedrock)
101 | --timeout The socket timeout in milliseconds (default: ${DEFAULT_TIMEOUT})
102 |
103 | P.S. Don't use -j and -b at the same time!`);
104 | }
105 |
106 | function printInterestingFacts() {
107 | console.log(`Some interesting facts about MOTDs on bedrock:
108 | - so far they seem to exclusively use legacy color codes
109 | - the random style has a special impl for periods, they turn into animated
110 | colons that warp up and down rapidly
111 | - motd_2 is ignored? client displays "motd_1 - v{version}", where the
112 | appended version text is considered part of motd_1 for color code processing
113 | - motd_2 seems to mainly be used to return the server software in use (e.g. PocketMine-MP)`);
114 | }
115 |
116 | function getDefaultPort(args) {
117 | return args.j ? JAVA_DEFAULT_PORT : BEDROCK_DEFAULT_PORT;
118 | }
119 |
120 | async function pingJavaServer(host, port, timeout) {
121 | const data = await pingJava(host, { port, timeout });
122 | console.log(`Host: ${host}
123 | Version: ${data.version.name} (protocol: ${data.version.protocol})
124 | Players: ${data.players?.online}/${data.players?.max}
125 | Description: ${
126 | typeof data.description === "string"
127 | ? data.description
128 | : data.description?.text
129 | }`);
130 | }
131 |
132 | async function pingBedrockServer(host, port, timeout) {
133 | const data = await pingBedrock(host, { port, timeout });
134 | console.log(`Host: ${host}
135 | Edition: ${data.edition}
136 | Version: ${data.version.minecraft} (protocol: ${data.version.protocol})
137 | Players: ${data.players.online}/${data.players.max}
138 | Name: ${data.name}
139 | Gamemode: ${data.gamemode}`);
140 | }
141 |
--------------------------------------------------------------------------------
/index.js:
--------------------------------------------------------------------------------
1 | export { pingJava } from './lib/java.js';
2 | export { pingBedrock } from './lib/bedrock.js';
--------------------------------------------------------------------------------
/lib/bedrock.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Implementation of the RakNet ping/pong protocol.
3 | * @see https://minecraft.wiki/w/RakNet
4 | */
5 |
6 | "use strict";
7 |
8 | import dgram from "node:dgram";
9 | import crypto from "node:crypto";
10 | import createDebug from "debug";
11 |
12 | const debug = createDebug("mineping:bedrock");
13 |
14 | const MAGIC = "00ffff00fefefefefdfdfdfd12345678";
15 | const START_TIME = Date.now();
16 | const UNCONNECTED_PONG = 0x1c;
17 |
18 | /**
19 | * Representation of raw, semicolon-delimited MOTD string.
20 | * This struct directly mirrors the fields and order from the server response.
21 | * See [`Unconnected Pong Documentation`](https://minecraft.wiki/w/RakNet#Unconnected_Pong) for more details.
22 | * @typedef {object} BedrockMotd
23 | * @property {string} edition - The edition of the server (MCPE or MCEE).
24 | * @property {string} name - The primary name of the server (first line of MOTD).
25 | * @property {number} protocol - The protocol version.
26 | * @property {string} version - The game version (e.g., "1.21.2").
27 | * @property {number} playerCount - The current number of players online.
28 | * @property {number} playerMax - The maximum number of players allowed.
29 | * @property {bigint} serverGuid - The server's GUID.
30 | * @property {string} subName - The secondary name of the server (second line of MOTD).
31 | * @property {string} gamemode - The default gamemode (e.g., "Survival").
32 | * @property {boolean} [nintendoLimited] - Whether the server is Nintendo limited.
33 | * @property {number} [port] - The server's IPv4 port, if provided.
34 | * @property {number} [ipv6Port] - The server's IPv6 port, if provided.
35 | * @property {boolean} [editorMode] - Whether the server is in editor mode, if provided. See [Minecraft Editor Mode Documentation](https://learn.microsoft.com/en-us/minecraft/creator/documents/bedrockeditor/editoroverview?view=minecraft-bedrock-stable) for more details.
36 | */
37 |
38 | /**
39 | * Represents the structured and user-friendly response from a server ping.
40 | * This is the public-facing object that users of the library will receive.
41 | * @typedef {object} BedrockPingResponse
42 | * @property {string} edition - The edition of the server (MCPE or MCEE).
43 | * @property {string} name - The primary name of the server (first line of MOTD).
44 | * @property {string} levelName - The name of the world or level being hosted.
45 | * @property {string} gamemode - The default gamemode of the server.
46 | * @property {{ protocol: number, minecraft: string }} version - Game and protocol versions.
47 | * @property {{ online: number, max: number }} players - Current and maximum player counts.
48 | * @property {{ v4?: number, v6?: number }} port - Announced IPv4 and IPv6 ports.
49 | * @property {bigint} guid - The server's unique 64-bit GUID.
50 | * @property {boolean} [isNintendoLimited] - True if the server restricts Nintendo Switch players.
51 | * @property {boolean} [isEditorModeEnabled] - True if the server is in editor mode. See [Minecraft Editor Mode Documentation](https://learn.microsoft.com/en-us/minecraft/creator/documents/bedrockeditor/editoroverview?view=minecraft-bedrock-stable) for more details.
52 | */
53 |
54 | /**
55 | * @typedef {object} BedrockPingOptions
56 | * @property {number} [port=19132] - The server port to ping.
57 | * @property {number} [timeout=5000] - The timeout in milliseconds for the request.
58 | */
59 |
60 | /**
61 | * Creates an Unconnected Ping packet.
62 | * See [Unconnected Ping Documentation](https://minecraft.wiki/w/RakNet#Unconnected_Ping) for more details.
63 | * @param {number} timestamp - The current time delta since the script started.
64 | * @returns {Buffer}
65 | */
66 | const createUnconnectedPingFrame = (timestamp) => {
67 | const buffer = Buffer.alloc(33);
68 | buffer.writeUInt8(0x01, 0); // Packet ID
69 | buffer.writeBigInt64LE(BigInt(timestamp), 1); // Timestamp
70 | Buffer.from(MAGIC, "hex").copy(buffer, 9); // OFFLINE_MESSAGE_DATA_ID (Magic bytes)
71 | Buffer.from(crypto.randomBytes(8)).copy(buffer, 25); // Client GUID
72 | return buffer;
73 | };
74 |
75 | /**
76 | * Parses the semicolon-delimited MOTD string into a structured object.
77 | * @param {string} motdString - The raw MOTD string from the server.
78 | * @returns {BedrockMotd} The parsed internal MOTD object.
79 | * @throws {Error} If the MOTD string is missing required fields.
80 | */
81 | const parseMotd = (motdString) => {
82 | const parts = motdString.split(";");
83 |
84 | if (parts.length < 5) {
85 | throw new Error(
86 | `Invalid MOTD format: Expected at least 5 fields, but got ${parts.length}.`
87 | );
88 | }
89 |
90 | const [
91 | edition,
92 | name,
93 | protocolStr,
94 | version,
95 | playerCountStr,
96 | playerMaxStr,
97 | serverGuidStr,
98 | subName,
99 | gamemode,
100 | nintendoLimitedStr,
101 | port,
102 | ipv6Port,
103 | editorModeStr,
104 | ] = parts;
105 |
106 | let nintendoLimited;
107 | if (nintendoLimitedStr === "0") {
108 | nintendoLimited = true;
109 | } else if (nintendoLimitedStr === "1") {
110 | nintendoLimited = false;
111 | }
112 |
113 | return {
114 | edition,
115 | name,
116 | protocol: Number(protocolStr),
117 | version,
118 | playerCount: Number(playerCountStr),
119 | playerMax: Number(playerMaxStr),
120 | serverGuid: BigInt(serverGuidStr),
121 | subName,
122 | gamemode,
123 | nintendoLimited,
124 | port: port ? Number(port) : undefined,
125 | ipv6Port: ipv6Port ? Number(ipv6Port) : undefined,
126 | editorMode: editorModeStr ? Boolean(Number(editorModeStr)) : undefined,
127 | };
128 | };
129 |
130 | /**
131 | * Transforms the raw MOTD object into a user-friendly, nested structure.
132 | * @param {BedrockMotd} motd - The parsed MOTD object.
133 | * @returns {BedrockPingResponse} The final, user-facing response object.
134 | */
135 | const transformMotd = (motd) => {
136 | return {
137 | edition: motd.edition,
138 | name: motd.name,
139 | levelName: motd.subName,
140 | gamemode: motd.gamemode,
141 | version: {
142 | protocol: motd.protocol,
143 | minecraft: motd.version,
144 | },
145 | players: {
146 | online: motd.playerCount,
147 | max: motd.playerMax,
148 | },
149 | port: {
150 | v4: motd.port,
151 | v6: motd.ipv6Port,
152 | },
153 | guid: motd.serverGuid,
154 | isNintendoLimited: motd.nintendoLimited,
155 | isEditorModeEnabled: motd.editorMode,
156 | };
157 | };
158 |
159 | /**
160 | * Extracts the MOTD string from an Unconnected Pong packet and parses it.
161 | * @param {Buffer} pongPacket - The raw pong packet from the server.
162 | * @returns {BedrockPingResponse} The final response object.
163 | * @throws {Error} If the packet is malformed.
164 | */
165 | const parseUnconnectedPong = (pongPacket) => {
166 | if (!Buffer.isBuffer(pongPacket) || pongPacket.length < 35) {
167 | throw new Error("Invalid pong packet: buffer is too small.");
168 | }
169 |
170 | const packetId = pongPacket.readUInt8(0);
171 | if (packetId !== UNCONNECTED_PONG) {
172 | throw new Error(
173 | `Unexpected packet ID: 0x${packetId.toString(16)}. Expected 0x1c.`
174 | );
175 | }
176 |
177 | // The MOTD string is prefixed with its length as a 16-bit big-endian integer
178 | const motdLength = pongPacket.readUInt16BE(33);
179 | const motdOffset = 35;
180 |
181 | if (motdOffset + motdLength > pongPacket.length) {
182 | throw new Error("Malformed pong packet: MOTD length exceeds buffer size.");
183 | }
184 |
185 | const motdString = pongPacket.toString(
186 | "utf-8",
187 | motdOffset,
188 | motdOffset + motdLength
189 | );
190 | debug("received raw MOTD string: %s", motdString);
191 |
192 | const rawMotd = parseMotd(motdString);
193 | const motd = transformMotd(rawMotd);
194 | return motd;
195 | };
196 |
197 | /**
198 | * Asynchronously pings a Minecraft Bedrock server.
199 | * @param {string} host - The IP address or hostname of the server.
200 | * @param {BedrockPingOptions} [options={}] - Optional configuration.
201 | * @returns {Promise} A promise that resolves with the server's parsed MOTD.
202 | */
203 | export async function pingBedrock(host, options = {}) {
204 | if (!host) {
205 | throw new Error("Host argument is required.");
206 | }
207 |
208 | const { port = 19132, timeout = 5000 } = options;
209 | debug("pinging Bedrock server %s:%d with %dms timeout", host, port, timeout);
210 |
211 | return new Promise((resolve, reject) => {
212 | const socket = dgram.createSocket("udp4");
213 |
214 | // Prevent cleanup tasks from running more than once
215 | // in case of multiple error callbacks
216 | let isCleanupCompleted = false;
217 |
218 | // Set a manual timeout interval to ensure
219 | // the connection will NEVER hang regardless of internal state
220 | const timeoutTask = setTimeout(() => {
221 | socket.emit("error", new Error("Socket timeout"));
222 | }, timeout);
223 |
224 | // Idempotent function to handle cleanup tasks, we can safely call it multiple times without side effects
225 | const cleanup = () => {
226 | if (isCleanupCompleted) return;
227 | isCleanupCompleted = true;
228 | debug("cleaning up resources for %s:%d", host, port);
229 | clearTimeout(timeoutTask);
230 | socket.close();
231 | };
232 |
233 | // Generic error handler
234 | socket.on("error", (err) => {
235 | debug("socket error for %s:%d - %s", host, port, err.message);
236 | cleanup();
237 | reject(err);
238 | });
239 |
240 | socket.on("message", (pongPacket) => {
241 | debug("received %d bytes from %s:%d", pongPacket.length, host, port);
242 | try {
243 | const motd = parseUnconnectedPong(pongPacket);
244 | cleanup();
245 | resolve(motd);
246 | } catch (err) {
247 | socket.emit("error", err);
248 | }
249 | });
250 |
251 | try {
252 | const pingPacket = createUnconnectedPingFrame(Date.now() - START_TIME);
253 | debug("sending Unconnected Ping packet to %s:%d", host, port);
254 | debug("packet: %o", pingPacket);
255 | socket.send(pingPacket, 0, pingPacket.length, port, host);
256 | } catch (err) {
257 | // Handle any immediate, synchronous errors that might occur when sending the ping packet
258 | socket.emit("error", err);
259 | }
260 | });
261 | }
262 |
--------------------------------------------------------------------------------
/lib/java.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Implementation of the Java Minecraft ping protocol.
3 | * @see https://minecraft.wiki/w/Java_Edition_protocol/Server_List_Ping
4 | */
5 |
6 | "use strict";
7 |
8 | import { createConnection, isIP } from "node:net";
9 | import { Resolver } from "node:dns/promises";
10 | import createDebug from "debug";
11 | import * as varint from "./varint.js";
12 |
13 | const debug = createDebug("mineping:java");
14 |
15 | /**
16 | * Represents the structured and user-friendly response from a server ping.
17 | * The fields and their optionality are based on the protocol documentation.
18 | * See [Status Response Documentation](https://minecraft.wiki/w/Java_Edition_protocol/Server_List_Ping#Status_Response) for more details.
19 | * @typedef {object} JavaPingResponse
20 | * @property {{ name: string, protocol: number }} version - Contains the server's version name and protocol number.
21 | * @property {{ max: number, online: number, sample?: Array<{ name: string, id: string }> }} [players] - Player count and a sample of online players.
22 | * @property {object | string} [description] - The server's Message of the Day (MOTD).
23 | * @property {string} [favicon] - A Base64-encoded 64x64 PNG image data URI.
24 | * @property {boolean} [enforcesSecureChat] - True if the server requires clients to have a Mojang-signed public key.
25 | * @property {boolean} [preventsChatReports] - True if a mod is installed to disable chat reporting.
26 | */
27 |
28 | /**
29 | * @typedef {object} JavaPingOptions
30 | * @property {number} [port=25565] - The fallback port if an SRV record is not found.
31 | * @property {number} [timeout=5000] - The connection timeout in milliseconds.
32 | * @property {number} [protocolVersion=-1] - The protocol version to use in the handshake. `-1` is for auto-detection.
33 | */
34 |
35 | /**
36 | * Creates the Handshake packet.
37 | * @param {string} host The hostname to connect to
38 | * @param {number} port The port to connect to
39 | * @param {number} protocolVersion The protocol version to use
40 | * @returns {Buffer} The complete Handshake packet
41 | */
42 | function createHandshakePacket(host, port, protocolVersion) {
43 | const hostBuffer = varint.encodeString(host);
44 |
45 | const payload = [
46 | varint.encodeVarInt(0x00), // Packet ID
47 | varint.encodeVarInt(protocolVersion),
48 | varint.encodeVarInt(hostBuffer.length),
49 | hostBuffer,
50 | varint.encodeUShort(port),
51 | varint.encodeVarInt(1), // Next state: 1 for Status
52 | ];
53 |
54 | return varint.concatPackets(payload);
55 | }
56 |
57 | /**
58 | * Creates the Status Request packet.
59 | * @returns {Buffer} The complete Status Request packet
60 | */
61 | function createStatusRequestPacket() {
62 | const payload = [
63 | varint.encodeVarInt(0x00), // Packet ID
64 | ];
65 | return varint.concatPackets(payload);
66 | }
67 |
68 | /**
69 | * Attempts to parse the server status response from the buffer.
70 | * @param {Buffer} buffer The incoming data buffer
71 | * @returns {{ response: JavaPingResponse, remainder: Buffer } | null} The parsed response and the remaining buffer, or null if the packet is incomplete
72 | */
73 | function processResponse(buffer) {
74 | let offset = 0;
75 |
76 | try {
77 | const packetLengthResult = varint.decodeVarInt(buffer, offset);
78 | const packetLength = packetLengthResult.value;
79 | offset += packetLengthResult.bytesRead;
80 |
81 | // Check if the full packet has arrived yet.
82 | if (buffer.length < offset + packetLength) {
83 | debug("packet incomplete, waiting for more data");
84 | return null; // Incomplete packet, wait for more data.
85 | }
86 |
87 | const packetIdResult = varint.decodeVarInt(buffer, offset);
88 | if (packetIdResult.value !== 0x00) {
89 | throw new Error(
90 | `Unexpected packet ID: ${packetIdResult.value}. Expected 0x00.`
91 | );
92 | }
93 | offset += packetIdResult.bytesRead;
94 |
95 | const jsonLengthResult = varint.decodeVarInt(buffer, offset);
96 | const jsonLength = jsonLengthResult.value;
97 | offset += jsonLengthResult.bytesRead;
98 |
99 | if (buffer.length < offset + jsonLength) {
100 | debug("JSON string incomplete, waiting for more data");
101 | return null; // Incomplete JSON string, wait for more data.
102 | }
103 |
104 | const jsonString = buffer
105 | .subarray(offset, offset + jsonLength)
106 | .toString("utf8");
107 | debug("received raw JSON response");
108 | const response = JSON.parse(jsonString);
109 |
110 | // Return the response and any data that came after this packet.
111 | const remainder = buffer.subarray(offset + jsonLength);
112 |
113 | return { response, remainder };
114 | } catch (err) {
115 | // If the buffer is too short for a VarInt, it's a recoverable state.
116 | if (err instanceof varint.VarIntError) {
117 | if (err.code === varint.ERR_VARINT_BUFFER_UNDERFLOW) {
118 | debug("buffer underflow while parsing VarInt, waiting for more data");
119 | return null; // Wait for more data.
120 | }
121 | // For malformed VarInts or JSON, throw the error to reject the promise.
122 | throw err;
123 | }
124 | throw err;
125 | }
126 | }
127 |
128 | /**
129 | * Asynchronously Pings a Minecraft Java Edition server.
130 | * This function performs an SRV lookup and then attempts to connect and retrieve the server status.
131 | * @param {string} host - The server address to ping.
132 | * @param {JavaPingOptions} [options={}] - Optional configuration.
133 | * @returns {Promise} A promise that resolves with the server's status.
134 | */
135 | export async function pingJava(host, options = {}) {
136 | if (typeof host !== "string" || host.trim() === "") {
137 | throw new Error("Host argument is required.");
138 | }
139 |
140 | const {
141 | port: fallbackPort = 25565,
142 | timeout = 5000,
143 | protocolVersion = -1,
144 | } = options;
145 | debug("pinging Java server %s with options: %o", host, options);
146 |
147 | let targetHost = host;
148 | let targetPort = fallbackPort;
149 |
150 | // A list of hostnames that should never have an SRV lookup.
151 | const nonSrvLookableHostnames = ["localhost"];
152 |
153 | // Check if the host is a valid IP address (v4 or v6).
154 | // net.isIP() returns 0 for invalid IPs, 4 for IPv4, and 6 for IPv6.
155 | const isDirectIp = isIP(host) !== 0;
156 | const isNonLookableHostname = nonSrvLookableHostnames.includes(
157 | host.toLowerCase()
158 | );
159 |
160 | if (isDirectIp || isNonLookableHostname) {
161 | debug(
162 | "host '%s' is a direct IP or a non-lookable hostname, skipping SRV lookup.",
163 | host
164 | );
165 | } else {
166 | try {
167 | debug(
168 | "attempting SRV lookup for _minecraft._tcp.%s with %dms timeout",
169 | host,
170 | timeout
171 | );
172 | const resolver = new Resolver({ timeout, tries: 3 });
173 | const srvRecords = await resolver.resolveSrv(`_minecraft._tcp.${host}`);
174 | if (srvRecords.length > 0) {
175 | targetHost = srvRecords[0].name;
176 | targetPort = srvRecords[0].port;
177 | debug(
178 | "SRV lookup successful, new target: %s:%d",
179 | targetHost,
180 | targetPort
181 | );
182 | }
183 | } catch (err) {
184 | // Common errors like ENODATA, ENOTFOUND, or a DNS timeout (ETIMEOUT) are expected
185 | // when a server does not have an SRV record, so we ignore them and proceed.
186 | const nonFatalDnsCodes = ["ENODATA", "ENOTFOUND", "ETIMEOUT"];
187 | if (
188 | err instanceof Error &&
189 | "code" in err &&
190 | nonFatalDnsCodes.includes(err.code)
191 | ) {
192 | debug("SRV lookup for %s failed (%s), using fallback.", host, err.code);
193 | } else {
194 | // Re-throw anything else to fail the operation
195 | debug("SRV lookup for %s failed unexpectedly, re-throwing.", host, err);
196 | throw err;
197 | }
198 | }
199 | }
200 |
201 | return new Promise((resolve, reject) => {
202 | debug("creating TCP connection to %s:%d", targetHost, targetPort);
203 | const socket = createConnection({ host: targetHost, port: targetPort });
204 |
205 | // Prevent cleanup tasks from running more than once
206 | // in case of multiple error callbacks
207 | let isCleanupCompleted = false;
208 |
209 | // Set a manual timeout interval to ensure
210 | // the connection will NEVER hang regardless of internal state
211 | const timeoutTask = setTimeout(() => {
212 | socket.emit("error", new Error("Socket timeout"));
213 | }, timeout);
214 |
215 | // Idempotent function to handle cleanup tasks, we can safely call it multiple times without side effects
216 | const cleanup = () => {
217 | if (isCleanupCompleted) return;
218 | isCleanupCompleted = true;
219 | debug("cleaning up resources for %s:%d", targetHost, targetPort);
220 | clearTimeout(timeoutTask);
221 | socket.destroy();
222 | };
223 |
224 | // #setNoDelay instantly flushes data during read/writes
225 | // This prevents the runtime from delaying the write at all
226 | socket.setNoDelay(true);
227 |
228 | // Generic error handler
229 | socket.on("error", (err) => {
230 | debug("socket error for %s:%d - %s", targetHost, targetPort, err.message);
231 | cleanup();
232 | reject(err);
233 | });
234 |
235 | socket.on("close", () => {
236 | if (!isCleanupCompleted) {
237 | debug("socket for %s:%d closed prematurely", targetHost, targetPort);
238 | cleanup();
239 | reject(new Error("Socket closed unexpectedly without a response."));
240 | }
241 | });
242 |
243 | socket.on("connect", () => {
244 | debug(
245 | "socket connected to %s:%d, sending packets...",
246 | targetHost,
247 | targetPort
248 | );
249 | try {
250 | const handshakePacket = createHandshakePacket(
251 | host,
252 | targetPort,
253 | protocolVersion
254 | );
255 | const statusRequestPacket = createStatusRequestPacket();
256 | socket.write(handshakePacket);
257 | socket.write(statusRequestPacket);
258 | } catch (err) {
259 | // Handle synchronous errors during packet creation/writing
260 | socket.emit("error", err);
261 | }
262 | });
263 |
264 | let incomingBuffer = Buffer.alloc(0);
265 |
266 | socket.on("data", (data) => {
267 | debug(
268 | "received %d bytes of data, total buffer size is now %d bytes",
269 | data.length,
270 | incomingBuffer.length + data.length
271 | );
272 | incomingBuffer = Buffer.concat([incomingBuffer, data]);
273 |
274 | try {
275 | const result = processResponse(incomingBuffer);
276 | if (result) {
277 | debug("successfully parsed full response");
278 | // We successfully parsed a response. Clean up before resolving.
279 | cleanup();
280 | resolve(result.response);
281 | }
282 | // If result is null, we just wait for more data to arrive.
283 | } catch (err) {
284 | socket.emit("error", err);
285 | }
286 | });
287 | });
288 | }
289 |
--------------------------------------------------------------------------------
/lib/varint.js:
--------------------------------------------------------------------------------
1 | // https://minecraft.wiki/w/Java_Edition_protocol/Data_types
2 |
3 | "use strict";
4 |
5 | export const ERR_VARINT_BUFFER_UNDERFLOW = "VARINT_BUFFER_UNDERFLOW";
6 | export const ERR_VARINT_MALFORMED = "VARINT_MALFORMED";
7 | export const ERR_VARINT_ENCODE_TOO_LARGE = "VARINT_ENCODE_TOO_LARGE";
8 |
9 | export class VarIntError extends Error {
10 | /**
11 | * @param {string} message The error message.
12 | * @param {string} code The error code.
13 | */
14 | constructor(message, code) {
15 | super(message);
16 | this.name = "VarIntError";
17 | this.code = code;
18 | }
19 | }
20 |
21 | /**
22 | * Encodes an integer into a VarInt buffer.
23 | * VarInts are never longer than 5 bytes for the Minecraft protocol.
24 | * @param {number} value The integer to encode
25 | * @returns {Buffer} The encoded VarInt as a buffer
26 | * @throws {VarIntError} if the value is too large to be encoded
27 | */
28 | export function encodeVarInt(value) {
29 | const buf = Buffer.alloc(5);
30 | let written = 0;
31 | let val = value;
32 |
33 | while (true) {
34 | const byte = val & 0x7f;
35 | val >>>= 7;
36 |
37 | if (val === 0) {
38 | buf.writeUInt8(byte, written++);
39 | break;
40 | }
41 |
42 | buf.writeUInt8(byte | 0x80, written++);
43 |
44 | if (written >= 5 && val > 0) {
45 | throw new VarIntError(
46 | "Value too large for a 5-byte VarInt",
47 | ERR_VARINT_ENCODE_TOO_LARGE
48 | );
49 | }
50 | }
51 |
52 | return buf.subarray(0, written);
53 | }
54 |
55 | /**
56 | * Encodes a string into a UTF-8 buffer.
57 | * @param {string} value The string to encode
58 | * @returns {Buffer}
59 | */
60 | export function encodeString(value) {
61 | return Buffer.from(value, "utf-8");
62 | }
63 |
64 | /**
65 | * Encodes an unsigned short (16-bit big-endian) into a 2-byte buffer.
66 | * @param {number} value The number to encode
67 | * @returns {Buffer}
68 | */
69 | export function encodeUShort(value) {
70 | const buf = Buffer.alloc(2);
71 | buf.writeUInt16BE(value, 0);
72 | return buf;
73 | }
74 |
75 | /**
76 | * Creates a Minecraft-style packet by concatenating chunks and prefixing the total length as a VarInt.
77 | * @param {Buffer[]} chunks An array of buffers to include in the packet payload
78 | * @returns {Buffer} The complete packet with its length prefix
79 | */
80 | export function concatPackets(chunks) {
81 | const payload = Buffer.concat(chunks);
82 | const lengthPrefix = encodeVarInt(payload.length);
83 | return Buffer.concat([lengthPrefix, payload]);
84 | }
85 |
86 | /**
87 | * Decodes a VarInt from a buffer.
88 | * Returns the decoded value and the number of bytes it consumed.
89 | * @param {Buffer} buffer The buffer to read from
90 | * @param {number} [offset=0] The starting offset in the buffer
91 | * @returns {{ value: number, bytesRead: number }}
92 | * @throws {VarIntError} if the buffer is too short or the VarInt is malformed
93 | */
94 | export function decodeVarInt(buffer, offset = 0) {
95 | if (offset >= buffer.length) {
96 | throw new VarIntError(
97 | "Buffer underflow: Cannot decode VarInt at or beyond buffer length.",
98 | ERR_VARINT_BUFFER_UNDERFLOW
99 | );
100 | }
101 |
102 | // Fast path for single-byte VarInts, which are very common.
103 | const firstByte = buffer.readUInt8(offset);
104 | if ((firstByte & 0x80) === 0) {
105 | return { value: firstByte, bytesRead: 1 };
106 | }
107 |
108 | let val = firstByte & 0x7f; // Get the first 7 bits
109 | let position = 7; // Bit position for the next byte's data
110 | let bytesRead = 1; // We've read one byte so far
111 | let currentOffset = offset + 1; // Start reading from the next
112 |
113 | // Max 4 more bytes (total 5 bytes for a VarInt)
114 | for (let i = 0; i < 4; i++) {
115 | if (currentOffset >= buffer.length) {
116 | throw new VarIntError(
117 | "Buffer underflow: Incomplete VarInt, expected more bytes.",
118 | ERR_VARINT_BUFFER_UNDERFLOW
119 | );
120 | }
121 |
122 | const byte = buffer.readUInt8(currentOffset);
123 | bytesRead++;
124 | currentOffset++;
125 |
126 | val |= (byte & 0x7f) << position;
127 | position += 7;
128 |
129 | if ((byte & 0x80) === 0) {
130 | return { value: val, bytesRead: bytesRead };
131 | }
132 | }
133 |
134 | throw new VarIntError(
135 | "VarInt is too big or malformed: 5 bytes read with continuation bit still set.",
136 | ERR_VARINT_MALFORMED
137 | );
138 | }
139 |
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@minescope/mineping",
3 | "version": "1.7.0-beta.1",
4 | "lockfileVersion": 3,
5 | "requires": true,
6 | "packages": {
7 | "": {
8 | "name": "@minescope/mineping",
9 | "version": "1.7.0-beta.1",
10 | "license": "MIT",
11 | "dependencies": {
12 | "debug": "^4.4.1"
13 | },
14 | "devDependencies": {
15 | "@types/debug": "^4.1.12",
16 | "@types/node": "^24.0.3",
17 | "typescript": "^5.8.3",
18 | "vitest": "^3.2.3"
19 | },
20 | "engines": {
21 | "node": ">=14"
22 | }
23 | },
24 | "node_modules/@esbuild/aix-ppc64": {
25 | "version": "0.25.5",
26 | "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.5.tgz",
27 | "integrity": "sha512-9o3TMmpmftaCMepOdA5k/yDw8SfInyzWWTjYTFCX3kPSDJMROQTb8jg+h9Cnwnmm1vOzvxN7gIfB5V2ewpjtGA==",
28 | "cpu": [
29 | "ppc64"
30 | ],
31 | "dev": true,
32 | "license": "MIT",
33 | "optional": true,
34 | "os": [
35 | "aix"
36 | ],
37 | "engines": {
38 | "node": ">=18"
39 | }
40 | },
41 | "node_modules/@esbuild/android-arm": {
42 | "version": "0.25.5",
43 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.5.tgz",
44 | "integrity": "sha512-AdJKSPeEHgi7/ZhuIPtcQKr5RQdo6OO2IL87JkianiMYMPbCtot9fxPbrMiBADOWWm3T2si9stAiVsGbTQFkbA==",
45 | "cpu": [
46 | "arm"
47 | ],
48 | "dev": true,
49 | "license": "MIT",
50 | "optional": true,
51 | "os": [
52 | "android"
53 | ],
54 | "engines": {
55 | "node": ">=18"
56 | }
57 | },
58 | "node_modules/@esbuild/android-arm64": {
59 | "version": "0.25.5",
60 | "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.5.tgz",
61 | "integrity": "sha512-VGzGhj4lJO+TVGV1v8ntCZWJktV7SGCs3Pn1GRWI1SBFtRALoomm8k5E9Pmwg3HOAal2VDc2F9+PM/rEY6oIDg==",
62 | "cpu": [
63 | "arm64"
64 | ],
65 | "dev": true,
66 | "license": "MIT",
67 | "optional": true,
68 | "os": [
69 | "android"
70 | ],
71 | "engines": {
72 | "node": ">=18"
73 | }
74 | },
75 | "node_modules/@esbuild/android-x64": {
76 | "version": "0.25.5",
77 | "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.5.tgz",
78 | "integrity": "sha512-D2GyJT1kjvO//drbRT3Hib9XPwQeWd9vZoBJn+bu/lVsOZ13cqNdDeqIF/xQ5/VmWvMduP6AmXvylO/PIc2isw==",
79 | "cpu": [
80 | "x64"
81 | ],
82 | "dev": true,
83 | "license": "MIT",
84 | "optional": true,
85 | "os": [
86 | "android"
87 | ],
88 | "engines": {
89 | "node": ">=18"
90 | }
91 | },
92 | "node_modules/@esbuild/darwin-arm64": {
93 | "version": "0.25.5",
94 | "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.5.tgz",
95 | "integrity": "sha512-GtaBgammVvdF7aPIgH2jxMDdivezgFu6iKpmT+48+F8Hhg5J/sfnDieg0aeG/jfSvkYQU2/pceFPDKlqZzwnfQ==",
96 | "cpu": [
97 | "arm64"
98 | ],
99 | "dev": true,
100 | "license": "MIT",
101 | "optional": true,
102 | "os": [
103 | "darwin"
104 | ],
105 | "engines": {
106 | "node": ">=18"
107 | }
108 | },
109 | "node_modules/@esbuild/darwin-x64": {
110 | "version": "0.25.5",
111 | "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.5.tgz",
112 | "integrity": "sha512-1iT4FVL0dJ76/q1wd7XDsXrSW+oLoquptvh4CLR4kITDtqi2e/xwXwdCVH8hVHU43wgJdsq7Gxuzcs6Iq/7bxQ==",
113 | "cpu": [
114 | "x64"
115 | ],
116 | "dev": true,
117 | "license": "MIT",
118 | "optional": true,
119 | "os": [
120 | "darwin"
121 | ],
122 | "engines": {
123 | "node": ">=18"
124 | }
125 | },
126 | "node_modules/@esbuild/freebsd-arm64": {
127 | "version": "0.25.5",
128 | "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.5.tgz",
129 | "integrity": "sha512-nk4tGP3JThz4La38Uy/gzyXtpkPW8zSAmoUhK9xKKXdBCzKODMc2adkB2+8om9BDYugz+uGV7sLmpTYzvmz6Sw==",
130 | "cpu": [
131 | "arm64"
132 | ],
133 | "dev": true,
134 | "license": "MIT",
135 | "optional": true,
136 | "os": [
137 | "freebsd"
138 | ],
139 | "engines": {
140 | "node": ">=18"
141 | }
142 | },
143 | "node_modules/@esbuild/freebsd-x64": {
144 | "version": "0.25.5",
145 | "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.5.tgz",
146 | "integrity": "sha512-PrikaNjiXdR2laW6OIjlbeuCPrPaAl0IwPIaRv+SMV8CiM8i2LqVUHFC1+8eORgWyY7yhQY+2U2fA55mBzReaw==",
147 | "cpu": [
148 | "x64"
149 | ],
150 | "dev": true,
151 | "license": "MIT",
152 | "optional": true,
153 | "os": [
154 | "freebsd"
155 | ],
156 | "engines": {
157 | "node": ">=18"
158 | }
159 | },
160 | "node_modules/@esbuild/linux-arm": {
161 | "version": "0.25.5",
162 | "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.5.tgz",
163 | "integrity": "sha512-cPzojwW2okgh7ZlRpcBEtsX7WBuqbLrNXqLU89GxWbNt6uIg78ET82qifUy3W6OVww6ZWobWub5oqZOVtwolfw==",
164 | "cpu": [
165 | "arm"
166 | ],
167 | "dev": true,
168 | "license": "MIT",
169 | "optional": true,
170 | "os": [
171 | "linux"
172 | ],
173 | "engines": {
174 | "node": ">=18"
175 | }
176 | },
177 | "node_modules/@esbuild/linux-arm64": {
178 | "version": "0.25.5",
179 | "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.5.tgz",
180 | "integrity": "sha512-Z9kfb1v6ZlGbWj8EJk9T6czVEjjq2ntSYLY2cw6pAZl4oKtfgQuS4HOq41M/BcoLPzrUbNd+R4BXFyH//nHxVg==",
181 | "cpu": [
182 | "arm64"
183 | ],
184 | "dev": true,
185 | "license": "MIT",
186 | "optional": true,
187 | "os": [
188 | "linux"
189 | ],
190 | "engines": {
191 | "node": ">=18"
192 | }
193 | },
194 | "node_modules/@esbuild/linux-ia32": {
195 | "version": "0.25.5",
196 | "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.5.tgz",
197 | "integrity": "sha512-sQ7l00M8bSv36GLV95BVAdhJ2QsIbCuCjh/uYrWiMQSUuV+LpXwIqhgJDcvMTj+VsQmqAHL2yYaasENvJ7CDKA==",
198 | "cpu": [
199 | "ia32"
200 | ],
201 | "dev": true,
202 | "license": "MIT",
203 | "optional": true,
204 | "os": [
205 | "linux"
206 | ],
207 | "engines": {
208 | "node": ">=18"
209 | }
210 | },
211 | "node_modules/@esbuild/linux-loong64": {
212 | "version": "0.25.5",
213 | "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.5.tgz",
214 | "integrity": "sha512-0ur7ae16hDUC4OL5iEnDb0tZHDxYmuQyhKhsPBV8f99f6Z9KQM02g33f93rNH5A30agMS46u2HP6qTdEt6Q1kg==",
215 | "cpu": [
216 | "loong64"
217 | ],
218 | "dev": true,
219 | "license": "MIT",
220 | "optional": true,
221 | "os": [
222 | "linux"
223 | ],
224 | "engines": {
225 | "node": ">=18"
226 | }
227 | },
228 | "node_modules/@esbuild/linux-mips64el": {
229 | "version": "0.25.5",
230 | "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.5.tgz",
231 | "integrity": "sha512-kB/66P1OsHO5zLz0i6X0RxlQ+3cu0mkxS3TKFvkb5lin6uwZ/ttOkP3Z8lfR9mJOBk14ZwZ9182SIIWFGNmqmg==",
232 | "cpu": [
233 | "mips64el"
234 | ],
235 | "dev": true,
236 | "license": "MIT",
237 | "optional": true,
238 | "os": [
239 | "linux"
240 | ],
241 | "engines": {
242 | "node": ">=18"
243 | }
244 | },
245 | "node_modules/@esbuild/linux-ppc64": {
246 | "version": "0.25.5",
247 | "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.5.tgz",
248 | "integrity": "sha512-UZCmJ7r9X2fe2D6jBmkLBMQetXPXIsZjQJCjgwpVDz+YMcS6oFR27alkgGv3Oqkv07bxdvw7fyB71/olceJhkQ==",
249 | "cpu": [
250 | "ppc64"
251 | ],
252 | "dev": true,
253 | "license": "MIT",
254 | "optional": true,
255 | "os": [
256 | "linux"
257 | ],
258 | "engines": {
259 | "node": ">=18"
260 | }
261 | },
262 | "node_modules/@esbuild/linux-riscv64": {
263 | "version": "0.25.5",
264 | "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.5.tgz",
265 | "integrity": "sha512-kTxwu4mLyeOlsVIFPfQo+fQJAV9mh24xL+y+Bm6ej067sYANjyEw1dNHmvoqxJUCMnkBdKpvOn0Ahql6+4VyeA==",
266 | "cpu": [
267 | "riscv64"
268 | ],
269 | "dev": true,
270 | "license": "MIT",
271 | "optional": true,
272 | "os": [
273 | "linux"
274 | ],
275 | "engines": {
276 | "node": ">=18"
277 | }
278 | },
279 | "node_modules/@esbuild/linux-s390x": {
280 | "version": "0.25.5",
281 | "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.5.tgz",
282 | "integrity": "sha512-K2dSKTKfmdh78uJ3NcWFiqyRrimfdinS5ErLSn3vluHNeHVnBAFWC8a4X5N+7FgVE1EjXS1QDZbpqZBjfrqMTQ==",
283 | "cpu": [
284 | "s390x"
285 | ],
286 | "dev": true,
287 | "license": "MIT",
288 | "optional": true,
289 | "os": [
290 | "linux"
291 | ],
292 | "engines": {
293 | "node": ">=18"
294 | }
295 | },
296 | "node_modules/@esbuild/linux-x64": {
297 | "version": "0.25.5",
298 | "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.5.tgz",
299 | "integrity": "sha512-uhj8N2obKTE6pSZ+aMUbqq+1nXxNjZIIjCjGLfsWvVpy7gKCOL6rsY1MhRh9zLtUtAI7vpgLMK6DxjO8Qm9lJw==",
300 | "cpu": [
301 | "x64"
302 | ],
303 | "dev": true,
304 | "license": "MIT",
305 | "optional": true,
306 | "os": [
307 | "linux"
308 | ],
309 | "engines": {
310 | "node": ">=18"
311 | }
312 | },
313 | "node_modules/@esbuild/netbsd-arm64": {
314 | "version": "0.25.5",
315 | "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.5.tgz",
316 | "integrity": "sha512-pwHtMP9viAy1oHPvgxtOv+OkduK5ugofNTVDilIzBLpoWAM16r7b/mxBvfpuQDpRQFMfuVr5aLcn4yveGvBZvw==",
317 | "cpu": [
318 | "arm64"
319 | ],
320 | "dev": true,
321 | "license": "MIT",
322 | "optional": true,
323 | "os": [
324 | "netbsd"
325 | ],
326 | "engines": {
327 | "node": ">=18"
328 | }
329 | },
330 | "node_modules/@esbuild/netbsd-x64": {
331 | "version": "0.25.5",
332 | "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.5.tgz",
333 | "integrity": "sha512-WOb5fKrvVTRMfWFNCroYWWklbnXH0Q5rZppjq0vQIdlsQKuw6mdSihwSo4RV/YdQ5UCKKvBy7/0ZZYLBZKIbwQ==",
334 | "cpu": [
335 | "x64"
336 | ],
337 | "dev": true,
338 | "license": "MIT",
339 | "optional": true,
340 | "os": [
341 | "netbsd"
342 | ],
343 | "engines": {
344 | "node": ">=18"
345 | }
346 | },
347 | "node_modules/@esbuild/openbsd-arm64": {
348 | "version": "0.25.5",
349 | "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.5.tgz",
350 | "integrity": "sha512-7A208+uQKgTxHd0G0uqZO8UjK2R0DDb4fDmERtARjSHWxqMTye4Erz4zZafx7Di9Cv+lNHYuncAkiGFySoD+Mw==",
351 | "cpu": [
352 | "arm64"
353 | ],
354 | "dev": true,
355 | "license": "MIT",
356 | "optional": true,
357 | "os": [
358 | "openbsd"
359 | ],
360 | "engines": {
361 | "node": ">=18"
362 | }
363 | },
364 | "node_modules/@esbuild/openbsd-x64": {
365 | "version": "0.25.5",
366 | "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.5.tgz",
367 | "integrity": "sha512-G4hE405ErTWraiZ8UiSoesH8DaCsMm0Cay4fsFWOOUcz8b8rC6uCvnagr+gnioEjWn0wC+o1/TAHt+It+MpIMg==",
368 | "cpu": [
369 | "x64"
370 | ],
371 | "dev": true,
372 | "license": "MIT",
373 | "optional": true,
374 | "os": [
375 | "openbsd"
376 | ],
377 | "engines": {
378 | "node": ">=18"
379 | }
380 | },
381 | "node_modules/@esbuild/sunos-x64": {
382 | "version": "0.25.5",
383 | "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.5.tgz",
384 | "integrity": "sha512-l+azKShMy7FxzY0Rj4RCt5VD/q8mG/e+mDivgspo+yL8zW7qEwctQ6YqKX34DTEleFAvCIUviCFX1SDZRSyMQA==",
385 | "cpu": [
386 | "x64"
387 | ],
388 | "dev": true,
389 | "license": "MIT",
390 | "optional": true,
391 | "os": [
392 | "sunos"
393 | ],
394 | "engines": {
395 | "node": ">=18"
396 | }
397 | },
398 | "node_modules/@esbuild/win32-arm64": {
399 | "version": "0.25.5",
400 | "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.5.tgz",
401 | "integrity": "sha512-O2S7SNZzdcFG7eFKgvwUEZ2VG9D/sn/eIiz8XRZ1Q/DO5a3s76Xv0mdBzVM5j5R639lXQmPmSo0iRpHqUUrsxw==",
402 | "cpu": [
403 | "arm64"
404 | ],
405 | "dev": true,
406 | "license": "MIT",
407 | "optional": true,
408 | "os": [
409 | "win32"
410 | ],
411 | "engines": {
412 | "node": ">=18"
413 | }
414 | },
415 | "node_modules/@esbuild/win32-ia32": {
416 | "version": "0.25.5",
417 | "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.5.tgz",
418 | "integrity": "sha512-onOJ02pqs9h1iMJ1PQphR+VZv8qBMQ77Klcsqv9CNW2w6yLqoURLcgERAIurY6QE63bbLuqgP9ATqajFLK5AMQ==",
419 | "cpu": [
420 | "ia32"
421 | ],
422 | "dev": true,
423 | "license": "MIT",
424 | "optional": true,
425 | "os": [
426 | "win32"
427 | ],
428 | "engines": {
429 | "node": ">=18"
430 | }
431 | },
432 | "node_modules/@esbuild/win32-x64": {
433 | "version": "0.25.5",
434 | "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.5.tgz",
435 | "integrity": "sha512-TXv6YnJ8ZMVdX+SXWVBo/0p8LTcrUYngpWjvm91TMjjBQii7Oz11Lw5lbDV5Y0TzuhSJHwiH4hEtC1I42mMS0g==",
436 | "cpu": [
437 | "x64"
438 | ],
439 | "dev": true,
440 | "license": "MIT",
441 | "optional": true,
442 | "os": [
443 | "win32"
444 | ],
445 | "engines": {
446 | "node": ">=18"
447 | }
448 | },
449 | "node_modules/@jridgewell/sourcemap-codec": {
450 | "version": "1.5.0",
451 | "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz",
452 | "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==",
453 | "dev": true,
454 | "license": "MIT"
455 | },
456 | "node_modules/@rollup/rollup-android-arm-eabi": {
457 | "version": "4.43.0",
458 | "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.43.0.tgz",
459 | "integrity": "sha512-Krjy9awJl6rKbruhQDgivNbD1WuLb8xAclM4IR4cN5pHGAs2oIMMQJEiC3IC/9TZJ+QZkmZhlMO/6MBGxPidpw==",
460 | "cpu": [
461 | "arm"
462 | ],
463 | "dev": true,
464 | "license": "MIT",
465 | "optional": true,
466 | "os": [
467 | "android"
468 | ]
469 | },
470 | "node_modules/@rollup/rollup-android-arm64": {
471 | "version": "4.43.0",
472 | "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.43.0.tgz",
473 | "integrity": "sha512-ss4YJwRt5I63454Rpj+mXCXicakdFmKnUNxr1dLK+5rv5FJgAxnN7s31a5VchRYxCFWdmnDWKd0wbAdTr0J5EA==",
474 | "cpu": [
475 | "arm64"
476 | ],
477 | "dev": true,
478 | "license": "MIT",
479 | "optional": true,
480 | "os": [
481 | "android"
482 | ]
483 | },
484 | "node_modules/@rollup/rollup-darwin-arm64": {
485 | "version": "4.43.0",
486 | "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.43.0.tgz",
487 | "integrity": "sha512-eKoL8ykZ7zz8MjgBenEF2OoTNFAPFz1/lyJ5UmmFSz5jW+7XbH1+MAgCVHy72aG59rbuQLcJeiMrP8qP5d/N0A==",
488 | "cpu": [
489 | "arm64"
490 | ],
491 | "dev": true,
492 | "license": "MIT",
493 | "optional": true,
494 | "os": [
495 | "darwin"
496 | ]
497 | },
498 | "node_modules/@rollup/rollup-darwin-x64": {
499 | "version": "4.43.0",
500 | "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.43.0.tgz",
501 | "integrity": "sha512-SYwXJgaBYW33Wi/q4ubN+ldWC4DzQY62S4Ll2dgfr/dbPoF50dlQwEaEHSKrQdSjC6oIe1WgzosoaNoHCdNuMg==",
502 | "cpu": [
503 | "x64"
504 | ],
505 | "dev": true,
506 | "license": "MIT",
507 | "optional": true,
508 | "os": [
509 | "darwin"
510 | ]
511 | },
512 | "node_modules/@rollup/rollup-freebsd-arm64": {
513 | "version": "4.43.0",
514 | "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.43.0.tgz",
515 | "integrity": "sha512-SV+U5sSo0yujrjzBF7/YidieK2iF6E7MdF6EbYxNz94lA+R0wKl3SiixGyG/9Klab6uNBIqsN7j4Y/Fya7wAjQ==",
516 | "cpu": [
517 | "arm64"
518 | ],
519 | "dev": true,
520 | "license": "MIT",
521 | "optional": true,
522 | "os": [
523 | "freebsd"
524 | ]
525 | },
526 | "node_modules/@rollup/rollup-freebsd-x64": {
527 | "version": "4.43.0",
528 | "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.43.0.tgz",
529 | "integrity": "sha512-J7uCsiV13L/VOeHJBo5SjasKiGxJ0g+nQTrBkAsmQBIdil3KhPnSE9GnRon4ejX1XDdsmK/l30IYLiAaQEO0Cg==",
530 | "cpu": [
531 | "x64"
532 | ],
533 | "dev": true,
534 | "license": "MIT",
535 | "optional": true,
536 | "os": [
537 | "freebsd"
538 | ]
539 | },
540 | "node_modules/@rollup/rollup-linux-arm-gnueabihf": {
541 | "version": "4.43.0",
542 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.43.0.tgz",
543 | "integrity": "sha512-gTJ/JnnjCMc15uwB10TTATBEhK9meBIY+gXP4s0sHD1zHOaIh4Dmy1X9wup18IiY9tTNk5gJc4yx9ctj/fjrIw==",
544 | "cpu": [
545 | "arm"
546 | ],
547 | "dev": true,
548 | "license": "MIT",
549 | "optional": true,
550 | "os": [
551 | "linux"
552 | ]
553 | },
554 | "node_modules/@rollup/rollup-linux-arm-musleabihf": {
555 | "version": "4.43.0",
556 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.43.0.tgz",
557 | "integrity": "sha512-ZJ3gZynL1LDSIvRfz0qXtTNs56n5DI2Mq+WACWZ7yGHFUEirHBRt7fyIk0NsCKhmRhn7WAcjgSkSVVxKlPNFFw==",
558 | "cpu": [
559 | "arm"
560 | ],
561 | "dev": true,
562 | "license": "MIT",
563 | "optional": true,
564 | "os": [
565 | "linux"
566 | ]
567 | },
568 | "node_modules/@rollup/rollup-linux-arm64-gnu": {
569 | "version": "4.43.0",
570 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.43.0.tgz",
571 | "integrity": "sha512-8FnkipasmOOSSlfucGYEu58U8cxEdhziKjPD2FIa0ONVMxvl/hmONtX/7y4vGjdUhjcTHlKlDhw3H9t98fPvyA==",
572 | "cpu": [
573 | "arm64"
574 | ],
575 | "dev": true,
576 | "license": "MIT",
577 | "optional": true,
578 | "os": [
579 | "linux"
580 | ]
581 | },
582 | "node_modules/@rollup/rollup-linux-arm64-musl": {
583 | "version": "4.43.0",
584 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.43.0.tgz",
585 | "integrity": "sha512-KPPyAdlcIZ6S9C3S2cndXDkV0Bb1OSMsX0Eelr2Bay4EsF9yi9u9uzc9RniK3mcUGCLhWY9oLr6er80P5DE6XA==",
586 | "cpu": [
587 | "arm64"
588 | ],
589 | "dev": true,
590 | "license": "MIT",
591 | "optional": true,
592 | "os": [
593 | "linux"
594 | ]
595 | },
596 | "node_modules/@rollup/rollup-linux-loongarch64-gnu": {
597 | "version": "4.43.0",
598 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.43.0.tgz",
599 | "integrity": "sha512-HPGDIH0/ZzAZjvtlXj6g+KDQ9ZMHfSP553za7o2Odegb/BEfwJcR0Sw0RLNpQ9nC6Gy8s+3mSS9xjZ0n3rhcYg==",
600 | "cpu": [
601 | "loong64"
602 | ],
603 | "dev": true,
604 | "license": "MIT",
605 | "optional": true,
606 | "os": [
607 | "linux"
608 | ]
609 | },
610 | "node_modules/@rollup/rollup-linux-powerpc64le-gnu": {
611 | "version": "4.43.0",
612 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.43.0.tgz",
613 | "integrity": "sha512-gEmwbOws4U4GLAJDhhtSPWPXUzDfMRedT3hFMyRAvM9Mrnj+dJIFIeL7otsv2WF3D7GrV0GIewW0y28dOYWkmw==",
614 | "cpu": [
615 | "ppc64"
616 | ],
617 | "dev": true,
618 | "license": "MIT",
619 | "optional": true,
620 | "os": [
621 | "linux"
622 | ]
623 | },
624 | "node_modules/@rollup/rollup-linux-riscv64-gnu": {
625 | "version": "4.43.0",
626 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.43.0.tgz",
627 | "integrity": "sha512-XXKvo2e+wFtXZF/9xoWohHg+MuRnvO29TI5Hqe9xwN5uN8NKUYy7tXUG3EZAlfchufNCTHNGjEx7uN78KsBo0g==",
628 | "cpu": [
629 | "riscv64"
630 | ],
631 | "dev": true,
632 | "license": "MIT",
633 | "optional": true,
634 | "os": [
635 | "linux"
636 | ]
637 | },
638 | "node_modules/@rollup/rollup-linux-riscv64-musl": {
639 | "version": "4.43.0",
640 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.43.0.tgz",
641 | "integrity": "sha512-ruf3hPWhjw6uDFsOAzmbNIvlXFXlBQ4nk57Sec8E8rUxs/AI4HD6xmiiasOOx/3QxS2f5eQMKTAwk7KHwpzr/Q==",
642 | "cpu": [
643 | "riscv64"
644 | ],
645 | "dev": true,
646 | "license": "MIT",
647 | "optional": true,
648 | "os": [
649 | "linux"
650 | ]
651 | },
652 | "node_modules/@rollup/rollup-linux-s390x-gnu": {
653 | "version": "4.43.0",
654 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.43.0.tgz",
655 | "integrity": "sha512-QmNIAqDiEMEvFV15rsSnjoSmO0+eJLoKRD9EAa9rrYNwO/XRCtOGM3A5A0X+wmG+XRrw9Fxdsw+LnyYiZWWcVw==",
656 | "cpu": [
657 | "s390x"
658 | ],
659 | "dev": true,
660 | "license": "MIT",
661 | "optional": true,
662 | "os": [
663 | "linux"
664 | ]
665 | },
666 | "node_modules/@rollup/rollup-linux-x64-gnu": {
667 | "version": "4.43.0",
668 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.43.0.tgz",
669 | "integrity": "sha512-jAHr/S0iiBtFyzjhOkAics/2SrXE092qyqEg96e90L3t9Op8OTzS6+IX0Fy5wCt2+KqeHAkti+eitV0wvblEoQ==",
670 | "cpu": [
671 | "x64"
672 | ],
673 | "dev": true,
674 | "license": "MIT",
675 | "optional": true,
676 | "os": [
677 | "linux"
678 | ]
679 | },
680 | "node_modules/@rollup/rollup-linux-x64-musl": {
681 | "version": "4.43.0",
682 | "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.43.0.tgz",
683 | "integrity": "sha512-3yATWgdeXyuHtBhrLt98w+5fKurdqvs8B53LaoKD7P7H7FKOONLsBVMNl9ghPQZQuYcceV5CDyPfyfGpMWD9mQ==",
684 | "cpu": [
685 | "x64"
686 | ],
687 | "dev": true,
688 | "license": "MIT",
689 | "optional": true,
690 | "os": [
691 | "linux"
692 | ]
693 | },
694 | "node_modules/@rollup/rollup-win32-arm64-msvc": {
695 | "version": "4.43.0",
696 | "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.43.0.tgz",
697 | "integrity": "sha512-wVzXp2qDSCOpcBCT5WRWLmpJRIzv23valvcTwMHEobkjippNf+C3ys/+wf07poPkeNix0paTNemB2XrHr2TnGw==",
698 | "cpu": [
699 | "arm64"
700 | ],
701 | "dev": true,
702 | "license": "MIT",
703 | "optional": true,
704 | "os": [
705 | "win32"
706 | ]
707 | },
708 | "node_modules/@rollup/rollup-win32-ia32-msvc": {
709 | "version": "4.43.0",
710 | "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.43.0.tgz",
711 | "integrity": "sha512-fYCTEyzf8d+7diCw8b+asvWDCLMjsCEA8alvtAutqJOJp/wL5hs1rWSqJ1vkjgW0L2NB4bsYJrpKkiIPRR9dvw==",
712 | "cpu": [
713 | "ia32"
714 | ],
715 | "dev": true,
716 | "license": "MIT",
717 | "optional": true,
718 | "os": [
719 | "win32"
720 | ]
721 | },
722 | "node_modules/@rollup/rollup-win32-x64-msvc": {
723 | "version": "4.43.0",
724 | "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.43.0.tgz",
725 | "integrity": "sha512-SnGhLiE5rlK0ofq8kzuDkM0g7FN1s5VYY+YSMTibP7CqShxCQvqtNxTARS4xX4PFJfHjG0ZQYX9iGzI3FQh5Aw==",
726 | "cpu": [
727 | "x64"
728 | ],
729 | "dev": true,
730 | "license": "MIT",
731 | "optional": true,
732 | "os": [
733 | "win32"
734 | ]
735 | },
736 | "node_modules/@types/chai": {
737 | "version": "5.2.2",
738 | "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.2.tgz",
739 | "integrity": "sha512-8kB30R7Hwqf40JPiKhVzodJs2Qc1ZJ5zuT3uzw5Hq/dhNCl3G3l83jfpdI1e20BP348+fV7VIL/+FxaXkqBmWg==",
740 | "dev": true,
741 | "license": "MIT",
742 | "dependencies": {
743 | "@types/deep-eql": "*"
744 | }
745 | },
746 | "node_modules/@types/debug": {
747 | "version": "4.1.12",
748 | "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.12.tgz",
749 | "integrity": "sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==",
750 | "dev": true,
751 | "license": "MIT",
752 | "dependencies": {
753 | "@types/ms": "*"
754 | }
755 | },
756 | "node_modules/@types/deep-eql": {
757 | "version": "4.0.2",
758 | "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz",
759 | "integrity": "sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==",
760 | "dev": true,
761 | "license": "MIT"
762 | },
763 | "node_modules/@types/estree": {
764 | "version": "1.0.8",
765 | "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz",
766 | "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==",
767 | "dev": true,
768 | "license": "MIT"
769 | },
770 | "node_modules/@types/ms": {
771 | "version": "2.1.0",
772 | "resolved": "https://registry.npmjs.org/@types/ms/-/ms-2.1.0.tgz",
773 | "integrity": "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA==",
774 | "dev": true,
775 | "license": "MIT"
776 | },
777 | "node_modules/@types/node": {
778 | "version": "24.0.3",
779 | "resolved": "https://registry.npmjs.org/@types/node/-/node-24.0.3.tgz",
780 | "integrity": "sha512-R4I/kzCYAdRLzfiCabn9hxWfbuHS573x+r0dJMkkzThEa7pbrcDWK+9zu3e7aBOouf+rQAciqPFMnxwr0aWgKg==",
781 | "dev": true,
782 | "license": "MIT",
783 | "dependencies": {
784 | "undici-types": "~7.8.0"
785 | }
786 | },
787 | "node_modules/@vitest/expect": {
788 | "version": "3.2.3",
789 | "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-3.2.3.tgz",
790 | "integrity": "sha512-W2RH2TPWVHA1o7UmaFKISPvdicFJH+mjykctJFoAkUw+SPTJTGjUNdKscFBrqM7IPnCVu6zihtKYa7TkZS1dkQ==",
791 | "dev": true,
792 | "license": "MIT",
793 | "dependencies": {
794 | "@types/chai": "^5.2.2",
795 | "@vitest/spy": "3.2.3",
796 | "@vitest/utils": "3.2.3",
797 | "chai": "^5.2.0",
798 | "tinyrainbow": "^2.0.0"
799 | },
800 | "funding": {
801 | "url": "https://opencollective.com/vitest"
802 | }
803 | },
804 | "node_modules/@vitest/mocker": {
805 | "version": "3.2.3",
806 | "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-3.2.3.tgz",
807 | "integrity": "sha512-cP6fIun+Zx8he4rbWvi+Oya6goKQDZK+Yq4hhlggwQBbrlOQ4qtZ+G4nxB6ZnzI9lyIb+JnvyiJnPC2AGbKSPA==",
808 | "dev": true,
809 | "license": "MIT",
810 | "dependencies": {
811 | "@vitest/spy": "3.2.3",
812 | "estree-walker": "^3.0.3",
813 | "magic-string": "^0.30.17"
814 | },
815 | "funding": {
816 | "url": "https://opencollective.com/vitest"
817 | },
818 | "peerDependencies": {
819 | "msw": "^2.4.9",
820 | "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0"
821 | },
822 | "peerDependenciesMeta": {
823 | "msw": {
824 | "optional": true
825 | },
826 | "vite": {
827 | "optional": true
828 | }
829 | }
830 | },
831 | "node_modules/@vitest/pretty-format": {
832 | "version": "3.2.3",
833 | "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-3.2.3.tgz",
834 | "integrity": "sha512-yFglXGkr9hW/yEXngO+IKMhP0jxyFw2/qys/CK4fFUZnSltD+MU7dVYGrH8rvPcK/O6feXQA+EU33gjaBBbAng==",
835 | "dev": true,
836 | "license": "MIT",
837 | "dependencies": {
838 | "tinyrainbow": "^2.0.0"
839 | },
840 | "funding": {
841 | "url": "https://opencollective.com/vitest"
842 | }
843 | },
844 | "node_modules/@vitest/runner": {
845 | "version": "3.2.3",
846 | "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-3.2.3.tgz",
847 | "integrity": "sha512-83HWYisT3IpMaU9LN+VN+/nLHVBCSIUKJzGxC5RWUOsK1h3USg7ojL+UXQR3b4o4UBIWCYdD2fxuzM7PQQ1u8w==",
848 | "dev": true,
849 | "license": "MIT",
850 | "dependencies": {
851 | "@vitest/utils": "3.2.3",
852 | "pathe": "^2.0.3",
853 | "strip-literal": "^3.0.0"
854 | },
855 | "funding": {
856 | "url": "https://opencollective.com/vitest"
857 | }
858 | },
859 | "node_modules/@vitest/snapshot": {
860 | "version": "3.2.3",
861 | "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-3.2.3.tgz",
862 | "integrity": "sha512-9gIVWx2+tysDqUmmM1L0hwadyumqssOL1r8KJipwLx5JVYyxvVRfxvMq7DaWbZZsCqZnu/dZedaZQh4iYTtneA==",
863 | "dev": true,
864 | "license": "MIT",
865 | "dependencies": {
866 | "@vitest/pretty-format": "3.2.3",
867 | "magic-string": "^0.30.17",
868 | "pathe": "^2.0.3"
869 | },
870 | "funding": {
871 | "url": "https://opencollective.com/vitest"
872 | }
873 | },
874 | "node_modules/@vitest/spy": {
875 | "version": "3.2.3",
876 | "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-3.2.3.tgz",
877 | "integrity": "sha512-JHu9Wl+7bf6FEejTCREy+DmgWe+rQKbK+y32C/k5f4TBIAlijhJbRBIRIOCEpVevgRsCQR2iHRUH2/qKVM/plw==",
878 | "dev": true,
879 | "license": "MIT",
880 | "dependencies": {
881 | "tinyspy": "^4.0.3"
882 | },
883 | "funding": {
884 | "url": "https://opencollective.com/vitest"
885 | }
886 | },
887 | "node_modules/@vitest/utils": {
888 | "version": "3.2.3",
889 | "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-3.2.3.tgz",
890 | "integrity": "sha512-4zFBCU5Pf+4Z6v+rwnZ1HU1yzOKKvDkMXZrymE2PBlbjKJRlrOxbvpfPSvJTGRIwGoahaOGvp+kbCoxifhzJ1Q==",
891 | "dev": true,
892 | "license": "MIT",
893 | "dependencies": {
894 | "@vitest/pretty-format": "3.2.3",
895 | "loupe": "^3.1.3",
896 | "tinyrainbow": "^2.0.0"
897 | },
898 | "funding": {
899 | "url": "https://opencollective.com/vitest"
900 | }
901 | },
902 | "node_modules/assertion-error": {
903 | "version": "2.0.1",
904 | "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz",
905 | "integrity": "sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==",
906 | "dev": true,
907 | "license": "MIT",
908 | "engines": {
909 | "node": ">=12"
910 | }
911 | },
912 | "node_modules/cac": {
913 | "version": "6.7.14",
914 | "resolved": "https://registry.npmjs.org/cac/-/cac-6.7.14.tgz",
915 | "integrity": "sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==",
916 | "dev": true,
917 | "license": "MIT",
918 | "engines": {
919 | "node": ">=8"
920 | }
921 | },
922 | "node_modules/chai": {
923 | "version": "5.2.0",
924 | "resolved": "https://registry.npmjs.org/chai/-/chai-5.2.0.tgz",
925 | "integrity": "sha512-mCuXncKXk5iCLhfhwTc0izo0gtEmpz5CtG2y8GiOINBlMVS6v8TMRc5TaLWKS6692m9+dVVfzgeVxR5UxWHTYw==",
926 | "dev": true,
927 | "license": "MIT",
928 | "dependencies": {
929 | "assertion-error": "^2.0.1",
930 | "check-error": "^2.1.1",
931 | "deep-eql": "^5.0.1",
932 | "loupe": "^3.1.0",
933 | "pathval": "^2.0.0"
934 | },
935 | "engines": {
936 | "node": ">=12"
937 | }
938 | },
939 | "node_modules/check-error": {
940 | "version": "2.1.1",
941 | "resolved": "https://registry.npmjs.org/check-error/-/check-error-2.1.1.tgz",
942 | "integrity": "sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw==",
943 | "dev": true,
944 | "license": "MIT",
945 | "engines": {
946 | "node": ">= 16"
947 | }
948 | },
949 | "node_modules/debug": {
950 | "version": "4.4.1",
951 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.1.tgz",
952 | "integrity": "sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ==",
953 | "license": "MIT",
954 | "dependencies": {
955 | "ms": "^2.1.3"
956 | },
957 | "engines": {
958 | "node": ">=6.0"
959 | },
960 | "peerDependenciesMeta": {
961 | "supports-color": {
962 | "optional": true
963 | }
964 | }
965 | },
966 | "node_modules/deep-eql": {
967 | "version": "5.0.2",
968 | "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-5.0.2.tgz",
969 | "integrity": "sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==",
970 | "dev": true,
971 | "license": "MIT",
972 | "engines": {
973 | "node": ">=6"
974 | }
975 | },
976 | "node_modules/es-module-lexer": {
977 | "version": "1.7.0",
978 | "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.7.0.tgz",
979 | "integrity": "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==",
980 | "dev": true,
981 | "license": "MIT"
982 | },
983 | "node_modules/esbuild": {
984 | "version": "0.25.5",
985 | "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.5.tgz",
986 | "integrity": "sha512-P8OtKZRv/5J5hhz0cUAdu/cLuPIKXpQl1R9pZtvmHWQvrAUVd0UNIPT4IB4W3rNOqVO0rlqHmCIbSwxh/c9yUQ==",
987 | "dev": true,
988 | "hasInstallScript": true,
989 | "license": "MIT",
990 | "bin": {
991 | "esbuild": "bin/esbuild"
992 | },
993 | "engines": {
994 | "node": ">=18"
995 | },
996 | "optionalDependencies": {
997 | "@esbuild/aix-ppc64": "0.25.5",
998 | "@esbuild/android-arm": "0.25.5",
999 | "@esbuild/android-arm64": "0.25.5",
1000 | "@esbuild/android-x64": "0.25.5",
1001 | "@esbuild/darwin-arm64": "0.25.5",
1002 | "@esbuild/darwin-x64": "0.25.5",
1003 | "@esbuild/freebsd-arm64": "0.25.5",
1004 | "@esbuild/freebsd-x64": "0.25.5",
1005 | "@esbuild/linux-arm": "0.25.5",
1006 | "@esbuild/linux-arm64": "0.25.5",
1007 | "@esbuild/linux-ia32": "0.25.5",
1008 | "@esbuild/linux-loong64": "0.25.5",
1009 | "@esbuild/linux-mips64el": "0.25.5",
1010 | "@esbuild/linux-ppc64": "0.25.5",
1011 | "@esbuild/linux-riscv64": "0.25.5",
1012 | "@esbuild/linux-s390x": "0.25.5",
1013 | "@esbuild/linux-x64": "0.25.5",
1014 | "@esbuild/netbsd-arm64": "0.25.5",
1015 | "@esbuild/netbsd-x64": "0.25.5",
1016 | "@esbuild/openbsd-arm64": "0.25.5",
1017 | "@esbuild/openbsd-x64": "0.25.5",
1018 | "@esbuild/sunos-x64": "0.25.5",
1019 | "@esbuild/win32-arm64": "0.25.5",
1020 | "@esbuild/win32-ia32": "0.25.5",
1021 | "@esbuild/win32-x64": "0.25.5"
1022 | }
1023 | },
1024 | "node_modules/estree-walker": {
1025 | "version": "3.0.3",
1026 | "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz",
1027 | "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==",
1028 | "dev": true,
1029 | "license": "MIT",
1030 | "dependencies": {
1031 | "@types/estree": "^1.0.0"
1032 | }
1033 | },
1034 | "node_modules/expect-type": {
1035 | "version": "1.2.1",
1036 | "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.2.1.tgz",
1037 | "integrity": "sha512-/kP8CAwxzLVEeFrMm4kMmy4CCDlpipyA7MYLVrdJIkV0fYF0UaigQHRsxHiuY/GEea+bh4KSv3TIlgr+2UL6bw==",
1038 | "dev": true,
1039 | "license": "Apache-2.0",
1040 | "engines": {
1041 | "node": ">=12.0.0"
1042 | }
1043 | },
1044 | "node_modules/fdir": {
1045 | "version": "6.4.6",
1046 | "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.4.6.tgz",
1047 | "integrity": "sha512-hiFoqpyZcfNm1yc4u8oWCf9A2c4D3QjCrks3zmoVKVxpQRzmPNar1hUJcBG2RQHvEVGDN+Jm81ZheVLAQMK6+w==",
1048 | "dev": true,
1049 | "license": "MIT",
1050 | "peerDependencies": {
1051 | "picomatch": "^3 || ^4"
1052 | },
1053 | "peerDependenciesMeta": {
1054 | "picomatch": {
1055 | "optional": true
1056 | }
1057 | }
1058 | },
1059 | "node_modules/fsevents": {
1060 | "version": "2.3.3",
1061 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
1062 | "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
1063 | "dev": true,
1064 | "hasInstallScript": true,
1065 | "license": "MIT",
1066 | "optional": true,
1067 | "os": [
1068 | "darwin"
1069 | ],
1070 | "engines": {
1071 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
1072 | }
1073 | },
1074 | "node_modules/js-tokens": {
1075 | "version": "9.0.1",
1076 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.1.tgz",
1077 | "integrity": "sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==",
1078 | "dev": true,
1079 | "license": "MIT"
1080 | },
1081 | "node_modules/loupe": {
1082 | "version": "3.1.3",
1083 | "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.1.3.tgz",
1084 | "integrity": "sha512-kkIp7XSkP78ZxJEsSxW3712C6teJVoeHHwgo9zJ380de7IYyJ2ISlxojcH2pC5OFLewESmnRi/+XCDIEEVyoug==",
1085 | "dev": true,
1086 | "license": "MIT"
1087 | },
1088 | "node_modules/magic-string": {
1089 | "version": "0.30.17",
1090 | "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.17.tgz",
1091 | "integrity": "sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA==",
1092 | "dev": true,
1093 | "license": "MIT",
1094 | "dependencies": {
1095 | "@jridgewell/sourcemap-codec": "^1.5.0"
1096 | }
1097 | },
1098 | "node_modules/ms": {
1099 | "version": "2.1.3",
1100 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
1101 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
1102 | "license": "MIT"
1103 | },
1104 | "node_modules/nanoid": {
1105 | "version": "3.3.11",
1106 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz",
1107 | "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==",
1108 | "dev": true,
1109 | "funding": [
1110 | {
1111 | "type": "github",
1112 | "url": "https://github.com/sponsors/ai"
1113 | }
1114 | ],
1115 | "license": "MIT",
1116 | "bin": {
1117 | "nanoid": "bin/nanoid.cjs"
1118 | },
1119 | "engines": {
1120 | "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
1121 | }
1122 | },
1123 | "node_modules/pathe": {
1124 | "version": "2.0.3",
1125 | "resolved": "https://registry.npmjs.org/pathe/-/pathe-2.0.3.tgz",
1126 | "integrity": "sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==",
1127 | "dev": true,
1128 | "license": "MIT"
1129 | },
1130 | "node_modules/pathval": {
1131 | "version": "2.0.0",
1132 | "resolved": "https://registry.npmjs.org/pathval/-/pathval-2.0.0.tgz",
1133 | "integrity": "sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA==",
1134 | "dev": true,
1135 | "license": "MIT",
1136 | "engines": {
1137 | "node": ">= 14.16"
1138 | }
1139 | },
1140 | "node_modules/picocolors": {
1141 | "version": "1.1.1",
1142 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
1143 | "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
1144 | "dev": true,
1145 | "license": "ISC"
1146 | },
1147 | "node_modules/picomatch": {
1148 | "version": "4.0.2",
1149 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz",
1150 | "integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==",
1151 | "dev": true,
1152 | "license": "MIT",
1153 | "engines": {
1154 | "node": ">=12"
1155 | },
1156 | "funding": {
1157 | "url": "https://github.com/sponsors/jonschlinkert"
1158 | }
1159 | },
1160 | "node_modules/postcss": {
1161 | "version": "8.5.5",
1162 | "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.5.tgz",
1163 | "integrity": "sha512-d/jtm+rdNT8tpXuHY5MMtcbJFBkhXE6593XVR9UoGCH8jSFGci7jGvMGH5RYd5PBJW+00NZQt6gf7CbagJCrhg==",
1164 | "dev": true,
1165 | "funding": [
1166 | {
1167 | "type": "opencollective",
1168 | "url": "https://opencollective.com/postcss/"
1169 | },
1170 | {
1171 | "type": "tidelift",
1172 | "url": "https://tidelift.com/funding/github/npm/postcss"
1173 | },
1174 | {
1175 | "type": "github",
1176 | "url": "https://github.com/sponsors/ai"
1177 | }
1178 | ],
1179 | "license": "MIT",
1180 | "dependencies": {
1181 | "nanoid": "^3.3.11",
1182 | "picocolors": "^1.1.1",
1183 | "source-map-js": "^1.2.1"
1184 | },
1185 | "engines": {
1186 | "node": "^10 || ^12 || >=14"
1187 | }
1188 | },
1189 | "node_modules/rollup": {
1190 | "version": "4.43.0",
1191 | "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.43.0.tgz",
1192 | "integrity": "sha512-wdN2Kd3Twh8MAEOEJZsuxuLKCsBEo4PVNLK6tQWAn10VhsVewQLzcucMgLolRlhFybGxfclbPeEYBaP6RvUFGg==",
1193 | "dev": true,
1194 | "license": "MIT",
1195 | "dependencies": {
1196 | "@types/estree": "1.0.7"
1197 | },
1198 | "bin": {
1199 | "rollup": "dist/bin/rollup"
1200 | },
1201 | "engines": {
1202 | "node": ">=18.0.0",
1203 | "npm": ">=8.0.0"
1204 | },
1205 | "optionalDependencies": {
1206 | "@rollup/rollup-android-arm-eabi": "4.43.0",
1207 | "@rollup/rollup-android-arm64": "4.43.0",
1208 | "@rollup/rollup-darwin-arm64": "4.43.0",
1209 | "@rollup/rollup-darwin-x64": "4.43.0",
1210 | "@rollup/rollup-freebsd-arm64": "4.43.0",
1211 | "@rollup/rollup-freebsd-x64": "4.43.0",
1212 | "@rollup/rollup-linux-arm-gnueabihf": "4.43.0",
1213 | "@rollup/rollup-linux-arm-musleabihf": "4.43.0",
1214 | "@rollup/rollup-linux-arm64-gnu": "4.43.0",
1215 | "@rollup/rollup-linux-arm64-musl": "4.43.0",
1216 | "@rollup/rollup-linux-loongarch64-gnu": "4.43.0",
1217 | "@rollup/rollup-linux-powerpc64le-gnu": "4.43.0",
1218 | "@rollup/rollup-linux-riscv64-gnu": "4.43.0",
1219 | "@rollup/rollup-linux-riscv64-musl": "4.43.0",
1220 | "@rollup/rollup-linux-s390x-gnu": "4.43.0",
1221 | "@rollup/rollup-linux-x64-gnu": "4.43.0",
1222 | "@rollup/rollup-linux-x64-musl": "4.43.0",
1223 | "@rollup/rollup-win32-arm64-msvc": "4.43.0",
1224 | "@rollup/rollup-win32-ia32-msvc": "4.43.0",
1225 | "@rollup/rollup-win32-x64-msvc": "4.43.0",
1226 | "fsevents": "~2.3.2"
1227 | }
1228 | },
1229 | "node_modules/rollup/node_modules/@types/estree": {
1230 | "version": "1.0.7",
1231 | "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.7.tgz",
1232 | "integrity": "sha512-w28IoSUCJpidD/TGviZwwMJckNESJZXFu7NBZ5YJ4mEUnNraUn9Pm8HSZm/jDF1pDWYKspWE7oVphigUPRakIQ==",
1233 | "dev": true,
1234 | "license": "MIT"
1235 | },
1236 | "node_modules/siginfo": {
1237 | "version": "2.0.0",
1238 | "resolved": "https://registry.npmjs.org/siginfo/-/siginfo-2.0.0.tgz",
1239 | "integrity": "sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==",
1240 | "dev": true,
1241 | "license": "ISC"
1242 | },
1243 | "node_modules/source-map-js": {
1244 | "version": "1.2.1",
1245 | "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
1246 | "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
1247 | "dev": true,
1248 | "license": "BSD-3-Clause",
1249 | "engines": {
1250 | "node": ">=0.10.0"
1251 | }
1252 | },
1253 | "node_modules/stackback": {
1254 | "version": "0.0.2",
1255 | "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz",
1256 | "integrity": "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==",
1257 | "dev": true,
1258 | "license": "MIT"
1259 | },
1260 | "node_modules/std-env": {
1261 | "version": "3.9.0",
1262 | "resolved": "https://registry.npmjs.org/std-env/-/std-env-3.9.0.tgz",
1263 | "integrity": "sha512-UGvjygr6F6tpH7o2qyqR6QYpwraIjKSdtzyBdyytFOHmPZY917kwdwLG0RbOjWOnKmnm3PeHjaoLLMie7kPLQw==",
1264 | "dev": true,
1265 | "license": "MIT"
1266 | },
1267 | "node_modules/strip-literal": {
1268 | "version": "3.0.0",
1269 | "resolved": "https://registry.npmjs.org/strip-literal/-/strip-literal-3.0.0.tgz",
1270 | "integrity": "sha512-TcccoMhJOM3OebGhSBEmp3UZ2SfDMZUEBdRA/9ynfLi8yYajyWX3JiXArcJt4Umh4vISpspkQIY8ZZoCqjbviA==",
1271 | "dev": true,
1272 | "license": "MIT",
1273 | "dependencies": {
1274 | "js-tokens": "^9.0.1"
1275 | },
1276 | "funding": {
1277 | "url": "https://github.com/sponsors/antfu"
1278 | }
1279 | },
1280 | "node_modules/tinybench": {
1281 | "version": "2.9.0",
1282 | "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz",
1283 | "integrity": "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==",
1284 | "dev": true,
1285 | "license": "MIT"
1286 | },
1287 | "node_modules/tinyexec": {
1288 | "version": "0.3.2",
1289 | "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.2.tgz",
1290 | "integrity": "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==",
1291 | "dev": true,
1292 | "license": "MIT"
1293 | },
1294 | "node_modules/tinyglobby": {
1295 | "version": "0.2.14",
1296 | "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.14.tgz",
1297 | "integrity": "sha512-tX5e7OM1HnYr2+a2C/4V0htOcSQcoSTH9KgJnVvNm5zm/cyEWKJ7j7YutsH9CxMdtOkkLFy2AHrMci9IM8IPZQ==",
1298 | "dev": true,
1299 | "license": "MIT",
1300 | "dependencies": {
1301 | "fdir": "^6.4.4",
1302 | "picomatch": "^4.0.2"
1303 | },
1304 | "engines": {
1305 | "node": ">=12.0.0"
1306 | },
1307 | "funding": {
1308 | "url": "https://github.com/sponsors/SuperchupuDev"
1309 | }
1310 | },
1311 | "node_modules/tinypool": {
1312 | "version": "1.1.0",
1313 | "resolved": "https://registry.npmjs.org/tinypool/-/tinypool-1.1.0.tgz",
1314 | "integrity": "sha512-7CotroY9a8DKsKprEy/a14aCCm8jYVmR7aFy4fpkZM8sdpNJbKkixuNjgM50yCmip2ezc8z4N7k3oe2+rfRJCQ==",
1315 | "dev": true,
1316 | "license": "MIT",
1317 | "engines": {
1318 | "node": "^18.0.0 || >=20.0.0"
1319 | }
1320 | },
1321 | "node_modules/tinyrainbow": {
1322 | "version": "2.0.0",
1323 | "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-2.0.0.tgz",
1324 | "integrity": "sha512-op4nsTR47R6p0vMUUoYl/a+ljLFVtlfaXkLQmqfLR1qHma1h/ysYk4hEXZ880bf2CYgTskvTa/e196Vd5dDQXw==",
1325 | "dev": true,
1326 | "license": "MIT",
1327 | "engines": {
1328 | "node": ">=14.0.0"
1329 | }
1330 | },
1331 | "node_modules/tinyspy": {
1332 | "version": "4.0.3",
1333 | "resolved": "https://registry.npmjs.org/tinyspy/-/tinyspy-4.0.3.tgz",
1334 | "integrity": "sha512-t2T/WLB2WRgZ9EpE4jgPJ9w+i66UZfDc8wHh0xrwiRNN+UwH98GIJkTeZqX9rg0i0ptwzqW+uYeIF0T4F8LR7A==",
1335 | "dev": true,
1336 | "license": "MIT",
1337 | "engines": {
1338 | "node": ">=14.0.0"
1339 | }
1340 | },
1341 | "node_modules/typescript": {
1342 | "version": "5.8.3",
1343 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.8.3.tgz",
1344 | "integrity": "sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ==",
1345 | "dev": true,
1346 | "license": "Apache-2.0",
1347 | "bin": {
1348 | "tsc": "bin/tsc",
1349 | "tsserver": "bin/tsserver"
1350 | },
1351 | "engines": {
1352 | "node": ">=14.17"
1353 | }
1354 | },
1355 | "node_modules/undici-types": {
1356 | "version": "7.8.0",
1357 | "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.8.0.tgz",
1358 | "integrity": "sha512-9UJ2xGDvQ43tYyVMpuHlsgApydB8ZKfVYTsLDhXkFL/6gfkp+U8xTGdh8pMJv1SpZna0zxG1DwsKZsreLbXBxw==",
1359 | "dev": true,
1360 | "license": "MIT"
1361 | },
1362 | "node_modules/vite": {
1363 | "version": "6.3.5",
1364 | "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.5.tgz",
1365 | "integrity": "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ==",
1366 | "dev": true,
1367 | "license": "MIT",
1368 | "dependencies": {
1369 | "esbuild": "^0.25.0",
1370 | "fdir": "^6.4.4",
1371 | "picomatch": "^4.0.2",
1372 | "postcss": "^8.5.3",
1373 | "rollup": "^4.34.9",
1374 | "tinyglobby": "^0.2.13"
1375 | },
1376 | "bin": {
1377 | "vite": "bin/vite.js"
1378 | },
1379 | "engines": {
1380 | "node": "^18.0.0 || ^20.0.0 || >=22.0.0"
1381 | },
1382 | "funding": {
1383 | "url": "https://github.com/vitejs/vite?sponsor=1"
1384 | },
1385 | "optionalDependencies": {
1386 | "fsevents": "~2.3.3"
1387 | },
1388 | "peerDependencies": {
1389 | "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0",
1390 | "jiti": ">=1.21.0",
1391 | "less": "*",
1392 | "lightningcss": "^1.21.0",
1393 | "sass": "*",
1394 | "sass-embedded": "*",
1395 | "stylus": "*",
1396 | "sugarss": "*",
1397 | "terser": "^5.16.0",
1398 | "tsx": "^4.8.1",
1399 | "yaml": "^2.4.2"
1400 | },
1401 | "peerDependenciesMeta": {
1402 | "@types/node": {
1403 | "optional": true
1404 | },
1405 | "jiti": {
1406 | "optional": true
1407 | },
1408 | "less": {
1409 | "optional": true
1410 | },
1411 | "lightningcss": {
1412 | "optional": true
1413 | },
1414 | "sass": {
1415 | "optional": true
1416 | },
1417 | "sass-embedded": {
1418 | "optional": true
1419 | },
1420 | "stylus": {
1421 | "optional": true
1422 | },
1423 | "sugarss": {
1424 | "optional": true
1425 | },
1426 | "terser": {
1427 | "optional": true
1428 | },
1429 | "tsx": {
1430 | "optional": true
1431 | },
1432 | "yaml": {
1433 | "optional": true
1434 | }
1435 | }
1436 | },
1437 | "node_modules/vite-node": {
1438 | "version": "3.2.3",
1439 | "resolved": "https://registry.npmjs.org/vite-node/-/vite-node-3.2.3.tgz",
1440 | "integrity": "sha512-gc8aAifGuDIpZHrPjuHyP4dpQmYXqWw7D1GmDnWeNWP654UEXzVfQ5IHPSK5HaHkwB/+p1atpYpSdw/2kOv8iQ==",
1441 | "dev": true,
1442 | "license": "MIT",
1443 | "dependencies": {
1444 | "cac": "^6.7.14",
1445 | "debug": "^4.4.1",
1446 | "es-module-lexer": "^1.7.0",
1447 | "pathe": "^2.0.3",
1448 | "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0"
1449 | },
1450 | "bin": {
1451 | "vite-node": "vite-node.mjs"
1452 | },
1453 | "engines": {
1454 | "node": "^18.0.0 || ^20.0.0 || >=22.0.0"
1455 | },
1456 | "funding": {
1457 | "url": "https://opencollective.com/vitest"
1458 | }
1459 | },
1460 | "node_modules/vitest": {
1461 | "version": "3.2.3",
1462 | "resolved": "https://registry.npmjs.org/vitest/-/vitest-3.2.3.tgz",
1463 | "integrity": "sha512-E6U2ZFXe3N/t4f5BwUaVCKRLHqUpk1CBWeMh78UT4VaTPH/2dyvH6ALl29JTovEPu9dVKr/K/J4PkXgrMbw4Ww==",
1464 | "dev": true,
1465 | "license": "MIT",
1466 | "dependencies": {
1467 | "@types/chai": "^5.2.2",
1468 | "@vitest/expect": "3.2.3",
1469 | "@vitest/mocker": "3.2.3",
1470 | "@vitest/pretty-format": "^3.2.3",
1471 | "@vitest/runner": "3.2.3",
1472 | "@vitest/snapshot": "3.2.3",
1473 | "@vitest/spy": "3.2.3",
1474 | "@vitest/utils": "3.2.3",
1475 | "chai": "^5.2.0",
1476 | "debug": "^4.4.1",
1477 | "expect-type": "^1.2.1",
1478 | "magic-string": "^0.30.17",
1479 | "pathe": "^2.0.3",
1480 | "picomatch": "^4.0.2",
1481 | "std-env": "^3.9.0",
1482 | "tinybench": "^2.9.0",
1483 | "tinyexec": "^0.3.2",
1484 | "tinyglobby": "^0.2.14",
1485 | "tinypool": "^1.1.0",
1486 | "tinyrainbow": "^2.0.0",
1487 | "vite": "^5.0.0 || ^6.0.0 || ^7.0.0-0",
1488 | "vite-node": "3.2.3",
1489 | "why-is-node-running": "^2.3.0"
1490 | },
1491 | "bin": {
1492 | "vitest": "vitest.mjs"
1493 | },
1494 | "engines": {
1495 | "node": "^18.0.0 || ^20.0.0 || >=22.0.0"
1496 | },
1497 | "funding": {
1498 | "url": "https://opencollective.com/vitest"
1499 | },
1500 | "peerDependencies": {
1501 | "@edge-runtime/vm": "*",
1502 | "@types/debug": "^4.1.12",
1503 | "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0",
1504 | "@vitest/browser": "3.2.3",
1505 | "@vitest/ui": "3.2.3",
1506 | "happy-dom": "*",
1507 | "jsdom": "*"
1508 | },
1509 | "peerDependenciesMeta": {
1510 | "@edge-runtime/vm": {
1511 | "optional": true
1512 | },
1513 | "@types/debug": {
1514 | "optional": true
1515 | },
1516 | "@types/node": {
1517 | "optional": true
1518 | },
1519 | "@vitest/browser": {
1520 | "optional": true
1521 | },
1522 | "@vitest/ui": {
1523 | "optional": true
1524 | },
1525 | "happy-dom": {
1526 | "optional": true
1527 | },
1528 | "jsdom": {
1529 | "optional": true
1530 | }
1531 | }
1532 | },
1533 | "node_modules/why-is-node-running": {
1534 | "version": "2.3.0",
1535 | "resolved": "https://registry.npmjs.org/why-is-node-running/-/why-is-node-running-2.3.0.tgz",
1536 | "integrity": "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==",
1537 | "dev": true,
1538 | "license": "MIT",
1539 | "dependencies": {
1540 | "siginfo": "^2.0.0",
1541 | "stackback": "0.0.2"
1542 | },
1543 | "bin": {
1544 | "why-is-node-running": "cli.js"
1545 | },
1546 | "engines": {
1547 | "node": ">=8"
1548 | }
1549 | }
1550 | }
1551 | }
1552 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@minescope/mineping",
3 | "version": "2.0.0",
4 | "description": "Ping both Minecraft Bedrock and Java servers.",
5 | "main": "index.js",
6 | "type": "module",
7 | "types": "types/index.d.ts",
8 | "scripts": {
9 | "test": "vitest run",
10 | "test:watch": "vitest",
11 | "types:check": "tsc --noEmit",
12 | "types:build": "tsc"
13 | },
14 | "repository": {
15 | "type": "git",
16 | "url": "git://github.com/minescope/mineping.git"
17 | },
18 | "license": "MIT",
19 | "keywords": [
20 | "minecraft",
21 | "raknet",
22 | "node",
23 | "mcpe",
24 | "mcbe",
25 | "ping",
26 | "bedrock"
27 | ],
28 | "author": {
29 | "name": "Timofey Gelazoniya",
30 | "email": "timofey@z4n.me",
31 | "url": "https://zeldon.ru"
32 | },
33 | "engines": {
34 | "node": ">=14"
35 | },
36 | "dependencies": {
37 | "debug": "^4.4.1"
38 | },
39 | "devDependencies": {
40 | "@types/debug": "^4.1.12",
41 | "@types/node": "^24.0.3",
42 | "typescript": "^5.8.3",
43 | "vitest": "^3.2.3"
44 | }
45 | }
46 |
--------------------------------------------------------------------------------
/test/bedrock.test.js:
--------------------------------------------------------------------------------
1 | import dgram from "node:dgram";
2 | import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
3 | import { pingBedrock } from "../lib/bedrock.js";
4 |
5 | vi.mock("node:dgram");
6 |
7 | describe("bedrock.js", () => {
8 | let mockSocket;
9 |
10 | beforeEach(() => {
11 | // A store for event handlers, closed over by the mockSocket.
12 | const handlers = {};
13 |
14 | // Create a stateful mock socket to simulate EventEmitter.
15 | mockSocket = {
16 | send: vi.fn(),
17 | close: vi.fn(),
18 | on: vi.fn((event, handler) => {
19 | handlers[event] = handler;
20 | }),
21 | emit: vi.fn((event, ...args) => {
22 | if (handlers[event]) {
23 | handlers[event](...args);
24 | }
25 | }),
26 | };
27 |
28 | dgram.createSocket = vi.fn().mockReturnValue(mockSocket);
29 | vi.useFakeTimers();
30 | });
31 |
32 | afterEach(() => {
33 | vi.restoreAllMocks();
34 | vi.useRealTimers();
35 | });
36 |
37 | it("should ping a 3rd party server and parse MOTD", async () => {
38 | const host = "play.example.com";
39 | const options = { port: 25565, timeout: 10000 };
40 | const pingPromise = pingBedrock(host, options);
41 |
42 | const motd =
43 | "MCPE;§l§bOasys§fPE §eГриф§7, §cДуэли§7, §aКейсы;0;1337;1070;1999;-138584171542148188;oasys-pe.ru;Adventure;1";
44 | const mockPongPacket = createMockPongPacket(motd);
45 |
46 | mockSocket.emit("message", mockPongPacket);
47 |
48 | const result = await pingPromise;
49 |
50 | expect(dgram.createSocket).toHaveBeenCalledWith("udp4");
51 | expect(mockSocket.send).toHaveBeenCalledWith(
52 | expect.any(Buffer),
53 | 0,
54 | 33,
55 | options.port,
56 | host
57 | );
58 | expect(mockSocket.close).toHaveBeenCalled();
59 | expect(result).toEqual({
60 | edition: "MCPE",
61 | name: "§l§bOasys§fPE §eГриф§7, §cДуэли§7, §aКейсы",
62 | levelName: "oasys-pe.ru",
63 | gamemode: "Adventure",
64 | version: {
65 | protocol: 0,
66 | minecraft: "1337",
67 | },
68 | players: {
69 | online: 1070,
70 | max: 1999,
71 | },
72 | port: {
73 | v4: undefined,
74 | v6: undefined,
75 | },
76 | guid: -138584171542148188n,
77 | isNintendoLimited: false,
78 | isEditorModeEnabled: undefined,
79 | });
80 | });
81 |
82 | it("should ping a BDS server with default `server.properties` and parse MOTD", async () => {
83 | const host = "play.example.com";
84 | const options = { port: 25565, timeout: 10000 };
85 | const pingPromise = pingBedrock(host, options);
86 |
87 | const motd =
88 | "MCPE;Dedicated Server;800;1.21.84;0;10;11546321190880321782;Bedrock level;Survival;1;19132;19133;0;";
89 | const mockPongPacket = createMockPongPacket(motd);
90 |
91 | mockSocket.emit("message", mockPongPacket);
92 |
93 | const result = await pingPromise;
94 |
95 | expect(dgram.createSocket).toHaveBeenCalledWith("udp4");
96 | expect(mockSocket.send).toHaveBeenCalledWith(
97 | expect.any(Buffer),
98 | 0,
99 | 33,
100 | options.port,
101 | host
102 | );
103 | expect(mockSocket.close).toHaveBeenCalled();
104 | expect(result).toEqual({
105 | edition: "MCPE",
106 | name: "Dedicated Server",
107 | levelName: "Bedrock level",
108 | gamemode: "Survival",
109 | version: {
110 | protocol: 800,
111 | minecraft: "1.21.84",
112 | },
113 | players: {
114 | online: 0,
115 | max: 10,
116 | },
117 | port: {
118 | v4: 19132,
119 | v6: 19133,
120 | },
121 | guid: 11546321190880321782n,
122 | isNintendoLimited: false,
123 | isEditorModeEnabled: false,
124 | });
125 | });
126 |
127 | describe("errors", () => {
128 | it("should throw an error if host is not provided", async () => {
129 | await expect(pingBedrock(null)).rejects.toThrow(
130 | "Host argument is required"
131 | );
132 | });
133 |
134 | it("should reject on socket timeout", async () => {
135 | const pingPromise = pingBedrock("play.example.com", { timeout: 1000 });
136 |
137 | vi.advanceTimersByTime(1000);
138 |
139 | await expect(pingPromise).rejects.toThrow("Socket timeout");
140 | expect(mockSocket.close).toHaveBeenCalled();
141 | });
142 |
143 | it("should reject on a generic socket error", async () => {
144 | const pingPromise = pingBedrock("play.example.com");
145 |
146 | // Simulate a DNS or network error by emitting it.
147 | mockSocket.emit("error", new Error("EHOSTUNREACH"));
148 |
149 | await expect(pingPromise).rejects.toThrow("EHOSTUNREACH");
150 | });
151 |
152 | it("should only reject once, even if multiple errors occur", async () => {
153 | const pingPromise = pingBedrock("play.example.com");
154 |
155 | // Fire a socket error first.
156 | mockSocket.emit("error", new Error("First error"));
157 |
158 | // Then, try to trigger another error by sending a bad message.
159 | mockSocket.emit("message", Buffer.alloc(0));
160 |
161 | await expect(pingPromise).rejects.toThrow("First error");
162 | expect(mockSocket.close).toHaveBeenCalledTimes(1);
163 | });
164 | });
165 | });
166 |
167 | function createMockPongPacket(motd) {
168 | const motdBuffer = Buffer.from(motd, "utf-8");
169 | const packet = Buffer.alloc(35 + motdBuffer.length);
170 | packet.writeUInt8(0x1c, 0);
171 | packet.writeBigInt64LE(BigInt(Date.now()), 1);
172 | Buffer.from("00ffff00fefefefefdfdfdfd12345678", "hex").copy(packet, 9);
173 | packet.writeUInt16BE(motdBuffer.length, 33);
174 | motdBuffer.copy(packet, 35);
175 | return packet;
176 | }
177 |
--------------------------------------------------------------------------------
/test/java.test.js:
--------------------------------------------------------------------------------
1 | import net from "node:net";
2 | import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
3 | import { pingJava } from "../lib/java.js";
4 | import * as varint from "../lib/varint.js";
5 |
6 | const mockResolveSrv = vi.fn();
7 |
8 | vi.mock("node:net");
9 | vi.mock("node:dns/promises", () => ({
10 | Resolver: vi.fn().mockImplementation(() => ({
11 | resolveSrv: mockResolveSrv,
12 | })),
13 | }));
14 |
15 | describe("pingJava", () => {
16 | let mockSocket;
17 |
18 | beforeEach(() => {
19 | // Reset mocks before each test.
20 | mockResolveSrv.mockClear();
21 | // Simulate no SRV record found by default.
22 | mockResolveSrv.mockResolvedValue([]);
23 |
24 | const mockHandlers = {};
25 | mockSocket = {
26 | write: vi.fn(),
27 | // Make `destroy` emit 'error' if an error is passed.
28 | destroy: vi.fn((err) => {
29 | if (err) {
30 | mockSocket.emit("error", err);
31 | }
32 | }),
33 | setNoDelay: vi.fn(),
34 | on: vi.fn((event, handler) => (mockHandlers[event] = handler)),
35 | emit: vi.fn((event, ...args) => mockHandlers[event]?.(...args)),
36 | };
37 | net.createConnection.mockReturnValue(mockSocket);
38 | vi.useFakeTimers();
39 | });
40 |
41 | afterEach(() => {
42 | vi.restoreAllMocks();
43 | vi.useRealTimers();
44 | });
45 |
46 | it("should ping a server and handle a chunked response", async () => {
47 | const host = "mc.hypixel.net";
48 | const options = { port: 25565 };
49 | const pingPromise = pingJava(host, options);
50 |
51 | // Allow the async SRV lookup to complete
52 | await vi.runAllTicks();
53 |
54 | expect(net.createConnection).toHaveBeenCalledWith({
55 | host: host,
56 | port: options.port,
57 | });
58 |
59 | expect(net.createConnection).toHaveBeenCalledWith({
60 | host,
61 | port: options.port,
62 | });
63 |
64 | mockSocket.emit("connect");
65 | expect(mockSocket.write).toHaveBeenCalledTimes(2);
66 |
67 | const mockResponse = {
68 | version: { name: "1.21", protocol: 765 },
69 | players: { max: 20, online: 5, sample: [] },
70 | description: "A Minecraft Server",
71 | };
72 |
73 | const fullPacket = createMockJavaResponse(mockResponse);
74 | const chunk1 = fullPacket.subarray(0, 10);
75 | const chunk2 = fullPacket.subarray(10);
76 |
77 | // Simulate receiving data in chunks
78 | mockSocket.emit("data", chunk1);
79 | mockSocket.emit("data", chunk2);
80 |
81 | const result = await pingPromise;
82 | expect(result).toEqual(mockResponse);
83 | });
84 |
85 | describe("errors", () => {
86 | it("should throw an error if host is not provided", async () => {
87 | await expect(pingJava(null)).rejects.toThrow("Host argument is required");
88 | });
89 |
90 | it("should reject on socket timeout", async () => {
91 | const pingPromise = pingJava("localhost", { timeout: 1000 });
92 | await vi.runAllTicks();
93 | mockSocket.emit("connect");
94 | vi.advanceTimersByTime(1000);
95 | await expect(pingPromise).rejects.toThrow("Socket timeout");
96 | });
97 |
98 | it("should reject on connection error", async () => {
99 | const pingPromise = pingJava("localhost");
100 | await vi.runAllTicks();
101 | mockSocket.emit("error", new Error("ECONNREFUSED"));
102 | await expect(pingPromise).rejects.toThrow("ECONNREFUSED");
103 | });
104 |
105 | it("should reject if the socket closes prematurely without a response", async () => {
106 | const pingPromise = pingJava("localhost");
107 |
108 | // Allow the initial async operations to complete
109 | await vi.runAllTicks();
110 |
111 | // Simulate the server accepting the connection and then immediately closing it
112 | mockSocket.emit("connect");
113 | mockSocket.emit("close");
114 |
115 | // The promise should reject with our specific 'close' handler message
116 | await expect(pingPromise).rejects.toThrow(
117 | "Socket closed unexpectedly without a response."
118 | );
119 | });
120 |
121 | it("should only reject once, even if multiple errors occur", async () => {
122 | const pingPromise = pingJava("localhost");
123 | await vi.runAllTicks();
124 | mockSocket.emit("error", new Error("First error"));
125 | mockSocket.emit("error", new Error("Second error")); // Should be ignored
126 | await expect(pingPromise).rejects.toThrow("First error");
127 | });
128 | });
129 | });
130 |
131 | /**
132 | * Creates a mock Java status response packet according to the protocol.
133 | * Structure: [Overall Length] [Packet ID] [JSON Length] [JSON String]
134 | * @param {object} response The JSON response object
135 | * @returns {Buffer}
136 | */
137 | function createMockJavaResponse(response) {
138 | const jsonString = JSON.stringify(response);
139 | const jsonBuffer = varint.encodeString(jsonString);
140 | const jsonLength = varint.encodeVarInt(jsonBuffer.length);
141 | const packetId = varint.encodeVarInt(0x00);
142 | const payloadParts = [packetId, jsonLength, jsonBuffer];
143 | return varint.concatPackets(payloadParts);
144 | }
145 |
--------------------------------------------------------------------------------
/test/varint.test.js:
--------------------------------------------------------------------------------
1 | import { describe, it, expect } from "vitest";
2 | import * as varint from "../lib/varint.js";
3 |
4 | describe("varint.js", () => {
5 | it("should encode and decode integers symmetrically (round-trip)", () => {
6 | const testValues = [
7 | 0,
8 | 1,
9 | 127, // Max 1-byte
10 | 128, // Min 2-byte
11 | 255,
12 | 16383, // Max 2-byte
13 | 16384, // Min 3-byte
14 | 2147483647, // Max signed 32-bit int
15 | -1, // Critical edge case (encodes as max unsigned int)
16 | ];
17 |
18 | testValues.forEach((value) => {
19 | const encoded = varint.encodeVarInt(value);
20 | const { value: decoded } = varint.decodeVarInt(encoded, 0);
21 | expect(decoded, `Value ${value} failed round-trip`).toBe(value);
22 | });
23 | });
24 |
25 | it("should decode an integer from a non-zero offset", () => {
26 | // [255 (invalid varint), 128 (valid varint), 127 (valid varint)]
27 | const buffer = Buffer.from([0xff, 0x80, 0x01, 0x7f]);
28 | const { value: decoded } = varint.decodeVarInt(buffer, 1);
29 | expect(decoded).toBe(128);
30 | });
31 |
32 | it("should throw an error for a malformed varint that is too long", () => {
33 | const invalidBuffer = Buffer.from([0x80, 0x80, 0x80, 0x80, 0x80, 0x80]);
34 | expect(() => varint.decodeVarInt(invalidBuffer, 0)).toThrow(
35 | "VarInt is too big or malformed"
36 | );
37 | });
38 |
39 | it("should encode 16-bit unsigned shorts in big-endian format", () => {
40 | expect(varint.encodeUShort(0)).toEqual(Buffer.from([0x00, 0x00]));
41 | expect(varint.encodeUShort(256)).toEqual(Buffer.from([0x01, 0x00]));
42 | expect(varint.encodeUShort(65535)).toEqual(Buffer.from([0xff, 0xff]));
43 | });
44 |
45 | it("should correctly assemble a Minecraft packet with a length prefix", () => {
46 | const payloadParts = [
47 | varint.encodeVarInt(0), // protocol
48 | varint.encodeString("mc.example.com"), // host
49 | varint.encodeUShort(25565), // port
50 | ];
51 | const payload = Buffer.concat(payloadParts);
52 | const finalPacket = varint.concatPackets(payloadParts);
53 | const { value: decodedPacketLength, bytesRead } = varint.decodeVarInt(
54 | finalPacket,
55 | 0
56 | );
57 | expect(decodedPacketLength).toBe(payload.length);
58 | const decodedPayload = finalPacket.subarray(bytesRead);
59 | expect(decodedPayload).toEqual(payload);
60 | });
61 | });
62 |
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "include": ["lib/**/*.js", "index.js"],
3 | "compilerOptions": {
4 | "allowJs": true,
5 | "declaration": true,
6 | "emitDeclarationOnly": true,
7 | "outDir": "types",
8 | "removeComments": false
9 | }
10 | }
--------------------------------------------------------------------------------
/types/index.d.ts:
--------------------------------------------------------------------------------
1 | export { pingJava } from "./lib/java.js";
2 | export { pingBedrock } from "./lib/bedrock.js";
3 |
--------------------------------------------------------------------------------
/types/lib/bedrock.d.ts:
--------------------------------------------------------------------------------
1 | /**
2 | * Asynchronously pings a Minecraft Bedrock server.
3 | * @param {string} host - The IP address or hostname of the server.
4 | * @param {BedrockPingOptions} [options={}] - Optional configuration.
5 | * @returns {Promise} A promise that resolves with the server's parsed MOTD.
6 | */
7 | export function pingBedrock(host: string, options?: BedrockPingOptions): Promise;
8 | /**
9 | * Representation of raw, semicolon-delimited MOTD string.
10 | * This struct directly mirrors the fields and order from the server response.
11 | * See [`Unconnected Pong Documentation`](https://minecraft.wiki/w/RakNet#Unconnected_Pong) for more details.
12 | */
13 | export type BedrockMotd = {
14 | /**
15 | * - The edition of the server (MCPE or MCEE).
16 | */
17 | edition: string;
18 | /**
19 | * - The primary name of the server (first line of MOTD).
20 | */
21 | name: string;
22 | /**
23 | * - The protocol version.
24 | */
25 | protocol: number;
26 | /**
27 | * - The game version (e.g., "1.21.2").
28 | */
29 | version: string;
30 | /**
31 | * - The current number of players online.
32 | */
33 | playerCount: number;
34 | /**
35 | * - The maximum number of players allowed.
36 | */
37 | playerMax: number;
38 | /**
39 | * - The server's GUID.
40 | */
41 | serverGuid: bigint;
42 | /**
43 | * - The secondary name of the server (second line of MOTD).
44 | */
45 | subName: string;
46 | /**
47 | * - The default gamemode (e.g., "Survival").
48 | */
49 | gamemode: string;
50 | /**
51 | * - Whether the server is Nintendo limited.
52 | */
53 | nintendoLimited?: boolean;
54 | /**
55 | * - The server's IPv4 port, if provided.
56 | */
57 | port?: number;
58 | /**
59 | * - The server's IPv6 port, if provided.
60 | */
61 | ipv6Port?: number;
62 | /**
63 | * - Whether the server is in editor mode, if provided. See [Minecraft Editor Mode Documentation](https://learn.microsoft.com/en-us/minecraft/creator/documents/bedrockeditor/editoroverview?view=minecraft-bedrock-stable) for more details.
64 | */
65 | editorMode?: boolean;
66 | };
67 | /**
68 | * Represents the structured and user-friendly response from a server ping.
69 | * This is the public-facing object that users of the library will receive.
70 | */
71 | export type BedrockPingResponse = {
72 | /**
73 | * - The edition of the server (MCPE or MCEE).
74 | */
75 | edition: string;
76 | /**
77 | * - The primary name of the server (first line of MOTD).
78 | */
79 | name: string;
80 | /**
81 | * - The name of the world or level being hosted.
82 | */
83 | levelName: string;
84 | /**
85 | * - The default gamemode of the server.
86 | */
87 | gamemode: string;
88 | /**
89 | * - Game and protocol versions.
90 | */
91 | version: {
92 | protocol: number;
93 | minecraft: string;
94 | };
95 | /**
96 | * - Current and maximum player counts.
97 | */
98 | players: {
99 | online: number;
100 | max: number;
101 | };
102 | /**
103 | * - Announced IPv4 and IPv6 ports.
104 | */
105 | port: {
106 | v4?: number;
107 | v6?: number;
108 | };
109 | /**
110 | * - The server's unique 64-bit GUID.
111 | */
112 | guid: bigint;
113 | /**
114 | * - True if the server restricts Nintendo Switch players.
115 | */
116 | isNintendoLimited?: boolean;
117 | /**
118 | * - True if the server is in editor mode. See [Minecraft Editor Mode Documentation](https://learn.microsoft.com/en-us/minecraft/creator/documents/bedrockeditor/editoroverview?view=minecraft-bedrock-stable) for more details.
119 | */
120 | isEditorModeEnabled?: boolean;
121 | };
122 | export type BedrockPingOptions = {
123 | /**
124 | * - The server port to ping.
125 | */
126 | port?: number;
127 | /**
128 | * - The timeout in milliseconds for the request.
129 | */
130 | timeout?: number;
131 | };
132 |
--------------------------------------------------------------------------------
/types/lib/java.d.ts:
--------------------------------------------------------------------------------
1 | /**
2 | * Asynchronously Pings a Minecraft Java Edition server.
3 | * This function performs an SRV lookup and then attempts to connect and retrieve the server status.
4 | * @param {string} host - The server address to ping.
5 | * @param {JavaPingOptions} [options={}] - Optional configuration.
6 | * @returns {Promise} A promise that resolves with the server's status.
7 | */
8 | export function pingJava(host: string, options?: JavaPingOptions): Promise;
9 | /**
10 | * Represents the structured and user-friendly response from a server ping.
11 | * The fields and their optionality are based on the protocol documentation.
12 | * See [Status Response Documentation](https://minecraft.wiki/w/Java_Edition_protocol/Server_List_Ping#Status_Response) for more details.
13 | */
14 | export type JavaPingResponse = {
15 | /**
16 | * - Contains the server's version name and protocol number.
17 | */
18 | version: {
19 | name: string;
20 | protocol: number;
21 | };
22 | /**
23 | * - Player count and a sample of online players.
24 | */
25 | players?: {
26 | max: number;
27 | online: number;
28 | sample?: Array<{
29 | name: string;
30 | id: string;
31 | }>;
32 | };
33 | /**
34 | * - The server's Message of the Day (MOTD).
35 | */
36 | description?: object | string;
37 | /**
38 | * - A Base64-encoded 64x64 PNG image data URI.
39 | */
40 | favicon?: string;
41 | /**
42 | * - True if the server requires clients to have a Mojang-signed public key.
43 | */
44 | enforcesSecureChat?: boolean;
45 | /**
46 | * - True if a mod is installed to disable chat reporting.
47 | */
48 | preventsChatReports?: boolean;
49 | };
50 | export type JavaPingOptions = {
51 | /**
52 | * - The fallback port if an SRV record is not found.
53 | */
54 | port?: number;
55 | /**
56 | * - The connection timeout in milliseconds.
57 | */
58 | timeout?: number;
59 | /**
60 | * - The protocol version to use in the handshake. `-1` is for auto-detection.
61 | */
62 | protocolVersion?: number;
63 | };
64 |
--------------------------------------------------------------------------------
/types/lib/varint.d.ts:
--------------------------------------------------------------------------------
1 | /**
2 | * Encodes an integer into a VarInt buffer.
3 | * VarInts are never longer than 5 bytes for the Minecraft protocol.
4 | * @param {number} value The integer to encode
5 | * @returns {Buffer} The encoded VarInt as a buffer
6 | * @throws {VarIntError} if the value is too large to be encoded
7 | */
8 | export function encodeVarInt(value: number): Buffer;
9 | /**
10 | * Encodes a string into a UTF-8 buffer.
11 | * @param {string} value The string to encode
12 | * @returns {Buffer}
13 | */
14 | export function encodeString(value: string): Buffer;
15 | /**
16 | * Encodes an unsigned short (16-bit big-endian) into a 2-byte buffer.
17 | * @param {number} value The number to encode
18 | * @returns {Buffer}
19 | */
20 | export function encodeUShort(value: number): Buffer;
21 | /**
22 | * Creates a Minecraft-style packet by concatenating chunks and prefixing the total length as a VarInt.
23 | * @param {Buffer[]} chunks An array of buffers to include in the packet payload
24 | * @returns {Buffer} The complete packet with its length prefix
25 | */
26 | export function concatPackets(chunks: Buffer[]): Buffer;
27 | /**
28 | * Decodes a VarInt from a buffer.
29 | * Returns the decoded value and the number of bytes it consumed.
30 | * @param {Buffer} buffer The buffer to read from
31 | * @param {number} [offset=0] The starting offset in the buffer
32 | * @returns {{ value: number, bytesRead: number }}
33 | * @throws {VarIntError} if the buffer is too short or the VarInt is malformed
34 | */
35 | export function decodeVarInt(buffer: Buffer, offset?: number): {
36 | value: number;
37 | bytesRead: number;
38 | };
39 | export const ERR_VARINT_BUFFER_UNDERFLOW: "VARINT_BUFFER_UNDERFLOW";
40 | export const ERR_VARINT_MALFORMED: "VARINT_MALFORMED";
41 | export const ERR_VARINT_ENCODE_TOO_LARGE: "VARINT_ENCODE_TOO_LARGE";
42 | export class VarIntError extends Error {
43 | /**
44 | * @param {string} message The error message.
45 | * @param {string} code The error code.
46 | */
47 | constructor(message: string, code: string);
48 | code: string;
49 | }
50 |
--------------------------------------------------------------------------------