diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json new file mode 100644 index 0000000..d749c77 --- /dev/null +++ b/.devcontainer/devcontainer.json @@ -0,0 +1,43 @@ +// For format details, see https://aka.ms/devcontainer.json. For config options, see the +// README at: https://github.com/devcontainers/templates/tree/main/src/typescript-node +{ + // Or use a Dockerfile or Docker Compose file. More info: https://containers.dev/guide/dockerfile + // "image": "mcr.microsoft.com/devcontainers/typescript-node:1-22-bookworm", + + // Features to add to the dev container. More info: https://containers.dev/features. + // "features": { + // "ghcr.io/devcontainers/features/node:1": {} + // }, + + "postCreateCommand": { + "dependencies": "npm install", + "build": "npm run build" + }, + + "postAttachCommand": "# Welcome to your Codespace! Run `npm run start` to start development.", + + // Configure tool-specific properties. + "customizations": { + "vscode": { + "extensions": ["dbaeumer.vscode-eslint", "esbenp.prettier-vscode"], + "settings": { + "editor.formatOnSave": true, + "[javascript]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[typescript]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[json]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[jsonl]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[jsonc]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + } + } + } + } +} diff --git a/.editorconfig b/.editorconfig index 4b43aa8..a091831 100644 --- a/.editorconfig +++ b/.editorconfig @@ -2,12 +2,10 @@ root = true [*] indent_style = space -indent_size = 4 +indent_size = 2 end_of_line = lf charset = utf-8 trim_trailing_whitespace = true insert_final_newline = true [*.md] trim_trailing_whitespace = false -[*.json] -indent_size = 2 diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml new file mode 100644 index 0000000..1132a56 --- /dev/null +++ b/.github/workflows/main.yml @@ -0,0 +1,32 @@ +name: Build & Test main + +on: + push: + branches: [main] + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Setup Node.js + uses: actions/setup-node@v3 + with: + node-version: "24" + + - name: Install dependencies + run: npm install + + - name: Build + run: npm run build + + - name: Run tests + run: npm run test:coverage + + - name: Coveralls + uses: coverallsapp/github-action@v2 + with: + github-token: ${{ secrets.GITHUB_TOKEN }} diff --git a/.github/workflows/pr.yml b/.github/workflows/pr.yml new file mode 100644 index 0000000..efe1e21 --- /dev/null +++ b/.github/workflows/pr.yml @@ -0,0 +1,26 @@ +name: Build & Test PR + +on: + pull_request: + +jobs: + build-pr: + runs-on: ubuntu-latest + + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Setup Node.js + uses: actions/setup-node@v3 + with: + node-version: "20" + + - name: Install dependencies + run: npm install + + - name: Build + run: npm run build + + - name: Run tests + run: npm run test:coverage diff --git a/.gitignore b/.gitignore index 551b248..9cab6c1 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,5 @@ /node_modules -/dist +/lib /test_dist /coverage /.nyc_output diff --git a/.nycrc.json b/.nycrc.json deleted file mode 100644 index 30a09be..0000000 --- a/.nycrc.json +++ /dev/null @@ -1,8 +0,0 @@ -{ - "cache": false, - "extension": [".ts"], - "include": [ - "lib/**/*.ts", - "test_dist/**/*.js" - ] -} diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index 7fef8cc..0000000 --- a/.travis.yml +++ /dev/null @@ -1,12 +0,0 @@ -language: node_js -node_js: - - 12 - - 11 - - 10 -install: - - npm install -script: - - npm run build - - npm test -after_success: - - npm run test:coveralls diff --git a/README.md b/README.md index 7dfd49e..041d939 100644 --- a/README.md +++ b/README.md @@ -1,33 +1,34 @@ -[![Build Status](https://travis-ci.org/Novage/wt-tracker.svg?branch=master)](https://travis-ci.org/Novage/wt-tracker) -[![Coverage Status](https://coveralls.io/repos/github/Novage/wt-tracker/badge.svg?branch=master)](https://coveralls.io/github/Novage/wt-tracker?branch=master) +[![Build Status](https://github.com/Novage/wt-tracker/actions/workflows/main.yml/badge.svg)](https://github.com/Novage/wt-tracker/actions/workflows/main.yml) +[![Coverage Status](https://coveralls.io/repos/github/Novage/wt-tracker/badge.svg?branch=main)](https://coveralls.io/github/Novage/wt-tracker?branch=main) + # wt-tracker + High-performance WebTorrent tracker. WebTorrent tracker is a required component of [WebTorrent](https://github.com/webtorrent/webtorrent) and [P2P Media Loader](https://github.com/Novage/p2p-media-loader) (peer-to-peer networks for web browsers) to do [WebRTC](https://en.wikipedia.org/wiki/WebRTC) signaling - exchanging connection data (i.e. [SDP](https://en.wikipedia.org/wiki/Session_Description_Protocol)) between peers - joining them into swarms. ## Features -* handles more than 40k WebSocket Secure (HTTPS) peers on a VPS with only 2 GiB memory and 1 virtual CPU thanks to [uWebSockets.js](https://github.com/uNetworking/uWebSockets.js) I/O backend and perfomance optimizations in the code -* handles ws:// (HTTP) and wss:// (HTTPS) connections simultaneously -* IPv4 and IPv6 support -* robust and well-tested: CI, unit tests, static code analyzis, 100% TypeScript -* supports tracker "scrape" extension -* statistics under /stats.json URL +- handles up to 30k WebSocket Secure (HTTPS) peers on a VPS with only 2 GiB memory and 1 virtual CPU thanks to [uWebSockets.js](https://github.com/uNetworking/uWebSockets.js) I/O backend and perfomance optimizations in the code +- handles ws:// (HTTP) and wss:// (HTTPS) connections simultaneously +- IPv4 and IPv6 support +- robust and well-tested: CI, unit tests, static code analyzis, 100% TypeScript +- supports tracker "scrape" extension +- statistics under /stats.json URL ## Related projects -* [P2P Media Loader](https://github.com/Novage/p2p-media-loader) - an open-source engine for P2P streaming of live and on demand video directly in a web browser HTML page -* [Novage, LLC](https://novage.com.ua/) - P2P development, support & consulting -* [WebTorrent](https://github.com/webtorrent/webtorrent) - streaming torrent client for the web https://webtorrent.io -* [uWebSockets.js](https://github.com/uNetworking/uWebSockets.js) - the Node.js bindings to µWebSockets, one of the most efficient web servers available +- [P2P Media Loader](https://github.com/Novage/p2p-media-loader) - an open-source engine for P2P streaming of live and on demand video directly in a web browser HTML page +- [Novage, LLC](https://novage.com.ua/) - P2P development, support & consulting +- [WebTorrent](https://github.com/webtorrent/webtorrent) - streaming torrent client for the web https://webtorrent.io +- [uWebSockets.js](https://github.com/uNetworking/uWebSockets.js) - the Node.js bindings to µWebSockets, one of the most efficient web servers available ## Build instructions -Node.js 10+ is required. +Node.js 22.18.0+ is required for direct TypeScript support. ```sh npm install -npm run build ``` ## Run instructions @@ -39,7 +40,14 @@ npm run build or ```sh -node dist/run-uws-tracker.js [config.json] +node src/run-tracker.ts [config.json] +``` + +or + +```sh +npm run build +node lib/run-tracker.js [config.json] ``` or @@ -52,15 +60,15 @@ npm start [config.json] See [config.json](sample/config.json) -|Name|Type|Description| -|----|----|-----------| -|servers.websockets.path|string|URL pattern for the WebSockets endpoint| -|servers.websockets.maxPayloadLength|number|The maximum length of received message| -|servers.websockets.midleTimeout|number|The maximum amount of seconds that may pass without sending or getting a message. Being idle for more than this, and the connection is severed.| -|servers.websockets.compression|0,1,2|0 = no compression, 1 = shared compressor, 2 = dedicated compressor (see [details](https://github.com/uNetworking/uWebSockets/blob/master/misc/READMORE.md#settings))| -|servers.websockets.maxConnections|number|The maximum number of WebSocket connections. 0 = no limit.| -|tracker.maxOffers|number|The maximum number of client's WebRTC SDP offers that are processed| -|tracker.announceInterval|number|Desired announce interval in seconds required from the clients| +| Name | Type | Description | +| ----------------------------------- | ------ | --------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| servers.websockets.path | string | URL pattern for the WebSockets endpoint | +| servers.websockets.maxPayloadLength | number | The maximum length of received message | +| servers.websockets.midleTimeout | number | The maximum amount of seconds that may pass without sending or getting a message. Being idle for more than this, and the connection is severed. | +| servers.websockets.compression | 0,1,2 | 0 = no compression, 1 = shared compressor, 2 = dedicated compressor (see [details](https://github.com/uNetworking/uWebSockets/blob/master/misc/READMORE.md#settings)) | +| servers.websockets.maxConnections | number | The maximum number of WebSocket connections. 0 = no limit. | +| tracker.maxOffers | number | The maximum number of client's WebRTC SDP offers that are processed | +| tracker.announceInterval | number | Desired announce interval in seconds required from the clients | ## Index HTML page diff --git a/bin/wt-tracker b/bin/wt-tracker index 17f387d..2a9686a 100755 --- a/bin/wt-tracker +++ b/bin/wt-tracker @@ -1,3 +1,3 @@ #!/usr/bin/env node -require("../dist/run-uws-tracker.js"); +import "../src/run-tracker.ts"; diff --git a/eslint.config.ts b/eslint.config.ts new file mode 100644 index 0000000..8fc4c39 --- /dev/null +++ b/eslint.config.ts @@ -0,0 +1,61 @@ +import globals from "globals"; +import eslint from "@eslint/js"; +import tsEslint from "typescript-eslint"; +import { defineConfig } from "eslint/config"; + +export default defineConfig([ + eslint.configs.recommended, + tsEslint.configs.strictTypeChecked, + tsEslint.configs.stylisticTypeChecked, + { + languageOptions: { + globals: { + ...globals.node, + }, + + ecmaVersion: 2022, + sourceType: "module", + + parserOptions: { + project: ["tsconfig.json"], // TODO: change to tsconfig.ling.json after fixing linter issues + }, + }, + + rules: { + "@typescript-eslint/consistent-type-definitions": "off", + "@typescript-eslint/no-unused-vars": [ + "warn", + { argsIgnorePattern: "^_" }, + ], + "@typescript-eslint/restrict-template-expressions": [ + "error", + { allowNumber: true }, + ], + "@typescript-eslint/no-confusing-void-expression": [ + "error", + { ignoreArrowShorthand: true }, + ], + "@typescript-eslint/no-unnecessary-condition": [ + "error", + { allowConstantLoopConditions: true }, + ], + + "no-var": "error", + "no-alert": "warn", + "prefer-const": "error", + "prefer-spread": "error", + "no-multi-assign": "error", + "prefer-template": "error", + "object-shorthand": "error", + "no-nested-ternary": "error", + "no-array-constructor": "error", + "prefer-object-spread": "error", + "prefer-arrow-callback": "error", + "prefer-destructuring": ["error", { object: true, array: false }], + "no-console": "warn", + curly: ["warn", "multi-line", "consistent"], + "no-debugger": "warn", + "spaced-comment": ["warn", "always", { markers: ["/"] }], + }, + }, +]); diff --git a/lib/fast-tracker.ts b/lib/fast-tracker.ts deleted file mode 100644 index 530b92a..0000000 --- a/lib/fast-tracker.ts +++ /dev/null @@ -1,398 +0,0 @@ -/** - * Copyright 2019 Novage LLC. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -import { Tracker, PeerContext, TrackerError } from "./tracker"; - -import * as Debug from "debug"; - -const debug = Debug("wt-tracker:fast-tracker"); -const debugEnabled = debug.enabled; - -export class FastTracker implements Tracker { - private _swarms = new Map(); - private _peers = new Map(); - - constructor(readonly settings: any = {}) { - this.settings = { - maxOffers: 20, - announceInterval: 120, - ...(settings ? settings : {}), - }; - } - - public get swarms(): ReadonlyMap }> { - return this._swarms; - } - - public processMessage(json: any, peer: PeerContext) { - const action: any = json.action; - - if (action === "announce") { - const event: any = json.event; - - if (event === undefined) { - if (json.answer === undefined) { - this.processAnnounce(json, peer); - } else { - this.processAnswer(json, peer); - } - } else if (event === "started") { - this.processAnnounce(json, peer); - } else if (event === "stopped") { - this.processStop(json, peer); - } else if (event === "completed") { - this.processAnnounce(json, peer, true); - } else { - throw new TrackerError("unknown announce event"); - } - } else if (action === "scrape") { - this.processScrape(json, peer); - } else { - throw new TrackerError("unknown action"); - } - } - - public disconnectPeer(peer: PeerContext) { - const peerId = peer.id; - if (peerId === undefined) { - return; - } - - if (debugEnabled) { - debug("disconnect peer:", Buffer.from(peerId).toString("hex")); - } - - // tslint:disable-next-line:forin - for (const infoHash in peer) { - const swarm: any = (peer as any)[infoHash]; - - if (!(swarm instanceof Swarm)) { - continue; - } - - swarm.removePeer(peer); - delete (peer as any)[infoHash]; - - if (debugEnabled) { - debug("disconnect peer: peer", Buffer.from(peerId).toString("hex"), "removed from swarm", Buffer.from(infoHash).toString("hex")); - } - - if (swarm.peers.length === 0) { - if (debugEnabled) { - debug("disconnect peer: swarm removed (empty)", Buffer.from(swarm.infoHash).toString("hex")); - } - this._swarms.delete(swarm.infoHash); - } - } - - this._peers.delete(peerId); - peer.id = undefined; - } - - private processAnnounce(json: any, peer: PeerContext, completed: boolean = false) { - const infoHash = json.info_hash; - const peerId = json.peer_id; - let swarm: Swarm | undefined; - - if (peer.id === undefined) { - if (typeof peerId !== "string") { - throw new TrackerError("announce: peer_id field is missing or wrong"); - } - - peer.id = peerId; - - const oldPeer = this._peers.get(peerId); - if (oldPeer !== undefined) { - this.disconnectPeer(oldPeer); - } - - this._peers.set(peerId, peer); - } else if (peer.id !== peerId) { - throw new TrackerError("announce: different peer_id on the same connection"); - } else { - swarm = (peer as any)[infoHash]; - } - - const isPeerCompleted = (completed || json.left === 0); - - if (swarm === undefined) { - swarm = this.addPeerToSwarm(peer, infoHash, isPeerCompleted); - } else if (swarm instanceof Swarm) { - if (debugEnabled) { - debug("announce: peer", Buffer.from(peer.id).toString("hex"), "in swarm", Buffer.from(infoHash).toString("hex")); - } - - if (isPeerCompleted) { - swarm.setCompleted(peer); - } - } else { - throw new TrackerError("announce: illegal info_hash field"); - } - - peer.sendMessage({ - action: "announce", - interval: this.settings.announceInterval, - info_hash: infoHash, - complete: swarm.completedCount, - incomplete: swarm.peers.length - swarm.completedCount, - }, peer); - - this.sendOffersToPeers(json, swarm.peers, peer, infoHash); - } - - private addPeerToSwarm(peer: PeerContext, infoHash: any, completed: boolean): Swarm { - let swarm = this._swarms.get(infoHash); - - if (swarm === undefined) { - if (typeof infoHash !== "string") { - throw new TrackerError("announce: info_hash field is missing or wrong"); - } - - if (debugEnabled) { - debug("announce: swarm created:", Buffer.from(infoHash).toString("hex")); - } - - swarm = new Swarm(infoHash); - this._swarms.set(infoHash, swarm); - } - - if (debugEnabled) { - debug("announce: peer", Buffer.from(peer.id!).toString("hex"), "added to swarm", Buffer.from(infoHash).toString("hex")); - } - - swarm.addPeer(peer, completed); - (peer as any)[infoHash] = swarm; - return swarm; - } - - // tslint:disable-next-line:cognitive-complexity - private sendOffersToPeers(json: any, peers: ReadonlyArray, peer: PeerContext, infoHash: string) { - if (peers.length <= 1) { - return; - } - - const offers: any = json.offers; - if (offers == undefined) { - return; - } else if (!(offers instanceof Array)) { - throw new TrackerError("announce: offers field is not an array"); - } - - const numwant = json.numwant; - if (!Number.isInteger(numwant)) { - return; - } - - const countPeersToSend = peers.length - 1; - const countOffersToSend = Math.min(countPeersToSend, offers.length, this.settings.maxOffers, numwant); - - if (countOffersToSend === countPeersToSend) { - // we have offers for all the peers from the swarm - send offers to all - const offersIterator = offers.values(); - for (const toPeer of peers) { - if (toPeer !== peer) { - sendOffer(offersIterator.next().value, peer.id!, toPeer, infoHash); - } - } - } else { - // send offers to random peers - let peerIndex = Math.floor(Math.random() * peers.length); - - for (let i = 0; i < countOffersToSend; i++) { - const toPeer = peers[peerIndex]; - - if (toPeer === peer) { - i--; // do one more iteration - } else { - sendOffer(offers[i], peer.id!, toPeer, infoHash); - } - - peerIndex++; - if (peerIndex === peers.length) { - peerIndex = 0; - } - } - } - - debug("announce: sent offers", countOffersToSend < 0 ? 0 : countOffersToSend); - } - - private processAnswer(json: any, peer: PeerContext) { - const toPeerId = json.to_peer_id; - const toPeer = this._peers.get(toPeerId); - if (toPeer === undefined) { - throw new TrackerError("answer: to_peer_id is not in the swarm"); - } - - json.peer_id = peer.id; - delete json.to_peer_id; - toPeer.sendMessage(json, toPeer); - - if (debugEnabled) { - debug("answer: from peer", Buffer.from(peer.id!).toString("hex"), "to peer", Buffer.from(toPeerId).toString("hex")); - } - } - - private processStop(json: any, peer: PeerContext) { - const infoHash: string = json.info_hash; - const swarm = (peer as any)[infoHash] as Swarm | undefined; - - if (!(swarm instanceof Swarm)) { - debug("stop event: peer not in the swarm"); - return; - } - - if (debugEnabled) { - debug("stop event: peer", Buffer.from(peer.id!).toString("hex"), "removed from swarm", Buffer.from(infoHash).toString("hex")); - } - - swarm.removePeer(peer); - delete (peer as any)[infoHash]; - - if (swarm.peers.length === 0) { - if (debugEnabled) { - debug("stop event: swarm removed (empty)", Buffer.from(infoHash).toString("hex")); - } - this._swarms.delete(infoHash); - } - } - - private processScrape(json: any, peer: PeerContext) { - const infoHash: any = json.info_hash; - const files: {[key: string]: { complete: number, incomplete: number, downloaded: number}} = {}; - - if (infoHash === undefined) { - for (const swarm of this._swarms.values()) { - files[swarm.infoHash] = { - complete: swarm.completedCount, - incomplete: swarm.peers.length - swarm.completedCount, - downloaded: swarm.completedCount, - }; - } - } else if (infoHash instanceof Array) { - for (const singleInfoHash of infoHash) { - const swarm = this._swarms.get(singleInfoHash); - if (swarm !== undefined) { - files[singleInfoHash] = { - complete: swarm.completedCount, - incomplete: swarm.peers.length - swarm.completedCount, - downloaded: swarm.completedCount, - }; - } else if (typeof singleInfoHash === "string") { - files[singleInfoHash] = { - complete: 0, - incomplete: 0, - downloaded: 0, - }; - } - } - } else { - const swarm = this._swarms.get(infoHash); - if (swarm !== undefined) { - files[infoHash] = { - complete: swarm.completedCount, - incomplete: swarm.peers.length - swarm.completedCount, - downloaded: swarm.completedCount, - }; - } else if (typeof infoHash === "string") { - files[infoHash] = { - complete: 0, - incomplete: 0, - downloaded: 0, - }; - } - } - - peer.sendMessage({ - action: "scrape", - files: files, - }, peer); - } -} - -// tslint:disable-next-line:max-classes-per-file -class Swarm { - public completedCount = 0; - - private _peers: PeerContext[] = []; - private completedPeers?: Set; - - constructor(readonly infoHash: string) {} - - public addPeer(peer: PeerContext, completed: boolean) { - this._peers.push(peer); - if (completed) { - if (this.completedPeers === undefined) { - this.completedPeers = new Set(); - } - this.completedPeers.add(peer.id!); - this.completedCount++; - } - } - - public removePeer(peer: PeerContext) { - const index = this._peers.indexOf(peer); - - if ((this.completedPeers !== undefined) && this.completedPeers.delete(peer.id!)) { - this.completedCount--; - } - - // Delete peerId from array without calling splice - const last = this._peers.pop()!; - if (index < this._peers.length) { - this._peers[index] = last; - } - } - - public setCompleted(peer: PeerContext) { - if (this.completedPeers === undefined) { - this.completedPeers = new Set(); - } - - if (!this.completedPeers.has(peer.id!)) { - this.completedPeers.add(peer.id!); - this.completedCount++; - } - } - - public get peers(): ReadonlyArray { - return this._peers; - } -} - -function sendOffer(offerItem: {offer?: { sdp?: string }, offer_id?: string } | null, fromPeerId: string, toPeer: PeerContext, infoHash: string) { - if (offerItem === null) { - throw new TrackerError("announce: wrong offer item format"); - } - - const offer = offerItem.offer; - const offerId = offerItem.offer_id; - - if (offer == undefined) { - throw new TrackerError("announce: wrong offer item field format"); - } - - toPeer.sendMessage({ - action: "announce", - info_hash: infoHash, - offer_id: offerId, // offerId is not validated to be a string - peer_id: fromPeerId, - offer: { - type: "offer", - sdp: offer.sdp, // offer.sdp is not validated to be a string - }, - }, toPeer); -} diff --git a/lib/run-uws-tracker.ts b/lib/run-uws-tracker.ts deleted file mode 100644 index f165bf5..0000000 --- a/lib/run-uws-tracker.ts +++ /dev/null @@ -1,153 +0,0 @@ -/** - * Copyright 2019 Novage LLC. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -import { UWebSocketsTracker } from "./uws-tracker"; -import { FastTracker } from "./fast-tracker"; -import { readFileSync } from "fs"; -import { HttpResponse, HttpRequest } from "uWebSockets.js"; -import { Tracker } from "./tracker"; -import * as Debug from "debug"; - -const debugRequests = Debug("wt-tracker:uws-tracker-requests"); -const debugRequestsEnabled = debugRequests.enabled; - -// tslint:disable:no-console - -async function main() { - let settingsFileData: any; - - if (process.argv[2]) { - try { - settingsFileData = readFileSync(process.argv[2]); - } catch (e) { - console.error("failed to read configuration file:", e.toString()); - return; - } - } else { - try { - settingsFileData = readFileSync("config.json"); - } catch (e) { - if (e.code !== "ENOENT") { - console.error("failed to read configuration file:", e.toString()); - return; - } - } - } - - let settings: any; - try { - settings = (settingsFileData !== undefined) ? JSON.parse(settingsFileData.toString()) : {}; - } catch (e) { - console.error("failed to parse JSON configuration file:", e.toString()); - return; - } - - const serversSettings = (settings.servers === undefined ? [{}] : settings.servers); - - if (!(serversSettings instanceof Array)) { - console.error("failed to parse JSON configuration file: 'servers' property should be an array"); - return; - } - - const tracker = new FastTracker(settings.tracker); - - try { - await runServers(serversSettings, tracker, settings.websocketsAccess); - } catch (e) { - console.error("failed to start the web server:", e.toString()); - } -} - -async function runServers(serversSettings: any[], tracker: Tracker, websocketsAccess: any) { - const servers: UWebSocketsTracker[] = []; - let indexHtml: Buffer | undefined; - try { - indexHtml = readFileSync("index.html"); - } catch (e) { - if (e.code !== "ENOENT") { - throw e; - } - } - - for (const serverSettings of serversSettings) { - serverSettings.access = websocketsAccess; - const server = new UWebSocketsTracker(tracker, serverSettings); - - server.app - .get("/", (response: HttpResponse, request: HttpRequest) => { - debugRequest(server, request); - - if (indexHtml !== undefined) { - response.end(indexHtml); - } else { - const status = "404 Not Found"; - response.writeStatus(status).end(status); - } - }) - .get("/stats.json", (response: HttpResponse, request: HttpRequest) => { - debugRequest(server, request); - - const swarms = tracker.swarms; - let peersCount = 0; - for (const swarm of swarms.values()) { - peersCount += swarm.peers.length; - } - - const serversStats = new Array<{ server: string, webSocketsCount: number }>(); - for (const serverForStats of servers) { - const settings = serverForStats.settings; - serversStats.push({ - server: `${settings.server.host}:${settings.server.port}`, - webSocketsCount: serverForStats.stats.webSocketsCount, - }); - } - - response.writeHeader("Content-Type", "application/json") - .end(JSON.stringify({ - torrentsCount: swarms.size, - peersCount: peersCount, - servers: serversStats, - memory: process.memoryUsage(), - })); - }) - .any("/*", (response: HttpResponse, request: HttpRequest) => { - debugRequest(server, request); - - const status = "404 Not Found"; - response.writeStatus(status).end(status); - }); - - servers.push(server); - await server.run(); - console.info(`listening ${server.settings.server.host}:${server.settings.server.port}`); - } -} - -function debugRequest(server: UWebSocketsTracker, request: HttpRequest) { - if (debugRequestsEnabled) { - debugRequests(server.settings.server.host, server.settings.server.port, - "request method:", request.getMethod(), "url:", request.getUrl(), - "query:", request.getQuery()); - } -} - -(async () => { - try { - await main(); - } catch (e) { - console.error(e); - } -})(); diff --git a/lib/uws-tracker.ts b/lib/uws-tracker.ts deleted file mode 100644 index bd7b2c6..0000000 --- a/lib/uws-tracker.ts +++ /dev/null @@ -1,230 +0,0 @@ -/** - * Copyright 2019 Novage LLC. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -import { App, SSLApp, WebSocket, HttpRequest, TemplatedApp } from "uWebSockets.js"; -import { Tracker, TrackerError } from "./tracker"; -import { StringDecoder } from "string_decoder"; -import * as Debug from "debug"; - -const debugWebSockets = Debug("wt-tracker:uws-tracker"); -const debugWebSocketsEnabled = debugWebSockets.enabled; - -const debugMessages = Debug("wt-tracker:uws-tracker-messages"); -const debugMessagesEnabled = debugMessages.enabled; - -const debugRequests = Debug("wt-tracker:uws-tracker-requests"); -const debugRequestsEnabled = debugRequests.enabled; - -const decoder = new StringDecoder(); - -export class UWebSocketsTracker { - private _app: TemplatedApp; - private webSocketsCount: number = 0; - private validateOrigin = false; - private maxConnections = 0; - - get app() { - return this._app; - } - - get stats() { - return { - webSocketsCount: this.webSocketsCount, - }; - } - - constructor(readonly tracker: Tracker, readonly settings: any = {}) { - this.settings = { - server: { - port: 8000, - host: "0.0.0.0", - ...((settings && settings.server) ? settings.server : {}), - }, - websockets: { - path: "/*", - maxPayloadLength: 64 * 1024, - idleTimeout: 240, - compression: 1, - maxConnections: 0, - ...((settings && settings.websockets) ? settings.websockets : {}), - }, - access: { - allowOrigins: undefined, - denyOrigins: undefined, - denyEmptyOrigin: false, - ...((settings && settings.access) ? settings.access : {}), - }, - }; - - if (this.settings.websockets.maxConnections !== undefined) { - this.maxConnections = this.settings.websockets.maxConnections; - } - - this.validateAccess(); - - this._app = this.settings.server.key_file_name === undefined - ? App(this.settings.server) - : SSLApp(this.settings.server); - - this.buildApplication(); - } - - public async run() { - return new Promise((resolve, reject) => { - this._app.listen(this.settings.server.host, this.settings.server.port, (token: any) => { - if (token) { - resolve(); - } else { - reject(new Error(`failed to listen to ${this.settings.server.host}:${this.settings.server.port}`)); - } - }); - }); - } - - // tslint:disable-next-line:cognitive-complexity - private validateAccess() { - if (this.settings.access.allowOrigins !== undefined) { - if (this.settings.access.denyOrigins !== undefined) { - throw new Error("allowOrigins and denyOrigins can't be set simultaneously"); - } else if (!(this.settings.access.allowOrigins instanceof Array)) { - throw new Error("allowOrigins configuration paramenters should be an array of strings"); - } - } else if ((this.settings.access.denyOrigins !== undefined) && !(this.settings.access.denyOrigins instanceof Array)) { - throw new Error("denyOrigins configuration paramenters should be an array of strings"); - } - - const origins: string[] | undefined = (this.settings.access.allowOrigins === undefined - ? this.settings.access.denyOrigins - : this.settings.access.allowOrigins); - - if (origins !== undefined) { - for (const origin of origins) { - if (typeof origin !== "string") { - throw new Error("allowOrigins and denyOrigins configuration paramenters should be arrays of strings"); - } - } - } - - if (this.settings.access.denyEmptyOrigin || this.settings.access.allowOrigins || this.settings.access.denyOrigins) { - this.validateOrigin = true; - } - } - - private buildApplication() { - this._app - .ws(this.settings.websockets.path, { - compression: this.settings.websockets.compression, - maxPayloadLength: this.settings.websockets.maxPayloadLength, - idleTimeout: this.settings.websockets.idleTimeout, - open: this.onOpen, - drain: (ws: WebSocket) => { - if (debugWebSocketsEnabled) { - debugWebSockets("drain", ws.getBufferedAmount()); - } - }, - message: this.onMessage, - close: this.onClose, - }); - } - - private onOpen = (ws: WebSocket, request: HttpRequest) => { - this.webSocketsCount++; - - if ((this.maxConnections !== 0) && (this.webSocketsCount > this.maxConnections)) { - if (debugRequestsEnabled) { - debugRequests(this.settings.server.host, this.settings.server.port, - "ws-denied-max-connections url:", request.getUrl(), "query:", request.getQuery(), - "origin:", request.getHeader("origin"), "total:", this.webSocketsCount); - } - ws.close(); - return; - } - - if (debugWebSocketsEnabled) { - debugWebSockets("connected via URL", request.getUrl()); - } - - if (this.validateOrigin) { - const origin = request.getHeader("origin"); - if ((this.settings.access.denyEmptyOrigin && origin.length === 0) || - (this.settings.access.denyOrigins && (this.settings.access.denyOrigins as string[]).includes(origin)) || - (this.settings.access.allowOrigins && !(this.settings.access.allowOrigins as string[]).includes(origin))) { - if (debugRequestsEnabled) { - debugRequests(this.settings.server.host, this.settings.server.port, - "ws-denied url:", request.getUrl(), "query:", request.getQuery(), - "origin:", origin, "total:", this.webSocketsCount); - } - ws.close(); - return; - } - } - - if (debugRequestsEnabled) { - debugRequests(this.settings.server.host, this.settings.server.port, - "ws-open url:", request.getUrl(), "query:", request.getQuery(), - "origin:", request.getHeader("origin"), "total:", this.webSocketsCount); - } - } - - private onMessage = (ws: WebSocket, message: ArrayBuffer, isBinary: boolean) => { - debugWebSockets("message of size", message.byteLength); - - let json: any; - try { - json = JSON.parse(decoder.end(new Uint8Array(message) as any)); - } catch (e) { - debugWebSockets("failed to parse JSON message", e); - ws.close(); - return; - } - - if (ws.sendMessage === undefined) { - ws.sendMessage = sendMessage; - } - - if (debugMessagesEnabled) { - debugMessages("in", ws.id !== undefined ? Buffer.from(ws.id).toString("hex") : "unknown peer", json); - } - - try { - this.tracker.processMessage(json, ws as any); - } catch (e) { - if (e instanceof TrackerError) { - debugWebSockets("failed to process message from the peer:", e); - ws.close(); - } else { - throw e; - } - } - } - - private onClose = (ws: WebSocket, code: number, message: ArrayBuffer) => { - this.webSocketsCount--; - - if (ws.sendMessage !== undefined) { - this.tracker.disconnectPeer(ws as any); - } - - debugWebSockets("closed with code", code); - } -} - -function sendMessage(json: any, ws: WebSocket) { - ws.send(JSON.stringify(json), false, false); - if (debugMessagesEnabled) { - debugMessages("out", ws.id !== undefined ? Buffer.from(ws.id).toString("hex") : "unknown peer", json); - } -} diff --git a/package-lock.json b/package-lock.json index bf9e2b3..c7316cd 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,2418 +1,3844 @@ { "name": "wt-tracker", "version": "0.0.1", - "lockfileVersion": 1, + "lockfileVersion": 3, "requires": true, - "dependencies": { - "@babel/code-frame": { - "version": "7.5.5", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.5.5.tgz", - "integrity": "sha512-27d4lZoomVyo51VegxI20xZPuSHusqbQag/ztrBC7wegWoQ1nLREPVSKSW8byhTlzTKyNE4ifaTA6lCp7JjpFw==", - "dev": true, - "requires": { - "@babel/highlight": "^7.0.0" + "packages": { + "": { + "name": "wt-tracker", + "version": "0.0.1", + "license": "Apache-2.0", + "dependencies": { + "debug": "^4.4.3", + "uWebSockets.js": "github:uNetworking/uWebSockets.js#v20.56.0" + }, + "bin": { + "wt-tracker": "bin/wt-tracker" + }, + "devDependencies": { + "@types/debug": "^4.1.12", + "@types/node": "^24.10.1", + "@types/ws": "^8.18.1", + "@vitest/coverage-v8": "^4.0.10", + "globals": "^16.5.0", + "prettier": "^3.6.2", + "rimraf": "^6.1.0", + "ts-mockito": "^2.6.1", + "typescript": "^5.9.3", + "typescript-eslint": "^8.47.0", + "vitest": "^4.0.10", + "ws": "^8.18.3" + }, + "engines": { + "node": ">=16.0.0" } }, - "@babel/generator": { - "version": "7.5.5", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.5.5.tgz", - "integrity": "sha512-ETI/4vyTSxTzGnU2c49XHv2zhExkv9JHLTwDAFz85kmcwuShvYG2H08FwgIguQf4JC75CBnXAUM5PqeF4fj0nQ==", + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", "dev": true, - "requires": { - "@babel/types": "^7.5.5", - "jsesc": "^2.5.1", - "lodash": "^4.17.13", - "source-map": "^0.5.0", - "trim-right": "^1.0.1" + "license": "MIT", + "engines": { + "node": ">=6.9.0" } }, - "@babel/helper-function-name": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.1.0.tgz", - "integrity": "sha512-A95XEoCpb3TO+KZzJ4S/5uW5fNe26DjBGqf1o9ucyLyCmi1dXq/B3c8iaWTfBk3VvetUxl16e8tIrd5teOCfGw==", + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", "dev": true, - "requires": { - "@babel/helper-get-function-arity": "^7.0.0", - "@babel/template": "^7.1.0", - "@babel/types": "^7.0.0" + "license": "MIT", + "engines": { + "node": ">=6.9.0" } }, - "@babel/helper-get-function-arity": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.0.0.tgz", - "integrity": "sha512-r2DbJeg4svYvt3HOS74U4eWKsUAMRH01Z1ds1zx8KNTPtpTL5JAsdFv8BNyOpVqdFhHkkRDIg5B4AsxmkjAlmQ==", + "node_modules/@babel/parser": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz", + "integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==", "dev": true, - "requires": { - "@babel/types": "^7.0.0" + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.5" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" } }, - "@babel/helper-split-export-declaration": { - "version": "7.4.4", - "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.4.4.tgz", - "integrity": "sha512-Ro/XkzLf3JFITkW6b+hNxzZ1n5OQ80NvIUdmHspih1XAhtN3vPTuUFT4eQnela+2MaZ5ulH+iyP513KJrxbN7Q==", + "node_modules/@babel/types": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz", + "integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==", "dev": true, - "requires": { - "@babel/types": "^7.4.4" + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" } }, - "@babel/highlight": { - "version": "7.5.0", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.5.0.tgz", - "integrity": "sha512-7dV4eu9gBxoM0dAnj/BCFDW9LFU0zvTrkq0ugM7pnHEgguOEeOz1so2ZghEdzviYzQEED0r4EAgpsBChKy1TRQ==", + "node_modules/@bcoe/v8-coverage": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-1.0.2.tgz", + "integrity": "sha512-6zABk/ECA/QYSCQ1NGiVwwbQerUCZ+TQbp64Q3AgmfNvurHH0j8TtXa1qbShXA6qqkpAj4V5W8pP6mLe1mcMqA==", "dev": true, - "requires": { - "chalk": "^2.0.0", - "esutils": "^2.0.2", - "js-tokens": "^4.0.0" + "license": "MIT", + "engines": { + "node": ">=18" } }, - "@babel/parser": { - "version": "7.5.5", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.5.5.tgz", - "integrity": "sha512-E5BN68cqR7dhKan1SfqgPGhQ178bkVKpXTPEXnFJBrEt8/DKRZlybmy+IgYLTeN7tp1R5Ccmbm2rBk17sHYU3g==", - "dev": true - }, - "@babel/template": { - "version": "7.4.4", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.4.4.tgz", - "integrity": "sha512-CiGzLN9KgAvgZsnivND7rkA+AeJ9JB0ciPOD4U59GKbQP2iQl+olF1l76kJOupqidozfZ32ghwBEJDhnk9MEcw==", + "node_modules/@esbuild/aix-ppc64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.12.tgz", + "integrity": "sha512-Hhmwd6CInZ3dwpuGTF8fJG6yoWmsToE+vYgD4nytZVxcu1ulHpUQRAB1UJ8+N1Am3Mz4+xOByoQoSZf4D+CpkA==", + "cpu": [ + "ppc64" + ], "dev": true, - "requires": { - "@babel/code-frame": "^7.0.0", - "@babel/parser": "^7.4.4", - "@babel/types": "^7.4.4" + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.12.tgz", + "integrity": "sha512-VJ+sKvNA/GE7Ccacc9Cha7bpS8nyzVv0jdVgwNDaR4gDMC/2TTRc33Ip8qrNYUcpkOHUT5OZ0bUcNNVZQ9RLlg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.12.tgz", + "integrity": "sha512-6AAmLG7zwD1Z159jCKPvAxZd4y/VTO0VkprYy+3N2FtJ8+BQWFXU+OxARIwA46c5tdD9SsKGZ/1ocqBS/gAKHg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.12.tgz", + "integrity": "sha512-5jbb+2hhDHx5phYR2By8GTWEzn6I9UqR11Kwf22iKbNpYrsmRB18aX/9ivc5cabcUiAT/wM+YIZ6SG9QO6a8kg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz", + "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.12.tgz", + "integrity": "sha512-HQ9ka4Kx21qHXwtlTUVbKJOAnmG1ipXhdWTmNXiPzPfWKpXqASVcWdnf2bnL73wgjNrFXAa3yYvBSd9pzfEIpA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.12.tgz", + "integrity": "sha512-gA0Bx759+7Jve03K1S0vkOu5Lg/85dou3EseOGUes8flVOGxbhDDh/iZaoek11Y8mtyKPGF3vP8XhnkDEAmzeg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.12.tgz", + "integrity": "sha512-TGbO26Yw2xsHzxtbVFGEXBFH0FRAP7gtcPE7P5yP7wGy7cXK2oO7RyOhL5NLiqTlBh47XhmIUXuGciXEqYFfBQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.12.tgz", + "integrity": "sha512-lPDGyC1JPDou8kGcywY0YILzWlhhnRjdof3UlcoqYmS9El818LLfJJc3PXXgZHrHCAKs/Z2SeZtDJr5MrkxtOw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.12.tgz", + "integrity": "sha512-8bwX7a8FghIgrupcxb4aUmYDLp8pX06rGh5HqDT7bB+8Rdells6mHvrFHHW2JAOPZUbnjUpKTLg6ECyzvas2AQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.12.tgz", + "integrity": "sha512-0y9KrdVnbMM2/vG8KfU0byhUN+EFCny9+8g202gYqSSVMonbsCfLjUO+rCci7pM0WBEtz+oK/PIwHkzxkyharA==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.12.tgz", + "integrity": "sha512-h///Lr5a9rib/v1GGqXVGzjL4TMvVTv+s1DPoxQdz7l/AYv6LDSxdIwzxkrPW438oUXiDtwM10o9PmwS/6Z0Ng==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.12.tgz", + "integrity": "sha512-iyRrM1Pzy9GFMDLsXn1iHUm18nhKnNMWscjmp4+hpafcZjrr2WbT//d20xaGljXDBYHqRcl8HnxbX6uaA/eGVw==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.12.tgz", + "integrity": "sha512-9meM/lRXxMi5PSUqEXRCtVjEZBGwB7P/D4yT8UG/mwIdze2aV4Vo6U5gD3+RsoHXKkHCfSxZKzmDssVlRj1QQA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.12.tgz", + "integrity": "sha512-Zr7KR4hgKUpWAwb1f3o5ygT04MzqVrGEGXGLnj15YQDJErYu/BGg+wmFlIDOdJp0PmB0lLvxFIOXZgFRrdjR0w==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.12.tgz", + "integrity": "sha512-MsKncOcgTNvdtiISc/jZs/Zf8d0cl/t3gYWX8J9ubBnVOwlk65UIEEvgBORTiljloIWnBzLs4qhzPkJcitIzIg==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.12.tgz", + "integrity": "sha512-uqZMTLr/zR/ed4jIGnwSLkaHmPjOjJvnm6TVVitAa08SLS9Z0VM8wIRx7gWbJB5/J54YuIMInDquWyYvQLZkgw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.12.tgz", + "integrity": "sha512-xXwcTq4GhRM7J9A8Gv5boanHhRa/Q9KLVmcyXHCTaM4wKfIpWkdXiMog/KsnxzJ0A1+nD+zoecuzqPmCRyBGjg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.12.tgz", + "integrity": "sha512-Ld5pTlzPy3YwGec4OuHh1aCVCRvOXdH8DgRjfDy/oumVovmuSzWfnSJg+VtakB9Cm0gxNO9BzWkj6mtO1FMXkQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.12.tgz", + "integrity": "sha512-fF96T6KsBo/pkQI950FARU9apGNTSlZGsv1jZBAlcLL1MLjLNIWPBkj5NlSz8aAzYKg+eNqknrUJ24QBybeR5A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.12.tgz", + "integrity": "sha512-MZyXUkZHjQxUvzK7rN8DJ3SRmrVrke8ZyRusHlP+kuwqTcfWLyqMOE3sScPPyeIXN/mDJIfGXvcMqCgYKekoQw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openharmony-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.12.tgz", + "integrity": "sha512-rm0YWsqUSRrjncSXGA7Zv78Nbnw4XL6/dzr20cyrQf7ZmRcsovpcRBdhD43Nuk3y7XIoW2OxMVvwuRvk9XdASg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.12.tgz", + "integrity": "sha512-3wGSCDyuTHQUzt0nV7bocDy72r2lI33QL3gkDNGkod22EsYl04sMf0qLb8luNKTOmgF/eDEDP5BFNwoBKH441w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.12.tgz", + "integrity": "sha512-rMmLrur64A7+DKlnSuwqUdRKyd3UE7oPJZmnljqEptesKM8wx9J8gx5u0+9Pq0fQQW8vqeKebwNXdfOyP+8Bsg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.12.tgz", + "integrity": "sha512-HkqnmmBoCbCwxUKKNPBixiWDGCpQGVsrQfJoVGYLPT41XWF8lHuE5N6WhVia2n4o5QK5M4tYr21827fNhi4byQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.12.tgz", + "integrity": "sha512-alJC0uCZpTFrSL0CCDjcgleBXPnCrEAhTBILpeAp7M/OFgoqtAetfBzX0xM00MUsVVPpVjlPuMbREqnZCXaTnA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" } }, - "@babel/traverse": { - "version": "7.5.5", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.5.5.tgz", - "integrity": "sha512-MqB0782whsfffYfSjH4TM+LMjrJnhCNEDMDIjeTpl+ASaUvxcjoiVCo/sM1GhS1pHOXYfWVCYneLjMckuUxDaQ==", + "node_modules/@eslint-community/eslint-utils": { + "version": "4.9.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.0.tgz", + "integrity": "sha512-ayVFHdtZ+hsq1t2Dy24wCmGXGe4q9Gu3smhLYALJrr473ZH27MsnSL+LKUlimp4BWJqMDMLmPpx/Q9R3OAlL4g==", "dev": true, - "requires": { - "@babel/code-frame": "^7.5.5", - "@babel/generator": "^7.5.5", - "@babel/helper-function-name": "^7.1.0", - "@babel/helper-split-export-declaration": "^7.4.4", - "@babel/parser": "^7.5.5", - "@babel/types": "^7.5.5", - "debug": "^4.1.0", - "globals": "^11.1.0", - "lodash": "^4.17.13" + "license": "MIT", + "dependencies": { + "eslint-visitor-keys": "^3.4.3" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" } }, - "@babel/types": { - "version": "7.5.5", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.5.5.tgz", - "integrity": "sha512-s63F9nJioLqOlW3UkyMd+BYhXt44YuaFm/VV0VwuteqjYwRrObkU7ra9pY4wAJR3oXi8hJrMcrcJdO/HH33vtw==", + "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", "dev": true, - "requires": { - "esutils": "^2.0.2", - "lodash": "^4.17.13", - "to-fast-properties": "^2.0.0" - } - }, - "@types/chai": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.2.0.tgz", - "integrity": "sha512-zw8UvoBEImn392tLjxoavuonblX/4Yb9ha4KBU10FirCfwgzhKO0dvyJSF9ByxV1xK1r2AgnAi/tvQaLgxQqxA==", - "dev": true - }, - "@types/debug": { - "version": "4.1.5", - "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.5.tgz", - "integrity": "sha512-Q1y515GcOdTHgagaVFhHnIFQ38ygs/kmxdNpvpou+raI9UO3YZcHDngBSYKQklcKlvA7iuQlmIKbzvmxcOE9CQ==", - "dev": true - }, - "@types/mocha": { - "version": "5.2.7", - "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-5.2.7.tgz", - "integrity": "sha512-NYrtPht0wGzhwe9+/idPaBB+TqkY9AhTvOLMkThm0IoEfLaiVQZwBwyJ5puCkO3AUCWrmcoePjp2mbFocKy4SQ==", - "dev": true - }, - "@types/node": { - "version": "12.7.2", - "resolved": "https://registry.npmjs.org/@types/node/-/node-12.7.2.tgz", - "integrity": "sha512-dyYO+f6ihZEtNPDcWNR1fkoTDf3zAK3lAABDze3mz6POyIercH0lEUawUFXlG8xaQZmm1yEBON/4TsYv/laDYg==", - "dev": true - }, - "@types/ws": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/@types/ws/-/ws-6.0.3.tgz", - "integrity": "sha512-yBTM0P05Tx9iXGq00BbJPo37ox68R5vaGTXivs6RGh/BQ6QP5zqZDGWdAO6JbRE/iR1l80xeGAwCQS2nMV9S/w==", - "dev": true, - "requires": { - "@types/node": "*" + "license": "Apache-2.0", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "ajv": { - "version": "6.10.2", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.2.tgz", - "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", + "node_modules/@eslint-community/regexpp": { + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", "dev": true, - "requires": { - "fast-deep-equal": "^2.0.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" } }, - "ansi-colors": { - "version": "3.2.3", - "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.3.tgz", - "integrity": "sha512-LEHHyuhlPY3TmuUYMh2oz89lTShfvgbmzaBcxve9t/9Wuy7Dwf4yoAKcND7KFT1HAQfqZ12qtc+DUrBMeKF9nw==", - "dev": true - }, - "ansi-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", - "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", - "dev": true - }, - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "node_modules/@eslint/config-array": { + "version": "0.21.1", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.21.1.tgz", + "integrity": "sha512-aw1gNayWpdI/jSYVgzN5pL0cfzU02GT3NBpeT/DXbx1/1x7ZKxFPd9bwrzygx/qiwIQiJ1sw/zD8qY/kRvlGHA==", "dev": true, - "requires": { - "color-convert": "^1.9.0" + "license": "Apache-2.0", + "dependencies": { + "@eslint/object-schema": "^2.1.7", + "debug": "^4.3.1", + "minimatch": "^3.1.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "append-transform": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/append-transform/-/append-transform-1.0.0.tgz", - "integrity": "sha512-P009oYkeHyU742iSZJzZZywj4QRJdnTWffaKuJQLablCZ1uz6/cW4yaRgcDaoQ+uwOxxnt0gRUcwfsNP2ri0gw==", + "node_modules/@eslint/config-helpers": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.4.2.tgz", + "integrity": "sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==", "dev": true, - "requires": { - "default-require-extensions": "^2.0.0" + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^0.17.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "archy": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/archy/-/archy-1.0.0.tgz", - "integrity": "sha1-+cjBN1fMHde8N5rHeyxipcKGjEA=", - "dev": true - }, - "argparse": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", - "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "node_modules/@eslint/core": { + "version": "0.17.0", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.17.0.tgz", + "integrity": "sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==", "dev": true, - "requires": { - "sprintf-js": "~1.0.2" + "license": "Apache-2.0", + "dependencies": { + "@types/json-schema": "^7.0.15" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "asn1": { - "version": "0.2.4", - "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", - "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", + "node_modules/@eslint/eslintrc": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.1.tgz", + "integrity": "sha512-gtF186CXhIl1p4pJNGZw8Yc6RlshoePRvE0X91oPGb3vZ8pM3qOS9W9NGPat9LziaBV7XrJWGylNQXkGcnM3IQ==", "dev": true, - "requires": { - "safer-buffer": "~2.1.0" + "license": "MIT", + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^10.0.1", + "globals": "^14.0.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "assert-plus": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", - "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", - "dev": true - }, - "assertion-error": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz", - "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==", - "dev": true - }, - "async-limiter": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", - "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==", - "dev": true - }, - "asynckit": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", - "dev": true - }, - "aws-sign2": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", - "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=", - "dev": true - }, - "aws4": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.8.0.tgz", - "integrity": "sha512-ReZxvNHIOv88FlT7rxcXIIC0fPt4KZqZbOlivyWtXLt8ESx84zd3kMC6iK5jVeS2qt+g7ftS7ye4fi06X5rtRQ==", - "dev": true - }, - "balanced-match": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", - "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", - "dev": true - }, - "bcrypt-pbkdf": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", - "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", + "node_modules/@eslint/eslintrc/node_modules/globals": { + "version": "14.0.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", + "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", "dev": true, - "requires": { - "tweetnacl": "^0.14.3" + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "node_modules/@eslint/js": { + "version": "9.39.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.39.1.tgz", + "integrity": "sha512-S26Stp4zCy88tH94QbBv3XCuzRQiZ9yXofEILmglYTh/Ug/a9/umqvgFtYBAo3Lp0nsI/5/qH1CCrbdK3AP1Tw==", "dev": true, - "requires": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "browser-stdout": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", - "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", - "dev": true - }, - "builtin-modules": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", - "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", - "dev": true - }, - "caching-transform": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/caching-transform/-/caching-transform-3.0.2.tgz", - "integrity": "sha512-Mtgcv3lh3U0zRii/6qVgQODdPA4G3zhG+jtbCWj39RXuUFTMzH0vcdMtaJS1jPowd+It2Pqr6y3NJMQqOqCE2w==", - "dev": true, - "requires": { - "hasha": "^3.0.0", - "make-dir": "^2.0.0", - "package-hash": "^3.0.0", - "write-file-atomic": "^2.4.2" - } - }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - }, - "caseless": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", - "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", - "dev": true - }, - "chai": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/chai/-/chai-4.2.0.tgz", - "integrity": "sha512-XQU3bhBukrOsQCuwZndwGcCVQHyZi53fQ6Ys1Fym7E4olpIqqZZhhoFJoaKVvV17lWQoXYwgWN2nF5crA8J2jw==", - "dev": true, - "requires": { - "assertion-error": "^1.1.0", - "check-error": "^1.0.2", - "deep-eql": "^3.0.1", - "get-func-name": "^2.0.0", - "pathval": "^1.1.0", - "type-detect": "^4.0.5" - } - }, - "chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", - "dev": true, - "requires": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, - "dependencies": { - "supports-color": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", - "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - } + "funding": { + "url": "https://eslint.org/donate" } }, - "check-error": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz", - "integrity": "sha1-V00xLt2Iu13YkS6Sht1sCu1KrII=", - "dev": true - }, - "cliui": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-4.1.0.tgz", - "integrity": "sha512-4FG+RSG9DL7uEwRUZXZn3SS34DiDPfzP0VOiEwtUWlE+AR2EIg+hSyvrIgUUfhdgR/UkAeW2QHgeP+hWrXs7jQ==", + "node_modules/@eslint/object-schema": { + "version": "2.1.7", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.7.tgz", + "integrity": "sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==", "dev": true, - "requires": { - "string-width": "^2.1.1", - "strip-ansi": "^4.0.0", - "wrap-ansi": "^2.0.0" + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "code-point-at": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", - "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=", - "dev": true - }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "node_modules/@eslint/plugin-kit": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.4.1.tgz", + "integrity": "sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==", "dev": true, - "requires": { - "color-name": "1.1.3" + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^0.17.0", + "levn": "^0.4.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", - "dev": true - }, - "combined-stream": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", - "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "node_modules/@humanfs/core": { + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.1.tgz", + "integrity": "sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==", "dev": true, - "requires": { - "delayed-stream": "~1.0.0" + "license": "Apache-2.0", + "engines": { + "node": ">=18.18.0" } }, - "commander": { - "version": "2.20.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.0.tgz", - "integrity": "sha512-7j2y+40w61zy6YC2iRNpUe/NwhNyoXrYpHMrSunaMG64nRnaf96zO/KMQR4OyN/UnE5KLyEBnKHd4aG3rskjpQ==", - "dev": true + "node_modules/@humanfs/node": { + "version": "0.16.7", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.7.tgz", + "integrity": "sha512-/zUx+yOsIrG4Y43Eh2peDeKCxlRt/gET6aHfaKpuq267qXdYDFViVHfMaLyygZOnl0kGWxFIgsBy8QFuTLUXEQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanfs/core": "^0.19.1", + "@humanwhocodes/retry": "^0.4.0" + }, + "engines": { + "node": ">=18.18.0" + } }, - "commondir": { + "node_modules/@humanwhocodes/module-importer": { "version": "1.0.1", - "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", - "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", - "dev": true - }, - "concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", - "dev": true - }, - "convert-source-map": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.6.0.tgz", - "integrity": "sha512-eFu7XigvxdZ1ETfbgPBohgyQ/Z++C0eEhTor0qRwBw9unw+L0/6V8wkSuGgzdThkiS5lSpdptOQPD8Ak40a+7A==", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", "dev": true, - "requires": { - "safe-buffer": "~5.1.1" + "license": "Apache-2.0", + "engines": { + "node": ">=12.22" }, - "dependencies": { - "safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", - "dev": true - } + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" } }, - "core-util-is": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", - "dev": true - }, - "coveralls": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/coveralls/-/coveralls-3.0.6.tgz", - "integrity": "sha512-Pgh4v3gCI4T/9VijVrm8Ym5v0OgjvGLKj3zTUwkvsCiwqae/p6VLzpsFNjQS2i6ewV7ef+DjFJ5TSKxYt/mCrA==", + "node_modules/@humanwhocodes/retry": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz", + "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==", "dev": true, - "requires": { - "growl": "~> 1.10.0", - "js-yaml": "^3.13.1", - "lcov-parse": "^0.0.10", - "log-driver": "^1.2.7", - "minimist": "^1.2.0", - "request": "^2.86.0" + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" } }, - "cp-file": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/cp-file/-/cp-file-6.2.0.tgz", - "integrity": "sha512-fmvV4caBnofhPe8kOcitBwSn2f39QLjnAnGq3gO9dfd75mUytzKNZB1hde6QHunW2Rt+OwuBOMc3i1tNElbszA==", + "node_modules/@isaacs/balanced-match": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/@isaacs/balanced-match/-/balanced-match-4.0.1.tgz", + "integrity": "sha512-yzMTt9lEb8Gv7zRioUilSglI0c0smZ9k5D65677DLWLtWJaXIS3CqcGyUFByYKlnUj6TkjLVs54fBl6+TiGQDQ==", "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "make-dir": "^2.0.0", - "nested-error-stacks": "^2.0.0", - "pify": "^4.0.1", - "safe-buffer": "^5.0.1" + "license": "MIT", + "engines": { + "node": "20 || >=22" } }, - "cross-spawn": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", - "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "node_modules/@isaacs/brace-expansion": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@isaacs/brace-expansion/-/brace-expansion-5.0.0.tgz", + "integrity": "sha512-ZT55BDLV0yv0RBm2czMiZ+SqCGO7AvmOM3G/w2xhVPH+te0aKgFjmBvGlL1dH+ql2tgGO3MVrbb3jCKyvpgnxA==", "dev": true, - "requires": { - "nice-try": "^1.0.4", - "path-key": "^2.0.1", - "semver": "^5.5.0", - "shebang-command": "^1.2.0", - "which": "^1.2.9" + "license": "MIT", + "dependencies": { + "@isaacs/balanced-match": "^4.0.1" + }, + "engines": { + "node": "20 || >=22" } }, - "dashdash": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", - "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", + "node_modules/@isaacs/cliui": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", + "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", "dev": true, - "requires": { - "assert-plus": "^1.0.0" + "license": "ISC", + "dependencies": { + "string-width": "^5.1.2", + "string-width-cjs": "npm:string-width@^4.2.0", + "strip-ansi": "^7.0.1", + "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", + "wrap-ansi": "^8.1.0", + "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" + }, + "engines": { + "node": ">=12" } }, - "debug": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", - "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", - "requires": { - "ms": "^2.1.1" + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0.0" } }, - "decamelize": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", - "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", - "dev": true + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", + "dev": true, + "license": "MIT" }, - "deep-eql": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz", - "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==", + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", "dev": true, - "requires": { - "type-detect": "^4.0.0" + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" } }, - "default-require-extensions": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/default-require-extensions/-/default-require-extensions-2.0.0.tgz", - "integrity": "sha1-9fj7sYp9bVCyH2QfZJ67Uiz+JPc=", + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", "dev": true, - "requires": { - "strip-bom": "^3.0.0" + "license": "MIT", + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" } }, - "define-properties": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", - "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", "dev": true, - "requires": { - "object-keys": "^1.0.12" + "license": "MIT", + "engines": { + "node": ">= 8" } }, - "delayed-stream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", - "dev": true - }, - "diff": { - "version": "3.5.0", - "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", - "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", - "dev": true - }, - "ecc-jsbn": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", - "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", "dev": true, - "requires": { - "jsbn": "~0.1.0", - "safer-buffer": "^2.1.0" + "license": "MIT", + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" } }, - "emoji-regex": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", - "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", - "dev": true - }, - "end-of-stream": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.1.tgz", - "integrity": "sha512-1MkrZNvWTKCaigbn+W15elq2BB/L22nqrSY5DKlo3X6+vclJm8Bb5djXJBmEX6fS3+zCh/F4VBK5Z2KxJt4s2Q==", + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.53.2.tgz", + "integrity": "sha512-yDPzwsgiFO26RJA4nZo8I+xqzh7sJTZIWQOxn+/XOdPE31lAvLIYCKqjV+lNH/vxE2L2iH3plKxDCRK6i+CwhA==", + "cpu": [ + "arm" + ], "dev": true, - "requires": { - "once": "^1.4.0" - } + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.53.2.tgz", + "integrity": "sha512-k8FontTxIE7b0/OGKeSN5B6j25EuppBcWM33Z19JoVT7UTXFSo3D9CdU39wGTeb29NO3XxpMNauh09B+Ibw+9g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.53.2.tgz", + "integrity": "sha512-A6s4gJpomNBtJ2yioj8bflM2oogDwzUiMl2yNJ2v9E7++sHrSrsQ29fOfn5DM/iCzpWcebNYEdXpaK4tr2RhfQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.53.2.tgz", + "integrity": "sha512-e6XqVmXlHrBlG56obu9gDRPW3O3hLxpwHpLsBJvuI8qqnsrtSZ9ERoWUXtPOkY8c78WghyPHZdmPhHLWNdAGEw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-freebsd-arm64": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.53.2.tgz", + "integrity": "sha512-v0E9lJW8VsrwPux5Qe5CwmH/CF/2mQs6xU1MF3nmUxmZUCHazCjLgYvToOk+YuuUqLQBio1qkkREhxhc656ViA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-freebsd-x64": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.53.2.tgz", + "integrity": "sha512-ClAmAPx3ZCHtp6ysl4XEhWU69GUB1D+s7G9YjHGhIGCSrsg00nEGRRZHmINYxkdoJehde8VIsDC5t9C0gb6yqA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.53.2.tgz", + "integrity": "sha512-EPlb95nUsz6Dd9Qy13fI5kUPXNSljaG9FiJ4YUGU1O/Q77i5DYFW5KR8g1OzTcdZUqQQ1KdDqsTohdFVwCwjqg==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm-musleabihf": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.53.2.tgz", + "integrity": "sha512-BOmnVW+khAUX+YZvNfa0tGTEMVVEerOxN0pDk2E6N6DsEIa2Ctj48FOMfNDdrwinocKaC7YXUZ1pHlKpnkja/Q==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.53.2.tgz", + "integrity": "sha512-Xt2byDZ+6OVNuREgBXr4+CZDJtrVso5woFtpKdGPhpTPHcNG7D8YXeQzpNbFRxzTVqJf7kvPMCub/pcGUWgBjA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.53.2.tgz", + "integrity": "sha512-+LdZSldy/I9N8+klim/Y1HsKbJ3BbInHav5qE9Iy77dtHC/pibw1SR/fXlWyAk0ThnpRKoODwnAuSjqxFRDHUQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-loong64-gnu": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.53.2.tgz", + "integrity": "sha512-8ms8sjmyc1jWJS6WdNSA23rEfdjWB30LH8Wqj0Cqvv7qSHnvw6kgMMXRdop6hkmGPlyYBdRPkjJnj3KCUHV/uQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-ppc64-gnu": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.53.2.tgz", + "integrity": "sha512-3HRQLUQbpBDMmzoxPJYd3W6vrVHOo2cVW8RUo87Xz0JPJcBLBr5kZ1pGcQAhdZgX9VV7NbGNipah1omKKe23/g==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.53.2.tgz", + "integrity": "sha512-fMjKi+ojnmIvhk34gZP94vjogXNNUKMEYs+EDaB/5TG/wUkoeua7p7VCHnE6T2Tx+iaghAqQX8teQzcvrYpaQA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-musl": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.53.2.tgz", + "integrity": "sha512-XuGFGU+VwUUV5kLvoAdi0Wz5Xbh2SrjIxCtZj6Wq8MDp4bflb/+ThZsVxokM7n0pcbkEr2h5/pzqzDYI7cCgLQ==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.53.2.tgz", + "integrity": "sha512-w6yjZF0P+NGzWR3AXWX9zc0DNEGdtvykB03uhonSHMRa+oWA6novflo2WaJr6JZakG2ucsyb+rvhrKac6NIy+w==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.53.2.tgz", + "integrity": "sha512-yo8d6tdfdeBArzC7T/PnHd7OypfI9cbuZzPnzLJIyKYFhAQ8SvlkKtKBMbXDxe1h03Rcr7u++nFS7tqXz87Gtw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.53.2.tgz", + "integrity": "sha512-ah59c1YkCxKExPP8O9PwOvs+XRLKwh/mV+3YdKqQ5AMQ0r4M4ZDuOrpWkUaqO7fzAHdINzV9tEVu8vNw48z0lA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-openharmony-arm64": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.53.2.tgz", + "integrity": "sha512-4VEd19Wmhr+Zy7hbUsFZ6YXEiP48hE//KPLCSVNY5RMGX2/7HZ+QkN55a3atM1C/BZCGIgqN+xrVgtdak2S9+A==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.53.2.tgz", + "integrity": "sha512-IlbHFYc/pQCgew/d5fslcy1KEaYVCJ44G8pajugd8VoOEI8ODhtb/j8XMhLpwHCMB3yk2J07ctup10gpw2nyMA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.53.2.tgz", + "integrity": "sha512-lNlPEGgdUfSzdCWU176ku/dQRnA7W+Gp8d+cWv73jYrb8uT7HTVVxq62DUYxjbaByuf1Yk0RIIAbDzp+CnOTFg==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-gnu": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.53.2.tgz", + "integrity": "sha512-S6YojNVrHybQis2lYov1sd+uj7K0Q05NxHcGktuMMdIQ2VixGwAfbJ23NnlvvVV1bdpR2m5MsNBViHJKcA4ADw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.53.2.tgz", + "integrity": "sha512-k+/Rkcyx//P6fetPoLMb8pBeqJBNGx81uuf7iljX9++yNBVRDQgD04L+SVXmXmh5ZP4/WOp4mWF0kmi06PW2tA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] }, - "error-ex": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", - "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "node_modules/@standard-schema/spec": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@standard-schema/spec/-/spec-1.0.0.tgz", + "integrity": "sha512-m2bOd0f2RT9k8QJx1JN85cZYyH1RqFBdlwtkSlf4tBDYLCiiZnv1fIIwacK6cqwXavOydf0NPToMQgpKq+dVlA==", "dev": true, - "requires": { - "is-arrayish": "^0.2.1" - } + "license": "MIT" }, - "es-abstract": { - "version": "1.13.0", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.13.0.tgz", - "integrity": "sha512-vDZfg/ykNxQVwup/8E1BZhVzFfBxs9NqMzGcvIJrqg5k2/5Za2bWo40dK2J1pgLngZ7c+Shh8lwYtLGyrwPutg==", + "node_modules/@types/chai": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.3.tgz", + "integrity": "sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==", "dev": true, - "requires": { - "es-to-primitive": "^1.2.0", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "is-callable": "^1.1.4", - "is-regex": "^1.0.4", - "object-keys": "^1.0.12" + "license": "MIT", + "dependencies": { + "@types/deep-eql": "*", + "assertion-error": "^2.0.1" } }, - "es-to-primitive": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.0.tgz", - "integrity": "sha512-qZryBOJjV//LaxLTV6UC//WewneB3LcXOL9NP++ozKVXsIIIpm/2c13UDiD9Jp2eThsecw9m3jPqDwTyobcdbg==", + "node_modules/@types/debug": { + "version": "4.1.12", + "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.12.tgz", + "integrity": "sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==", "dev": true, - "requires": { - "is-callable": "^1.1.4", - "is-date-object": "^1.0.1", - "is-symbol": "^1.0.2" + "license": "MIT", + "peer": true, + "dependencies": { + "@types/ms": "*" } }, - "es6-error": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/es6-error/-/es6-error-4.1.1.tgz", - "integrity": "sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg==", - "dev": true - }, - "escape-string-regexp": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", - "dev": true - }, - "esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true - }, - "esutils": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", - "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "dev": true + "node_modules/@types/deep-eql": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz", + "integrity": "sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==", + "dev": true, + "license": "MIT" }, - "execa": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", - "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", - "dev": true, - "requires": { - "cross-spawn": "^6.0.0", - "get-stream": "^4.0.0", - "is-stream": "^1.1.0", - "npm-run-path": "^2.0.0", - "p-finally": "^1.0.0", - "signal-exit": "^3.0.0", - "strip-eof": "^1.0.0" - } - }, - "extend": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", - "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", - "dev": true - }, - "extsprintf": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", - "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", - "dev": true - }, - "fast-deep-equal": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", - "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=", - "dev": true + "node_modules/@types/estree": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", + "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", + "dev": true, + "license": "MIT" }, - "fast-json-stable-stringify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", - "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=", - "dev": true + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true, + "license": "MIT" }, - "find-cache-dir": { + "node_modules/@types/ms": { "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", - "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", + "resolved": "https://registry.npmjs.org/@types/ms/-/ms-2.1.0.tgz", + "integrity": "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA==", "dev": true, - "requires": { - "commondir": "^1.0.1", - "make-dir": "^2.0.0", - "pkg-dir": "^3.0.0" - } + "license": "MIT" }, - "find-up": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", - "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "node_modules/@types/node": { + "version": "24.10.1", + "resolved": "https://registry.npmjs.org/@types/node/-/node-24.10.1.tgz", + "integrity": "sha512-GNWcUTRBgIRJD5zj+Tq0fKOJ5XZajIiBroOF0yvj2bSU1WvNdYS/dn9UxwsujGW4JX06dnHyjV2y9rRaybH0iQ==", "dev": true, - "requires": { - "locate-path": "^3.0.0" + "license": "MIT", + "dependencies": { + "undici-types": "~7.16.0" } }, - "flat": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/flat/-/flat-4.1.0.tgz", - "integrity": "sha512-Px/TiLIznH7gEDlPXcUD4KnBusa6kR6ayRUVcnEAbreRIuhkqow/mun59BuRXwoYk7ZQOLW1ZM05ilIvK38hFw==", + "node_modules/@types/ws": { + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.18.1.tgz", + "integrity": "sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg==", "dev": true, - "requires": { - "is-buffer": "~2.0.3" + "license": "MIT", + "dependencies": { + "@types/node": "*" } }, - "foreground-child": { - "version": "1.5.6", - "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-1.5.6.tgz", - "integrity": "sha1-T9ca0t/elnibmApcCilZN8svXOk=", + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.47.0.tgz", + "integrity": "sha512-fe0rz9WJQ5t2iaLfdbDc9T80GJy0AeO453q8C3YCilnGozvOyCG5t+EZtg7j7D88+c3FipfP/x+wzGnh1xp8ZA==", "dev": true, - "requires": { - "cross-spawn": "^4", - "signal-exit": "^3.0.0" - }, + "license": "MIT", "dependencies": { - "cross-spawn": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-4.0.2.tgz", - "integrity": "sha1-e5JHYhwjrf3ThWAEqCPL45dCTUE=", - "dev": true, - "requires": { - "lru-cache": "^4.0.1", - "which": "^1.2.9" - } - } + "@eslint-community/regexpp": "^4.10.0", + "@typescript-eslint/scope-manager": "8.47.0", + "@typescript-eslint/type-utils": "8.47.0", + "@typescript-eslint/utils": "8.47.0", + "@typescript-eslint/visitor-keys": "8.47.0", + "graphemer": "^1.4.0", + "ignore": "^7.0.0", + "natural-compare": "^1.4.0", + "ts-api-utils": "^2.1.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^8.47.0", + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" } }, - "forever-agent": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", - "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", - "dev": true - }, - "form-data": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", - "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", + "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": { + "version": "7.0.5", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz", + "integrity": "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==", "dev": true, - "requires": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.6", - "mime-types": "^2.1.12" + "license": "MIT", + "engines": { + "node": ">= 4" } }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", - "dev": true - }, - "function-bind": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", - "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", - "dev": true - }, - "get-caller-file": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", - "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", - "dev": true - }, - "get-func-name": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.0.tgz", - "integrity": "sha1-6td0q+5y4gQJQzoGY2YCPdaIekE=", - "dev": true - }, - "get-stream": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", - "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "node_modules/@typescript-eslint/parser": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.47.0.tgz", + "integrity": "sha512-lJi3PfxVmo0AkEY93ecfN+r8SofEqZNGByvHAI3GBLrvt1Cw6H5k1IM02nSzu0RfUafr2EvFSw0wAsZgubNplQ==", "dev": true, - "requires": { - "pump": "^3.0.0" + "license": "MIT", + "peer": true, + "dependencies": { + "@typescript-eslint/scope-manager": "8.47.0", + "@typescript-eslint/types": "8.47.0", + "@typescript-eslint/typescript-estree": "8.47.0", + "@typescript-eslint/visitor-keys": "8.47.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" } }, - "getpass": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", - "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", + "node_modules/@typescript-eslint/project-service": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.47.0.tgz", + "integrity": "sha512-2X4BX8hUeB5JcA1TQJ7GjcgulXQ+5UkNb0DL8gHsHUHdFoiCTJoYLTpib3LtSDPZsRET5ygN4qqIWrHyYIKERA==", "dev": true, - "requires": { - "assert-plus": "^1.0.0" + "license": "MIT", + "dependencies": { + "@typescript-eslint/tsconfig-utils": "^8.47.0", + "@typescript-eslint/types": "^8.47.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.0.0" } }, - "glob": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", - "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", + "node_modules/@typescript-eslint/scope-manager": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.47.0.tgz", + "integrity": "sha512-a0TTJk4HXMkfpFkL9/WaGTNuv7JWfFTQFJd6zS9dVAjKsojmv9HT55xzbEpnZoY+VUb+YXLMp+ihMLz/UlZfDg==", "dev": true, - "requires": { - "fs.realpath": "^1.0.0", - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "^3.0.4", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.47.0", + "@typescript-eslint/visitor-keys": "8.47.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" } }, - "globals": { - "version": "11.12.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", - "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", - "dev": true - }, - "graceful-fs": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.2.tgz", - "integrity": "sha512-IItsdsea19BoLC7ELy13q1iJFNmd7ofZH5+X/pJr90/nRoPEX0DJo1dHDbgtYWOhJhcCgMDTOw84RZ72q6lB+Q==", - "dev": true - }, - "growl": { - "version": "1.10.5", - "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz", - "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==", - "dev": true - }, - "handlebars": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.1.2.tgz", - "integrity": "sha512-nvfrjqvt9xQ8Z/w0ijewdD/vvWDTOweBUm96NTr66Wfvo1mJenBLwcYmPs3TIBP5ruzYGD7Hx/DaM9RmhroGPw==", + "node_modules/@typescript-eslint/tsconfig-utils": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.47.0.tgz", + "integrity": "sha512-ybUAvjy4ZCL11uryalkKxuT3w3sXJAuWhOoGS3T/Wu+iUu1tGJmk5ytSY8gbdACNARmcYEB0COksD2j6hfGK2g==", "dev": true, - "requires": { - "neo-async": "^2.6.0", - "optimist": "^0.6.1", - "source-map": "^0.6.1", - "uglify-js": "^3.1.4" + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.0.0" } }, - "har-schema": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", - "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", - "dev": true - }, - "har-validator": { - "version": "5.1.3", - "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.3.tgz", - "integrity": "sha512-sNvOCzEQNr/qrvJgc3UG/kD4QtlHycrzwS+6mfTrrSq97BvaYcPZZI1ZSqGSPR73Cxn4LKTD4PttRwfU7jWq5g==", + "node_modules/@typescript-eslint/type-utils": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.47.0.tgz", + "integrity": "sha512-QC9RiCmZ2HmIdCEvhd1aJELBlD93ErziOXXlHEZyuBo3tBiAZieya0HLIxp+DoDWlsQqDawyKuNEhORyku+P8A==", "dev": true, - "requires": { - "ajv": "^6.5.5", - "har-schema": "^2.0.0" + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.47.0", + "@typescript-eslint/typescript-estree": "8.47.0", + "@typescript-eslint/utils": "8.47.0", + "debug": "^4.3.4", + "ts-api-utils": "^2.1.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" } }, - "has": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", - "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "node_modules/@typescript-eslint/types": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.47.0.tgz", + "integrity": "sha512-nHAE6bMKsizhA2uuYZbEbmp5z2UpffNrPEqiKIeN7VsV6UY/roxanWfoRrf6x/k9+Obf+GQdkm0nPU+vnMXo9A==", "dev": true, - "requires": { - "function-bind": "^1.1.1" + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" } }, - "has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", - "dev": true - }, - "has-symbols": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.0.tgz", - "integrity": "sha1-uhqPGvKg/DllD1yFA2dwQSIGO0Q=", - "dev": true - }, - "hasha": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/hasha/-/hasha-3.0.0.tgz", - "integrity": "sha1-UqMvq4Vp1BymmmH/GiFPjrfIvTk=", + "node_modules/@typescript-eslint/typescript-estree": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.47.0.tgz", + "integrity": "sha512-k6ti9UepJf5NpzCjH31hQNLHQWupTRPhZ+KFF8WtTuTpy7uHPfeg2NM7cP27aCGajoEplxJDFVCEm9TGPYyiVg==", "dev": true, - "requires": { - "is-stream": "^1.0.1" + "license": "MIT", + "dependencies": { + "@typescript-eslint/project-service": "8.47.0", + "@typescript-eslint/tsconfig-utils": "8.47.0", + "@typescript-eslint/types": "8.47.0", + "@typescript-eslint/visitor-keys": "8.47.0", + "debug": "^4.3.4", + "fast-glob": "^3.3.2", + "is-glob": "^4.0.3", + "minimatch": "^9.0.4", + "semver": "^7.6.0", + "ts-api-utils": "^2.1.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.0.0" } }, - "he": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", - "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", - "dev": true - }, - "hosted-git-info": { - "version": "2.8.4", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.4.tgz", - "integrity": "sha512-pzXIvANXEFrc5oFFXRMkbLPQ2rXRoDERwDLyrcUxGhaZhgP54BBSl9Oheh7Vv0T090cszWBxPjkQQ5Sq1PbBRQ==", - "dev": true - }, - "http-signature": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", - "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", + "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", + "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", "dev": true, - "requires": { - "assert-plus": "^1.0.0", - "jsprim": "^1.2.2", - "sshpk": "^1.7.0" + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" } }, - "immutable": { - "version": "3.8.2", - "resolved": "https://registry.npmjs.org/immutable/-/immutable-3.8.2.tgz", - "integrity": "sha1-wkOZUUVbs5kT2vKBN28VMOEErfM=", - "dev": true - }, - "imurmurhash": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", - "dev": true + "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } }, - "inflight": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "node_modules/@typescript-eslint/utils": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.47.0.tgz", + "integrity": "sha512-g7XrNf25iL4TJOiPqatNuaChyqt49a/onq5YsJ9+hXeugK+41LVg7AxikMfM02PC6jbNtZLCJj6AUcQXJS/jGQ==", "dev": true, - "requires": { - "once": "^1.3.0", - "wrappy": "1" + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.7.0", + "@typescript-eslint/scope-manager": "8.47.0", + "@typescript-eslint/types": "8.47.0", + "@typescript-eslint/typescript-estree": "8.47.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" } }, - "inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "dev": true + "node_modules/@typescript-eslint/visitor-keys": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.47.0.tgz", + "integrity": "sha512-SIV3/6eftCy1bNzCQoPmbWsRLujS8t5iDIZ4spZOBHqrM+yfX2ogg8Tt3PDTAVKw3sSCiUgg30uOAvK2r9zGjQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.47.0", + "eslint-visitor-keys": "^4.2.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } }, - "invert-kv": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-2.0.0.tgz", - "integrity": "sha512-wPVv/y/QQ/Uiirj/vh3oP+1Ww+AWehmi1g5fFWGPF6IpCBCDVrhgHRMvrLfdYcwDh3QJbGXDW4JAuzxElLSqKA==", - "dev": true - }, - "is-arrayish": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", - "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", - "dev": true - }, - "is-buffer": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.3.tgz", - "integrity": "sha512-U15Q7MXTuZlrbymiz95PJpZxu8IlipAp4dtS3wOdgPXx3mqBnslrWU14kxfHB+Py/+2PVKSr37dMAgM2A4uArw==", - "dev": true - }, - "is-callable": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.4.tgz", - "integrity": "sha512-r5p9sxJjYnArLjObpjA4xu5EKI3CuKHkJXMhT7kwbpUyIFD1n5PMAsoPvWnvtZiNz7LjkYDRZhd7FlI0eMijEA==", - "dev": true - }, - "is-date-object": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.1.tgz", - "integrity": "sha1-mqIOtq7rv/d/vTPnTKAbM1gdOhY=", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", - "dev": true - }, - "is-regex": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.4.tgz", - "integrity": "sha1-VRdIm1RwkbCTDglWVM7SXul+lJE=", + "node_modules/@vitest/coverage-v8": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/@vitest/coverage-v8/-/coverage-v8-4.0.10.tgz", + "integrity": "sha512-g+brmtoKa/sAeIohNJnnWhnHtU6GuqqVOSQ4SxDIPcgZWZyhJs5RmF5LpqXs8Kq64lANP+vnbn5JLzhLj/G56g==", "dev": true, - "requires": { - "has": "^1.0.1" + "license": "MIT", + "dependencies": { + "@bcoe/v8-coverage": "^1.0.2", + "@vitest/utils": "4.0.10", + "ast-v8-to-istanbul": "^0.3.8", + "debug": "^4.4.3", + "istanbul-lib-coverage": "^3.2.2", + "istanbul-lib-report": "^3.0.1", + "istanbul-lib-source-maps": "^5.0.6", + "istanbul-reports": "^3.2.0", + "magicast": "^0.5.1", + "std-env": "^3.10.0", + "tinyrainbow": "^3.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "@vitest/browser": "4.0.10", + "vitest": "4.0.10" + }, + "peerDependenciesMeta": { + "@vitest/browser": { + "optional": true + } } }, - "is-stream": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", - "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", - "dev": true - }, - "is-symbol": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.2.tgz", - "integrity": "sha512-HS8bZ9ox60yCJLH9snBpIwv9pYUAkcuLhSA1oero1UB5y9aiQpRA8y2ex945AOtCZL1lJDeIk3G5LthswI46Lw==", + "node_modules/@vitest/expect": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-4.0.10.tgz", + "integrity": "sha512-3QkTX/lK39FBNwARCQRSQr0TP9+ywSdxSX+LgbJ2M1WmveXP72anTbnp2yl5fH+dU6SUmBzNMrDHs80G8G2DZg==", "dev": true, - "requires": { - "has-symbols": "^1.0.0" + "license": "MIT", + "dependencies": { + "@standard-schema/spec": "^1.0.0", + "@types/chai": "^5.2.2", + "@vitest/spy": "4.0.10", + "@vitest/utils": "4.0.10", + "chai": "^6.2.1", + "tinyrainbow": "^3.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" } }, - "is-typedarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", - "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", - "dev": true - }, - "isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", - "dev": true - }, - "isstream": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", - "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=", - "dev": true - }, - "istanbul-lib-coverage": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.5.tgz", - "integrity": "sha512-8aXznuEPCJvGnMSRft4udDRDtb1V3pkQkMMI5LI+6HuQz5oQ4J2UFn1H82raA3qJtyOLkkwVqICBQkjnGtn5mA==", - "dev": true - }, - "istanbul-lib-hook": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/istanbul-lib-hook/-/istanbul-lib-hook-2.0.7.tgz", - "integrity": "sha512-vrRztU9VRRFDyC+aklfLoeXyNdTfga2EI3udDGn4cZ6fpSXpHLV9X6CHvfoMCPtggg8zvDDmC4b9xfu0z6/llA==", - "dev": true, - "requires": { - "append-transform": "^1.0.0" - } - }, - "istanbul-lib-instrument": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-3.3.0.tgz", - "integrity": "sha512-5nnIN4vo5xQZHdXno/YDXJ0G+I3dAm4XgzfSVTPLQpj/zAV2dV6Juy0yaf10/zrJOJeHoN3fraFe+XRq2bFVZA==", - "dev": true, - "requires": { - "@babel/generator": "^7.4.0", - "@babel/parser": "^7.4.3", - "@babel/template": "^7.4.0", - "@babel/traverse": "^7.4.3", - "@babel/types": "^7.4.0", - "istanbul-lib-coverage": "^2.0.5", - "semver": "^6.0.0" - }, - "dependencies": { - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true + "node_modules/@vitest/mocker": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-4.0.10.tgz", + "integrity": "sha512-e2OfdexYkjkg8Hh3L9NVEfbwGXq5IZbDovkf30qW2tOh7Rh9sVtmSr2ztEXOFbymNxS4qjzLXUQIvATvN4B+lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/spy": "4.0.10", + "estree-walker": "^3.0.3", + "magic-string": "^0.30.21" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "msw": "^2.4.9", + "vite": "^6.0.0 || ^7.0.0-0" + }, + "peerDependenciesMeta": { + "msw": { + "optional": true + }, + "vite": { + "optional": true } } }, - "istanbul-lib-report": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-2.0.8.tgz", - "integrity": "sha512-fHBeG573EIihhAblwgxrSenp0Dby6tJMFR/HvlerBsrCTD5bkUuoNtn3gVh29ZCS824cGGBPn7Sg7cNk+2xUsQ==", + "node_modules/@vitest/pretty-format": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-4.0.10.tgz", + "integrity": "sha512-99EQbpa/zuDnvVjthwz5bH9o8iPefoQZ63WV8+bsRJZNw3qQSvSltfut8yu1Jc9mqOYi7pEbsKxYTi/rjaq6PA==", "dev": true, - "requires": { - "istanbul-lib-coverage": "^2.0.5", - "make-dir": "^2.1.0", - "supports-color": "^6.1.0" - }, + "license": "MIT", "dependencies": { - "supports-color": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", - "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - } + "tinyrainbow": "^3.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" } }, - "istanbul-lib-source-maps": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-3.0.6.tgz", - "integrity": "sha512-R47KzMtDJH6X4/YW9XTx+jrLnZnscW4VpNN+1PViSYTejLVPWv7oov+Duf8YQSPyVRUvueQqz1TcsC6mooZTXw==", + "node_modules/@vitest/runner": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-4.0.10.tgz", + "integrity": "sha512-EXU2iSkKvNwtlL8L8doCpkyclw0mc/t4t9SeOnfOFPyqLmQwuceMPA4zJBa6jw0MKsZYbw7kAn+gl7HxrlB8UQ==", "dev": true, - "requires": { - "debug": "^4.1.1", - "istanbul-lib-coverage": "^2.0.5", - "make-dir": "^2.1.0", - "rimraf": "^2.6.3", - "source-map": "^0.6.1" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } + "license": "MIT", + "dependencies": { + "@vitest/utils": "4.0.10", + "pathe": "^2.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" } }, - "istanbul-reports": { - "version": "2.2.6", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-2.2.6.tgz", - "integrity": "sha512-SKi4rnMyLBKe0Jy2uUdx28h8oG7ph2PPuQPvIAh31d+Ci+lSiEu4C+h3oBPuJ9+mPKhOyW0M8gY4U5NM1WLeXA==", + "node_modules/@vitest/snapshot": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-4.0.10.tgz", + "integrity": "sha512-2N4X2ZZl7kZw0qeGdQ41H0KND96L3qX1RgwuCfy6oUsF2ISGD/HpSbmms+CkIOsQmg2kulwfhJ4CI0asnZlvkg==", "dev": true, - "requires": { - "handlebars": "^4.1.2" + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "4.0.10", + "magic-string": "^0.30.21", + "pathe": "^2.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" } }, - "js-tokens": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", - "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", - "dev": true - }, - "js-yaml": { - "version": "3.13.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", - "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", + "node_modules/@vitest/spy": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-4.0.10.tgz", + "integrity": "sha512-AsY6sVS8OLb96GV5RoG8B6I35GAbNrC49AO+jNRF9YVGb/g9t+hzNm1H6kD0NDp8tt7VJLs6hb7YMkDXqu03iw==", "dev": true, - "requires": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" + "license": "MIT", + "funding": { + "url": "https://opencollective.com/vitest" } }, - "jsbn": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", - "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", - "dev": true - }, - "jsesc": { - "version": "2.5.2", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", - "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", - "dev": true - }, - "json-parse-better-errors": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", - "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", - "dev": true - }, - "json-schema": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", - "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=", - "dev": true - }, - "json-schema-traverse": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", - "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true - }, - "json-stringify-safe": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", - "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", - "dev": true - }, - "jsprim": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", - "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", + "node_modules/@vitest/utils": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-4.0.10.tgz", + "integrity": "sha512-kOuqWnEwZNtQxMKg3WmPK1vmhZu9WcoX69iwWjVz+jvKTsF1emzsv3eoPcDr6ykA3qP2bsCQE7CwqfNtAVzsmg==", "dev": true, - "requires": { - "assert-plus": "1.0.0", - "extsprintf": "1.3.0", - "json-schema": "0.2.3", - "verror": "1.10.0" + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "4.0.10", + "tinyrainbow": "^3.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" } }, - "lcid": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/lcid/-/lcid-2.0.0.tgz", - "integrity": "sha512-avPEb8P8EGnwXKClwsNUgryVjllcRqtMYa49NTsbQagYuT1DcXnl1915oxWjoyGrXR6zH/Y0Zc96xWsPcoDKeA==", + "node_modules/acorn": { + "version": "8.15.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", + "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", "dev": true, - "requires": { - "invert-kv": "^2.0.0" + "license": "MIT", + "peer": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" } }, - "lcov-parse": { - "version": "0.0.10", - "resolved": "https://registry.npmjs.org/lcov-parse/-/lcov-parse-0.0.10.tgz", - "integrity": "sha1-GwuP+ayceIklBYK3C3ExXZ2m2aM=", - "dev": true + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } }, - "load-json-file": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-4.0.0.tgz", - "integrity": "sha1-L19Fq5HjMhYjT9U62rZo607AmTs=", + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "parse-json": "^4.0.0", - "pify": "^3.0.0", - "strip-bom": "^3.0.0" - }, + "license": "MIT", "dependencies": { - "pify": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", - "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", - "dev": true - } + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" } }, - "locate-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", - "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "node_modules/ansi-regex": { + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.2.2.tgz", + "integrity": "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg==", "dev": true, - "requires": { - "p-locate": "^3.0.0", - "path-exists": "^3.0.0" + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-regex?sponsor=1" } }, - "lodash": { - "version": "4.17.15", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", - "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", - "dev": true + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } }, - "lodash.flattendeep": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/lodash.flattendeep/-/lodash.flattendeep-4.4.0.tgz", - "integrity": "sha1-+wMJF/hqMTTlvJvsDWngAT3f7bI=", - "dev": true + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true, + "license": "Python-2.0" }, - "log-driver": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/log-driver/-/log-driver-1.2.7.tgz", - "integrity": "sha512-U7KCmLdqsGHBLeWqYlFA0V0Sl6P08EE1ZrmA9cxjUE0WVqT9qnyVDPz1kzpFEP0jdJuFnasWIfSd7fsaNXkpbg==", - "dev": true + "node_modules/assertion-error": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz", + "integrity": "sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + } }, - "log-symbols": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", - "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", + "node_modules/ast-v8-to-istanbul": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/ast-v8-to-istanbul/-/ast-v8-to-istanbul-0.3.8.tgz", + "integrity": "sha512-szgSZqUxI5T8mLKvS7WTjF9is+MVbOeLADU73IseOcrqhxr/VAvy6wfoVE39KnKzA7JRhjF5eUagNlHwvZPlKQ==", "dev": true, - "requires": { - "chalk": "^2.0.1" + "license": "MIT", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.31", + "estree-walker": "^3.0.3", + "js-tokens": "^9.0.1" } }, - "lru-cache": { - "version": "4.1.5", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.5.tgz", - "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", "dev": true, - "requires": { - "pseudomap": "^1.0.2", - "yallist": "^2.1.2" + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" } }, - "make-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", - "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", "dev": true, - "requires": { - "pify": "^4.0.1", - "semver": "^5.6.0" + "license": "MIT", + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" } }, - "map-age-cleaner": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/map-age-cleaner/-/map-age-cleaner-0.1.3.tgz", - "integrity": "sha512-bJzx6nMoP6PDLPBFmg7+xRKeFZvFboMrGlxmNj9ClvX53KrmvM5bXFXEWjbz4cz1AFn+jWJ9z/DJSz7hrs0w3w==", + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", "dev": true, - "requires": { - "p-defer": "^1.0.0" + "license": "MIT", + "engines": { + "node": ">=6" } }, - "mem": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/mem/-/mem-4.3.0.tgz", - "integrity": "sha512-qX2bG48pTqYRVmDB37rn/6PT7LcR8T7oAX3bf99u1Tt1nzxYfxkgqDwUwolPlXweM0XzBOBFzSx4kfp7KP1s/w==", + "node_modules/chai": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/chai/-/chai-6.2.1.tgz", + "integrity": "sha512-p4Z49OGG5W/WBCPSS/dH3jQ73kD6tiMmUM+bckNK6Jr5JHMG3k9bg/BvKR8lKmtVBKmOiuVaV2ws8s9oSbwysg==", "dev": true, - "requires": { - "map-age-cleaner": "^0.1.1", - "mimic-fn": "^2.0.0", - "p-is-promise": "^2.0.0" + "license": "MIT", + "engines": { + "node": ">=18" } }, - "merge-source-map": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/merge-source-map/-/merge-source-map-1.1.0.tgz", - "integrity": "sha512-Qkcp7P2ygktpMPh2mCQZaf3jhN6D3Z/qVZHSdWvQ+2Ef5HgRAPBO57A77+ENm0CPx2+1Ce/MYKi3ymqdfuqibw==", + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", "dev": true, - "requires": { - "source-map": "^0.6.1" - }, + "license": "MIT", "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" } }, - "mime-db": { - "version": "1.40.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.40.0.tgz", - "integrity": "sha512-jYdeOMPy9vnxEqFRRo6ZvTZ8d9oPb+k18PKoYNYUe2stVEBPPwsln/qWzdbmaIvnhZ9v2P+CuecK+fpUfsV2mA==", - "dev": true - }, - "mime-types": { - "version": "2.1.24", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.24.tgz", - "integrity": "sha512-WaFHS3MCl5fapm3oLxU4eYDw77IQM2ACcxQ9RIxfaC3ooc6PFuBMGZZsYpvoXS5D5QTWPieo1jjLdAm3TBP3cQ==", + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", "dev": true, - "requires": { - "mime-db": "1.40.0" + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" } }, - "mimic-fn": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", - "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", - "dev": true - }, - "minimatch": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", - "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true, - "requires": { - "brace-expansion": "^1.1.7" - } + "license": "MIT" }, - "minimist": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", - "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", - "dev": true + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true, + "license": "MIT" }, - "mkdirp": { - "version": "0.5.1", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", - "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", "dev": true, - "requires": { - "minimist": "0.0.8" - }, + "license": "MIT", "dependencies": { - "minimist": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", - "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", - "dev": true - } + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" } }, - "mocha": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/mocha/-/mocha-6.2.0.tgz", - "integrity": "sha512-qwfFgY+7EKAAUAdv7VYMZQknI7YJSGesxHyhn6qD52DV8UcSZs5XwCifcZGMVIE4a5fbmhvbotxC0DLQ0oKohQ==", - "dev": true, - "requires": { - "ansi-colors": "3.2.3", - "browser-stdout": "1.3.1", - "debug": "3.2.6", - "diff": "3.5.0", - "escape-string-regexp": "1.0.5", - "find-up": "3.0.0", - "glob": "7.1.3", - "growl": "1.10.5", - "he": "1.2.0", - "js-yaml": "3.13.1", - "log-symbols": "2.2.0", - "minimatch": "3.0.4", - "mkdirp": "0.5.1", - "ms": "2.1.1", - "node-environment-flags": "1.0.5", - "object.assign": "4.1.0", - "strip-json-comments": "2.0.1", - "supports-color": "6.0.0", - "which": "1.3.1", - "wide-align": "1.1.3", - "yargs": "13.2.2", - "yargs-parser": "13.0.0", - "yargs-unparser": "1.5.0" - }, - "dependencies": { - "debug": { - "version": "3.2.6", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", - "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "ms": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", - "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", - "dev": true + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true } } }, - "ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" - }, - "neo-async": { - "version": "2.6.1", - "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.1.tgz", - "integrity": "sha512-iyam8fBuCUpWeKPGpaNMetEocMt364qkCsfL9JuhjXX6dRnguRVOfk2GZaDpPjcOKiiXCPINZC1GczQ7iTq3Zw==", - "dev": true - }, - "nested-error-stacks": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/nested-error-stacks/-/nested-error-stacks-2.1.0.tgz", - "integrity": "sha512-AO81vsIO1k1sM4Zrd6Hu7regmJN1NSiAja10gc4bX3F0wd+9rQmcuHQaHVQCYIEC8iFXnE+mavh23GOt7wBgug==", - "dev": true + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true, + "license": "MIT" }, - "nice-try": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", - "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", - "dev": true + "node_modules/eastasianwidth": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", + "dev": true, + "license": "MIT" }, - "node-environment-flags": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/node-environment-flags/-/node-environment-flags-1.0.5.tgz", - "integrity": "sha512-VNYPRfGfmZLx0Ye20jWzHUjyTW/c+6Wq+iLhDzUI4XmhrDd9l/FozXV3F2xOaXjvp0co0+v1YSR3CMP6g+VvLQ==", + "node_modules/emoji-regex": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", "dev": true, - "requires": { - "object.getownpropertydescriptors": "^2.0.3", - "semver": "^5.7.0" - } + "license": "MIT" }, - "normalize-package-data": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", - "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", + "node_modules/es-module-lexer": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.7.0.tgz", + "integrity": "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==", "dev": true, - "requires": { - "hosted-git-info": "^2.1.4", - "resolve": "^1.10.0", - "semver": "2 || 3 || 4 || 5", - "validate-npm-package-license": "^3.0.1" - } + "license": "MIT" }, - "npm-run-path": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", - "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", + "node_modules/esbuild": { + "version": "0.25.12", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz", + "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.25.12", + "@esbuild/android-arm": "0.25.12", + "@esbuild/android-arm64": "0.25.12", + "@esbuild/android-x64": "0.25.12", + "@esbuild/darwin-arm64": "0.25.12", + "@esbuild/darwin-x64": "0.25.12", + "@esbuild/freebsd-arm64": "0.25.12", + "@esbuild/freebsd-x64": "0.25.12", + "@esbuild/linux-arm": "0.25.12", + "@esbuild/linux-arm64": "0.25.12", + "@esbuild/linux-ia32": "0.25.12", + "@esbuild/linux-loong64": "0.25.12", + "@esbuild/linux-mips64el": "0.25.12", + "@esbuild/linux-ppc64": "0.25.12", + "@esbuild/linux-riscv64": "0.25.12", + "@esbuild/linux-s390x": "0.25.12", + "@esbuild/linux-x64": "0.25.12", + "@esbuild/netbsd-arm64": "0.25.12", + "@esbuild/netbsd-x64": "0.25.12", + "@esbuild/openbsd-arm64": "0.25.12", + "@esbuild/openbsd-x64": "0.25.12", + "@esbuild/openharmony-arm64": "0.25.12", + "@esbuild/sunos-x64": "0.25.12", + "@esbuild/win32-arm64": "0.25.12", + "@esbuild/win32-ia32": "0.25.12", + "@esbuild/win32-x64": "0.25.12" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", "dev": true, - "requires": { - "path-key": "^2.0.0" + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "number-is-nan": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", - "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", - "dev": true - }, - "nyc": { - "version": "14.1.1", - "resolved": "https://registry.npmjs.org/nyc/-/nyc-14.1.1.tgz", - "integrity": "sha512-OI0vm6ZGUnoGZv/tLdZ2esSVzDwUC88SNs+6JoSOMVxA+gKMB8Tk7jBwgemLx4O40lhhvZCVw1C+OYLOBOPXWw==", - "dev": true, - "requires": { - "archy": "^1.0.0", - "caching-transform": "^3.0.2", - "convert-source-map": "^1.6.0", - "cp-file": "^6.2.0", - "find-cache-dir": "^2.1.0", - "find-up": "^3.0.0", - "foreground-child": "^1.5.6", - "glob": "^7.1.3", - "istanbul-lib-coverage": "^2.0.5", - "istanbul-lib-hook": "^2.0.7", - "istanbul-lib-instrument": "^3.3.0", - "istanbul-lib-report": "^2.0.8", - "istanbul-lib-source-maps": "^3.0.6", - "istanbul-reports": "^2.2.4", - "js-yaml": "^3.13.1", - "make-dir": "^2.1.0", - "merge-source-map": "^1.1.0", - "resolve-from": "^4.0.0", - "rimraf": "^2.6.3", - "signal-exit": "^3.0.2", - "spawn-wrap": "^1.4.2", - "test-exclude": "^5.2.3", - "uuid": "^3.3.2", - "yargs": "^13.2.2", - "yargs-parser": "^13.0.0" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } + "node_modules/eslint": { + "version": "9.39.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.39.1.tgz", + "integrity": "sha512-BhHmn2yNOFA9H9JmmIVKJmd288g9hrVRDkdoIgRCRuSySRUHH7r/DI6aAXW9T1WwUuY3DFgrcaqB+deURBLR5g==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.8.0", + "@eslint-community/regexpp": "^4.12.1", + "@eslint/config-array": "^0.21.1", + "@eslint/config-helpers": "^0.4.2", + "@eslint/core": "^0.17.0", + "@eslint/eslintrc": "^3.3.1", + "@eslint/js": "9.39.1", + "@eslint/plugin-kit": "^0.4.1", + "@humanfs/node": "^0.16.6", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.4.2", + "@types/estree": "^1.0.6", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.6", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^8.4.0", + "eslint-visitor-keys": "^4.2.1", + "espree": "^10.4.0", + "esquery": "^1.5.0", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^8.0.0", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "jiti": "*" + }, + "peerDependenciesMeta": { + "jiti": { + "optional": true } } }, - "oauth-sign": { - "version": "0.9.0", - "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", - "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", - "dev": true - }, - "object-keys": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", - "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", - "dev": true - }, - "object.assign": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz", - "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==", + "node_modules/eslint-scope": { + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz", + "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==", "dev": true, - "requires": { - "define-properties": "^1.1.2", - "function-bind": "^1.1.1", - "has-symbols": "^1.0.0", - "object-keys": "^1.0.11" + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "object.getownpropertydescriptors": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.0.3.tgz", - "integrity": "sha1-h1jIRvW0B62rDyNuCYbxSwUcqhY=", + "node_modules/eslint-visitor-keys": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", + "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", "dev": true, - "requires": { - "define-properties": "^1.1.2", - "es-abstract": "^1.5.1" + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "node_modules/espree": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz", + "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==", "dev": true, - "requires": { - "wrappy": "1" + "license": "BSD-2-Clause", + "dependencies": { + "acorn": "^8.15.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^4.2.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "optimist": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/optimist/-/optimist-0.6.1.tgz", - "integrity": "sha1-2j6nRob6IaGaERwybpDrFaAZZoY=", + "node_modules/esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", "dev": true, - "requires": { - "minimist": "~0.0.1", - "wordwrap": "~0.0.2" - }, + "license": "BSD-3-Clause", "dependencies": { - "minimist": { - "version": "0.0.10", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.10.tgz", - "integrity": "sha1-3j+YVD2/lggr5IrRoMfNqDYwHc8=", - "dev": true - } + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" } }, - "os-homedir": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", - "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=", - "dev": true - }, - "os-locale": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-3.1.0.tgz", - "integrity": "sha512-Z8l3R4wYWM40/52Z+S265okfFj8Kt2cC2MKY+xNi3kFs+XGI7WXu/I309QQQYbRW4ijiZ+yxs9pqEhJh0DqW3Q==", + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", "dev": true, - "requires": { - "execa": "^1.0.0", - "lcid": "^2.0.0", - "mem": "^4.0.0" + "license": "BSD-2-Clause", + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" } }, - "p-defer": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-defer/-/p-defer-1.0.0.tgz", - "integrity": "sha1-n26xgvbJqozXQwBKfU+WsZaw+ww=", - "dev": true - }, - "p-finally": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", - "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", - "dev": true - }, - "p-is-promise": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-2.1.0.tgz", - "integrity": "sha512-Y3W0wlRPK8ZMRbNq97l4M5otioeA5lm1z7bkNkxCka8HSPjR0xRWmpCmc9utiaLP9Jb1eD8BgeIxTW4AIF45Pg==", - "dev": true - }, - "p-limit": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.2.1.tgz", - "integrity": "sha512-85Tk+90UCVWvbDavCLKPOLC9vvY8OwEX/RtKF+/1OADJMVlFfEHOiMTPVyxg7mk/dKa+ipdHm0OUkTvCpMTuwg==", + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", "dev": true, - "requires": { - "p-try": "^2.0.0" + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" } }, - "p-locate": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", - "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "node_modules/estree-walker": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz", + "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==", "dev": true, - "requires": { - "p-limit": "^2.0.0" + "license": "MIT", + "dependencies": { + "@types/estree": "^1.0.0" } }, - "p-try": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", - "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", - "dev": true - }, - "package-hash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/package-hash/-/package-hash-3.0.0.tgz", - "integrity": "sha512-lOtmukMDVvtkL84rJHI7dpTYq+0rli8N2wlnqUcBuDWCfVhRUfOmnR9SsoHFMLpACvEV60dX7rd0rFaYDZI+FA==", + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", "dev": true, - "requires": { - "graceful-fs": "^4.1.15", - "hasha": "^3.0.0", - "lodash.flattendeep": "^4.4.0", - "release-zalgo": "^1.0.0" + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.10.0" } }, - "parse-json": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", - "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "node_modules/expect-type": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.2.2.tgz", + "integrity": "sha512-JhFGDVJ7tmDJItKhYgJCGLOWjuK9vPxiXoUFLwLDc99NlmklilbiQJwoctZtt13+xMw91MCk/REan6MWHqDjyA==", "dev": true, - "requires": { - "error-ex": "^1.3.1", - "json-parse-better-errors": "^1.0.1" + "license": "Apache-2.0", + "engines": { + "node": ">=12.0.0" } }, - "path-exists": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", - "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", - "dev": true - }, - "path-is-absolute": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", - "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", - "dev": true - }, - "path-key": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", - "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", - "dev": true - }, - "path-parse": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", - "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", - "dev": true + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true, + "license": "MIT" }, - "path-type": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", - "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "node_modules/fast-glob": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", + "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", "dev": true, - "requires": { - "pify": "^3.0.0" - }, + "license": "MIT", "dependencies": { - "pify": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", - "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", - "dev": true - } + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.8" + }, + "engines": { + "node": ">=8.6.0" } }, - "pathval": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.0.tgz", - "integrity": "sha1-uULm1L3mUwBe9rcTYd74cn0GReA=", - "dev": true + "node_modules/fast-glob/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } }, - "performance-now": { + "node_modules/fast-json-stable-stringify": { "version": "2.1.0", - "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", - "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=", - "dev": true + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true, + "license": "MIT" }, - "pify": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", - "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", - "dev": true + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true, + "license": "MIT" }, - "pkg-dir": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", - "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "node_modules/fastq": { + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", + "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", "dev": true, - "requires": { - "find-up": "^3.0.0" + "license": "ISC", + "dependencies": { + "reusify": "^1.0.4" } }, - "pseudomap": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", - "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=", - "dev": true - }, - "psl": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/psl/-/psl-1.3.0.tgz", - "integrity": "sha512-avHdspHO+9rQTLbv1RO+MPYeP/SzsCoxofjVnHanETfQhTJrmB0HlDoW+EiN/R+C0BZ+gERab9NY0lPN2TxNag==", - "dev": true - }, - "pump": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", - "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "node_modules/file-entry-cache": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", "dev": true, - "requires": { - "end-of-stream": "^1.1.0", - "once": "^1.3.1" + "license": "MIT", + "dependencies": { + "flat-cache": "^4.0.0" + }, + "engines": { + "node": ">=16.0.0" } }, - "punycode": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", - "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", - "dev": true - }, - "qs": { - "version": "6.5.2", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", - "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==", - "dev": true - }, - "read-pkg": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-3.0.0.tgz", - "integrity": "sha1-nLxoaXj+5l0WwA4rGcI3/Pbjg4k=", + "node_modules/fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", "dev": true, - "requires": { - "load-json-file": "^4.0.0", - "normalize-package-data": "^2.3.2", - "path-type": "^3.0.0" + "license": "MIT", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" } }, - "read-pkg-up": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-4.0.0.tgz", - "integrity": "sha512-6etQSH7nJGsK0RbG/2TeDzZFa8shjQ1um+SwQQ5cwKy0dhSXdOncEhb1CPpvQG4h7FyOV6EB6YlV0yJvZQNAkA==", + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", "dev": true, - "requires": { - "find-up": "^3.0.0", - "read-pkg": "^3.0.0" + "license": "MIT", + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "release-zalgo": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/release-zalgo/-/release-zalgo-1.0.0.tgz", - "integrity": "sha1-CXALflB0Mpc5Mw5TXFqQ+2eFFzA=", - "dev": true, - "requires": { - "es6-error": "^4.0.1" - } - }, - "request": { - "version": "2.88.0", - "resolved": "https://registry.npmjs.org/request/-/request-2.88.0.tgz", - "integrity": "sha512-NAqBSrijGLZdM0WZNsInLJpkJokL72XYjUpnB0iwsRgxh7dB6COrHnTBNwN0E+lHDAJzu7kLAkDeY08z2/A0hg==", - "dev": true, - "requires": { - "aws-sign2": "~0.7.0", - "aws4": "^1.8.0", - "caseless": "~0.12.0", - "combined-stream": "~1.0.6", - "extend": "~3.0.2", - "forever-agent": "~0.6.1", - "form-data": "~2.3.2", - "har-validator": "~5.1.0", - "http-signature": "~1.2.0", - "is-typedarray": "~1.0.0", - "isstream": "~0.1.2", - "json-stringify-safe": "~5.0.1", - "mime-types": "~2.1.19", - "oauth-sign": "~0.9.0", - "performance-now": "^2.1.0", - "qs": "~6.5.2", - "safe-buffer": "^5.1.2", - "tough-cookie": "~2.4.3", - "tunnel-agent": "^0.6.0", - "uuid": "^3.3.2" - } - }, - "require-directory": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", - "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", - "dev": true - }, - "require-main-filename": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", - "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", - "dev": true - }, - "resolve": { - "version": "1.12.0", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.12.0.tgz", - "integrity": "sha512-B/dOmuoAik5bKcD6s6nXDCjzUKnaDvdkRyAk6rsmsKLipWj4797iothd7jmmUhWTfinVMU+wc56rYKsit2Qy4w==", + "node_modules/flat-cache": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", + "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", "dev": true, - "requires": { - "path-parse": "^1.0.6" + "license": "MIT", + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.4" + }, + "engines": { + "node": ">=16" } }, - "resolve-from": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", - "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", - "dev": true + "node_modules/flatted": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", + "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", + "dev": true, + "license": "ISC" }, - "rimraf": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.0.tgz", - "integrity": "sha512-NDGVxTsjqfunkds7CqsOiEnxln4Bo7Nddl3XhS4pXg5OzwkLqJ971ZVAAnB+DDLnF76N+VnDEiBHaVV8I06SUg==", + "node_modules/foreground-child": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.1.tgz", + "integrity": "sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==", "dev": true, - "requires": { - "glob": "^7.1.3" + "license": "ISC", + "dependencies": { + "cross-spawn": "^7.0.6", + "signal-exit": "^4.0.1" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, - "safe-buffer": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.0.tgz", - "integrity": "sha512-fZEwUGbVl7kouZs1jCdMLdt95hdIv0ZeHg6L7qPeciMZhZ+/gdesW4wgTARkrFWEpspjEATAzUGPG8N2jJiwbg==", - "dev": true - }, - "safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", - "dev": true - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "set-blocking": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", - "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", - "dev": true - }, - "shebang-command": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", - "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", "dev": true, - "requires": { - "shebang-regex": "^1.0.0" + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" } }, - "shebang-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", - "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", - "dev": true - }, - "signal-exit": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", - "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", - "dev": true - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - }, - "spawn-wrap": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/spawn-wrap/-/spawn-wrap-1.4.3.tgz", - "integrity": "sha512-IgB8md0QW/+tWqcavuFgKYR/qIRvJkRLPJDFaoXtLLUaVcCDK0+HeFTkmQHj3eprcYhc+gOl0aEA1w7qZlYezw==", - "dev": true, - "requires": { - "foreground-child": "^1.5.6", - "mkdirp": "^0.5.0", - "os-homedir": "^1.0.1", - "rimraf": "^2.6.2", - "signal-exit": "^3.0.2", - "which": "^1.3.0" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } + "node_modules/glob": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-11.1.0.tgz", + "integrity": "sha512-vuNwKSaKiqm7g0THUBu2x7ckSs3XJLXE+2ssL7/MfTGPLLcrJQ/4Uq1CjPTtO5cCIiRxqvN6Twy1qOwhL0Xjcw==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "foreground-child": "^3.3.1", + "jackspeak": "^4.1.1", + "minimatch": "^10.1.1", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^2.0.0" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, - "spdx-correct": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.0.tgz", - "integrity": "sha512-lr2EZCctC2BNR7j7WzJ2FpDznxky1sjfxvvYEyzxNyb6lZXHODmEoJeFu4JupYlkfha1KZpJyoqiJ7pgA1qq8Q==", + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", "dev": true, - "requires": { - "spdx-expression-parse": "^3.0.0", - "spdx-license-ids": "^3.0.0" + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" } }, - "spdx-exceptions": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.2.0.tgz", - "integrity": "sha512-2XQACfElKi9SlVb1CYadKDXvoajPgBVPn/gOQLrTvHdElaVhr7ZEbqJaRnJLVNeaI4cMEAgVCeBMKF6MWRDCRA==", - "dev": true + "node_modules/glob/node_modules/minimatch": { + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.1.1.tgz", + "integrity": "sha512-enIvLvRAFZYXJzkCYG5RKmPfrFArdLv+R+lbQ53BmIMLIry74bjKzX6iHAm8WYamJkhSSEabrWN5D97XnKObjQ==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "@isaacs/brace-expansion": "^5.0.0" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } }, - "spdx-expression-parse": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.0.tgz", - "integrity": "sha512-Yg6D3XpRD4kkOmTpdgbUiEJFKghJH03fiC1OPll5h/0sO6neh2jqRDVHOQ4o/LMea0tgCkbMgea5ip/e+MkWyg==", - "dev": true, - "requires": { - "spdx-exceptions": "^2.1.0", - "spdx-license-ids": "^3.0.0" - } - }, - "spdx-license-ids": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.5.tgz", - "integrity": "sha512-J+FWzZoynJEXGphVIS+XEh3kFSjZX/1i9gFBaWQcB+/tmpe2qUsSBABpcxqxnAxFdiUFEgAX1bjYGQvIZmoz9Q==", - "dev": true - }, - "sprintf-js": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", - "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", - "dev": true - }, - "sshpk": { - "version": "1.16.1", - "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", - "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", - "dev": true, - "requires": { - "asn1": "~0.2.3", - "assert-plus": "^1.0.0", - "bcrypt-pbkdf": "^1.0.0", - "dashdash": "^1.12.0", - "ecc-jsbn": "~0.1.1", - "getpass": "^0.1.1", - "jsbn": "~0.1.0", - "safer-buffer": "^2.0.2", - "tweetnacl": "~0.14.0" - } - }, - "string-width": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", - "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "node_modules/globals": { + "version": "16.5.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-16.5.0.tgz", + "integrity": "sha512-c/c15i26VrJ4IRt5Z89DnIzCGDn9EcebibhAOjw5ibqEHsE1wLUgkPn9RDmNcUKyU87GeaL633nyJ+pplFR2ZQ==", "dev": true, - "requires": { - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^4.0.0" + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "strip-ansi": { + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true, + "license": "MIT" + }, + "node_modules/has-flag": { "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", - "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true, - "requires": { - "ansi-regex": "^3.0.0" + "license": "MIT", + "engines": { + "node": ">=8" } }, - "strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", - "dev": true + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true, + "license": "MIT" }, - "strip-eof": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", - "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", - "dev": true + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } }, - "strip-json-comments": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", - "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", - "dev": true + "node_modules/import-fresh": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", + "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } }, - "supports-color": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.0.0.tgz", - "integrity": "sha512-on9Kwidc1IUQo+bQdhi8+Tijpo0e1SS6RoGo2guUwn5vdaxw8RXOF9Vb2ws+ihWOmh4JnCJOvaziZWP1VABaLg==", + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", "dev": true, - "requires": { - "has-flag": "^3.0.0" + "license": "MIT", + "engines": { + "node": ">=0.8.19" } }, - "test-exclude": { - "version": "5.2.3", - "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-5.2.3.tgz", - "integrity": "sha512-M+oxtseCFO3EDtAaGH7iiej3CBkzXqFMbzqYAACdzKui4eZA+pq3tZEwChvOdNfa7xxy8BfbmgJSIr43cC/+2g==", + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", "dev": true, - "requires": { - "glob": "^7.1.3", - "minimatch": "^3.0.4", - "read-pkg-up": "^4.0.0", - "require-main-filename": "^2.0.0" + "license": "MIT", + "engines": { + "node": ">=0.12.0" } }, - "to-fast-properties": { + "node_modules/isexe": { "version": "2.0.0", - "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", - "integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=", - "dev": true + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" }, - "tough-cookie": { - "version": "2.4.3", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.4.3.tgz", - "integrity": "sha512-Q5srk/4vDM54WJsJio3XNn6K2sCG+CQ8G5Wz6bZhRZoAe/+TxjWB/GlFAnYEbkYVlON9FMk/fE3h2RLpPXo4lQ==", + "node_modules/istanbul-lib-coverage": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.2.tgz", + "integrity": "sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==", "dev": true, - "requires": { - "psl": "^1.1.24", - "punycode": "^1.4.1" + "license": "BSD-3-Clause", + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", + "integrity": "sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^4.0.0", + "supports-color": "^7.1.0" }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "5.0.6", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-5.0.6.tgz", + "integrity": "sha512-yg2d+Em4KizZC5niWhQaIomgf5WlL4vOOjZ5xGCmF8SnPE/mDWWXgvRExdcpCgh9lLRRa1/fSYp2ymmbJ1pI+A==", + "dev": true, + "license": "BSD-3-Clause", "dependencies": { - "punycode": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", - "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=", - "dev": true - } + "@jridgewell/trace-mapping": "^0.3.23", + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0" + }, + "engines": { + "node": ">=10" } }, - "trim-right": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/trim-right/-/trim-right-1.0.1.tgz", - "integrity": "sha1-yy4SAwZ+DI3h9hQJS5/kVwTqYAM=", - "dev": true + "node_modules/istanbul-reports": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.2.0.tgz", + "integrity": "sha512-HGYWWS/ehqTV3xN10i23tkPkpH46MLCIMFNCaaKNavAXTF1RkqxawEPtnjnGZ6XKSInBKkiOA5BKS+aZiY3AvA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } }, - "ts-mockito": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/ts-mockito/-/ts-mockito-2.4.2.tgz", - "integrity": "sha512-3AqLVXxjfdwlo2eC+xrzFsc5rsPtKBBhJZAnxWmyBmgT/PC+K26RIxiT2QLKcqjcJqZnuGZkwfPMx2gN31lFnw==", + "node_modules/jackspeak": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-4.1.1.tgz", + "integrity": "sha512-zptv57P3GpL+O0I7VdMJNBZCu+BPHVQUk55Ft8/QCJjTVxrnJHuVuX/0Bl2A6/+2oyR/ZMEuFKwmzqqZ/U5nPQ==", "dev": true, - "requires": { - "lodash": "^4.17.5" + "license": "BlueOak-1.0.0", + "dependencies": { + "@isaacs/cliui": "^8.0.2" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, - "tslib": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.10.0.tgz", - "integrity": "sha512-qOebF53frne81cf0S9B41ByenJ3/IuH8yJKngAX35CmiZySA0khhkovshKK+jGCaMnVomla7gVlIcc3EvKPbTQ==", - "dev": true + "node_modules/js-tokens": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-9.0.1.tgz", + "integrity": "sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ==", + "dev": true, + "license": "MIT" }, - "tslint": { - "version": "5.19.0", - "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.19.0.tgz", - "integrity": "sha512-1LwwtBxfRJZnUvoS9c0uj8XQtAnyhWr9KlNvDIdB+oXyT+VpsOAaEhEgKi1HrZ8rq0ki/AAnbGSv4KM6/AfVZw==", + "node_modules/js-yaml": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz", + "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==", "dev": true, - "requires": { - "@babel/code-frame": "^7.0.0", - "builtin-modules": "^1.1.1", - "chalk": "^2.3.0", - "commander": "^2.12.1", - "diff": "^3.2.0", - "glob": "^7.1.1", - "js-yaml": "^3.13.1", - "minimatch": "^3.0.4", - "mkdirp": "^0.5.1", - "resolve": "^1.3.2", - "semver": "^5.3.0", - "tslib": "^1.8.0", - "tsutils": "^2.29.0" + "license": "MIT", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" } }, - "tslint-sonarts": { - "version": "1.9.0", - "resolved": "https://registry.npmjs.org/tslint-sonarts/-/tslint-sonarts-1.9.0.tgz", - "integrity": "sha512-CJWt+IiYI8qggb2O/JPkS6CkC5DY1IcqRsm9EHJ+AxoWK70lvtP7jguochyNDMP2vIz/giGdWCfEM39x/I/Vnw==", + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", "dev": true, - "requires": { - "immutable": "^3.8.2" + "license": "MIT" + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "json-buffer": "3.0.1" } }, - "tsutils": { - "version": "2.29.0", - "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", - "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", "dev": true, - "requires": { - "tslib": "^1.8.1" + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" } }, - "tunnel-agent": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", - "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", "dev": true, - "requires": { - "safe-buffer": "^5.0.1" + "license": "MIT", + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "tweetnacl": { - "version": "0.14.5", - "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", - "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", - "dev": true + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "dev": true, + "license": "MIT" }, - "type-detect": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", - "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", - "dev": true + "node_modules/lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true, + "license": "MIT" }, - "typescript": { - "version": "3.5.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.5.3.tgz", - "integrity": "sha512-ACzBtm/PhXBDId6a6sDJfroT2pOWt/oOnk4/dElG5G33ZL776N3Y6/6bKZJBFpd+b05F3Ct9qDjMeJmRWtE2/g==", - "dev": true + "node_modules/lru-cache": { + "version": "11.2.2", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-11.2.2.tgz", + "integrity": "sha512-F9ODfyqML2coTIsQpSkRHnLSZMtkU8Q+mSfcaIyKwy58u+8k5nvAYeiNhsyMARvzNcXJ9QfWVrcPsC9e9rAxtg==", + "dev": true, + "license": "ISC", + "engines": { + "node": "20 || >=22" + } }, - "uWebSockets.js": { - "version": "github:uNetworking/uWebSockets.js#e7552889a50a7ffa654c41c99a9bad46dd0024c5", - "from": "github:uNetworking/uWebSockets.js#v15.11.0" + "node_modules/magic-string": { + "version": "0.30.21", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz", + "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.5" + } }, - "uglify-js": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.6.0.tgz", - "integrity": "sha512-W+jrUHJr3DXKhrsS7NUVxn3zqMOFn0hL/Ei6v0anCIMoKC93TjcflTagwIHLW7SfMFfiQuktQyFVCFHGUE0+yg==", + "node_modules/magicast": { + "version": "0.5.1", + "resolved": "https://registry.npmjs.org/magicast/-/magicast-0.5.1.tgz", + "integrity": "sha512-xrHS24IxaLrvuo613F719wvOIv9xPHFWQHuvGUBmPnCA/3MQxKI3b+r7n1jAoDHmsbC5bRhTZYR77invLAxVnw==", "dev": true, - "optional": true, - "requires": { - "commander": "~2.20.0", - "source-map": "~0.6.1" + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.28.5", + "@babel/types": "^7.28.5", + "source-map-js": "^1.2.1" + } + }, + "node_modules/make-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", + "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", + "dev": true, + "license": "MIT", + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "license": "MIT", + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true, - "optional": true + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minipass": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", + "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=16 || 14 >=14.17" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "license": "MIT" + }, + "node_modules/nanoid": { + "version": "3.3.11", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz", + "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" } + ], + "license": "MIT", + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" } }, - "uri-js": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", - "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", "dev": true, - "requires": { - "punycode": "^2.1.0" + "license": "MIT" + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, + "engines": { + "node": ">= 0.8.0" } }, - "uuid": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.3.tgz", - "integrity": "sha512-pW0No1RGHgzlpHJO1nsVrHKpOEIxkGg1xB+v0ZmdNH5OAeAwzAVrCnI2/6Mtx+Uys6iaylxa+D3g4j63IKKjSQ==", - "dev": true + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } }, - "validate-npm-package-license": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", - "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", "dev": true, - "requires": { - "spdx-correct": "^3.0.0", - "spdx-expression-parse": "^3.0.0" + "license": "MIT", + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "verror": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", - "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", + "node_modules/package-json-from-dist": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", + "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==", + "dev": true, + "license": "BlueOak-1.0.0" + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", "dev": true, - "requires": { - "assert-plus": "^1.0.0", - "core-util-is": "1.0.2", - "extsprintf": "^1.2.0" + "license": "MIT", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" } }, - "which": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", - "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", "dev": true, - "requires": { - "isexe": "^2.0.0" + "license": "MIT", + "engines": { + "node": ">=8" } }, - "which-module": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", - "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", - "dev": true + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } }, - "wide-align": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", - "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", + "node_modules/path-scurry": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-2.0.1.tgz", + "integrity": "sha512-oWyT4gICAu+kaA7QWk/jvCHWarMKNs6pXOGWKDTr7cw4IGcUbW+PeTfbaQiLGheFRpjo6O9J0PmyMfQPjH71oA==", "dev": true, - "requires": { - "string-width": "^1.0.2 || 2" + "license": "BlueOak-1.0.0", + "dependencies": { + "lru-cache": "^11.0.0", + "minipass": "^7.1.2" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, - "wordwrap": { - "version": "0.0.3", - "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.3.tgz", - "integrity": "sha1-o9XabNXAvAAI03I0u68b7WMFkQc=", - "dev": true + "node_modules/pathe": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/pathe/-/pathe-2.0.3.tgz", + "integrity": "sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==", + "dev": true, + "license": "MIT" }, - "wrap-ansi": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz", - "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=", + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", "dev": true, - "requires": { - "string-width": "^1.0.1", - "strip-ansi": "^3.0.1" + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/postcss": { + "version": "8.5.6", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz", + "integrity": "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", "dependencies": { - "ansi-regex": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", - "dev": true + "nanoid": "^3.3.11", + "picocolors": "^1.1.1", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/prettier": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.6.2.tgz", + "integrity": "sha512-I7AIg5boAr5R0FFtJ6rCfD+LFsWHp81dolrFD8S79U9tb8Az2nGrJncnMSnys+bpQJfRUzqs9hnA81OAA3hCuQ==", + "dev": true, + "license": "MIT", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" }, - "is-fullwidth-code-point": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", - "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", - "dev": true, - "requires": { - "number-is-nan": "^1.0.0" - } + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/reusify": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", + "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-6.1.0.tgz", + "integrity": "sha512-DxdlA1bdNzkZK7JiNWH+BAx1x4tEJWoTofIopFo6qWUU94jYrFZ0ubY05TqH3nWPJ1nKa1JWVFDINZ3fnrle/A==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "glob": "^11.0.3", + "package-json-from-dist": "^1.0.1" + }, + "bin": { + "rimraf": "dist/esm/bin.mjs" + }, + "engines": { + "node": "20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rollup": { + "version": "4.53.2", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.53.2.tgz", + "integrity": "sha512-MHngMYwGJVi6Fmnk6ISmnk7JAHRNF0UkuucA0CUW3N3a4KnONPEZz+vUanQP/ZC/iY1Qkf3bwPWzyY84wEks1g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "1.0.8" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.53.2", + "@rollup/rollup-android-arm64": "4.53.2", + "@rollup/rollup-darwin-arm64": "4.53.2", + "@rollup/rollup-darwin-x64": "4.53.2", + "@rollup/rollup-freebsd-arm64": "4.53.2", + "@rollup/rollup-freebsd-x64": "4.53.2", + "@rollup/rollup-linux-arm-gnueabihf": "4.53.2", + "@rollup/rollup-linux-arm-musleabihf": "4.53.2", + "@rollup/rollup-linux-arm64-gnu": "4.53.2", + "@rollup/rollup-linux-arm64-musl": "4.53.2", + "@rollup/rollup-linux-loong64-gnu": "4.53.2", + "@rollup/rollup-linux-ppc64-gnu": "4.53.2", + "@rollup/rollup-linux-riscv64-gnu": "4.53.2", + "@rollup/rollup-linux-riscv64-musl": "4.53.2", + "@rollup/rollup-linux-s390x-gnu": "4.53.2", + "@rollup/rollup-linux-x64-gnu": "4.53.2", + "@rollup/rollup-linux-x64-musl": "4.53.2", + "@rollup/rollup-openharmony-arm64": "4.53.2", + "@rollup/rollup-win32-arm64-msvc": "4.53.2", + "@rollup/rollup-win32-ia32-msvc": "4.53.2", + "@rollup/rollup-win32-x64-gnu": "4.53.2", + "@rollup/rollup-win32-x64-msvc": "4.53.2", + "fsevents": "~2.3.2" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" }, - "string-width": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", - "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", - "dev": true, - "requires": { - "code-point-at": "^1.0.0", - "is-fullwidth-code-point": "^1.0.0", - "strip-ansi": "^3.0.0" - } + { + "type": "patreon", + "url": "https://www.patreon.com/feross" }, - "strip-ansi": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", - "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", - "dev": true, - "requires": { - "ansi-regex": "^2.0.0" - } + { + "type": "consulting", + "url": "https://feross.org/support" } + ], + "license": "MIT", + "dependencies": { + "queue-microtask": "^1.2.2" } }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", - "dev": true + "node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } }, - "write-file-atomic": { - "version": "2.4.3", - "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-2.4.3.tgz", - "integrity": "sha512-GaETH5wwsX+GcnzhPgKcKjJ6M2Cq3/iZp1WyY/X1CSqrW+jVNM9Y7D8EC2sM4ZG/V8wZlSniJnCKWPmBYAucRQ==", + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", "dev": true, - "requires": { - "graceful-fs": "^4.1.11", - "imurmurhash": "^0.1.4", - "signal-exit": "^3.0.2" + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/siginfo": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/siginfo/-/siginfo-2.0.0.tgz", + "integrity": "sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==", + "dev": true, + "license": "ISC" + }, + "node_modules/signal-exit": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", + "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/source-map-js": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", + "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/stackback": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz", + "integrity": "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==", + "dev": true, + "license": "MIT" + }, + "node_modules/std-env": { + "version": "3.10.0", + "resolved": "https://registry.npmjs.org/std-env/-/std-env-3.10.0.tgz", + "integrity": "sha512-5GS12FdOZNliM5mAOxFRg7Ir0pWz8MdpYm6AY6VPkGpbA7ZzmbzNcBJQ0GPvvyWgcY7QAhCgf9Uy89I03faLkg==", + "dev": true, + "license": "MIT" + }, + "node_modules/string-width": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", + "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", + "dev": true, + "license": "MIT", + "dependencies": { + "eastasianwidth": "^0.2.0", + "emoji-regex": "^9.2.2", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/string-width-cjs": { + "name": "string-width", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true, + "license": "MIT" + }, + "node_modules/string-width-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" } }, - "ws": { + "node_modules/strip-ansi": { "version": "7.1.2", - "resolved": "https://registry.npmjs.org/ws/-/ws-7.1.2.tgz", - "integrity": "sha512-gftXq3XI81cJCgkUiAVixA0raD9IVmXqsylCrjRygw4+UOOGzPoxnQ6r/CnVL9i+mDncJo94tSkyrtuuQVBmrg==", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.2.tgz", + "integrity": "sha512-gmBGslpoQJtgnMAvOVqGZpEz9dyoKTCzy2nfz/n8aIFhN/jCE/rCmcxabB6jOOHV+0WNnylOxaxBQPSvcWklhA==", "dev": true, - "requires": { - "async-limiter": "^1.0.0" + "license": "MIT", + "dependencies": { + "ansi-regex": "^6.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/strip-ansi?sponsor=1" } }, - "y18n": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", - "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==", - "dev": true - }, - "yallist": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", - "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=", - "dev": true - }, - "yargs": { - "version": "13.2.2", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.2.2.tgz", - "integrity": "sha512-WyEoxgyTD3w5XRpAQNYUB9ycVH/PQrToaTXdYXRdOXvEy1l19br+VJsc0vcO8PTGg5ro/l/GY7F/JMEBmI0BxA==", - "dev": true, - "requires": { - "cliui": "^4.0.0", - "find-up": "^3.0.0", - "get-caller-file": "^2.0.1", - "os-locale": "^3.1.0", - "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", - "string-width": "^3.0.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^13.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true + "node_modules/strip-ansi-cjs": { + "name": "strip-ansi", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tinybench": { + "version": "2.9.0", + "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz", + "integrity": "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinyexec": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.2.tgz", + "integrity": "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinyglobby": { + "version": "0.2.15", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.15.tgz", + "integrity": "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "fdir": "^6.5.0", + "picomatch": "^4.0.3" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" + } + }, + "node_modules/tinyglobby/node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/tinyglobby/node_modules/picomatch": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", + "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/tinyrainbow": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-3.0.3.tgz", + "integrity": "sha512-PSkbLUoxOFRzJYjjxHJt9xro7D+iilgMX/C9lawzVuYiIdcihh9DXmVibBe8lmcFrRi/VzlPjBxbN7rH24q8/Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/ts-api-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.1.0.tgz", + "integrity": "sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18.12" + }, + "peerDependencies": { + "typescript": ">=4.8.4" + } + }, + "node_modules/ts-mockito": { + "version": "2.6.1", + "resolved": "https://registry.npmjs.org/ts-mockito/-/ts-mockito-2.6.1.tgz", + "integrity": "sha512-qU9m/oEBQrKq5hwfbJ7MgmVN5Gu6lFnIGWvpxSjrqq6YYEVv+RwVFWySbZMBgazsWqv6ctAyVBpo9TmAxnOEKw==", + "dev": true, + "license": "MIT", + "dependencies": { + "lodash": "^4.17.5" + } + }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/typescript": { + "version": "5.9.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", + "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/typescript-eslint": { + "version": "8.47.0", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.47.0.tgz", + "integrity": "sha512-Lwe8i2XQ3WoMjua/r1PHrCTpkubPYJCAfOurtn+mtTzqB6jNd+14n9UN1bJ4s3F49x9ixAm0FLflB/JzQ57M8Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/eslint-plugin": "8.47.0", + "@typescript-eslint/parser": "8.47.0", + "@typescript-eslint/typescript-estree": "8.47.0", + "@typescript-eslint/utils": "8.47.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <6.0.0" + } + }, + "node_modules/undici-types": { + "version": "7.16.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz", + "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==", + "dev": true, + "license": "MIT" + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/uWebSockets.js": { + "version": "20.56.0", + "resolved": "git+ssh://git@github.com/uNetworking/uWebSockets.js.git#d04e707a1292928d50163ff7545e45c3e84c5ec3", + "license": "Apache-2.0" + }, + "node_modules/vite": { + "version": "7.2.2", + "resolved": "https://registry.npmjs.org/vite/-/vite-7.2.2.tgz", + "integrity": "sha512-BxAKBWmIbrDgrokdGZH1IgkIk/5mMHDreLDmCJ0qpyJaAteP8NvMhkwr/ZCQNqNH97bw/dANTE9PDzqwJghfMQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "esbuild": "^0.25.0", + "fdir": "^6.5.0", + "picomatch": "^4.0.3", + "postcss": "^8.5.6", + "rollup": "^4.43.0", + "tinyglobby": "^0.2.15" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^20.19.0 || >=22.12.0", + "jiti": ">=1.21.0", + "less": "^4.0.0", + "lightningcss": "^1.21.0", + "sass": "^1.70.0", + "sass-embedded": "^1.70.0", + "stylus": ">=0.54.8", + "sugarss": "^5.0.0", + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true }, - "string-width": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", - "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", - "dev": true, - "requires": { - "emoji-regex": "^7.0.1", - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^5.1.0" - } + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "dev": true, - "requires": { - "ansi-regex": "^4.1.0" - } + "terser": { + "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true } } }, - "yargs-parser": { - "version": "13.0.0", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.0.0.tgz", - "integrity": "sha512-w2LXjoL8oRdRQN+hOyppuXs+V/fVAYtpcrRxZuF7Kt/Oc+Jr2uAcVntaUTNT6w5ihoWfFDpNY8CPx1QskxZ/pw==", + "node_modules/vite/node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", "dev": true, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } } }, - "yargs-unparser": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-1.5.0.tgz", - "integrity": "sha512-HK25qidFTCVuj/D1VfNiEndpLIeJN78aqgR23nL3y4N0U/91cOAzqfHlF8n2BvoNDcZmJKin3ddNSvOxSr8flw==", + "node_modules/vite/node_modules/picomatch": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", + "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", "dev": true, - "requires": { - "flat": "^4.1.0", - "lodash": "^4.17.11", - "yargs": "^12.0.5" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=12" }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/vitest": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/vitest/-/vitest-4.0.10.tgz", + "integrity": "sha512-2Fqty3MM9CDwOVet/jaQalYlbcjATZwPYGcqpiYQqgQ/dLC7GuHdISKgTYIVF/kaishKxLzleKWWfbSDklyIKg==", + "dev": true, + "license": "MIT", + "peer": true, "dependencies": { - "get-caller-file": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.3.tgz", - "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==", - "dev": true + "@vitest/expect": "4.0.10", + "@vitest/mocker": "4.0.10", + "@vitest/pretty-format": "4.0.10", + "@vitest/runner": "4.0.10", + "@vitest/snapshot": "4.0.10", + "@vitest/spy": "4.0.10", + "@vitest/utils": "4.0.10", + "debug": "^4.4.3", + "es-module-lexer": "^1.7.0", + "expect-type": "^1.2.2", + "magic-string": "^0.30.21", + "pathe": "^2.0.3", + "picomatch": "^4.0.3", + "std-env": "^3.10.0", + "tinybench": "^2.9.0", + "tinyexec": "^0.3.2", + "tinyglobby": "^0.2.15", + "tinyrainbow": "^3.0.3", + "vite": "^6.0.0 || ^7.0.0", + "why-is-node-running": "^2.3.0" + }, + "bin": { + "vitest": "vitest.mjs" + }, + "engines": { + "node": "^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "@edge-runtime/vm": "*", + "@types/debug": "^4.1.12", + "@types/node": "^20.0.0 || ^22.0.0 || >=24.0.0", + "@vitest/browser-playwright": "4.0.10", + "@vitest/browser-preview": "4.0.10", + "@vitest/browser-webdriverio": "4.0.10", + "@vitest/ui": "4.0.10", + "happy-dom": "*", + "jsdom": "*" + }, + "peerDependenciesMeta": { + "@edge-runtime/vm": { + "optional": true + }, + "@types/debug": { + "optional": true + }, + "@types/node": { + "optional": true + }, + "@vitest/browser-playwright": { + "optional": true + }, + "@vitest/browser-preview": { + "optional": true + }, + "@vitest/browser-webdriverio": { + "optional": true + }, + "@vitest/ui": { + "optional": true }, - "require-main-filename": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz", - "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE=", - "dev": true + "happy-dom": { + "optional": true }, - "yargs": { - "version": "12.0.5", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-12.0.5.tgz", - "integrity": "sha512-Lhz8TLaYnxq/2ObqHDql8dX8CJi97oHxrjUcYtzKbbykPtVW9WB+poxI+NM2UIzsMgNCZTIf0AQwsjK5yMAqZw==", - "dev": true, - "requires": { - "cliui": "^4.0.0", - "decamelize": "^1.2.0", - "find-up": "^3.0.0", - "get-caller-file": "^1.0.1", - "os-locale": "^3.0.0", - "require-directory": "^2.1.1", - "require-main-filename": "^1.0.1", - "set-blocking": "^2.0.0", - "string-width": "^2.0.0", - "which-module": "^2.0.0", - "y18n": "^3.2.1 || ^4.0.0", - "yargs-parser": "^11.1.1" - } + "jsdom": { + "optional": true + } + } + }, + "node_modules/vitest/node_modules/picomatch": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", + "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/why-is-node-running": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/why-is-node-running/-/why-is-node-running-2.3.0.tgz", + "integrity": "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==", + "dev": true, + "license": "MIT", + "dependencies": { + "siginfo": "^2.0.0", + "stackback": "0.0.2" + }, + "bin": { + "why-is-node-running": "cli.js" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wrap-ansi": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", + "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^6.1.0", + "string-width": "^5.0.1", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs": { + "name": "wrap-ansi", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true, + "license": "MIT" + }, + "node_modules/wrap-ansi-cjs/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi/node_modules/ansi-styles": { + "version": "6.2.3", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.3.tgz", + "integrity": "sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/ws": { + "version": "8.18.3", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.18.3.tgz", + "integrity": "sha512-PEIGCY5tSlUt50cqyMXfCzX+oOPqN0vuGqWzbcJ2xvnkzkq46oOpz7dQaTDBdfICb4N14+GARUDw2XV2N4tvzg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true }, - "yargs-parser": { - "version": "11.1.1", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-11.1.1.tgz", - "integrity": "sha512-C6kB/WJDiaxONLJQnF8ccx9SEeoTTLek8RVbaOIsrAUS8VrBEXfmeSnCZxygc+XC2sNMBIwOOnfcxiynjHsVSQ==", - "dev": true, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } + "utf-8-validate": { + "optional": true } } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } } } } diff --git a/package.json b/package.json index 945bbfc..7f3ceef 100644 --- a/package.json +++ b/package.json @@ -5,10 +5,16 @@ "license": "Apache-2.0", "author": "Novage", "homepage": "https://github.com/Novage/wt-tracker", - "main": "dist/index.js", - "types": "dist/index.d.ts", + "exports": "./src/index.ts", + "types": "./src/index.ts", + "publishConfig": { + "exports": "lib/index.js", + "types": "lib/index.d.ts" + }, + "type": "module", + "sideEffects": false, "engines": { - "node": ">=10.0.0" + "node": ">=16.0.0" }, "bin": { "wt-tracker": "./bin/wt-tracker" @@ -25,42 +31,37 @@ "websockets" ], "scripts": { - "start": "node ./dist/run-uws-tracker.js", + "start": "node ./src/run-tracker.ts", + "start:worker-demo": "DEBUG=wt-tracker:tracker-worker-*,wt-tracker:multi-worker-tracker,wt-tracker:uws-tracker node ./src/run-worker-tracker.ts", "build": "npm run lint && npm run clean && npm run compile", - "compile": "tsc", - "lint": "tslint -c ./tslint.json -p ./tsconfig.tslint.json", - "clean": "rimraf dist", - "watch": "tsc --watch", - "test": "npm run test:clean && npm run test:compile && npm run test:run", - "test:run": "nyc mocha test_dist/**/*.test.js", - "test:compile": "tsc --project tsconfig.test.json", - "test:clean": "rimraf test_dist", - "test:coverage": "nyc report --reporter=lcov --reporter=text", - "test:coveralls": "nyc report --reporter=text-lcov | coveralls" + "compile": "tsc --noEmit false", + "lint": "eslint --flag unstable_native_nodejs_ts_config src", + "clean": "rimraf lib", + "test": "vitest", + "test:coverage": "vitest run --coverage", + "test:compile": "tsc --project tsconfig.test.json --noEmit false", + "test:clean": "rimraf test_dist" }, "repository": { "type": "git", "url": "https://github.com/Novage/wt-tracker.git" }, "dependencies": { - "debug": "^4.1.1", - "uWebSockets.js": "github:uNetworking/uWebSockets.js#v15.11.0" + "debug": "^4.4.3", + "uWebSockets.js": "github:uNetworking/uWebSockets.js#v20.56.0" }, "devDependencies": { - "@types/chai": "^4.2.0", - "@types/debug": "^4.1.5", - "@types/mocha": "^5.2.7", - "@types/node": "^12.7.2", - "@types/ws": "^6.0.3", - "chai": "^4.2.0", - "coveralls": "^3.0.6", - "mocha": "^6.2.0", - "nyc": "^14.1.1", - "rimraf": "^3.0.0", - "ts-mockito": "^2.4.2", - "tslint": "^5.19.0", - "tslint-sonarts": "^1.9.0", - "typescript": "^3.5.3", - "ws": "^7.1.2" + "@types/debug": "^4.1.12", + "@types/node": "^24.10.1", + "@types/ws": "^8.18.1", + "@vitest/coverage-v8": "^4.0.10", + "globals": "^16.5.0", + "prettier": "^3.6.2", + "rimraf": "^6.1.0", + "ts-mockito": "^2.6.1", + "typescript": "^5.9.3", + "typescript-eslint": "^8.47.0", + "vitest": "^4.0.10", + "ws": "^8.18.3" } } diff --git a/sample/config.json b/sample/config.json index 1acae53..5bcb6e7 100644 --- a/sample/config.json +++ b/sample/config.json @@ -1,5 +1,6 @@ { - "servers": [{ + "servers": [ + { "server": { "port": 8000, "host": "0.0.0.0" @@ -11,14 +12,17 @@ "compression": 1, "maxConnections": 0 } - }, { + }, + { "server": { - "port": 8433, + "port": 8443, "host": "0.0.0.0", "key_file_name": "misc/key.pem", "cert_file_name": "misc/cert.pem", "passphrase": "1234", "dh_params_file_name": "misc/params.dh", + "ca_file_name": "misc/ca.pem", + "ssl_ciphers": "DHE-RSA-AES256-GCM-SHA384:DHE-RSA-AES128-GCM-SHA256", "ssl_prefer_low_memory_usage": true }, "websockets": { diff --git a/src/build-uws-tracker.ts b/src/build-uws-tracker.ts new file mode 100644 index 0000000..de35b35 --- /dev/null +++ b/src/build-uws-tracker.ts @@ -0,0 +1,108 @@ +import type { HttpRequest, HttpResponse } from "uWebSockets.js"; +import { debugRequest } from "./debugRequest.ts"; +import type { + ServerItemSettings, + WebSocketsAccessSettings, +} from "./settings.ts"; +import type { Tracker } from "./tracker.ts"; +import { UWebSocketsTracker } from "./uws-tracker.ts"; +import type { UwsConnectionContext } from "./uws-tracker.ts"; + +type BuildServerParams = { + tracker: Tracker; + serverSettings: ServerItemSettings; + websocketsAccess: Partial | undefined; + indexHtml: Buffer | undefined; + getServersStats: () => Promise; +}; + +export function buildUwsTracker({ + tracker, + serverSettings, + websocketsAccess, + indexHtml, + getServersStats, +}: BuildServerParams): UWebSocketsTracker { + if (!(serverSettings instanceof Object)) { + throw Error( + "failed to parse JSON configuration file: 'servers' property should be an array of objects", + ); + } + + const server = new UWebSocketsTracker(tracker, { + ...serverSettings, + access: websocketsAccess, + }); + + server.app + .get("/", (response: HttpResponse, request: HttpRequest) => { + debugRequest(server, request); + + if (indexHtml === undefined) { + const status = "404 Not Found"; + response.writeStatus(status).end(status); + } else { + response.end(indexHtml); + } + }) + .get( + "/stats.json", + async (response: HttpResponse, request: HttpRequest) => { + debugRequest(server, request); + + response.onAborted(() => { + response.aborted = true; + }); + + const swarms = await tracker.getSwarms(); + const serversStats = await getServersStats(); + + if (!response.aborted) { + const peersCountPerInfoHashPerTracker: Record[] = []; + let peersCount = 0; + let torrentsCount = 0; + + for (const trackerSwarms of swarms) { + const peersCountPerInfoHash: Record = { + totalPeers: 0, + }; + + for (const swarm of trackerSwarms) { + peersCount += swarm.peersCount; + torrentsCount++; + + const infoHashHex = Buffer.from( + swarm.infoHash, + "binary", + ).toString("hex"); + + peersCountPerInfoHash[infoHashHex] = swarm.peersCount; + peersCountPerInfoHash.totalPeers += swarm.peersCount; + } + + peersCountPerInfoHashPerTracker.push(peersCountPerInfoHash); + } + + response.cork(() => { + response.writeHeader("Content-Type", "application/json").end( + JSON.stringify({ + torrentsCount, + peersCount, + servers: serversStats, + memory: process.memoryUsage(), + peersCountPerInfoHashPerTracker, + }), + ); + }); + } + }, + ) + .any("/*", (response: HttpResponse, request: HttpRequest) => { + debugRequest(server, request); + + const status = "404 Not Found"; + response.writeStatus(status).end(status); + }); + + return server; +} diff --git a/src/debugRequest.ts b/src/debugRequest.ts new file mode 100644 index 0000000..748ba19 --- /dev/null +++ b/src/debugRequest.ts @@ -0,0 +1,40 @@ +/** + * Copyright 2025 Novage LLC. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import Debug from "debug"; +import { UWebSocketsTracker } from "./uws-tracker.ts"; +import type { HttpRequest } from "uWebSockets.js"; + +const debugRequests = Debug("wt-tracker:uws-tracker-requests"); +const debugRequestsEnabled = debugRequests.enabled; + +export function debugRequest( + server: UWebSocketsTracker, + request: HttpRequest, +): void { + if (debugRequestsEnabled) { + debugRequests( + server.settings.server.host, + server.settings.server.port, + "request method:", + request.getMethod(), + "url:", + request.getUrl(), + "query:", + request.getQuery(), + ); + } +} diff --git a/src/fast-tracker.ts b/src/fast-tracker.ts new file mode 100644 index 0000000..13d3c68 --- /dev/null +++ b/src/fast-tracker.ts @@ -0,0 +1,542 @@ +/** + * Copyright 2019 Novage LLC. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import { threadId } from "node:worker_threads"; +import Debug from "debug"; +import type { Tracker } from "./tracker.ts"; +import { TrackerError } from "./tracker.ts"; + +const debugSuffix = threadId ? `-${threadId}` : ""; +const debug = Debug(`wt-tracker:fast-tracker${debugSuffix}`); +const debugEnabled = debug.enabled; + +interface Swarm> { + infoHash: string; + completedPeers?: Set; + peers: PeerContext[]; +} + +interface PeerContext> { + peerId: string; + connection: ConnectionContext; + lastAccessed: number; + swarm: Swarm; +} + +type UnknownObject = Record; + +export type FastTrackerSettings = { + maxOffers: number; + announceInterval: number; +}; + +export class FastTracker> + implements Tracker +{ + public readonly settings: FastTrackerSettings; + #sendMessage: (json: UnknownObject, connection: ConnectionContext) => void; + + readonly #swarms = new Map>(); + public get swarms() { + return this.#swarms; + } + + public getSwarms() { + const result = []; + + for (const swarm of this.#swarms.values()) { + result.push({ + infoHash: swarm.infoHash, + peersCount: swarm.peers.length, + }); + } + + return Promise.resolve([result]); + } + + readonly #peers = new Map>(); + public get peers() { + return this.#peers; + } + + #clearPeersInterval?: NodeJS.Timeout; + + #onRemovePeer?: (peerId: string, connection: ConnectionContext) => void; + set onRemovePeer( + callback: + | ((peerId: string, connection: ConnectionContext) => void) + | undefined, + ) { + this.#onRemovePeer = callback ?? undefined; + } + + public constructor( + settings: Partial | undefined, + sendMessage: (json: UnknownObject, connection: ConnectionContext) => void, + ) { + this.#sendMessage = sendMessage; + this.settings = { + maxOffers: 20, + announceInterval: 20, + ...settings, + }; + this.startClearPeersInterval(); + } + + private getOrCreateSwarm(infoHash: string) { + let swarm = this.#swarms.get(infoHash); + + if (swarm === undefined) { + if (typeof infoHash !== "string") { + throw new TrackerError("announce: info_hash field is missing or wrong"); + } + + if (debugEnabled) { + debug( + "announce: swarm created:", + Buffer.from(infoHash).toString("hex"), + ); + } + + swarm = { + infoHash, + peers: [], + }; + + this.#swarms.set(infoHash, swarm); + } + + return swarm; + } + + private addPeerToSwarm( + swarm: Swarm, + peer: PeerContext, + isPeerCompleted: boolean, + ) { + swarm.peers.push(peer); + if (isPeerCompleted) { + swarm.completedPeers ??= new Set(); + swarm.completedPeers.add(peer.peerId); + } + } + + private removePeerFromSwarm( + swarm: Swarm, + peer: PeerContext, + ) { + const peerIndex = swarm.peers.indexOf(peer); + + swarm.completedPeers?.delete(peer.peerId); + + // eslint-disable-next-line @typescript-eslint/no-non-null-assertion + const lastPeer = swarm.peers.pop()!; + if (peerIndex < swarm.peers.length) { + swarm.peers[peerIndex] = lastPeer; + } + + if (swarm.peers.length === 0) { + if (debugEnabled) { + debug( + "disconnect peer: swarm removed (empty)", + Buffer.from(swarm.infoHash).toString("hex"), + ); + } + this.#swarms.delete(swarm.infoHash); + } + } + + private setPeerCompletedInSwarm( + swarm: Swarm, + peer: PeerContext, + ) { + swarm.completedPeers ??= new Set(); + swarm.completedPeers.add(peer.peerId); + } + + private startClearPeersInterval(): void { + this.#clearPeersInterval = setInterval(() => { + const now = performance.now(); + for (const peer of this.#peers.values()) { + if ( + now - peer.lastAccessed > + this.settings.announceInterval * 2 * 1000 + ) { + if (debugEnabled) { + debug( + "remove by timeout peer:", + Buffer.from(peer.peerId).toString("hex"), + "swarm:", + Buffer.from(peer.swarm.infoHash).toString("hex"), + ); + } + this.removePeer(peer); + } + } + }, this.settings.announceInterval * 1000); + } + + private removePeer(peer: PeerContext) { + const { swarm } = peer; + + this.removePeerFromSwarm(swarm, peer); + this.#peers.delete(peer.peerId); + + // eslint-disable-next-line @typescript-eslint/no-dynamic-delete + delete peer.connection[peer.peerId]; + + this.#onRemovePeer?.(peer.peerId, peer.connection); + } + + public processMessage( + json: UnknownObject, + connection: ConnectionContext, + ): void { + const { action } = json; + + if (action === "announce") { + const { event } = json; + if (event === undefined) { + if (json.answer === undefined) { + this.processAnnounce(json, connection); + } else { + this.processAnswer(json); + } + } else if (event === "started") { + this.processAnnounce(json, connection); + } else if (event === "stopped") { + this.processStop(json); + } else if (event === "completed") { + this.processAnnounce(json, connection, true); + } else { + throw new TrackerError("unknown announce event"); + } + } else if (action === "scrape") { + this.processScrape(json, connection); + } else { + throw new TrackerError("unknown action"); + } + } + + public disconnect(connection: ConnectionContext): void { + // Connection closed - remove all peers + + for (const peerId in connection) { + const peer = connection[peerId] as + | PeerContext + | undefined; + + if (!peer?.peerId) continue; // Not a peer property + + if (debugEnabled) { + debug( + "disconnect peer:", + Buffer.from(peer.peerId).toString("hex"), + "swarm:", + Buffer.from(peer.swarm.infoHash).toString("hex"), + ); + } + + this.removePeer(peer); + } + } + + private processAnnounce( + json: UnknownObject, + connection: ConnectionContext, + completed = false, + ): void { + const infoHash = json.info_hash as string; + const peerId = json.peer_id as string; + let swarm: Swarm | undefined; + const isPeerCompleted = completed || json.left === 0; + + let peer = connection[peerId] as PeerContext | undefined; + + if (peer === undefined) { + const existingPeer = this.#peers.get(peerId); + + if (existingPeer) { + // The peer previously was on a different connection + + if (debugEnabled) { + debug( + "peer changed connection:", + Buffer.from(existingPeer.peerId).toString("hex"), + ); + } + + this.removePeer(existingPeer); + } + + swarm = this.getOrCreateSwarm(infoHash); + + peer = { + peerId, + connection, + lastAccessed: performance.now(), + swarm, + }; + + this.addPeerToSwarm(swarm, peer, isPeerCompleted); + + (connection as unknown as UnknownObject)[peerId] = peer; + this.#peers.set(peerId, peer); + } else if (peer.peerId === peerId) { + peer.lastAccessed = performance.now(); + + if (infoHash !== peer.swarm.infoHash) { + // Peer changes swarm + const oldSwarm = peer.swarm; + + this.removePeerFromSwarm(oldSwarm, peer); + + swarm = this.getOrCreateSwarm(infoHash); + peer.swarm = swarm; + this.addPeerToSwarm(swarm, peer, isPeerCompleted); + } else { + ({ swarm } = peer); + if (isPeerCompleted) { + this.setPeerCompletedInSwarm(swarm, peer); + } + } + } else { + throw new TrackerError("announce: peerId mismatch"); + } + + const complete = swarm.completedPeers?.size ?? 0; + + this.#sendMessage( + { + action: "announce", + interval: this.settings.announceInterval, + info_hash: infoHash, + complete, + incomplete: swarm.peers.length - complete, + }, + connection, + ); + + this.sendOffersToPeers(json, swarm.peers, peer, infoHash); + } + + private sendOffersToPeers( + json: UnknownObject, + peers: readonly PeerContext[], + peer: PeerContext, + infoHash: string, + ): void { + if (peers.length <= 1) { + return; + } + + const { offers } = json; + if (offers === undefined) { + return; + } else if (!(offers instanceof Array)) { + throw new TrackerError("announce: offers field is not an array"); + } + + const numwant = json.numwant as number; + if (!Number.isInteger(numwant)) { + return; + } + + const countPeersToSend = peers.length - 1; + const countOffersToSend = Math.min( + countPeersToSend, + offers.length, + this.settings.maxOffers, + numwant, + ); + + if (countOffersToSend === countPeersToSend) { + // we have offers for all the peers from the swarm - send offers to all + const offersIterator = offers.values(); + for (const toPeer of peers) { + if (toPeer !== peer) { + this.#sendMessage( + getSendOfferJson( + offersIterator.next().value, + peer.peerId, + infoHash, + ), + toPeer.connection, + ); + } + } + } else { + // send offers to random peers + let peerIndex = Math.floor(Math.random() * peers.length); + + for (let i = 0; i < countOffersToSend; i++) { + const toPeer = peers[peerIndex]; + + if (toPeer === peer) { + i--; // do one more iteration + } else { + this.#sendMessage( + getSendOfferJson(offers[i], peer.peerId, infoHash), + toPeer.connection, + ); + } + + peerIndex++; + if (peerIndex === peers.length) { + peerIndex = 0; + } + } + } + + if (debugEnabled) { + debug( + "announce: sent offers", + countOffersToSend < 0 ? 0 : countOffersToSend, + ); + } + } + + private processAnswer(json: UnknownObject): void { + const toPeerId = json.to_peer_id as string; + const toPeer = this.#peers.get(toPeerId); + if (toPeer === undefined) { + throw new TrackerError("answer: to_peer_id is not in the swarm"); + } + + delete json.to_peer_id; + this.#sendMessage(json, toPeer.connection); + + if (debugEnabled) { + debug( + "answer: from peer", + Buffer.from(json.peer_id as string).toString("hex"), + "to peer", + Buffer.from(toPeerId).toString("hex"), + ); + } + } + + private processStop(json: UnknownObject): void { + const peerId = json.peer_id as string; + + const peer = this.#peers.get(peerId); + if (peer) { + if (debugEnabled) { + debug( + "stop peer:", + Buffer.from(peer.peerId).toString("hex"), + "swarm:", + Buffer.from(peer.swarm.infoHash).toString("hex"), + ); + } + this.removePeer(peer); + } + } + + private processScrape( + json: UnknownObject, + connection: ConnectionContext, + ): void { + const infoHash = json.info_hash; + const files: Record< + string, + { + complete: number; + incomplete: number; + downloaded: number; + } + > = {}; + + if (infoHash === undefined) { + for (const swarm of this.#swarms.values()) { + const complete = swarm.completedPeers?.size ?? 0; + files[swarm.infoHash] = { + complete, + incomplete: swarm.peers.length - complete, + downloaded: complete, + }; + } + } else if (infoHash instanceof Array) { + for (const singleInfoHash of infoHash as unknown[]) { + const swarm = this.#swarms.get(singleInfoHash as string); + if (swarm !== undefined) { + const complete = swarm.completedPeers?.size ?? 0; + files[singleInfoHash as string] = { + complete, + incomplete: swarm.peers.length - complete, + downloaded: complete, + }; + } else if (typeof singleInfoHash === "string") { + files[singleInfoHash] = { + complete: 0, + incomplete: 0, + downloaded: 0, + }; + } + } + } else { + const swarm = this.#swarms.get(infoHash as string); + if (swarm !== undefined) { + const complete = swarm.completedPeers?.size ?? 0; + files[infoHash as string] = { + complete, + incomplete: swarm.peers.length - complete, + downloaded: complete, + }; + } else if (typeof infoHash === "string") { + files[infoHash] = { + complete: 0, + incomplete: 0, + downloaded: 0, + }; + } + } + + this.#sendMessage({ action: "scrape", files }, connection); + } + + public dispose() { + clearInterval(this.#clearPeersInterval); + } +} + +function getSendOfferJson( + offerItem: unknown, + fromPeerId: string, + infoHash: string, +) { + if (!(offerItem instanceof Object)) { + throw new TrackerError("announce: wrong offer item format"); + } + + const { offer } = offerItem as UnknownObject; + const offerId = (offerItem as UnknownObject).offer_id; + + if (!(offer instanceof Object)) { + throw new TrackerError("announce: wrong offer item field format"); + } + + return { + action: "announce", + info_hash: infoHash, + offer_id: offerId, // offerId is not validated to be a string + peer_id: fromPeerId, + offer: { + type: "offer", + sdp: (offer as UnknownObject).sdp, // offer.sdp is not validated to be a string + }, + }; +} diff --git a/lib/index.ts b/src/index.ts similarity index 79% rename from lib/index.ts rename to src/index.ts index 01157af..d48892a 100644 --- a/lib/index.ts +++ b/src/index.ts @@ -15,6 +15,6 @@ * limitations under the License. */ -export { UWebSocketsTracker } from "./uws-tracker"; -export { FastTracker } from "./fast-tracker"; -export { Tracker, PeerContext, TrackerError } from "./tracker"; +export * from "./uws-tracker.ts"; +export * from "./fast-tracker.ts"; +export * from "./tracker.ts"; diff --git a/src/multi-worker-socket-app/index.ts b/src/multi-worker-socket-app/index.ts new file mode 100644 index 0000000..0da7fcb --- /dev/null +++ b/src/multi-worker-socket-app/index.ts @@ -0,0 +1,149 @@ +/** + * Copyright 2019 Novage LLC. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import { App, SSLApp } from "uWebSockets.js"; +import type { Settings } from "../settings.ts"; +import { Worker } from "node:worker_threads"; +import type { + AppsStatsResponse, + ServerWorkerInMessage, + ServerWorkerOutMessage, + WorkerDataType, +} from "./types.ts"; +import { MultiWorkerTracker } from "../multi-worker-tracker/index.ts"; + +export async function runSocketWorkersApp(settings: Settings) { + // Create connections acceptors + + const acceptorAppPromises = settings.servers.map(async (serverSettings) => { + if (!(serverSettings instanceof Object)) { + throw Error( + "failed to parse JSON configuration file: 'servers' property should be an array of objects", + ); + } + + const appSettings = { + port: 8000, + host: "0.0.0.0", + ...serverSettings.server, + }; + + const app = + appSettings.key_file_name === undefined + ? App(appSettings) + : SSLApp(appSettings); + + await new Promise((resolve, reject) => { + app.listen( + appSettings.host, + appSettings.port, + (token: false | object) => { + if (token === false) { + reject( + new Error( + `failed to listen to ${appSettings.host}:${appSettings.port}`, + ), + ); + } else { + resolve(); + } + }, + ); + }); + + return app; + }); + + const acceptorApps = await Promise.all(acceptorAppPromises); + + // Create tracker workers + + const { buildWorkerPorts } = MultiWorkerTracker.buildWorkers( + settings.tracker, + ); + + // Create socket workers + + const socketWorkers: Worker[] = []; + + // TODO: number of workers configurable + + const moduleExtension = import.meta.filename.endsWith(".js") ? "js" : "ts"; + + for (let workerIndex = 0; workerIndex < 4; workerIndex++) { + // Ports between the socket worker and the tracker workers + const trackerPorts = buildWorkerPorts(); + + const worker = new Worker( + `${import.meta.dirname}/worker.${moduleExtension}`, + { + workerData: { + settings, + trackerPorts, + } satisfies WorkerDataType, + transferList: trackerPorts, + }, + ); + + socketWorkers.push(worker); + + // Bind acceptors and socket workers + + worker.on("message", (message: ServerWorkerOutMessage) => { + if (message.type === "appDescriptor") { + ( + acceptorApps[message.appIndex] as unknown as { + addChildAppDescriptor: (descriptor: unknown) => void; + } + ).addChildAppDescriptor(message.workerAppDescriptor); + } else if (message.type === "getAppsStats") { + const requestAppsStats = async () => { + const requestId = Math.random(); + const appsStats: AppsStatsResponse["stats"] = []; + + for (const worker of socketWorkers) { + const appStats = await new Promise<(typeof appsStats)[number]>( + (resolve) => { + const listener = (message: ServerWorkerOutMessage) => { + if (message.type === "appStats" && message.id === requestId) { + resolve(message.stats); + worker.removeListener("message", listener); + } + }; + + worker.addListener("message", listener); + worker.postMessage({ + type: "getAppStats", + id: requestId, + } satisfies ServerWorkerInMessage); + }, + ); + + appsStats.push(appStats); + } + + worker.postMessage({ + type: "appsStats", + id: message.id, + stats: appsStats, + } satisfies ServerWorkerInMessage); + }; + + void requestAppsStats(); + } + }); + } +} diff --git a/src/multi-worker-socket-app/types.ts b/src/multi-worker-socket-app/types.ts new file mode 100644 index 0000000..a615122 --- /dev/null +++ b/src/multi-worker-socket-app/types.ts @@ -0,0 +1,45 @@ +import type { Settings } from "../settings.ts"; +import { MessagePort } from "node:worker_threads"; + +export type WorkerDataType = { + settings: Settings; + trackerPorts: MessagePort[]; +}; + +export type AppDescriptorMessage = { + workerAppDescriptor: unknown; + appIndex: number; + type: "appDescriptor"; +}; + +export type AppStatsResponse = { + id: number; + type: "appStats"; + stats: { + threadId: number; + webSocketsCount: number; + }; +}; + +export type AppsStatsResponse = { + id: number; + type: "appsStats"; + stats: AppStatsResponse["stats"][]; +}; + +export type AppStatsRequest = { + id: number; + type: "getAppStats"; +}; + +export type AppsStatsRequest = { + id: number; + type: "getAppsStats"; +}; + +export type ServerWorkerOutMessage = + | AppDescriptorMessage + | AppStatsResponse + | AppsStatsRequest; + +export type ServerWorkerInMessage = AppStatsRequest | AppsStatsResponse; diff --git a/src/multi-worker-socket-app/worker.ts b/src/multi-worker-socket-app/worker.ts new file mode 100644 index 0000000..b078cff --- /dev/null +++ b/src/multi-worker-socket-app/worker.ts @@ -0,0 +1,125 @@ +import type { Settings } from "../settings.ts"; +import type { Tracker } from "../tracker.ts"; +import { sendMessage, UWebSocketsTracker } from "../uws-tracker.ts"; +import type { UwsConnectionContext } from "../uws-tracker.ts"; +import { readFileSync } from "fs"; +import { MessagePort } from "worker_threads"; +import { + isMainThread, + workerData, + parentPort, + threadId, +} from "node:worker_threads"; +import { MultiWorkerTracker } from "../multi-worker-tracker/index.ts"; +import { buildUwsTracker } from "../build-uws-tracker.ts"; +import type { + AppDescriptorMessage, + AppsStatsResponse, + ServerWorkerInMessage, + ServerWorkerOutMessage, + WorkerDataType, +} from "./types.ts"; + +// TODO: +// - test what host and port workers require +// - handle socket bind errors here +// - configure worker ports + +if (!isMainThread && parentPort) { + const { settings, trackerPorts } = workerData as WorkerDataType; + + const tracker = new MultiWorkerTracker(trackerPorts, sendMessage); + + await runSocketApp(tracker, settings, parentPort); +} + +async function runSocketApp( + tracker: Tracker, + settings: Settings, + parentPort: MessagePort, +): Promise { + let indexHtml: Buffer | undefined = undefined; + + try { + indexHtml = readFileSync("index.html"); + } catch (e) { + if ((e as { code?: string }).code !== "ENOENT") { + throw e; + } + } + + const servers: UWebSocketsTracker[] = []; + + const getAppsStats = async () => { + return new Promise((resolve) => { + const id = Math.random(); + + const listener = (message: ServerWorkerInMessage) => { + if (message.type === "appsStats" && message.id === id) { + resolve(message.stats); + parentPort.removeListener("message", listener); + } + }; + + parentPort.addListener("message", listener); + + parentPort.postMessage({ + id, + type: "getAppsStats", + } satisfies ServerWorkerOutMessage); + }); + }; + + const serverPromises = settings.servers.map( + async (serverSettings, appIndex) => { + const server = buildUwsTracker({ + tracker, + serverSettings: { + server: { + ...serverSettings.server, + port: (serverSettings.server?.port ?? 8000) + 10000, + }, + ...serverSettings, + }, + websocketsAccess: settings.websocketsAccess, + indexHtml, + getServersStats: getAppsStats, + }); + servers.push(server); + await server.run(); + + // The worker sends back its descriptor to the main acceptor + parentPort.postMessage({ + type: "appDescriptor", + appIndex, + workerAppDescriptor: ( + server.app as unknown as { getDescriptor: () => unknown } + ).getDescriptor(), + } satisfies AppDescriptorMessage); + }, + ); + + parentPort.on("message", (message: ServerWorkerInMessage) => { + if (message.type === "getAppStats") { + parentPort.postMessage({ + type: "appStats", + id: message.id, + stats: servers.reduce( + (acc, server) => { + return { + threadId, + webSocketsCount: + acc.webSocketsCount + server.stats.webSocketsCount, + }; + }, + { + threadId, + webSocketsCount: 0, + }, + ), + } satisfies ServerWorkerOutMessage); + } + }); + + await Promise.all(serverPromises); +} diff --git a/src/multi-worker-tracker/index.ts b/src/multi-worker-tracker/index.ts new file mode 100644 index 0000000..62497cd --- /dev/null +++ b/src/multi-worker-tracker/index.ts @@ -0,0 +1,242 @@ +/* +TODO: +- config validate via scheme, reuse scheme types +*/ + +import os from "node:os"; +import { Worker, MessagePort, threadId } from "node:worker_threads"; +import Debug from "debug"; +import { TrackerError } from "../tracker.ts"; +import type { Tracker } from "../tracker.ts"; +import type { + TrackerWorkerInEvent, + TrackerWorkerOutEvent, + WorkerDataType, +} from "./types.ts"; + +const debugSuffix = threadId ? `-${threadId}` : ""; +const debug = Debug(`wt-tracker:multi-worker-tracker${debugSuffix}`); + +type PeerPortWithConnection = MessagePort & { + peerId: string; + connection: ConnectionContext; + workerIndex: number; +}; + +export class MultiWorkerTracker< + ConnectionContext extends Record, +> implements Tracker +{ + #onRemovePeer?: (peerId: string, connection: ConnectionContext) => void; + set onRemovePeer( + callback: + | ((peerId: string, connection: ConnectionContext) => void) + | undefined, + ) { + this.#onRemovePeer = callback ?? undefined; + } + + #workerPorts: MessagePort[]; + #sendMessage: ( + json: Record, + connection: ConnectionContext, + ) => void; + + constructor( + workerPorts: MessagePort[], + sendMessage: ( + json: Record, + connection: ConnectionContext, + ) => void, + ) { + this.#workerPorts = workerPorts; + this.#sendMessage = sendMessage; + } + + async getSwarms() { + const swarms = []; + + for (const workerPort of this.#workerPorts) { + const stats = await new Promise((resolve) => { + const requestId = Math.random(); + + const listener = (event: Event) => { + const data = (event as MessageEvent).data as TrackerWorkerOutEvent; + + if ( + // data.type == "stats" && + data.id === requestId + ) { + resolve(data); + workerPort.removeEventListener("message", listener); + } + }; + + workerPort.addEventListener("message", listener); + + workerPort.postMessage({ + type: "getStats", + id: requestId, + } satisfies TrackerWorkerInEvent); + }); + + swarms.push(stats.swarms); + } + + return swarms; + } + + processMessage(json: Record, connection: ConnectionContext) { + // TODO: handle scrape requests by info_hash: undefined, string, array + + const infoHash = json.info_hash; + if (typeof infoHash !== "string" || infoHash.length < 4) { + throw new TrackerError("info_hash field is missing or wrong"); + } + + const peerId = json.peer_id; + if (typeof peerId !== "string" || !peerId) { + throw new TrackerError("peer_id field is missing or wrong"); + } + + const workerIndex = + (infoHash.charCodeAt(0) + + infoHash.charCodeAt(1) + + infoHash.charCodeAt(2) + + infoHash.charCodeAt(3)) % + this.#workerPorts.length; + + const workerPort = this.#workerPorts[workerIndex]; + + let peer = connection[peerId] as + | PeerPortWithConnection + | undefined; + + if (!peer || peer.workerIndex !== workerIndex) { + // Create or recreate peer on worker + + const { port1, port2 } = new MessageChannel(); + + const peerPort = port1 as PeerPortWithConnection; + peerPort.connection = connection; + peerPort.peerId = peerId; + peerPort.workerIndex = workerIndex; + + peerPort.addEventListener("message", this.processPeerPortMessage); + peerPort.addEventListener("close", this.processPeerPortClose); + + (connection as Record)[peerId] = peerPort; + + // Send new peer port to worker + workerPort.postMessage( + { type: "newPeer", port: port2 } satisfies TrackerWorkerInEvent, + [port2], + ); + + // Close existing peer if it is assigned to another worker + if (peer) peer.close(); + + peer = peerPort; + } + + // debug( + // "peer message in", + // peer.peerId, + // json.info_hash, + // json.action, + // json.event, + // ); + + // Send message to peer on worker + peer.postMessage(json); + } + + processPeerPortMessage = (event: Event) => { + const messageEvent = event as MessageEvent; + const peerPort = + messageEvent.target as PeerPortWithConnection; + const json = messageEvent.data as Record; + + // debug( + // "peer message out", + // peerPort.peerId, + // json.info_hash, + // json.action, + // json.event, + // ); + + this.#sendMessage(json, peerPort.connection); + }; + + processPeerPortClose = (event: Event) => { + const messageEvent = event as MessageEvent; + const peerPort = + messageEvent.target as PeerPortWithConnection; + + const isPeerRecreated = peerPort.connection[peerPort.peerId] !== peerPort; + + debug("peer close", peerPort.peerId, isPeerRecreated); + + if (isPeerRecreated) return; + + // eslint-disable-next-line @typescript-eslint/no-dynamic-delete + delete peerPort.connection[peerPort.peerId]; + this.#onRemovePeer?.(peerPort.peerId, peerPort.connection); + }; + + disconnect(connection: ConnectionContext) { + debug("disconnect"); + for (const peerId in connection) { + const peer = connection[peerId] as + | PeerPortWithConnection + | undefined; + if (!peer?.peerId) continue; // Not a peer property + + // Sends `close` message to ports on both sides + peer.close(); + } + } + + dispose() { + // do nothing + } + + public static buildWorkers( + // TODO: implement settings: at least configurable worker count + settings: Record = {}, + ) { + const workersCount = os.cpus().length; + debug("starting", workersCount, "workers"); + + const workers: Worker[] = []; + const moduleExtension = import.meta.filename.endsWith(".js") ? "js" : "ts"; + + for (let workerIndex = 0; workerIndex < workersCount; workerIndex++) { + workers.push( + new Worker(`${import.meta.dirname}/worker.${moduleExtension}`, { + workerData: { settings } satisfies WorkerDataType, + }), + ); + } + + return { + buildWorkerPorts: () => { + const ports = []; + + for (const worker of workers) { + const { port1, port2 } = new MessageChannel(); + worker.postMessage( + { type: "port", port: port2 } satisfies TrackerWorkerInEvent, + [port2], + ); + ports.push(port1); + } + + return ports; + }, + + terminateWorkers: () => + Promise.allSettled(workers.map((worker) => worker.terminate())), + }; + } +} diff --git a/src/multi-worker-tracker/types.ts b/src/multi-worker-tracker/types.ts new file mode 100644 index 0000000..30a1e13 --- /dev/null +++ b/src/multi-worker-tracker/types.ts @@ -0,0 +1,32 @@ +import type { FastTrackerSettings } from "../fast-tracker.ts"; +import { MessagePort } from "node:worker_threads"; + +export type WorkerDataType = { + settings?: Partial; +}; + +export type TrackerWorkerInEvent = + // Pot from socket worker to tracker worker + | { + type: "port"; + port: MessagePort; + } + // Stats request + | { + type: "getStats"; + id: number; + } + // Port from socket worker peer to tracker worker peer + | { + type: "newPeer"; + port: MessagePort; + }; + +export type SwarmsStats = { infoHash: string; peersCount: number }[]; + +export type TrackerWorkerOutEvent = { + type: "stats"; + id: number; + threadId: number; + swarms: SwarmsStats; +}; diff --git a/src/multi-worker-tracker/worker.ts b/src/multi-worker-tracker/worker.ts new file mode 100644 index 0000000..b2f31c6 --- /dev/null +++ b/src/multi-worker-tracker/worker.ts @@ -0,0 +1,116 @@ +import { + isMainThread, + workerData, + MessagePort, + threadId, + parentPort, +} from "node:worker_threads"; +import Debug from "debug"; +import { FastTracker } from "../fast-tracker.ts"; +import type { + TrackerWorkerInEvent, + TrackerWorkerOutEvent, + WorkerDataType, +} from "./types.ts"; +import { TrackerError } from "../tracker.ts"; + +type WorkerTrackerConnectionContext = MessagePort & + Record & { + peerId: string; + }; + +if (!isMainThread && parentPort) { + // Worker thread + + const { settings } = workerData as WorkerDataType; + + const debug = Debug(`wt-tracker:tracker-worker-${threadId}`); + + debug("worker started"); + + const tracker = new FastTracker(settings, sendMessage); + tracker.onRemovePeer = removePeer; + + parentPort.addEventListener("message", (event: Event) => { + const messageEvent = event as MessageEvent; + const data = messageEvent.data as TrackerWorkerInEvent | undefined; + + if (data?.type === "port") { + // Port from MultiWorkerTracker + const trackerPort = messageEvent.ports[0] as unknown as MessagePort; + trackerPort.addEventListener("message", processPortMessage); + } + }); + + function processPortMessage(event: Event) { + const messageEvent = event as MessageEvent; + const data = messageEvent.data as TrackerWorkerInEvent | undefined; + + if (data?.type === "newPeer") { + debug("new peer"); + // New peer port from MultiWorkerTracker + const peerPort = data.port; + peerPort.addEventListener("message", processPeerPortMessage); + peerPort.addEventListener("close", processPeerPortClose); + } else if (data?.type === "getStats") { + const swarms = []; + + for (const swarm of tracker.swarms.values()) { + swarms.push({ + infoHash: swarm.infoHash, + peersCount: swarm.peers.length, + }); + } + + (messageEvent.target as MessagePort).postMessage({ + type: "stats", + id: data.id, + threadId, + swarms, + } satisfies TrackerWorkerOutEvent); + } + } + + function processPeerPortMessage(event: Event) { + const messageEvent = event as MessageEvent; + const peerPort = messageEvent.target as WorkerTrackerConnectionContext; + const json = messageEvent.data as Record; + + peerPort.peerId = json.peer_id as string; + + try { + tracker.processMessage(json, peerPort); + } catch (e) { + if (e instanceof TrackerError) { + debug("failed to process message from the peer:", e); + removePeer(peerPort.peerId, peerPort); + } else { + throw e; + } + } + } + + function processPeerPortClose(event: Event) { + const messageEvent = event as MessageEvent; + const peerPort = messageEvent.target as WorkerTrackerConnectionContext; + + debug("peer close", peerPort.peerId, tracker.peers.size); + tracker.disconnect(peerPort); + } + + function sendMessage( + json: Record, + peerPort: WorkerTrackerConnectionContext, + ) { + // debug("peer message out", peerPort.peerId, json.info_hash, json.action); + peerPort.postMessage(json); + } + + function removePeer( + peerId: string, + peerPort: WorkerTrackerConnectionContext, + ) { + debug("peer remove", peerId, peerPort.peerId); + peerPort.close(); + } +} diff --git a/src/run-tracker.ts b/src/run-tracker.ts new file mode 100644 index 0000000..e1dbf51 --- /dev/null +++ b/src/run-tracker.ts @@ -0,0 +1,76 @@ +/** + * Copyright 2019 Novage LLC. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +/* eslint-disable no-console */ + +import { readFileSync } from "fs"; +import { sendMessage } from "./uws-tracker.ts"; +import { FastTracker } from "./fast-tracker.ts"; +import { validateSettings } from "./settings.ts"; +import { runSocketApp } from "./socket-app.ts"; + +async function main(): Promise { + let settingsFileData: Buffer | undefined = undefined; + + if (process.argv.length <= 2) { + try { + settingsFileData = readFileSync("config.json"); + } catch (e) { + if ((e as { code?: string }).code !== "ENOENT") { + console.error("failed to read configuration file:", e); + return; + } + } + } else { + try { + settingsFileData = readFileSync(process.argv[2]); + } catch (e) { + console.error("failed to read configuration file:", e); + return; + } + } + + let jsonSettings: Record | undefined = undefined; + + try { + jsonSettings = + settingsFileData === undefined + ? {} + : (JSON.parse(settingsFileData.toString()) as Record); + } catch (e) { + console.error("failed to parse JSON configuration file:", e); + return; + } + + const settings = validateSettings(jsonSettings); + if (settings === undefined) { + return; + } + + const tracker = new FastTracker(settings.tracker, sendMessage); + + try { + await runSocketApp(tracker, settings); + } catch (e) { + console.error("failed to start the web server:", e); + } +} + +try { + await main(); +} catch (e) { + console.error(e); +} diff --git a/src/run-worker-tracker.ts b/src/run-worker-tracker.ts new file mode 100644 index 0000000..472f07a --- /dev/null +++ b/src/run-worker-tracker.ts @@ -0,0 +1,72 @@ +/** + * Copyright 2019 Novage LLC. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +/* eslint-disable no-console */ + +import { readFileSync } from "fs"; +import { validateSettings } from "./settings.ts"; +import { runSocketWorkersApp } from "./multi-worker-socket-app/index.ts"; + +async function main(): Promise { + let settingsFileData: Buffer | undefined = undefined; + + if (process.argv.length <= 2) { + try { + settingsFileData = readFileSync("config.json"); + } catch (e) { + if ((e as { code?: string }).code !== "ENOENT") { + console.error("failed to read configuration file:", e); + return; + } + } + } else { + try { + settingsFileData = readFileSync(process.argv[2]); + } catch (e) { + console.error("failed to read configuration file:", e); + return; + } + } + + let jsonSettings: Record | undefined = undefined; + + try { + jsonSettings = + settingsFileData === undefined + ? {} + : (JSON.parse(settingsFileData.toString()) as Record); + } catch (e) { + console.error("failed to parse JSON configuration file:", e); + return; + } + + const settings = validateSettings(jsonSettings); + if (settings === undefined) { + return; + } + + try { + await runSocketWorkersApp(settings); + } catch (e) { + console.error("failed to start the web server:", e); + } +} + +try { + await main(); +} catch (e) { + console.error(e); +} diff --git a/src/settings.ts b/src/settings.ts new file mode 100644 index 0000000..96f18c1 --- /dev/null +++ b/src/settings.ts @@ -0,0 +1,110 @@ +/** + * Copyright 2025 Novage LLC. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +/* eslint-disable no-console */ +export interface Settings { + servers: ServerItemSettings[]; + tracker?: Record; + websocketsAccess?: Partial; +} + +export interface ServerItemSettings { + server?: Partial; + websockets?: Partial; +} + +export interface ServerSettings { + port: number; + host: string; + key_file_name?: string; + cert_file_name?: string; + passphrase?: string; + dh_params_file_name?: string; + ca_file_name?: string; + ssl_ciphers?: string; + ssl_prefer_low_memory_usage?: boolean; +} + +export interface WebSocketsSettings { + path: string; + maxPayloadLength: number; + idleTimeout: number; + compression: number; + maxConnections: number; +} + +export interface WebSocketsAccessSettings { + allowOrigins?: readonly string[]; + denyOrigins?: readonly string[]; + denyEmptyOrigin: boolean; +} + +export function validateSettings( + jsonSettings: Record, +): Settings | undefined { + if ( + jsonSettings.servers !== undefined && + !(jsonSettings.servers instanceof Array) + ) { + console.error( + "failed to parse JSON configuration file: 'servers' property should be an array", + ); + return undefined; + } + + const servers: object[] = []; + + if (jsonSettings.servers === undefined) { + servers.push({}); + } else { + for (const serverSettings of jsonSettings.servers) { + if (serverSettings instanceof Object) { + servers.push(serverSettings as object); + } else { + console.error( + "failed to parse JSON configuration file: 'servers' property should be an array of objects", + ); + return undefined; + } + } + } + + if ( + jsonSettings.tracker !== undefined && + !(jsonSettings.tracker instanceof Object) + ) { + console.error( + "failed to parse JSON configuration file: 'tracker' property should be an object", + ); + return undefined; + } + + if ( + jsonSettings.websocketsAccess !== undefined && + !(jsonSettings.websocketsAccess instanceof Object) + ) { + console.error( + "failed to parse JSON configuration file: 'websocketsAccess' property should be an object", + ); + return undefined; + } + + return { + servers, + tracker: jsonSettings.tracker as Record, + websocketsAccess: jsonSettings.websocketsAccess, + }; +} diff --git a/src/socket-app.ts b/src/socket-app.ts new file mode 100644 index 0000000..895644f --- /dev/null +++ b/src/socket-app.ts @@ -0,0 +1,67 @@ +/** + * Copyright 2019 Novage LLC. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +/* eslint-disable no-console */ + +import { readFileSync } from "fs"; +import { UWebSocketsTracker } from "./uws-tracker.ts"; +import type { UwsConnectionContext } from "./uws-tracker.ts"; +import type { Tracker } from "./tracker.ts"; +import type { Settings } from "./settings.ts"; +import { buildUwsTracker } from "./build-uws-tracker.ts"; + +export async function runSocketApp( + tracker: Tracker, + settings: Settings, +): Promise { + let indexHtml: Buffer | undefined = undefined; + + try { + indexHtml = readFileSync("index.html"); + } catch (e) { + if ((e as { code?: string }).code !== "ENOENT") { + throw e; + } + } + + const servers: UWebSocketsTracker[] = []; + + const getServersStats = async () => { + return Promise.resolve( + servers.map((server, index) => ({ + server: `${settings.servers[index].server?.host ?? "0.0.0.0"}:${settings.servers[index].server?.port ?? 8000}`, + webSocketsCount: server.stats.webSocketsCount, + })), + ); + }; + + const serverPromises = settings.servers.map(async (serverSettings) => { + const server = buildUwsTracker({ + tracker, + serverSettings, + websocketsAccess: settings.websocketsAccess, + indexHtml, + getServersStats, + }); + servers.push(server); + await server.run(); + console.info( + `listening ${server.settings.server.host}:${server.settings.server.port}`, + ); + }); + + await Promise.all(serverPromises); +} diff --git a/lib/tracker.ts b/src/tracker.ts similarity index 57% rename from lib/tracker.ts rename to src/tracker.ts index d56760b..8e98c43 100644 --- a/lib/tracker.ts +++ b/src/tracker.ts @@ -14,16 +14,20 @@ * limitations under the License. */ -export interface PeerContext { - id?: string; - sendMessage: (json: any, peer: PeerContext) => void; -} +export interface Tracker { + getSwarms: () => Promise<{ infoHash: string; peersCount: number }[][]>; + processMessage: ( + json: Record, + connection: ConnectionContext, + ) => void; + + disconnect: (connection: ConnectionContext) => void; -export interface Tracker { - readonly swarms: ReadonlyMap }>; - readonly settings: any; - processMessage(json: any, peer: PeerContext): void; - disconnectPeer(peer: PeerContext): void; + set onRemovePeer( + callback: + | ((peerId: string, connection: ConnectionContext) => void) + | undefined, + ); } -export class TrackerError extends Error { } +export class TrackerError extends Error {} diff --git a/src/uws-tracker.ts b/src/uws-tracker.ts new file mode 100644 index 0000000..241d8eb --- /dev/null +++ b/src/uws-tracker.ts @@ -0,0 +1,353 @@ +/** + * Copyright 2019 Novage LLC. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import { StringDecoder } from "string_decoder"; +import { App, SSLApp } from "uWebSockets.js"; +import type { + HttpRequest, + HttpResponse, + TemplatedApp, + WebSocket, + us_socket_context_t, +} from "uWebSockets.js"; +import Debug from "debug"; +import { TrackerError } from "./tracker.ts"; +import type { Tracker } from "./tracker.js"; +import type { + ServerSettings, + WebSocketsAccessSettings, + WebSocketsSettings, +} from "./settings.ts"; +import { threadId } from "node:worker_threads"; + +const debugSuffix = threadId ? `-${threadId}` : ""; + +const debugWebSockets = Debug(`wt-tracker:uws-tracker${debugSuffix}`); +const debugWebSocketsEnabled = debugWebSockets.enabled; + +const debugMessages = Debug(`wt-tracker:uws-tracker-messages${debugSuffix}`); +const debugMessagesEnabled = debugMessages.enabled; + +const debugRequests = Debug(`wt-tracker:uws-tracker-requests${debugSuffix}`); +const debugRequestsEnabled = debugRequests.enabled; + +const decoder = new StringDecoder(); + +export type UwsConnectionContext = { + ws?: WebSocket; +} & Record; + +export interface UwsTrackerSettings { + server: ServerSettings; + websockets: WebSocketsSettings; + access: WebSocketsAccessSettings; +} + +export interface PartialUwsTrackerSettings { + server?: Partial; + websockets?: Partial; + access?: Partial; +} + +export class UWebSocketsTracker { + public readonly settings: UwsTrackerSettings; + public readonly tracker: Readonly>; + + private webSocketsCount = 0; + private validateOrigin = false; + private readonly maxConnections: number; + + readonly #app: TemplatedApp; + + public constructor( + tracker: Readonly>, + settings: PartialUwsTrackerSettings, + ) { + this.tracker = tracker; + this.settings = { + server: { + port: 8000, + host: "0.0.0.0", + ...settings.server, + }, + websockets: { + path: "/*", + maxPayloadLength: 64 * 1024, + idleTimeout: 240, + compression: 1, + maxConnections: 0, + ...settings.websockets, + }, + access: { + allowOrigins: undefined, + denyOrigins: undefined, + denyEmptyOrigin: false, + ...settings.access, + }, + }; + + this.maxConnections = this.settings.websockets.maxConnections; + + this.validateAccess(); + + this.#app = + this.settings.server.key_file_name === undefined + ? App(this.settings.server) + : SSLApp(this.settings.server); + + this.buildApplication(); + } + + public get app(): TemplatedApp { + return this.#app; + } + + public get stats() { + return { + threadId, + webSocketsCount: this.webSocketsCount, + }; + } + + public async run(): Promise { + await new Promise((resolve, reject) => { + this.#app.listen( + this.settings.server.host, + this.settings.server.port, + (token: false | object) => { + if (token === false) { + reject( + new Error( + `failed to listen to ${this.settings.server.host}:${this.settings.server.port}`, + ), + ); + } else { + resolve(); + } + }, + ); + }); + } + + private validateAccess(): void { + if (this.settings.access.allowOrigins !== undefined) { + if (this.settings.access.denyOrigins !== undefined) { + throw new Error( + "allowOrigins and denyOrigins can't be set simultaneously", + ); + } else if (!(this.settings.access.allowOrigins instanceof Array)) { + throw new Error( + "allowOrigins configuration paramenters should be an array of strings", + ); + } + } else if ( + this.settings.access.denyOrigins !== undefined && + !(this.settings.access.denyOrigins instanceof Array) + ) { + throw new Error( + "denyOrigins configuration paramenters should be an array of strings", + ); + } + + const origins: readonly string[] | undefined = + this.settings.access.allowOrigins ?? this.settings.access.denyOrigins; + + if (origins !== undefined) { + for (const origin of origins) { + if (typeof origin !== "string") { + throw new Error( + "allowOrigins and denyOrigins configuration paramenters should be arrays of strings", + ); + } + } + } + + this.validateOrigin = + this.settings.access.denyEmptyOrigin || + this.settings.access.allowOrigins !== undefined || + this.settings.access.denyOrigins !== undefined; + } + + private buildApplication(): void { + this.#app.ws(this.settings.websockets.path, { + compression: this.settings.websockets.compression, + maxPayloadLength: this.settings.websockets.maxPayloadLength, + idleTimeout: this.settings.websockets.idleTimeout, + open: this.onOpen, + upgrade: this.onUpgrade, + drain: (ws: WebSocket) => { + if (debugWebSocketsEnabled) { + debugWebSockets("drain", ws.getBufferedAmount()); + } + }, + message: this.onMessage, + close: this.onClose, + }); + } + + private readonly onOpen = (ws: WebSocket): void => { + const userData = ws.getUserData(); + userData.ws = ws; + + this.webSocketsCount++; + }; + + private readonly onUpgrade = ( + response: HttpResponse, + request: HttpRequest, + context: us_socket_context_t, + ): void => { + if ( + this.maxConnections !== 0 && + this.webSocketsCount > this.maxConnections + ) { + if (debugRequestsEnabled) { + debugRequests( + this.settings.server.host, + this.settings.server.port, + "ws-denied-max-connections url:", + request.getUrl(), + "query:", + request.getQuery(), + "origin:", + request.getHeader("origin"), + "total:", + this.webSocketsCount, + ); + } + + response.close(); + return; + } + + if (debugWebSocketsEnabled) { + debugWebSockets("connected via URL", request.getUrl()); + } + + if (this.validateOrigin) { + const origin = request.getHeader("origin"); + + const shouldDeny = + (this.settings.access.denyEmptyOrigin && origin.length === 0) || + this.settings.access.denyOrigins?.includes(origin) === true || + this.settings.access.allowOrigins?.includes(origin) === false; + + if (shouldDeny) { + if (debugRequestsEnabled) { + debugRequests( + this.settings.server.host, + this.settings.server.port, + "ws-denied url:", + request.getUrl(), + "query:", + request.getQuery(), + "origin:", + origin, + "total:", + this.webSocketsCount, + ); + } + + response.close(); + return; + } + } + + if (debugRequestsEnabled) { + debugRequests( + this.settings.server.host, + this.settings.server.port, + "ws-open url:", + request.getUrl(), + "query:", + request.getQuery(), + "origin:", + request.getHeader("origin"), + "total:", + this.webSocketsCount, + ); + } + + response.upgrade( + {}, + request.getHeader("sec-websocket-key"), + request.getHeader("sec-websocket-protocol"), + request.getHeader("sec-websocket-extensions"), + context, + ); + }; + + private readonly onMessage = ( + ws: WebSocket, + message: ArrayBuffer, + ): void => { + debugWebSockets("message of size", message.byteLength); + + let json; + try { + json = JSON.parse( + decoder.end(new Uint8Array(message) as Buffer), + ) as Record; + } catch (e) { + debugWebSockets("failed to parse JSON message", e); + ws.close(); + return; + } + + if (debugMessagesEnabled) { + debugMessages("in", json); + } + + try { + const userData = ws.getUserData(); + this.tracker.processMessage(json, userData); + } catch (e) { + if (e instanceof TrackerError) { + debugWebSockets("failed to process message from the peer:", e); + ws.close(); + } else { + throw e; + } + } + }; + + private readonly onClose = ( + ws: WebSocket, + code: number, + ): void => { + this.webSocketsCount--; + + const userData = ws.getUserData() as + | ReturnType + | undefined; + + // Test that user data is really a connection context + if (userData?.ws) { + this.tracker.disconnect(userData); + } + + debugWebSockets("closed with code", code); + }; +} + +export function sendMessage(json: object, connection: UwsConnectionContext) { + // Connection without WebSocket is not possible here + // eslint-disable-next-line @typescript-eslint/no-non-null-assertion + connection.ws!.send(JSON.stringify(json), false, false); + if (debugMessagesEnabled) { + debugMessages("out", json); + } +} diff --git a/test/announce.test.ts b/test/announce.test.ts index 63a324f..71261be 100644 --- a/test/announce.test.ts +++ b/test/announce.test.ts @@ -14,454 +14,154 @@ * limitations under the License. */ -import { FastTracker } from "../lib/fast-tracker"; -import { PeerContext } from "../lib/tracker"; -import { expect } from "chai"; -import { mock, instance, anything, verify, capture, resetCalls } from "ts-mockito"; - -// tslint:disable:no-useless-cast -// tslint:disable:no-use-of-empty-return-value -// tslint:disable:no-unused-expression -// tslint:disable:no-big-function -// tslint:disable: no-shadowed-variable -class PeerContextClass implements PeerContext { - public id?: string; - public swarm1?: any; - public swarm2?: any; - public swarm3?: any; - public sendMessage: (json: any, peer: PeerContext) => void = () => {}; -} +import { FastTracker } from "../src/fast-tracker.ts"; +import { describe, it, expect } from "vitest"; describe("announce", () => { - it("should add peers to swarms on announce", () => { - - const tracker = new FastTracker(); - - const peer0 = new PeerContextClass(); - let announceMessage: any = { - action: "announce", - event: "started", - info_hash: "swarm1", - peer_id: "0", - offers: new Array(), - numwant: 100, - }; - tracker.processMessage(announceMessage, peer0); - - expect(tracker.swarms).to.have.all.keys("swarm1"); - expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(1); - expect(tracker.swarms.get("swarm1")!.peers).to.include.members([peer0]); - - const peer1 = new PeerContextClass(); - announceMessage = { - action: "announce", - info_hash: "swarm1", - peer_id: "1", - offers: new Array(), - numwant: 100, - }; - tracker.processMessage(announceMessage, peer1); - - expect(tracker.swarms).to.have.all.keys("swarm1"); - expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); - expect(tracker.swarms.get("swarm1")!.peers).to.include.members([peer0, peer1]); - - announceMessage = { - action: "announce", - event: "started", - info_hash: "swarm1", - peer_id: "1", - offers: new Array(), - numwant: 100, - }; - tracker.processMessage(announceMessage, peer1); - - expect(tracker.swarms).to.have.all.keys("swarm1"); - expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); - expect(tracker.swarms.get("swarm1")!.peers).to.include.members([peer0, peer1]); - - const peer2 = new PeerContextClass(); - announceMessage = { - action: "announce", - event: "completed", - info_hash: "swarm2", - peer_id: "2_0", - offers: new Array(), - numwant: 100, - }; - tracker.processMessage(announceMessage, peer2); - - expect(tracker.swarms).to.have.all.keys("swarm1", "swarm2"); - expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); - expect(tracker.swarms.get("swarm1")!.peers).to.include.members([peer0, peer1]); - expect(tracker.swarms.get("swarm2")!.peers).to.have.lengthOf(1); - expect(tracker.swarms.get("swarm2")!.peers).to.include.members([peer2]); - - const peer3 = new PeerContextClass(); - announceMessage = { - action: "announce", - event: "completed", - info_hash: "swarm2", - peer_id: "2_1", - offers: new Array(), - numwant: 100, - }; - tracker.processMessage(announceMessage, peer3); - - expect(tracker.swarms).to.have.all.keys("swarm1", "swarm2"); - expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); - expect(tracker.swarms.get("swarm1")!.peers).to.include.members([peer0, peer1]); - expect(tracker.swarms.get("swarm2")!.peers).to.have.lengthOf(2); - expect(tracker.swarms.get("swarm2")!.peers).to.include.members([peer2, peer3]); - - announceMessage = { - action: "announce", - event: "completed", - info_hash: "swarm2", - peer_id: "1", - offers: new Array(), - numwant: 100, - }; - tracker.processMessage(announceMessage, peer1); - - expect(tracker.swarms).to.have.all.keys("swarm1", "swarm2"); - expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); - expect(tracker.swarms.get("swarm1")!.peers).to.include.members([peer0, peer1]); - expect(tracker.swarms.get("swarm2")!.peers).to.have.lengthOf(3); - expect(tracker.swarms.get("swarm2")!.peers).to.include.members([peer1, peer2, peer3]); - - }); - - it("should send offers to peers in a swarm", () => { - - const tracker = new FastTracker(); - - const offers: any[] = []; - for (let i = 0; i < 10; i++) { - offers.push({ - offer: { sdp: "x" }, - offer_id: "y", - }); - } - - const mockedPeer0 = mock(PeerContextClass); - const peer0 = instance(mockedPeer0); - peer0.id = undefined; - peer0.swarm1 = undefined; - let announceMessage: any = { - action: "announce", - event: "started", - info_hash: "swarm1", - peer_id: "0", - offers: offers, - numwant: offers.length, - }; - tracker.processMessage(announceMessage, peer0); - - verify(mockedPeer0.sendMessage(anything(), peer0)).once(); - let [json] = capture(mockedPeer0.sendMessage).first(); - expect(json.info_hash).to.be.equal("swarm1"); - expect(json.complete).to.be.equal(0); - expect(json.incomplete).to.be.equal(1); - - resetCalls(mockedPeer0); - - const mockedPeer1 = mock(PeerContextClass); - const peer1 = instance(mockedPeer1); - peer1.id = undefined; - peer1.swarm1 = undefined; - peer1.swarm2 = undefined; - announceMessage = { - action: "announce", - event: "completed", - info_hash: "swarm1", - peer_id: "1", - offers: offers, - numwant: offers.length, - }; - tracker.processMessage(announceMessage, peer1); - - verify(mockedPeer1.sendMessage(anything(), peer1)).once(); - [json] = capture(mockedPeer1.sendMessage).first(); - expect(json.action).to.be.equal("announce"); - expect(json.info_hash).to.be.equal("swarm1"); - expect(json.complete).to.be.equal(1); - expect(json.incomplete).to.be.equal(1); - - verify(mockedPeer0.sendMessage(anything(), peer0)).once(); - [json] = capture(mockedPeer0.sendMessage).first(); - expect(json.action).to.be.equal("announce"); - expect(json.info_hash).to.be.equal("swarm1"); - expect(json.peer_id).to.be.equal("1"); - expect(json.offer_id).to.be.equal("y"); - expect(json.offer).to.exist; - expect(json.offer.type).to.be.equal("offer"); - expect(json.offer.sdp).to.be.equal("x"); - - resetCalls(mockedPeer0); - resetCalls(mockedPeer1); - - const mockedPeer2 = mock(PeerContextClass); - const peer2 = instance(mockedPeer2); - peer2.id = undefined; - peer2.swarm2 = undefined; - announceMessage = { - action: "announce", - event: "started", - info_hash: "swarm2", - peer_id: "2", - offers: offers, - numwant: offers.length, - }; - tracker.processMessage(announceMessage, peer2); - - verify(mockedPeer2.sendMessage(anything(), peer2)).once(); - [json] = capture(mockedPeer2.sendMessage).first(); - expect(json.action).to.be.equal("announce"); - expect(json.info_hash).to.be.equal("swarm2"); - expect(json.complete).to.be.equal(0); - expect(json.incomplete).to.be.equal(1); - - verify(mockedPeer0.sendMessage(anything(), peer0)).never(); - verify(mockedPeer1.sendMessage(anything(), peer1)).never(); - - resetCalls(mockedPeer0); - resetCalls(mockedPeer1); - resetCalls(mockedPeer2); - - const mockedPeer3 = mock(PeerContextClass); - const peer3 = instance(mockedPeer3); - peer3.id = undefined; - peer3.swarm2 = undefined; - announceMessage = { - action: "announce", - event: "completed", - info_hash: "swarm2", - peer_id: "3", - offers: offers, - numwant: offers.length, - }; - tracker.processMessage(announceMessage, peer3); - - verify(mockedPeer3.sendMessage(anything(), peer3)).once(); - const [json3] = capture(mockedPeer3.sendMessage).first(); - expect(json.action).to.be.equal("announce"); - expect(json3.info_hash).to.be.equal("swarm2"); - expect(json3.complete).to.be.equal(1); - expect(json3.incomplete).to.be.equal(1); - - verify(mockedPeer0.sendMessage(anything(), peer0)).never(); - verify(mockedPeer1.sendMessage(anything(), peer1)).never(); - verify(mockedPeer2.sendMessage(anything(), peer2)).once(); - [json] = capture(mockedPeer2.sendMessage).first(); - expect(json.action).to.be.equal("announce"); - expect(json.info_hash).to.be.equal("swarm2"); - expect(json.peer_id).to.be.equal("3"); - expect(json.offer_id).to.be.equal("y"); - expect(json.offer).to.exist; - expect(json.offer.type).to.be.equal("offer"); - expect(json.offer.sdp).to.be.equal("x"); - - resetCalls(mockedPeer0); - resetCalls(mockedPeer1); - resetCalls(mockedPeer2); - resetCalls(mockedPeer3); - - const mockedPeer4 = mock(PeerContextClass); - const peer4 = instance(mockedPeer4); - peer4.id = undefined; - peer4.swarm2 = undefined; - announceMessage = { - action: "announce", - event: "completed", - info_hash: "swarm2", - peer_id: "4", - offers: offers, - numwant: 1, - }; - tracker.processMessage(announceMessage, peer4); - - verify(mockedPeer4.sendMessage(anything(), peer4)).once(); - [json] = capture(mockedPeer4.sendMessage).first(); - expect(json.info_hash).to.be.equal("swarm2"); - expect(json.complete).to.be.equal(2); - expect(json.incomplete).to.be.equal(1); - - verify(mockedPeer0.sendMessage(anything(), peer0)).never(); - verify(mockedPeer1.sendMessage(anything(), peer1)).never(); - - try { - verify(mockedPeer2.sendMessage(anything(), peer2)).once(); - verify(mockedPeer3.sendMessage(anything(), peer3)).never(); - [json] = capture(mockedPeer2.sendMessage).first(); - } catch { - verify(mockedPeer3.sendMessage(anything(), peer3)).once(); - verify(mockedPeer2.sendMessage(anything(), peer2)).never(); - [json] = capture(mockedPeer3.sendMessage).first(); - } - expect(json.action).to.be.equal("announce"); - expect(json.info_hash).to.be.equal("swarm2"); - expect(json.peer_id).to.be.equal("4"); - expect(json.offer_id).to.be.equal("y"); - expect(json.offer).to.exist; - expect(json.offer.type).to.be.equal("offer"); - expect(json.offer.sdp).to.be.equal("x"); - - resetCalls(mockedPeer0); - resetCalls(mockedPeer1); - resetCalls(mockedPeer2); - resetCalls(mockedPeer3); - resetCalls(mockedPeer4); - - announceMessage = { - action: "announce", - event: "completed", - info_hash: "swarm2", - peer_id: "1", - offers: offers, - numwant: offers.length, - }; - tracker.processMessage(announceMessage, peer1); - - verify(mockedPeer0.sendMessage(anything(), peer0)).never(); - verify(mockedPeer1.sendMessage(anything(), peer1)).once(); - verify(mockedPeer2.sendMessage(anything(), peer2)).once(); - verify(mockedPeer3.sendMessage(anything(), peer3)).once(); - verify(mockedPeer4.sendMessage(anything(), peer4)).once(); - }); - - it("should process answer messages", () => { - - const tracker = new FastTracker(); - - const peer1 = { - sendMessage: (json: any) => { - if (!json.offer) { - return; - } - const answerMessage = { - action: "announce", - info_hash: json.info_hash, - peer_id: "1", - to_peer_id: json.peer_id, - answer: { - type: "answer", - sdp: "sdp1", - }, - offer_id: json.offer_id, - }; - tracker.processMessage(answerMessage, peer1); - }, - }; - let announceMessage: any = { - action: "announce", - event: "started", - info_hash: "swarm1", - peer_id: "1", - }; - tracker.processMessage(announceMessage, peer1); - - const peer2 = { - sendMessage: (json: any) => { - if (!json.offer) { - return; - } - const answerMessage = { - action: "announce", - info_hash: json.info_hash, - peer_id: "2", - to_peer_id: json.peer_id, - answer: { - type: "answer", - sdp: "sdp2", - }, - offer_id: json.offer_id, - }; - tracker.processMessage(answerMessage, peer2); - }, - }; - announceMessage = { - action: "announce", - event: "started", - info_hash: "swarm1", - peer_id: "2", - }; - tracker.processMessage(announceMessage, peer2); - - const peer3 = { - sendMessage: (json: any) => { - if (!json.offer) { - return; - } - const answerMessage = { - action: "announce", - info_hash: json.info_hash, - peer_id: "3", - to_peer_id: json.peer_id, - answer: { - type: "answer", - sdp: "sdp3", - }, - offer_id: json.offer_id, - }; - tracker.processMessage(answerMessage, peer3); - }, - }; - announceMessage = { - action: "announce", - event: "started", - info_hash: "swarm1", - peer_id: "3", - }; - tracker.processMessage(announceMessage, peer3); - - const mockedPeer0 = mock(PeerContextClass); - const peer0 = instance(mockedPeer0); - peer0.id = undefined; - peer0.swarm1 = undefined; - announceMessage = { - action: "announce", - event: "started", - info_hash: "swarm1", - peer_id: "0", - offers: [{ - offer: { sdp: "sdp01" }, - offer_id: "1", - }, { - offer: { sdp: "sdp02" }, - offer_id: "2", - }, { - offer: { sdp: "sdp03" }, - offer_id: "3", - }], - numwant: 100, - }; - tracker.processMessage(announceMessage, peer0); - - verify(mockedPeer0.sendMessage(anything(), peer0)).times(4); - - let [json] = capture(mockedPeer0.sendMessage).byCallIndex(1); - expect(json.action).to.be.equal("announce"); - expect(json.info_hash).to.be.equal("swarm1"); - expect(json.peer_id).to.be.equal("1"); - expect(json.offer_id).to.be.equal("1"); - expect(json.answer.type).to.be.equal("answer"); - expect(json.answer.sdp).to.be.equal("sdp1"); - - [json] = capture(mockedPeer0.sendMessage).byCallIndex(2); - expect(json.action).to.be.equal("announce"); - expect(json.info_hash).to.be.equal("swarm1"); - expect(json.peer_id).to.be.equal("2"); - expect(json.offer_id).to.be.equal("2"); - expect(json.answer.type).to.be.equal("answer"); - expect(json.answer.sdp).to.be.equal("sdp2"); - - [json] = capture(mockedPeer0.sendMessage).byCallIndex(3); - expect(json.action).to.be.equal("announce"); - expect(json.info_hash).to.be.equal("swarm1"); - expect(json.peer_id).to.be.equal("3"); - expect(json.offer_id).to.be.equal("3"); - expect(json.answer.type).to.be.equal("answer"); - expect(json.answer.sdp).to.be.equal("sdp3"); - }); + it("should add peers to swarms on announce", () => { + const tracker = new FastTracker<{}>(undefined, () => undefined); + + const peer0 = { + sendMessage: () => {}, + }; + let announceMessage = { + action: "announce", + event: "started" as string | undefined, + info_hash: "swarm1", + peer_id: "0", + offers: new Array(), + numwant: 100, + }; + tracker.processMessage(announceMessage, peer0); + + const peerContext0 = tracker.swarms + .get("swarm1") + ?.peers.find((pd) => pd.peerId === "0"); + + expect(tracker.swarms).to.have.all.keys("swarm1"); + expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(1); + expect(tracker.swarms.get("swarm1")!.peers).to.include.members([ + peerContext0, + ]); + + const peer1 = { + sendMessage: () => {}, + }; + announceMessage = { + action: "announce", + event: undefined, + info_hash: "swarm1", + peer_id: "1", + offers: new Array(), + numwant: 100, + }; + tracker.processMessage(announceMessage, peer1); + + const peerContext1 = tracker.swarms + .get("swarm1") + ?.peers.find((pd) => pd.peerId === "1"); + + expect(tracker.swarms).to.have.all.keys("swarm1"); + expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); + expect(tracker.swarms.get("swarm1")!.peers).to.include.members([ + peerContext0, + peerContext1, + ]); + + announceMessage = { + action: "announce", + event: "started", + info_hash: "swarm1", + peer_id: "1", + offers: new Array(), + numwant: 100, + }; + tracker.processMessage(announceMessage, peer1); + + expect(tracker.swarms).to.have.all.keys("swarm1"); + expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); + expect(tracker.swarms.get("swarm1")!.peers).to.include.members([ + peerContext0, + peerContext1, + ]); + + const peer2 = { + sendMessage: () => {}, + }; + announceMessage = { + action: "announce", + event: "completed", + info_hash: "swarm2", + peer_id: "2_0", + offers: new Array(), + numwant: 100, + }; + tracker.processMessage(announceMessage, peer2); + + const peerContext2_2 = tracker.swarms + .get("swarm2") + ?.peers.find((pd) => pd.peerId === "2_0"); + + expect(tracker.swarms).to.have.all.keys("swarm1", "swarm2"); + expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); + expect(tracker.swarms.get("swarm1")!.peers).to.include.members([ + peerContext0, + peerContext1, + ]); + expect(tracker.swarms.get("swarm2")!.peers).to.have.lengthOf(1); + expect(tracker.swarms.get("swarm2")!.peers).to.include.members([ + peerContext2_2, + ]); + + const peer3 = { + sendMessage: () => {}, + }; + announceMessage = { + action: "announce", + event: "completed", + info_hash: "swarm2", + peer_id: "2_1", + offers: new Array(), + numwant: 100, + }; + tracker.processMessage(announceMessage, peer3); + + const peerContext2_1 = tracker.swarms + .get("swarm2")! + .peers.find((pd) => pd.peerId === "2_1"); + + expect(tracker.swarms).to.have.all.keys("swarm1", "swarm2"); + expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(2); + expect(tracker.swarms.get("swarm1")!.peers).to.include.members([ + peerContext0, + peerContext1, + ]); + expect(tracker.swarms.get("swarm2")!.peers).to.have.lengthOf(2); + expect(tracker.swarms.get("swarm2")!.peers).to.include.members([ + peerContext2_1, + peerContext2_2, + ]); + + announceMessage = { + action: "announce", + event: "completed", + info_hash: "swarm2", + peer_id: "1", + offers: new Array(), + numwant: 100, + }; + tracker.processMessage(announceMessage, peer1); + + expect(tracker.swarms).to.have.all.keys("swarm1", "swarm2"); + expect(tracker.swarms.get("swarm1")!.peers).to.have.lengthOf(1); + expect(tracker.swarms.get("swarm1")!.peers).to.include.members([ + peerContext0, + ]); + expect(tracker.swarms.get("swarm2")!.peers).to.have.lengthOf(3); + expect(tracker.swarms.get("swarm2")!.peers).to.include.members([ + peerContext1, + peerContext2_1, + peerContext2_2, + ]); + }); }); diff --git a/test/load-tests/simultaneous-connections.ts b/test/load-tests/simultaneous-connections.ts index 4714dff..af0f206 100644 --- a/test/load-tests/simultaneous-connections.ts +++ b/test/load-tests/simultaneous-connections.ts @@ -14,24 +14,22 @@ * limitations under the License. */ -import * as WebSocket from "ws"; - -// tslint:disable:no-console +import WebSocket from "ws"; const peersCount = 10000; const swarmsCount = 10; const offersCount = 10; -const offers = new Array(); +const offers = new Array(); for (let o = 0; o < offersCount; o++) { - offers.push({ - offer: { - sdp: "asdfasdfasdfasdfasdfasdfasdf", - value: 1, - }, - offer_id: "taasdfasdfasdfasd", - }); + offers.push({ + offer: { + sdp: "asdfasdfasdfasdfasdfasdfasdf", + value: 1, + }, + offer_id: "taasdfasdfasdfasd", + }); } const closePromises = new Array>(); @@ -40,85 +38,94 @@ const connectPromises = new Array>(); console.log("creating", peersCount, "connections"); async function timeout(milliseconds: number) { - return new Promise(resolve => setTimeout(resolve, milliseconds)); + return new Promise((resolve) => setTimeout(resolve, milliseconds)); } -// tslint:disable-next-line:cognitive-complexity async function main() { - try { - for (let p = 0; p < peersCount; p++) { - const webSocket = new WebSocket("ws://localhost:8000/"); - - closePromises.push(new Promise((resolve, reject) => { - webSocket.on("close", resolve); - webSocket.on("error", e => reject(new Error(`Socket closed ${e}`))); - })); - - connectPromises.push(new Promise((resolve, reject) => { - webSocket.on("open", () => { - try { - webSocket.send(JSON.stringify({ - action: "announce", - event: "started", - info_hash: Math.floor(swarmsCount * Math.random()).toPrecision(19).toString(), - peer_id: p.toPrecision(19).toString(), - numwant: offersCount, - offers: offers, - })); - resolve(); - } catch (e) { - reject(new Error(`Send error ${e}`)); - } - }); - webSocket.on("message", message => { - const json = JSON.parse(message as string); - - if (!json.offer) { - return; - } - - try { - webSocket.send(JSON.stringify({ - action: "announce", - info_hash: json.info_hash, - peer_id: p.toPrecision(19).toString(), - to_peer_id: json.peer_id, - answer: { - type: "answer", - sdp: "xxxxxxxxxxxx", - }, - })); - } catch (e) { - reject(new Error(`Send error ${e}`)); - } - }); - webSocket.on("close", () => reject(new Error("Socket closed"))); - webSocket.on("error", e => reject(new Error(`Socket closed ${e}`))); - })); - - await timeout(10); - } - } catch (e) { - console.log("faied to create WebSocket connections", e); - return; - } - - try { - await Promise.all(connectPromises); - } catch (e) { - console.log("socket error:", e); - return; + try { + for (let p = 0; p < peersCount; p++) { + const webSocket = new WebSocket("ws://localhost:8000/"); + + closePromises.push( + new Promise((resolve, reject) => { + webSocket.on("close", resolve); + webSocket.on("error", (e) => reject(new Error(`Socket closed ${e}`))); + }), + ); + + connectPromises.push( + new Promise((resolve, reject) => { + webSocket.on("open", () => { + try { + webSocket.send( + JSON.stringify({ + action: "announce", + event: "started", + info_hash: Math.floor(swarmsCount * Math.random()) + .toPrecision(19) + .toString(), + peer_id: p.toPrecision(19).toString(), + numwant: offersCount, + offers: offers, + }), + ); + resolve(); + } catch (e) { + reject(new Error(`Send error ${e}`)); + } + }); + webSocket.on("message", (message) => { + const json = JSON.parse(message.toString()); // FIXME: may not work after dependencies update + + if (!json.offer) { + return; + } + + try { + webSocket.send( + JSON.stringify({ + action: "announce", + info_hash: json.info_hash, + peer_id: p.toPrecision(19).toString(), + to_peer_id: json.peer_id, + answer: { + type: "answer", + sdp: "xxxxxxxxxxxx", + }, + }), + ); + } catch (e) { + reject(new Error(`Send error ${e}`)); + } + }); + webSocket.on("close", () => reject(new Error("Socket closed"))); + webSocket.on("error", (e) => reject(new Error(`Socket closed ${e}`))); + }), + ); + + await timeout(10); } - - console.log("waiting for the connections to close"); - - try { - await Promise.all(closePromises); - } catch (e) { - console.log("socket error:", e); - } - - console.log("done"); + } catch (e) { + console.log("faied to create WebSocket connections", e); + return; + } + + try { + await Promise.all(connectPromises); + } catch (e) { + console.log("socket error:", e); + return; + } + + console.log("waiting for the connections to close"); + + try { + await Promise.all(closePromises); + } catch (e) { + console.log("socket error:", e); + } + + console.log("done"); } main(); diff --git a/test/memory/heap-usage.ts b/test/memory/heap-usage.ts index 17c66bc..9d4ae8d 100644 --- a/test/memory/heap-usage.ts +++ b/test/memory/heap-usage.ts @@ -14,69 +14,73 @@ * limitations under the License. */ - // tslint:disable - -import { FastTracker } from "../../lib/fast-tracker"; +import { FastTracker } from "../../src/fast-tracker.ts"; const peersCount = 100000; const swarmsCount = 1000000000; const message = { - action: "announce", - event: "started", - info_hash: "hash", - peer_id: "", - offers: new Array(), - numwant: 10, + action: "announce", + event: "started", + info_hash: "hash", + peer_id: "", + offers: new Array(), + numwant: 10, }; for (let o = 0; o < message.numwant; o++) { - message.offers.push({ - offer: { - sdp: "x", - value: 1, - }, - offer_id: "t", - }); + message.offers.push({ + offer: { + sdp: "x", + value: 1, + }, + offer_id: "t", + }); } -const tracker = new FastTracker(); +const tracker = new FastTracker<{}>(undefined, () => undefined); console.log("heap", process.memoryUsage()); -console.log("bytes per peer in average: " + process.memoryUsage().heapUsed / peersCount); +console.log( + "bytes per peer in average: " + process.memoryUsage().heapUsed / peersCount, +); console.log("\nadding peers to swarms"); -const peers: any[] = []; +const sockets: {}[] = []; for (let p = 0; p < peersCount; p++) { - message.peer_id = p.toPrecision(19).toString(); - message.info_hash = Math.floor(swarmsCount * Math.random()).toPrecision(19).toString(); - const peer = { - sendMessage: () => p, - }; - tracker.processMessage(message, peer); - peers.push(peer); + message.peer_id = p.toPrecision(19).toString(); + message.info_hash = Math.floor(swarmsCount * Math.random()) + .toPrecision(19) + .toString(); + const peer = {}; + tracker.processMessage(message, peer); + sockets.push(peer); } let peersCountAfter = 0; for (const swarm of tracker.swarms.values()) { - peersCountAfter += swarm.peers.length; + peersCountAfter += swarm.peers.length; } console.log("swarms:", tracker.swarms.size, "peers:", peersCountAfter); console.log("heap:", process.memoryUsage()); -console.log("bytes per peer in average: " + process.memoryUsage().heapUsed / peersCount); +console.log( + "bytes per peer in average: " + process.memoryUsage().heapUsed / peersCount, +); console.log("\nremoving peers"); -for (const peer of peers) { - tracker.disconnectPeer(peer); +for (const peer of sockets) { + tracker.disconnect(peer); } -peers.length = 0; +sockets.length = 0; if (global.gc) { - global.gc(); + global.gc(); } console.log("heap:", process.memoryUsage()); -console.log("bytes per peer in average:" + process.memoryUsage().heapUsed / peersCount); +console.log( + "bytes per peer in average:" + process.memoryUsage().heapUsed / peersCount, +); diff --git a/test/performance/announce.ts b/test/performance/announce.ts index 6dbf7e2..2c38999 100644 --- a/test/performance/announce.ts +++ b/test/performance/announce.ts @@ -14,10 +14,9 @@ * limitations under the License. */ -import { FastTracker } from "../../lib/fast-tracker"; -import { Tracker } from "../../lib/tracker"; - -// tslint:disable:no-console +import { FastTracker } from "../../src/fast-tracker.ts"; +import { Tracker } from "../../src/tracker.ts"; +import { UwsConnectionContext } from "../../src/uws-tracker.ts"; function sendMessage() {} const peersCount = 100000; @@ -25,98 +24,99 @@ const offersCount = 10; const peers: any[] = []; const message = { - action: "announce", - event: "started", - info_hash: "hashhashhashhashhash", - peer_id: "", - offers: new Array(), - numwant: offersCount, + action: "announce", + event: "started", + info_hash: "hashhashhashhashhash", + peer_id: "", + offers: new Array(), + numwant: offersCount, }; for (let i = 0; i < offersCount; i++) { - message.offers.push({ - offer: { - sdp: "x", - value: 1, - }, - offer_id: "t", - }); + message.offers.push({ + offer: { + sdp: "x", + value: 1, + }, + offer_id: "t", + }); } -function addingPeersToSwarm(tracker: Tracker) { - peers.length = 0; - for (let i = 0; i < peersCount; i++) { - peers.push({ - sendMessage: sendMessage, - _peerId: i.toPrecision(19), - }); - } - - console.time(`adding peers to a swarm ${tracker.constructor.name}`); - for (let i = 0; i < peersCount; i++) { - const peer = peers[i]; - message.peer_id = peer._peerId; - tracker.processMessage(message, peer); - } - console.timeEnd(`adding peers to a swarm ${tracker.constructor.name}`); +function addingPeersToSwarm(tracker: Tracker) { + peers.length = 0; + for (let i = 0; i < peersCount; i++) { + peers.push({ + sendMessage: sendMessage, + _peerId: i.toPrecision(19), + }); + } + + console.time(`adding peers to a swarm ${tracker.constructor.name}`); + for (let i = 0; i < peersCount; i++) { + const peer = peers[i]; + message.peer_id = peer._peerId; + tracker.processMessage(message, peer); + } + console.timeEnd(`adding peers to a swarm ${tracker.constructor.name}`); } -// tslint:disable-next-line:cognitive-complexity function addingPeersToSwarmReference() { - peers.length = 0; - for (let i = 0; i < peersCount; i++) { - peers.push({ - id: i.toPrecision(19), - }); - } - - const swarm = new Map(); - const peersOrdered: any[] = []; - let counter = 0; + peers.length = 0; + for (let i = 0; i < peersCount; i++) { + peers.push({ + id: i.toPrecision(19), + }); + } + + const swarm = new Map(); + const peersOrdered: any[] = []; + let counter = 0; + + console.time("adding peers to a swarm reference"); + for (let p = 0; p < peersCount; p++) { + const peer = peers[p]; + + swarm.set(peer.id, peer); + peersOrdered.push(peer); + + message.peer_id = peer._peerId; + const offers = message.offers; + const countOffersToSend = Math.min(swarm.size - 1, offers.length, 20); + if (countOffersToSend === swarm.size - 1) { + const offersIterator = offers.values(); + for (const toPeer of swarm.values()) { + if (toPeer !== peer) { + counter += offersIterator.next().value.offer.value; + } + } + } else { + let peerIndex = Math.floor(Math.random() * swarm.size); - console.time("adding peers to a swarm reference"); - for (let p = 0; p < peersCount; p++) { - const peer = peers[p]; + // send offers to random peers + for (let i = 0; i < countOffersToSend; i++) { + const toPeer = peersOrdered[peerIndex]; - swarm.set(peer.id, peer); - peersOrdered.push(peer); + peerIndex++; + if (peerIndex === swarm.size) { + peerIndex = 0; + } - message.peer_id = peer._peerId; - const offers = message.offers; - const countOffersToSend = Math.min(swarm.size - 1, offers.length, 20); - if (countOffersToSend === swarm.size - 1) { - const offersIterator = offers.values(); - for (const toPeer of swarm.values()) { - if (toPeer !== peer) { - counter += offersIterator.next().value.offer.value; - } - } + if (toPeer === peer) { + i--; // do one more iteration } else { - let peerIndex = Math.floor(Math.random() * swarm.size); - - // send offers to random peers - for (let i = 0; i < countOffersToSend; i++) { - const toPeer = peersOrdered[peerIndex]; - - peerIndex++; - if (peerIndex === swarm.size) { - peerIndex = 0; - } - - if (toPeer === peer) { - i--; // do one more iteration - } else { - counter += offers[i].offer.value; - } - } + counter += offers[i].offer.value; } + } } - console.timeEnd("adding peers to a swarm reference"); - console.log(counter); + } + console.timeEnd("adding peers to a swarm reference"); + console.log(counter); } for (let i = 0; i < 10; i++) { - addingPeersToSwarm(new FastTracker({})); - addingPeersToSwarmReference(); - console.log("---------"); + addingPeersToSwarm( + new FastTracker({}, () => undefined), + ); + addingPeersToSwarmReference(); + console.log("---------"); } diff --git a/test/simulation.test.ts b/test/simulation.test.ts index 54840df..6f90f10 100644 --- a/test/simulation.test.ts +++ b/test/simulation.test.ts @@ -14,94 +14,103 @@ * limitations under the License. */ -import { FastTracker } from "../lib/fast-tracker"; -import { PeerContext } from "../lib/tracker"; -import { expect } from "chai"; - -// tslint:disable:no-unused-expression +import { FastTracker } from "../src/fast-tracker.ts"; +import { describe, it, expect } from "vitest"; describe("simulation", () => { - it("should pass random simulations", () => { - const simulationsCount = 1000; - const torrentsCount = 2; - const peersCount = 200; - const offersCount = 10; - const sameIdPeersRatio = 0.1; - - const tracker = new FastTracker(); + it("should pass random simulations", () => { + const simulationsCount = 1000; + const torrentsCount = 2; + const peersCount = 200; + const offersCount = 10; + const sameIdPeersRatio = 0.1; - const peers: PeerContext[] = []; - const peersData: Array<{ infoHash?: string, peerId: string }> = []; + const tracker = new FastTracker<{}>(undefined, () => undefined); - for (let i = 0; i < peersCount; i++) { - peers.push({ - sendMessage: (json: any) => {}, - }); - peersData.push({ peerId: (i % Math.floor(peersCount * sameIdPeersRatio)).toString() }); - } + const sockets: {}[] = []; + const peersData: Array<{ infoHash?: string; peerId: string }> = []; - const announceMessage = { - action: "announce", - info_hash: "", - peer_id: "", - offers: new Array(), - numwant: 100, - }; + for (let i = 0; i < peersCount; i++) { + sockets.push({}); + peersData.push({ + peerId: (i % Math.floor(peersCount * sameIdPeersRatio)).toString(), + }); + } - for (let i = 0; i < offersCount; i++) { - announceMessage.offers.push({ - offer: { sdp: "x" }, - offer_id: i.toString(), - }); - } + const announceMessage = { + action: "announce", + info_hash: "", + peer_id: "", + offers: new Array(), + numwant: 100, + }; - function doIteration() { - const peerIndex = Math.floor(Math.random() * peers.length); - const peer = peers[peerIndex]; - const peerData = peersData[peerIndex]; + for (let i = 0; i < offersCount; i++) { + announceMessage.offers.push({ + offer: { sdp: "x" }, + offer_id: i.toString(), + }); + } - if (peerData.infoHash) { // peer has been assigned to a torrent - const random = Math.random(); - if (random < 0.05) { // leave torrent - tracker.processMessage({ - action: "announce", - event: "stopped", - info_hash: peerData.infoHash, - peer_id: peer.id, - }, peer); - peerData.infoHash = undefined; + function doIteration() { + const peerIndex = Math.floor(Math.random() * sockets.length); + const peer = sockets[peerIndex]; + const peerData = peersData[peerIndex]; - return; - } else if (random < 0.06) { // disconnect - tracker.disconnectPeer(peer); - peerData.infoHash = undefined; - peers[peerIndex] = { sendMessage: peer.sendMessage }; - return; - } else { // announce on the same torrent - announceMessage.peer_id = peerData.peerId; - announceMessage.info_hash = peerData.infoHash; - tracker.processMessage(announceMessage, peer); - return; - } - } + if (peerData.infoHash) { + // peer has been assigned to a torrent + const random = Math.random(); + if (random < 0.05) { + // leave torrent + tracker.processMessage( + { + action: "announce", + event: "stopped", + info_hash: peerData.infoHash, + peer_id: peerData.peerId, + }, + peer, + ); + peerData.infoHash = undefined; - // assign the peer to a torrent - announceMessage.peer_id = peerData.peerId; - announceMessage.info_hash = peerData.infoHash = Math.floor(Math.random() * torrentsCount).toString(); - tracker.processMessage(announceMessage, peer); + return; + } else if (random < 0.06) { + // disconnect + tracker.disconnect(peer); + peerData.infoHash = undefined; + sockets[peerIndex] = {}; + return; + } else { + // announce on the same torrent + announceMessage.peer_id = peerData.peerId; + announceMessage.info_hash = peerData.infoHash; + tracker.processMessage(announceMessage, peer); + return; } + } - for (let s = 0; s < simulationsCount; s++) { - doIteration(); - } + // assign the peer to a torrent + announceMessage.peer_id = peerData.peerId; + announceMessage.info_hash = peerData.infoHash = Math.floor( + Math.random() * torrentsCount, + ).toString(); + tracker.processMessage(announceMessage, peer); + } - for (const [swarmId, swarm] of tracker.swarms) { - expect(swarm.peers).to.be.not.empty; - for (const peer of swarm.peers.values()) { - const peerData = peersData[peers.indexOf(peer)]; - expect(peerData).to.exist; - expect(peerData.infoHash).to.be.equal(swarmId); - } - } - }); + for (let s = 0; s < simulationsCount; s++) { + doIteration(); + } + + for (const [swarmId, swarm] of tracker.swarms) { + expect(swarm.peers).to.be.not.empty; + for (const peer of swarm.peers.values()) { + const peerData = peersData.find( + (pd) => pd.peerId === peer.peerId && pd.infoHash === swarmId, + ); + + expect(peerData).to.exist; + expect(peerData?.infoHash).to.be.equal(swarmId); + } + } + }); }); diff --git a/tsconfig.json b/tsconfig.json index 30c149c..b2f1cdc 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -1,24 +1,34 @@ { "compilerOptions": { - "target": "es2017", - "module": "commonjs", - "lib": ["es2017"], + "target": "ESnext", + "module": "NodeNext", + "lib": ["ES2024"], + "moduleResolution": "NodeNext", + "types": ["node"], + + "skipLibCheck": true, + "resolveJsonModule": true, + "isolatedModules": true, + "esModuleInterop": true, + "allowImportingTsExtensions": true, + "rewriteRelativeImportExtensions": true, + "noEmit": true, + + "outDir": "./lib", + "strict": true, - "moduleResolution": "node", - "declaration": true, - "outDir": "./dist", + "noImplicitAny": true, "noUnusedLocals": true, "noUnusedParameters": false, - "strictNullChecks": true, - "forceConsistentCasingInFileNames": true, "noFallthroughCasesInSwitch": true, + "erasableSyntaxOnly": true, + "noUncheckedSideEffectImports": true, + "useDefineForClassFields": true, + "forceConsistentCasingInFileNames": true, "noImplicitReturns": true, - "noImplicitThis": true, - "noImplicitAny": true, - "types": ["node"] + + "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.tsbuildinfo" }, "compileOnSave": true, - "include": [ - "lib/**/*" - ] + "include": ["src/**/*"] } diff --git a/tsconfig.lint.json b/tsconfig.lint.json new file mode 100644 index 0000000..98263ff --- /dev/null +++ b/tsconfig.lint.json @@ -0,0 +1,4 @@ +{ + "extends": "./tsconfig.json", + "include": ["src/**/*", "test/**/*", ".eslintrc.cjs", "vitest.config.ts"] +} diff --git a/tsconfig.test.json b/tsconfig.test.json index 3c1d4c9..e86ac52 100644 --- a/tsconfig.test.json +++ b/tsconfig.test.json @@ -1,25 +1,8 @@ { + "extends": "./tsconfig.json", "compilerOptions": { - "target": "es2017", - "module": "commonjs", - "lib": ["es2017"], - "strict": true, - "moduleResolution": "node", - "declaration": true, "outDir": "./test_dist", - "noUnusedLocals": true, - "noUnusedParameters": false, - "strictNullChecks": true, - "forceConsistentCasingInFileNames": true, - "noFallthroughCasesInSwitch": true, - "noImplicitReturns": true, - "noImplicitThis": true, - "noImplicitAny": true, - "sourceMap": true, - "types": ["node", "mocha"] + "types": ["node"] }, - "compileOnSave": true, - "include": [ - "lib/**/*", "test/**/*" - ] + "include": ["src/tracker.ts", "src/fast-tracker.ts", "test/**/*"] } diff --git a/tsconfig.tslint.json b/tsconfig.tslint.json deleted file mode 100644 index 2708cc5..0000000 --- a/tsconfig.tslint.json +++ /dev/null @@ -1,23 +0,0 @@ -{ - "compilerOptions": { - "target": "es2017", - "module": "commonjs", - "strict": true, - "moduleResolution": "node", - "declaration": true, - "noEmit": true, - "noUnusedLocals": true, - "noUnusedParameters": false, - "strictNullChecks": true, - "forceConsistentCasingInFileNames": true, - "noFallthroughCasesInSwitch": true, - "noImplicitReturns": true, - "noImplicitThis": true, - "noImplicitAny": true, - "types": ["node"] - }, - "include": [ - "lib/**/*", - "test/**/*" - ] -} diff --git a/tslint.json b/tslint.json deleted file mode 100644 index 73dc9ab..0000000 --- a/tslint.json +++ /dev/null @@ -1,17 +0,0 @@ -{ - "rules": { - "indent": [true, "spaces", 4], - "triple-equals": [true, "allow-undefined-check"], - "object-literal-sort-keys": false, - "ordered-imports": false, - "arrow-parens": [true, "ban-single-arg-parens"], - "object-literal-shorthand": [true, "never"], - "interface-name": [true, "never-prefix"], - "max-line-length": [true, 150], - "no-empty": [true, "allow-empty-functions"], - "variable-name": [true, "ban-keywords", "check-format", "allow-leading-underscore"], - "no-implicit-dependencies": [true, ["chai", "ws", "ts-mockito"]] - }, - "defaultSeverity": "error", - "extends": ["tslint:latest", "tslint-sonarts"] -} diff --git a/vitest.config.ts b/vitest.config.ts new file mode 100644 index 0000000..321e77e --- /dev/null +++ b/vitest.config.ts @@ -0,0 +1,10 @@ +import { defineConfig } from "vitest/config"; + +export default defineConfig({ + test: { + include: ["test/*.{test,spec}.{js,ts,jsx,tsx}"], + coverage: { + include: ["src/fast-tracker.ts", "src/tracker.ts"], + }, + }, +});