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 5bbd5fd..0222e75 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,2513 +1,2970 @@ { "name": "wt-tracker", "version": "0.0.1", - "lockfileVersion": 1, + "lockfileVersion": 3, "requires": true, - "dependencies": { - "@babel/code-frame": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.8.3.tgz", - "integrity": "sha512-a9gxpmdXtZEInkCSHUJDLHZVBgb1QS0jhss4cPP93EW7s+uC5bikET2twEF3KV+7rDblJcmNvTR7VJejqd2C2g==", - "dev": true, - "requires": { - "@babel/highlight": "^7.8.3" - } - }, - "@babel/core": { - "version": "7.8.6", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.8.6.tgz", - "integrity": "sha512-Sheg7yEJD51YHAvLEV/7Uvw95AeWqYPL3Vk3zGujJKIhJ+8oLw2ALaf3hbucILhKsgSoADOvtKRJuNVdcJkOrg==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.8.3", - "@babel/generator": "^7.8.6", - "@babel/helpers": "^7.8.4", - "@babel/parser": "^7.8.6", - "@babel/template": "^7.8.6", - "@babel/traverse": "^7.8.6", - "@babel/types": "^7.8.6", - "convert-source-map": "^1.7.0", - "debug": "^4.1.0", - "gensync": "^1.0.0-beta.1", - "json5": "^2.1.0", - "lodash": "^4.17.13", - "resolve": "^1.3.2", - "semver": "^5.4.1", - "source-map": "^0.5.0" - } - }, - "@babel/generator": { - "version": "7.8.6", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.8.6.tgz", - "integrity": "sha512-4bpOR5ZBz+wWcMeVtcf7FbjcFzCp+817z2/gHNncIRcM9MmKzUhtWCYAq27RAfUrAFwb+OCG1s9WEaVxfi6cjg==", - "dev": true, - "requires": { - "@babel/types": "^7.8.6", - "jsesc": "^2.5.1", - "lodash": "^4.17.13", - "source-map": "^0.5.0" - } - }, - "@babel/helper-function-name": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.8.3.tgz", - "integrity": "sha512-BCxgX1BC2hD/oBlIFUgOCQDOPV8nSINxCwM3o93xP4P9Fq6aV5sgv2cOOITDMtCfQ+3PvHp3l689XZvAM9QyOA==", - "dev": true, - "requires": { - "@babel/helper-get-function-arity": "^7.8.3", - "@babel/template": "^7.8.3", - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-get-function-arity": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.8.3.tgz", - "integrity": "sha512-FVDR+Gd9iLjUMY1fzE2SR0IuaJToR4RkCDARVfsBBPSP53GEqSFjD8gNyxg246VUyc/ALRxFaAK8rVG7UT7xRA==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helper-split-export-declaration": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.8.3.tgz", - "integrity": "sha512-3x3yOeyBhW851hroze7ElzdkeRXQYQbFIb7gLK1WQYsw2GWDay5gAJNw1sWJ0VFP6z5J1whqeXH/WCdCjZv6dA==", - "dev": true, - "requires": { - "@babel/types": "^7.8.3" - } - }, - "@babel/helpers": { - "version": "7.8.4", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.8.4.tgz", - "integrity": "sha512-VPbe7wcQ4chu4TDQjimHv/5tj73qz88o12EPkO2ValS2QiQS/1F2SsjyIGNnAD0vF/nZS6Cf9i+vW6HIlnaR8w==", - "dev": true, - "requires": { - "@babel/template": "^7.8.3", - "@babel/traverse": "^7.8.4", - "@babel/types": "^7.8.3" - } - }, - "@babel/highlight": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.8.3.tgz", - "integrity": "sha512-PX4y5xQUvy0fnEVHrYOarRPXVWafSjTW9T0Hab8gVIawpl2Sj0ORyrygANq+KjcNlSSTw0YCLSNA8OyZ1I4yEg==", - "dev": true, - "requires": { - "chalk": "^2.0.0", - "esutils": "^2.0.2", - "js-tokens": "^4.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.68.0" + }, + "bin": { + "wt-tracker": "bin/wt-tracker" + }, + "devDependencies": { + "@eslint/js": "^10.0.1", + "@types/debug": "^4.1.13", + "@types/node": "^25.9.2", + "@types/ws": "^8.18.1", + "@vitest/coverage-v8": "^4.1.8", + "globals": "^17.6.0", + "prettier": "^3.8.3", + "rimraf": "^6.1.3", + "ts-mockito": "^2.6.1", + "typescript": "^6.0.3", + "typescript-eslint": "^8.60.1", + "vitest": "^4.1.8", + "ws": "^8.21.0" + }, + "engines": { + "node": ">=16.0.0" } }, - "@babel/parser": { - "version": "7.8.6", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.8.6.tgz", - "integrity": "sha512-trGNYSfwq5s0SgM1BMEB8hX3NDmO7EP2wsDGDexiaKMB92BaRpS+qZfpkMqUBhcsOTBwNy9B/jieo4ad/t/z2g==", - "dev": true - }, - "@babel/template": { - "version": "7.8.6", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.8.6.tgz", - "integrity": "sha512-zbMsPMy/v0PWFZEhQJ66bqjhH+z0JgMoBWuikXybgG3Gkd/3t5oQ1Rw2WQhnSrsOmsKXnZOx15tkC4qON/+JPg==", + "node_modules/@babel/helper-string-parser": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.29.7.tgz", + "integrity": "sha512-Pb5ijPrZ89GDH8223L4UP8i6QApWxs04RbPQJTeWDV0/keR2E36MeKnyr6LYmUUvqRRI+Iv87SuF1W6ErINzYw==", "dev": true, - "requires": { - "@babel/code-frame": "^7.8.3", - "@babel/parser": "^7.8.6", - "@babel/types": "^7.8.6" + "license": "MIT", + "engines": { + "node": ">=6.9.0" } }, - "@babel/traverse": { - "version": "7.8.6", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.8.6.tgz", - "integrity": "sha512-2B8l0db/DPi8iinITKuo7cbPznLCEk0kCxDoB9/N6gGNg/gxOXiR/IcymAFPiBwk5w6TtQ27w4wpElgp9btR9A==", + "node_modules/@babel/helper-validator-identifier": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.29.7.tgz", + "integrity": "sha512-qehxGkRj55h/ff8EMaJ+cYhyaKlHIxqYDn682wQD7RNp9UujOQsHog2uS0r2vzr4pW+sXf90NeeayjcNaX3fFg==", "dev": true, - "requires": { - "@babel/code-frame": "^7.8.3", - "@babel/generator": "^7.8.6", - "@babel/helper-function-name": "^7.8.3", - "@babel/helper-split-export-declaration": "^7.8.3", - "@babel/parser": "^7.8.6", - "@babel/types": "^7.8.6", - "debug": "^4.1.0", - "globals": "^11.1.0", - "lodash": "^4.17.13" + "license": "MIT", + "engines": { + "node": ">=6.9.0" } }, - "@babel/types": { - "version": "7.8.6", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.8.6.tgz", - "integrity": "sha512-wqz7pgWMIrht3gquyEFPVXeXCti72Rm8ep9b5tQKz9Yg9LzJA3HxosF1SB3Kc81KD1A3XBkkVYtJvCKS2Z/QrA==", + "node_modules/@babel/parser": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.7.tgz", + "integrity": "sha512-hnORnjP/1P/zFEndoeX+n+t1RwWRJiJpM/jO7FW32Kn9r5+sJB2JWOdYo4L6k78j15eCwY3Gm/7364B1EMwtNg==", "dev": true, - "requires": { - "esutils": "^2.0.2", - "lodash": "^4.17.13", - "to-fast-properties": "^2.0.0" - } - }, - "@istanbuljs/load-nyc-config": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.0.0.tgz", - "integrity": "sha512-ZR0rq/f/E4f4XcgnDvtMWXCUJpi8eO0rssVhmztsZqLIEFA9UUP9zmpE0VxlM+kv/E1ul2I876Fwil2ayptDVg==", - "dev": true, - "requires": { - "camelcase": "^5.3.1", - "find-up": "^4.1.0", - "js-yaml": "^3.13.1", - "resolve-from": "^5.0.0" - }, - "dependencies": { - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.0" - } - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.0" - } - }, - "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 - } + "license": "MIT", + "dependencies": { + "@babel/types": "^7.29.7" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" } }, - "@istanbuljs/schema": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.2.tgz", - "integrity": "sha512-tsAQNx32a8CoFhjhijUIhI4kccIAgmGhy8LZMZgGfmXcpMbPRUqn5LWmgRttILi6yeGmBJd2xsPkFMs0PzgPCw==", - "dev": true - }, - "@types/chai": { - "version": "4.2.9", - "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.2.9.tgz", - "integrity": "sha512-NeXgZj+MFL4izGqA4sapdYzkzQG+MtGra9vhQ58dnmDY++VgJaRUws+aLVV5zRJCYJl/8s9IjMmhiUw1WsKSmw==", - "dev": true - }, - "@types/color-name": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz", - "integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==", - "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": "7.0.1", - "resolved": "https://registry.npmjs.org/@types/mocha/-/mocha-7.0.1.tgz", - "integrity": "sha512-L/Nw/2e5KUaprNJoRA33oly+M8X8n0K+FwLTbYqwTcR14wdPWeRkigBLfSFpN/Asf9ENZTMZwLxjtjeYucAA4Q==", - "dev": true - }, - "@types/node": { - "version": "13.7.6", - "resolved": "https://registry.npmjs.org/@types/node/-/node-13.7.6.tgz", - "integrity": "sha512-eyK7MWD0R1HqVTp+PtwRgFeIsemzuj4gBFSQxfPHY5iMjS7474e5wq+VFgTcdpyHeNxyKSaetYAjdMLJlKoWqA==", - "dev": true - }, - "@types/ws": { - "version": "7.2.2", - "resolved": "https://registry.npmjs.org/@types/ws/-/ws-7.2.2.tgz", - "integrity": "sha512-oqnI3DbGCVI9zJ/WHdFo3CUE8jQ8CVQDUIKaDtlTcNeT4zs6UCg9Gvk5QrFx2QPkRszpM6yc8o0p4aGjCsTi+w==", - "dev": true, - "requires": { - "@types/node": "*" + "node_modules/@babel/types": { + "version": "7.29.7", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.7.tgz", + "integrity": "sha512-4zBIxpPzowiZpusoFkyGVwakdRJUyuH5PxQ/PrqghfdFWWasvnCdPfQXHrenDai+gyLARulZjZowCOj6fjT4pA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.29.7", + "@babel/helper-validator-identifier": "^7.29.7" + }, + "engines": { + "node": ">=6.9.0" } }, - "aggregate-error": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.0.1.tgz", - "integrity": "sha512-quoaXsZ9/BLNae5yiNoUz+Nhkwz83GhWwtYFglcjEQB2NDHCIpApbqXxIFnm4Pq/Nvhrsq5sYJFyohrrxnTGAA==", + "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": { - "clean-stack": "^2.0.0", - "indent-string": "^4.0.0" + "license": "MIT", + "engines": { + "node": ">=18" } }, - "ajv": { - "version": "6.12.0", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.0.tgz", - "integrity": "sha512-D6gFiFA0RRLyUbvijN74DWAjXSFxWKaWP7mldxkVhyhAV3+SWA9HEJPHQ2c9soIeTFJqcSdFDGFgdqs1iUU2Hw==", + "node_modules/@emnapi/core": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/core/-/core-1.10.0.tgz", + "integrity": "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw==", "dev": true, - "requires": { - "fast-deep-equal": "^3.1.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/wasi-threads": "1.2.1", + "tslib": "^2.4.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/@emnapi/runtime": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.10.0.tgz", + "integrity": "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA==", "dev": true, - "requires": { - "color-convert": "^1.9.0" + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" } }, - "anymatch": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", - "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", + "node_modules/@emnapi/wasi-threads": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz", + "integrity": "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==", "dev": true, - "requires": { - "normalize-path": "^3.0.0", - "picomatch": "^2.0.4" + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" } }, - "append-transform": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/append-transform/-/append-transform-2.0.0.tgz", - "integrity": "sha512-7yeyCEurROLQJFv5Xj4lEGTy0borxepjFv1g22oAdqFu//SrAlDl1O1Nxx15SH1RoliUml6p8dwJW9jvZughhg==", + "node_modules/@eslint-community/eslint-utils": { + "version": "4.9.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz", + "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==", "dev": true, - "requires": { - "default-require-extensions": "^3.0.0" + "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" } }, - "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-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": { - "sprintf-js": "~1.0.2" + "license": "Apache-2.0", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "asn1": { - "version": "0.2.4", - "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", - "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", + "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": { - "safer-buffer": "~2.1.0" + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" } }, - "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 - }, - "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.9.1", - "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.9.1.tgz", - "integrity": "sha512-wMHVg2EOHaMRxbzgFJ9gtjOOCrI80OHLG14rxi28XwOW8ux6IiEbRCGGGqCtdAIg4FQCbW20k9RsT4y3gJlFug==", - "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/config-array": { + "version": "0.23.5", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.23.5.tgz", + "integrity": "sha512-Y3kKLvC1dvTOT+oGlqNQ1XLqK6D1HU2YXPc52NmAlJZbMMWDzGYXMiPRJ8TYD39muD/OTjlZmNJ4ib7dvSrMBA==", "dev": true, - "requires": { - "tweetnacl": "^0.14.3" + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "@eslint/object-schema": "^3.0.5", + "debug": "^4.3.1", + "minimatch": "^10.2.4" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "binary-extensions": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz", - "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==", - "dev": true - }, - "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/config-helpers": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.6.0.tgz", + "integrity": "sha512-ii6Bw9jJ2zi2cWA2Z+9/QZ/+3DX6kwaV5Q986D/CdP3Lap3w/pgQZ373FV7byY/i7L4IRH/G43I5dz1ClsCbpA==", "dev": true, - "requires": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "@eslint/core": "^1.2.1" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "node_modules/@eslint/core": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-1.2.1.tgz", + "integrity": "sha512-MwcE1P+AZ4C6DWlpin/OmOA54mmIZ/+xZuJiQd4SyB29oAJjN30UW9wkKNptW2ctp4cEsvhlLY/CsQ1uoHDloQ==", "dev": true, - "requires": { - "fill-range": "^7.0.1" + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "@types/json-schema": "^7.0.15" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "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": "4.0.0", - "resolved": "https://registry.npmjs.org/caching-transform/-/caching-transform-4.0.0.tgz", - "integrity": "sha512-kpqOvwXnjjN44D89K5ccQC+RUrsy7jB/XLlRrx0D7/2HNcTPqzsb6XgYoErwko6QsV184CA2YgS1fxDiiDZMWA==", + "node_modules/@eslint/js": { + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-10.0.1.tgz", + "integrity": "sha512-zeR9k5pd4gxjZ0abRoIaxdc7I3nDktoXZk2qOv9gCNWx3mVwEn32VRhyLaRsDiJjTs0xq/T8mfPtyuXu7GWBcA==", "dev": true, - "requires": { - "hasha": "^5.0.0", - "make-dir": "^3.0.0", - "package-hash": "^4.0.0", - "write-file-atomic": "^3.0.0" + "license": "MIT", + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "eslint": "^10.0.0" + }, + "peerDependenciesMeta": { + "eslint": { + "optional": true + } } }, - "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==", + "node_modules/@eslint/object-schema": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-3.0.5.tgz", + "integrity": "sha512-vqTaUEgxzm+YDSdElad6PiRoX4t8VGDjCtt05zn4nU810UIx/uNEV7/lZJ6KwFThKZOzOxzXy48da+No7HZaMw==", "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" + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "node_modules/@eslint/plugin-kit": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.7.2.tgz", + "integrity": "sha512-+CNAzxglkrpNf/kKywqQfk74QjtceuOE7Qm+AF8miRvPF/wmmK5+OJOgVh3AVTT3RP2mH3+FOaxlE5v72owk0A==", "dev": true, - "requires": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" - }, + "license": "Apache-2.0", + "peer": true, "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" - } - } + "@eslint/core": "^1.2.1", + "levn": "^0.4.1" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "check-error": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz", - "integrity": "sha1-V00xLt2Iu13YkS6Sht1sCu1KrII=", - "dev": true - }, - "chokidar": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.3.0.tgz", - "integrity": "sha512-dGmKLDdT3Gdl7fBUe8XK+gAtGmzy5Fn0XkkWQuYxGIgWVPPse2CxFA5mtrlD0TOHaHjEUqkWNyP1XdHoJES/4A==", - "dev": true, - "requires": { - "anymatch": "~3.1.1", - "braces": "~3.0.2", - "fsevents": "~2.1.1", - "glob-parent": "~5.1.0", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.2.0" - } - }, - "clean-stack": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", - "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", - "dev": true - }, - "cliui": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", - "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", + "node_modules/@humanfs/core": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.2.tgz", + "integrity": "sha512-UhXNm+CFMWcbChXywFwkmhqjs3PRCmcSa/hfBgLIb7oQ5HNb1wS0icWsGtSAUNgefHeI+eBrA8I1fxmbHsGdvA==", "dev": true, - "requires": { - "string-width": "^3.1.0", - "strip-ansi": "^5.2.0", - "wrap-ansi": "^5.1.0" - }, + "license": "Apache-2.0", + "peer": true, "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 - }, - "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" - } - }, - "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" - } - } + "@humanfs/types": "^0.15.0" + }, + "engines": { + "node": ">=18.18.0" } }, - "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/@humanfs/node": { + "version": "0.16.8", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.8.tgz", + "integrity": "sha512-gE1eQNZ3R++kTzFUpdGlpmy8kDZD/MLyHqDwqjkVQI0JMdI1D51sy1H958PNXYkM2rAac7e5/CnIKZrHtPh3BQ==", "dev": true, - "requires": { - "color-name": "1.1.3" + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "@humanfs/core": "^0.19.2", + "@humanfs/types": "^0.15.0", + "@humanwhocodes/retry": "^0.4.0" + }, + "engines": { + "node": ">=18.18.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/types": { + "version": "0.15.0", + "resolved": "https://registry.npmjs.org/@humanfs/types/-/types-0.15.0.tgz", + "integrity": "sha512-ZZ1w0aoQkwuUuC7Yf+7sdeaNfqQiiLcSRbfI08oAxqLtpXQr9AIVX7Ay7HLDuiLYAaFPu8oBYNq/QIi9URHJ3Q==", "dev": true, - "requires": { - "delayed-stream": "~1.0.0" + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": ">=18.18.0" } }, - "commander": { - "version": "2.20.3", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", - "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", - "dev": true - }, - "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 + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } }, - "convert-source-map": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.7.0.tgz", - "integrity": "sha512-4FJkXzKXEDB1snCFZlLP4gpC3JILicCpGbzG9f9G7tGqGCzETQ2hWPrcinA9oU4wtf2biUaEH5065UnMeR33oA==", + "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": { - "safe-buffer": "~5.1.1" + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": ">=18.18" }, - "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 + "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" + } }, - "coveralls": { - "version": "3.0.9", - "resolved": "https://registry.npmjs.org/coveralls/-/coveralls-3.0.9.tgz", - "integrity": "sha512-nNBg3B1+4iDox5A5zqHKzUTiwl2ey4k2o0NEcVZYvl+GOSJdKBj4AJGKLv6h3SvWch7tABHePAQOSZWM9E2hMg==", + "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, - "requires": { - "js-yaml": "^3.13.1", - "lcov-parse": "^1.0.0", - "log-driver": "^1.2.7", - "minimist": "^1.2.0", - "request": "^2.88.0" + "license": "MIT" + }, + "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, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" } }, - "cross-spawn": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.1.tgz", - "integrity": "sha512-u7v4o84SwFpD32Z8IIcPZ6z1/ie24O6RU3RbtL5Y316l3KuHVPx9ItBgWQ6VlfAFnRnTtMUrsQ9MUUTuEZjogg==", + "node_modules/@napi-rs/wasm-runtime": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@napi-rs/wasm-runtime/-/wasm-runtime-1.1.4.tgz", + "integrity": "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow==", "dev": true, - "requires": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" - }, + "license": "MIT", + "optional": true, "dependencies": { - "which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", - "dev": true, - "requires": { - "isexe": "^2.0.0" - } - } + "@tybys/wasm-util": "^0.10.1" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Brooooooklyn" + }, + "peerDependencies": { + "@emnapi/core": "^1.7.1", + "@emnapi/runtime": "^1.7.1" } }, - "dashdash": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", - "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", + "node_modules/@oxc-project/types": { + "version": "0.133.0", + "resolved": "https://registry.npmjs.org/@oxc-project/types/-/types-0.133.0.tgz", + "integrity": "sha512-KzkdCd6Uxqnf6l3HOw1xfatAlUURA0g14cvBYFyJ5SaNOQbOUvBr9PKArcPcrNIeRsBdgcUzOGrhKveVpvOIGA==", "dev": true, - "requires": { - "assert-plus": "^1.0.0" + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/Boshen" } }, - "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/@rolldown/binding-android-arm64": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-android-arm64/-/binding-android-arm64-1.0.3.tgz", + "integrity": "sha512-454rs7jHngixp/NMxd5srYD57OnzSlZ/eFTETjORQHLwJG1lRtmNOJcBerZlfu4GjKqeq8aCCIQrMdHyhI51Hw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-darwin-arm64": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-arm64/-/binding-darwin-arm64-1.0.3.tgz", + "integrity": "sha512-PcAhP+ynjURNyy8SKGl5DQP94aGuB/7JrXJb/t7P+hanXvQVMWzUvRRhBAcg/lNRadBhoUPqSoP4xw5tR/KBEA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-darwin-x64": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-x64/-/binding-darwin-x64-1.0.3.tgz", + "integrity": "sha512-9YpfeUvSE2RS7wysJ81uOZkXJz7f7Q55H2Gvp3VEw/EsahqDtrphrZ0EwDLK5vvKOzaCrBsjF8JmnMLcUt78Gg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-freebsd-x64": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-freebsd-x64/-/binding-freebsd-x64-1.0.3.tgz", + "integrity": "sha512-yB1IlAsSNHncV6SCTL27/MVGR5htvQsoGxIv5KMGXALp+Ll1wYsn+x98M9MW7qa+NdSbvrrY7ANI4wLJ0n1e6g==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-arm-gnueabihf": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm-gnueabihf/-/binding-linux-arm-gnueabihf-1.0.3.tgz", + "integrity": "sha512-Yi30IVAAfLUCy2MseFjbB1jAMDl1VMCAas5StnYp8da9+CKvMd2H2cbEjWcw5NPaPqzvYkVIaF1nNUG+b7u/sw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-arm64-gnu": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-gnu/-/binding-linux-arm64-gnu-1.0.3.tgz", + "integrity": "sha512-jsO7R8To+AdlYgUmN5sHSCZbfhtMBkO0WUx8iORQnPcMMdgr7qM2DQmMwgabs3GhNztdmoKkMKQFHD6DTMCIQw==", + "cpu": [ + "arm64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-arm64-musl": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-musl/-/binding-linux-arm64-musl-1.0.3.tgz", + "integrity": "sha512-VWkUHwWriDciit80wleYwKILoR/KMvxh/IdwS/paX+ZgpuRpCrKLUdadJbc0NpBEiyhpYawsJ73j9aCvOH+f7Q==", + "cpu": [ + "arm64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-ppc64-gnu": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-ppc64-gnu/-/binding-linux-ppc64-gnu-1.0.3.tgz", + "integrity": "sha512-5f1laC0SlIR0yDbFCd8acUhvJIag6N3zC5P7oUPN6wX0aOma+uKJ0wBDH5aq7I1PVI2ttTlhJwzwRIBnLiSGEg==", + "cpu": [ + "ppc64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-s390x-gnu": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-s390x-gnu/-/binding-linux-s390x-gnu-1.0.3.tgz", + "integrity": "sha512-Iq4ko0r4XsgbrF/LunNgHtAGLRRVE2kXonAXQ/MV0mC6jQpMOhW1SvtZja2EhC/kd05++bP78dsqBeIQyYJ6Yg==", + "cpu": [ + "s390x" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-x64-gnu": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-gnu/-/binding-linux-x64-gnu-1.0.3.tgz", + "integrity": "sha512-B8m6tD5+/N5FeNQFbKlLA/2yVq9ycQP1SeedyEYYKWBNR3ZQbkvIUcNnDNM03lO1l5F2roiiFJGgvoLLyZXtSg==", + "cpu": [ + "x64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-linux-x64-musl": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-musl/-/binding-linux-x64-musl-1.0.3.tgz", + "integrity": "sha512-pSdpdUJHkuCxun9LE7jvgUB9qsRgaiyNNCX7m/AvHTcq67AiT/Yhoxvw5zPfhrM8k/BfP8ce/hMOpthKDpEUow==", + "cpu": [ + "x64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-openharmony-arm64": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-openharmony-arm64/-/binding-openharmony-arm64-1.0.3.tgz", + "integrity": "sha512-OXXS3RKJgX2uLwM+gYyuH5omcH8fL1LJs96pZGgtetVCahON57+d4SJHzTgZiOjxgGkSnpXpOsWuPDGAKAigEg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-wasm32-wasi": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-wasm32-wasi/-/binding-wasm32-wasi-1.0.3.tgz", + "integrity": "sha512-JTtb8BWFynicNSoPrehsCzBtOKjZ6jhMiPFEmOiuXg1Fl8dn2KHQob+GuPSGR0dryQa1PQJbzjF3dqO/whhjLg==", + "cpu": [ + "wasm32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/core": "1.10.0", + "@emnapi/runtime": "1.10.0", + "@napi-rs/wasm-runtime": "^1.1.4" + }, + "engines": { + "node": "^20.19.0 || >=22.12.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/@rolldown/binding-win32-arm64-msvc": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-arm64-msvc/-/binding-win32-arm64-msvc-1.0.3.tgz", + "integrity": "sha512-gEdFFEN70A/jxb2svrWsN3aDL7OUtmvlOy+6fa2jxG8K0wQ1ZbdeLGnidov6Yu5/733dI5ySfzFlQ/cb0bSz1g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-win32-x64-msvc": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-x64-msvc/-/binding-win32-x64-msvc-1.0.3.tgz", + "integrity": "sha512-eXB7CHuaQdqmJcc3koCNtNPmT/bj2gc999kUFgBxG8Ac0NdgXc4rkCHhqrgrhN3zddvvvrgzj1e90SuSfmyIXA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/pluginutils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.1.tgz", + "integrity": "sha512-2j9bGt5Jh8hj+vPtgzPtl72j0yRxHAyumoo6TNfAjsLB04UtpSvPbPcDcBMxz7n+9CYB0c1GxQFxYRg2jimqGw==", + "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/@standard-schema/spec": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@standard-schema/spec/-/spec-1.1.0.tgz", + "integrity": "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w==", "dev": true, - "requires": { - "type-detect": "^4.0.0" - } + "license": "MIT" }, - "default-require-extensions": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/default-require-extensions/-/default-require-extensions-3.0.0.tgz", - "integrity": "sha512-ek6DpXq/SCpvjhpFsLFRVtIxJCRw6fUR42lYMVZuUMK7n8eMz4Uh5clckdBjEpLhn/gEBZo7hDJnJcwdKLKQjg==", - "dev": true, - "requires": { - "strip-bom": "^4.0.0" - } - }, - "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==", - "dev": true, - "requires": { - "object-keys": "^1.0.12" - } - }, - "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=", - "dev": true, - "requires": { - "jsbn": "~0.1.0", - "safer-buffer": "^2.1.0" - } - }, - "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 - }, - "es-abstract": { - "version": "1.17.4", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.4.tgz", - "integrity": "sha512-Ae3um/gb8F0mui/jPL+QiqmglkUsaQf7FwBEHYIFkztkneosu9imhqHpBzQ3h1vit8t5iQ74t6PEVvphBZiuiQ==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.1.5", - "is-regex": "^1.0.5", - "object-inspect": "^1.7.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.0", - "string.prototype.trimleft": "^2.1.1", - "string.prototype.trimright": "^2.1.1" - } - }, - "es-to-primitive": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", - "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "node_modules/@tybys/wasm-util": { + "version": "0.10.2", + "resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.2.tgz", + "integrity": "sha512-RoBvJ2X0wuKlWFIjrwffGw1IqZHKQqzIchKaadZZfnNpsAYp2mM0h36JtPCjNDAHGgYez/15uMBpfGwchhiMgg==", "dev": true, - "requires": { - "is-callable": "^1.1.4", - "is-date-object": "^1.0.1", - "is-symbol": "^1.0.2" + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" } }, - "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 + "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, + "license": "MIT", + "dependencies": { + "@types/deep-eql": "*", + "assertion-error": "^2.0.1" + } }, - "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 + "node_modules/@types/debug": { + "version": "4.1.13", + "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.13.tgz", + "integrity": "sha512-KSVgmQmzMwPlmtljOomayoR89W4FynCAi3E8PPs7vmDVPe84hT+vGPKkJfThkmXs0x0jAaa9U8uW8bbfyS2fWw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/ms": "*" + } }, - "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" }, - "extend": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", - "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", - "dev": true + "node_modules/@types/esrecurse": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/@types/esrecurse/-/esrecurse-4.3.1.tgz", + "integrity": "sha512-xJBAbDifo5hpffDBuHl0Y8ywswbiAp/Wi7Y/GtAgSlZyIABppyurxVueOPE8LUQOxdlgi6Zqce7uoEpqNTeiUw==", + "dev": true, + "license": "MIT", + "peer": true }, - "extsprintf": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", - "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", - "dev": true + "node_modules/@types/estree": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.9.tgz", + "integrity": "sha512-GhdPgy1el4/ImP05X05Uw4cw2/M93BCUmnEvWZNStlCzEKME4Fkk+YpoA5OiHNQmoS7Cafb8Xa3Pya8m1Qrzeg==", + "dev": true, + "license": "MIT" }, - "fast-deep-equal": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.1.tgz", - "integrity": "sha512-8UEa58QDLauDNfpbrX55Q9jrGHThw2ZMdOky5Gl1CDtVeJDPVrG4Jxx1N8jw2gkWaff5UUuX1KJd+9zGe2B+ZA==", - "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", + "peer": true }, - "fast-json-stable-stringify": { + "node_modules/@types/ms": { "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", - "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", - "dev": true + "resolved": "https://registry.npmjs.org/@types/ms/-/ms-2.1.0.tgz", + "integrity": "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA==", + "dev": true, + "license": "MIT" }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "node_modules/@types/node": { + "version": "25.9.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-25.9.2.tgz", + "integrity": "sha512-G05zqtJhcDLb8uslf5EjCxXg9G1KQxiV8OS0R26IC//Eoyitzqe8z37I7cqvnZlrlSfgocQRfSn/AHBZJJFyGw==", "dev": true, - "requires": { - "to-regex-range": "^5.0.1" + "license": "MIT", + "dependencies": { + "undici-types": ">=7.24.0 <7.24.7" } }, - "find-cache-dir": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.0.tgz", - "integrity": "sha512-PtXtQb7IrD8O+h6Cq1dbpJH5NzD8+9keN1zZ0YlpDzl1PwXEJEBj6u1Xa92t1Hwluoozd9TNKul5Hi2iqpsWwg==", + "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": { - "commondir": "^1.0.1", - "make-dir": "^3.0.2", - "pkg-dir": "^4.1.0" + "license": "MIT", + "dependencies": { + "@types/node": "*" } }, - "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/@typescript-eslint/eslint-plugin": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.60.1.tgz", + "integrity": "sha512-JQ4S5GB0tfjO8BuJ4fcX+HodkzJjYBV+7OJ+wLygaX7OGQ7FudyHL4NSCA6ob+w3Yn+5MkKIozOwQhXeM7opVg==", "dev": true, - "requires": { - "locate-path": "^3.0.0" + "license": "MIT", + "dependencies": { + "@eslint-community/regexpp": "^4.12.2", + "@typescript-eslint/scope-manager": "8.60.1", + "@typescript-eslint/type-utils": "8.60.1", + "@typescript-eslint/utils": "8.60.1", + "@typescript-eslint/visitor-keys": "8.60.1", + "ignore": "^7.0.5", + "natural-compare": "^1.4.0", + "ts-api-utils": "^2.5.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.60.1", + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" } }, - "flat": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/flat/-/flat-4.1.0.tgz", - "integrity": "sha512-Px/TiLIznH7gEDlPXcUD4KnBusa6kR6ayRUVcnEAbreRIuhkqow/mun59BuRXwoYk7ZQOLW1ZM05ilIvK38hFw==", + "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": { - "is-buffer": "~2.0.3" + "license": "MIT", + "engines": { + "node": ">= 4" } }, - "foreground-child": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-2.0.0.tgz", - "integrity": "sha512-dCIq9FpEcyQyXKCkyzmlPTFNgrCzPudOe+mhvJU5zAtlBnGVy2yKxtfsxK2tQBThwq225jcvBjpw1Gr40uzZCA==", + "node_modules/@typescript-eslint/parser": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.60.1.tgz", + "integrity": "sha512-A0M6ua6H252bVjPvvtSgl2QA4+ET9S5Mtkb2GDyTxIhH/C4qDItT7RQNO5PhMC6NXGYXOR9dIalcDDgBKT7oFA==", "dev": true, - "requires": { - "cross-spawn": "^7.0.0", - "signal-exit": "^3.0.2" + "license": "MIT", + "dependencies": { + "@typescript-eslint/scope-manager": "8.60.1", + "@typescript-eslint/types": "8.60.1", + "@typescript-eslint/typescript-estree": "8.60.1", + "@typescript-eslint/visitor-keys": "8.60.1", + "debug": "^4.4.3" + }, + "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 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.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/project-service": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.60.1.tgz", + "integrity": "sha512-eXkTH2bxmXlqD1RnOPmLZ9ZM9D3VwSx04JOwBnP9RQ+yUA5a2Mu7SfW8uaV2Aon53NJzZlZYuX7tn91Izf+xaw==", "dev": true, - "requires": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.6", - "mime-types": "^2.1.12" + "license": "MIT", + "dependencies": { + "@typescript-eslint/tsconfig-utils": "^8.60.1", + "@typescript-eslint/types": "^8.60.1", + "debug": "^4.4.3" + }, + "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.1.0" } }, - "fromentries": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/fromentries/-/fromentries-1.2.0.tgz", - "integrity": "sha512-33X7H/wdfO99GdRLLgkjUrD4geAFdq/Uv0kl3HD4da6HDixd2GUg8Mw7dahLCV9r/EARkmtYBB6Tch4EEokFTQ==", - "dev": true - }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", - "dev": true - }, - "fsevents": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.2.tgz", - "integrity": "sha512-R4wDiBwZ0KzpgOWetKDug1FZcYhqYnUYKtfZYt4mD5SBz76q0KR4Q9o7GIPamsVPGmW3EYPPJ0dOOjvx32ldZA==", + "node_modules/@typescript-eslint/scope-manager": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.60.1.tgz", + "integrity": "sha512-gvI5OQoptnxQnchOirukCuQ55svJSTuD/4k5+pC267xyBtYry748R9/c3tYUzb/iE6RZfllRz2lVulLCHkTm4w==", "dev": true, - "optional": 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 - }, - "gensync": { - "version": "1.0.0-beta.1", - "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.1.tgz", - "integrity": "sha512-r8EC6NO1sngH/zdD9fiRDLdcgnbayXah+mLgManTaIZJqEC1MZstmnox8KpnI2/fxQwrp5OpCOYWLp4rBl4Jcg==", - "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 + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.60.1", + "@typescript-eslint/visitor-keys": "8.60.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } }, - "getpass": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", - "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", + "node_modules/@typescript-eslint/tsconfig-utils": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.60.1.tgz", + "integrity": "sha512-nh8w4qAteiKuZu3pSSzG/yGKpw0OlkrKnzFmbVRenKaD4qc+7i1GrmZaLVkr8rk4uipiPGMOW4YsM6WmKZ5CvA==", "dev": true, - "requires": { - "assert-plus": "^1.0.0" + "license": "MIT", + "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.1.0" } }, - "glob": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", - "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", - "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" - } - }, - "glob-parent": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.0.tgz", - "integrity": "sha512-qjtRgnIVmOfnKUE3NJAQEdk+lKrxfw8t5ke7SXtfMTHcjsBfOfWXCQfdb30zfDoZQ2IRSIiidmjtbHZPZ++Ihw==", - "dev": true, - "requires": { - "is-glob": "^4.0.1" - } - }, - "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.3", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.3.tgz", - "integrity": "sha512-a30VEBm4PEdx1dRB7MFK7BejejvCvBronbLjht+sHuGYj8PHs7M/5Z+rt5lw551vZ7yfTCj4Vuyy3mSJytDWRQ==", - "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 - }, - "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.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.60.1.tgz", + "integrity": "sha512-sdwTrpjosW7ANQYJ39ZBF1ZyEMEGVB2UsikrserVM/30a/F1dTLnu9bGxEdosugyu5caigjLrR2qiD11asjI1A==", "dev": true, - "requires": { - "ajv": "^6.5.5", - "har-schema": "^2.0.0" + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.60.1", + "@typescript-eslint/typescript-estree": "8.60.1", + "@typescript-eslint/utils": "8.60.1", + "debug": "^4.4.3", + "ts-api-utils": "^2.5.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 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.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.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.60.1.tgz", + "integrity": "sha512-4h0tY8ppCkdCzcrl2YM5M3my0xsE1Tf8om3owEu5oPWmXwkKRmk0j0LGDzYBGUcAlesEbxBhazqu/K4cu3Ug7w==", "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.1", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", - "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", - "dev": true - }, - "hasha": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/hasha/-/hasha-5.2.0.tgz", - "integrity": "sha512-2W+jKdQbAdSIrggA8Q35Br8qKadTrqCTC8+XZvBWepKDK6m9XkX6Iz1a2yh2KP01kzAR/dpuMeUnocoLYDcskw==", + "node_modules/@typescript-eslint/typescript-estree": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.60.1.tgz", + "integrity": "sha512-alpRkfG8hlVE5kdJW2GkfgDgXxold3e8e4l6EnmhRmRLbekgAPCCGDVD++sABy9FcgPFroq+uFcCSM1vR57Cew==", "dev": true, - "requires": { - "is-stream": "^2.0.0", - "type-fest": "^0.8.0" + "license": "MIT", + "dependencies": { + "@typescript-eslint/project-service": "8.60.1", + "@typescript-eslint/tsconfig-utils": "8.60.1", + "@typescript-eslint/types": "8.60.1", + "@typescript-eslint/visitor-keys": "8.60.1", + "debug": "^4.4.3", + "minimatch": "^10.2.2", + "semver": "^7.7.3", + "tinyglobby": "^0.2.15", + "ts-api-utils": "^2.5.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.1.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 - }, - "html-escaper": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.0.tgz", - "integrity": "sha512-a4u9BeERWGu/S8JiWEAQcdrg9v4QArtP9keViQjGMdff20fBdd8waotXaNmODqBe6uZ3Nafi7K/ho4gCQHV3Ig==", - "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/utils": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.60.1.tgz", + "integrity": "sha512-h2MPBLoNtjc3qZWfY3Tl51yPorQ2McHn8pJfcMNTcIvrrZrr90Ykffit0yjrPFWQcRcUxzH20+6OcVdW4yHtUg==", "dev": true, - "requires": { - "assert-plus": "^1.0.0", - "jsprim": "^1.2.2", - "sshpk": "^1.7.0" + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.9.1", + "@typescript-eslint/scope-manager": "8.60.1", + "@typescript-eslint/types": "8.60.1", + "@typescript-eslint/typescript-estree": "8.60.1" + }, + "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 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.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 - }, - "indent-string": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", - "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", - "dev": true - }, - "inflight": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "node_modules/@typescript-eslint/visitor-keys": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.60.1.tgz", + "integrity": "sha512-EbGRQg4FhrmwLodl+t3JNAnXHWVr9Vp+Zl1QBZVPY4ByfkzIT8cX3K6QWODHtkIZqqJVEWvhHSx3v5PDHsaQag==", "dev": true, - "requires": { - "once": "^1.3.0", - "wrappy": "1" + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.60.1", + "eslint-visitor-keys": "^5.0.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" } }, - "inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "dev": true - }, - "is-binary-path": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", - "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "node_modules/@vitest/coverage-v8": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@vitest/coverage-v8/-/coverage-v8-4.1.8.tgz", + "integrity": "sha512-lt3kovsyHwYe00wq4D1ti0Z974fWj4NLp6siqiyEufUpyFwK9Yhi7rBhac9JL5aA0zoMrJqc4vYPZRUnI7l7nw==", "dev": true, - "requires": { - "binary-extensions": "^2.0.0" + "license": "MIT", + "dependencies": { + "@bcoe/v8-coverage": "^1.0.2", + "@vitest/utils": "4.1.8", + "ast-v8-to-istanbul": "^1.0.0", + "istanbul-lib-coverage": "^3.2.2", + "istanbul-lib-report": "^3.0.1", + "istanbul-reports": "^3.2.0", + "magicast": "^0.5.2", + "obug": "^2.1.1", + "std-env": "^4.0.0-rc.1", + "tinyrainbow": "^3.1.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "@vitest/browser": "4.1.8", + "vitest": "4.1.8" + }, + "peerDependenciesMeta": { + "@vitest/browser": { + "optional": true + } } }, - "is-buffer": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.4.tgz", - "integrity": "sha512-Kq1rokWXOPXWuaMAqZiJW4XxsmD9zGx9q4aePabbn3qCRGedtH7Cm+zV8WETitMfu1wdh+Rvd6w5egwSngUX2A==", - "dev": true - }, - "is-callable": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.5.tgz", - "integrity": "sha512-ESKv5sMCJB2jnHTWZ3O5itG+O128Hsus4K4Qh1h2/cgn2vbgnLSVqfV46AeJA9D5EeeLa9w81KUXMtn34zhX+Q==", - "dev": true - }, - "is-date-object": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", - "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", - "dev": true - }, - "is-extglob": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", - "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 + "node_modules/@vitest/expect": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-4.1.8.tgz", + "integrity": "sha512-h3nDO677RDLEGlBxyQ5CW8RlMThSKSRLUePLOx09gNIWRL40edgA1GCZSZgf1W55MFAG6/Sw14KeaAnqv0NKdQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@standard-schema/spec": "^1.1.0", + "@types/chai": "^5.2.2", + "@vitest/spy": "4.1.8", + "@vitest/utils": "4.1.8", + "chai": "^6.2.2", + "tinyrainbow": "^3.1.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } }, - "is-glob": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", - "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", + "node_modules/@vitest/mocker": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-4.1.8.tgz", + "integrity": "sha512-LEiN/xe4OSIbKe9HQIp5OC24agGD9J5CnmMgsLohVVoOPWL9a2sBoR6VBx43jQZb7Kr1l4RCuyCJzcAa0+dojw==", "dev": true, - "requires": { - "is-extglob": "^2.1.1" + "license": "MIT", + "dependencies": { + "@vitest/spy": "4.1.8", + "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 || ^8.0.0" + }, + "peerDependenciesMeta": { + "msw": { + "optional": true + }, + "vite": { + "optional": true + } } }, - "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 + "node_modules/@vitest/pretty-format": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-4.1.8.tgz", + "integrity": "sha512-9GasEBxpZ1VYIpqHf/0+YGg121uSNwCKOJqIrTwWP/TB7DmFCiaBpNl3aPZzoLWfWkuqhbH8vJIVobZkvdo2cA==", + "dev": true, + "license": "MIT", + "dependencies": { + "tinyrainbow": "^3.1.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } }, - "is-regex": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.5.tgz", - "integrity": "sha512-vlKW17SNq44owv5AQR3Cq0bQPEb8+kF3UKZ2fiZNOWtztYE5i0CzCZxFDwO58qAOWtxdBRVO/V5Qin1wjCqFYQ==", + "node_modules/@vitest/runner": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-4.1.8.tgz", + "integrity": "sha512-EmVxeBAfMJvycdjd6Hm+RbFBbA9fKvo0Kx37hNpBYoYeavH3RNsBXWDooR1mgD52dCrxIIuP7UotpfiwOikvcg==", "dev": true, - "requires": { - "has": "^1.0.3" + "license": "MIT", + "dependencies": { + "@vitest/utils": "4.1.8", + "pathe": "^2.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" } }, - "is-stream": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", - "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==", - "dev": true + "node_modules/@vitest/snapshot": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-4.1.8.tgz", + "integrity": "sha512-acfZboRmAIf05DEKcBQy33VXojFJjtUdLyo7oOmV9kebb2xdU01UknNiPuPZoJZQyO7DF0gZdTGTpeAzET9QPQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "4.1.8", + "@vitest/utils": "4.1.8", + "magic-string": "^0.30.21", + "pathe": "^2.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } }, - "is-symbol": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", - "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", + "node_modules/@vitest/spy": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-4.1.8.tgz", + "integrity": "sha512-6EevtBp6OZOPF7bmz36HrGMeP3txgVSrgebWxHOafDXGkhIzfXK14f8KF6MuFfgXXUeHxmpD3BQxkV00/3s5mA==", "dev": true, - "requires": { - "has-symbols": "^1.0.1" + "license": "MIT", + "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 + "node_modules/@vitest/utils": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-4.1.8.tgz", + "integrity": "sha512-uOJamYALNhfJ6iolExyQM40yIQwDqYnkKtQ5VCiSe17E33H0aQ/u+1GlRuz4LZBk6Mm3sg90G9hEbmEt37C1Zg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "4.1.8", + "convert-source-map": "^2.0.0", + "tinyrainbow": "^3.1.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } }, - "is-windows": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", - "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", - "dev": true + "node_modules/acorn": { + "version": "8.16.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", + "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } }, - "isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", - "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", + "peer": true, + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } }, - "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 + "node_modules/ajv": { + "version": "6.15.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.15.0.tgz", + "integrity": "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "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" + } }, - "istanbul-lib-coverage": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.0.0.tgz", - "integrity": "sha512-UiUIqxMgRDET6eR+o5HbfRYP1l0hqkWOs7vNxC/mggutCMUIhWMm8gAHb8tHlyfD3/l6rlgNA5cKdDzEAf6hEg==", - "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" + } }, - "istanbul-lib-hook": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-hook/-/istanbul-lib-hook-3.0.0.tgz", - "integrity": "sha512-Pt/uge1Q9s+5VAZ+pCo16TYMWPBIl+oaNIjgLQxcX0itS6ueeaA+pEfThZpH8WxhFgCiEb8sAJY6MdUKgiIWaQ==", + "node_modules/ast-v8-to-istanbul": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/ast-v8-to-istanbul/-/ast-v8-to-istanbul-1.0.3.tgz", + "integrity": "sha512-jCMQ6ZylLPudp0CDfBmQBZUsrh1/8psbmu9ibeVWKuHWD0YrH9YABwlKu5kVEFoT0GCQQW9Z/SxfuEbbkGQCRg==", "dev": true, - "requires": { - "append-transform": "^2.0.0" + "license": "MIT", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.31", + "estree-walker": "^3.0.3", + "js-tokens": "^10.0.0" } }, - "istanbul-lib-instrument": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.1.tgz", - "integrity": "sha512-imIchxnodll7pvQBYOqUu88EufLCU56LMeFPZZM/fJZ1irYcYdqroaV+ACK1Ila8ls09iEYArp+nqyC6lW1Vfg==", - "dev": true, - "requires": { - "@babel/core": "^7.7.5", - "@babel/parser": "^7.7.5", - "@babel/template": "^7.7.4", - "@babel/traverse": "^7.7.4", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-coverage": "^3.0.0", - "semver": "^6.3.0" - }, + "node_modules/balanced-match": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", + "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "18 || 20 || >=22" + } + }, + "node_modules/brace-expansion": { + "version": "5.0.6", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.6.tgz", + "integrity": "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g==", + "dev": true, + "license": "MIT", "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 - } + "balanced-match": "^4.0.2" + }, + "engines": { + "node": "18 || 20 || >=22" } }, - "istanbul-lib-processinfo": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/istanbul-lib-processinfo/-/istanbul-lib-processinfo-2.0.2.tgz", - "integrity": "sha512-kOwpa7z9hme+IBPZMzQ5vdQj8srYgAtaRqeI48NGmAQ+/5yKiHLV0QbYqQpxsdEF0+w14SoB8YbnHKcXE2KnYw==", + "node_modules/chai": { + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/chai/-/chai-6.2.2.tgz", + "integrity": "sha512-NUPRluOfOiTKBKvWPtSD4PhFvWCqOi0BGStNWs57X9js7XGTprSmFoz5F0tWhR4WPjNeR9jXqdC7/UpSJTnlRg==", "dev": true, - "requires": { - "archy": "^1.0.0", - "cross-spawn": "^7.0.0", - "istanbul-lib-coverage": "^3.0.0-alpha.1", - "make-dir": "^3.0.0", - "p-map": "^3.0.0", - "rimraf": "^3.0.0", - "uuid": "^3.3.3" + "license": "MIT", + "engines": { + "node": ">=18" } }, - "istanbul-lib-report": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", - "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", + "node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", "dev": true, - "requires": { - "istanbul-lib-coverage": "^3.0.0", - "make-dir": "^3.0.0", - "supports-color": "^7.1.0" + "license": "MIT" + }, + "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, + "license": "MIT", + "peer": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" }, + "engines": { + "node": ">= 8" + } + }, + "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": { - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true - }, + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { "supports-color": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz", - "integrity": "sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g==", - "dev": true, - "requires": { - "has-flag": "^4.0.0" - } + "optional": true } } }, - "istanbul-lib-source-maps": { + "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", + "peer": true + }, + "node_modules/detect-libc": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", + "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=8" + } + }, + "node_modules/es-module-lexer": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-2.1.0.tgz", + "integrity": "sha512-n27zTYMjYu1aj4MjCWzSP7G9r75utsaoc8m61weK+W8JMBGGQybd43GstCXZ3WNmSFtGT9wi59qQTW6mhTR5LQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/escape-string-regexp": { "version": "4.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.0.tgz", - "integrity": "sha512-c16LpFRkR8vQXyHZ5nLpY35JZtzj1PQY1iZmesUbf1FZHbIupcWfjgOXBY9YHkLEQ6puz1u4Dgj6qmU/DisrZg==", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", "dev": true, - "requires": { - "debug": "^4.1.1", - "istanbul-lib-coverage": "^3.0.0", - "source-map": "^0.6.1" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=10" }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "10.4.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-10.4.1.tgz", + "integrity": "sha512-AyIKhnOBuOAdueD7RB3xB+YeAWScb9jHsJBgH2Hcde8InP5JYhqrRR6iTMHyTEwgENK54Cp44e4v8BwNhsuHuw==", + "dev": true, + "license": "MIT", + "peer": true, "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 + "@eslint-community/eslint-utils": "^4.8.0", + "@eslint-community/regexpp": "^4.12.2", + "@eslint/config-array": "^0.23.5", + "@eslint/config-helpers": "^0.6.0", + "@eslint/core": "^1.2.1", + "@eslint/plugin-kit": "^0.7.2", + "@humanfs/node": "^0.16.6", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.4.2", + "@types/estree": "^1.0.6", + "ajv": "^6.14.0", + "cross-spawn": "^7.0.6", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^9.1.2", + "eslint-visitor-keys": "^5.0.1", + "espree": "^11.2.0", + "esquery": "^1.7.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", + "minimatch": "^10.2.4", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "jiti": "*" + }, + "peerDependenciesMeta": { + "jiti": { + "optional": true } } }, - "istanbul-reports": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.0.0.tgz", - "integrity": "sha512-2osTcC8zcOSUkImzN2EWQta3Vdi4WjjKw99P2yWx5mLnigAM0Rd5uYFn1cf2i/Ois45GkNjaoTqc5CxgMSX80A==", + "node_modules/eslint-scope": { + "version": "9.1.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-9.1.2.tgz", + "integrity": "sha512-xS90H51cKw0jltxmvmHy2Iai1LIqrfbw57b79w/J7MfvDfkIkFZ+kj6zC3BjtUwh150HsSSdxXZcsuv72miDFQ==", "dev": true, - "requires": { - "html-escaper": "^2.0.0", - "istanbul-lib-report": "^3.0.0" + "license": "BSD-2-Clause", + "peer": true, + "dependencies": { + "@types/esrecurse": "^4.3.1", + "@types/estree": "^1.0.8", + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "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==", - "dev": true, - "requires": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" - } - }, - "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-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": { + "node_modules/eslint-visitor-keys": { "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 - }, - "json5": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.1.tgz", - "integrity": "sha512-l+3HXD0GEI3huGq1njuqtzYK8OYJyXMkOLtQ53pjWh89tvWS2h6l+1zMkYWqlb57+SiQodKZyvMEFb2X+KrFhQ==", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-5.0.1.tgz", + "integrity": "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA==", "dev": true, - "requires": { - "minimist": "^1.2.0" + "license": "Apache-2.0", + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "jsprim": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", - "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", + "node_modules/espree": { + "version": "11.2.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-11.2.0.tgz", + "integrity": "sha512-7p3DrVEIopW1B1avAGLuCSh1jubc01H2JHc8B4qqGblmg5gI9yumBgACjWo4JlIc04ufug4xJ3SQI8HkS/Rgzw==", "dev": true, - "requires": { - "assert-plus": "1.0.0", - "extsprintf": "1.3.0", - "json-schema": "0.2.3", - "verror": "1.10.0" + "license": "BSD-2-Clause", + "peer": true, + "dependencies": { + "acorn": "^8.16.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^5.0.1" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "lcov-parse": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/lcov-parse/-/lcov-parse-1.0.0.tgz", - "integrity": "sha1-6w1GtUER68VhrLTECO+TY73I9+A=", - "dev": true - }, - "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/esquery": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", + "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==", "dev": true, - "requires": { - "p-locate": "^3.0.0", - "path-exists": "^3.0.0" + "license": "BSD-3-Clause", + "peer": true, + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" } }, - "lodash": { - "version": "4.17.15", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", - "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", - "dev": true - }, - "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 - }, - "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/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, + "license": "BSD-2-Clause", + "peer": true, + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } }, - "log-symbols": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-3.0.0.tgz", - "integrity": "sha512-dSkNGuI7iG3mfvDzUuYZyvk5dD9ocYCYzNU6CYDE6+Xqd+gwme6Z00NS3dUh8mq/73HaEtT7m6W+yUPtU6BZnQ==", + "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": { - "chalk": "^2.4.2" + "license": "BSD-2-Clause", + "peer": true, + "engines": { + "node": ">=4.0" } }, - "make-dir": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.0.2.tgz", - "integrity": "sha512-rYKABKutXa6vXTXhoV18cBE7PaewPXHe/Bdq4v+ZLMhxbWApkFFplT0LcbMW+6BbjnQXzZ/sAvSE/JdguApG5w==", + "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": { - "semver": "^6.0.0" - }, + "license": "MIT", "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 - } + "@types/estree": "^1.0.0" } }, - "mime-db": { - "version": "1.43.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.43.0.tgz", - "integrity": "sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ==", - "dev": true - }, - "mime-types": { - "version": "2.1.26", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.26.tgz", - "integrity": "sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==", + "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": { - "mime-db": "1.43.0" + "license": "BSD-2-Clause", + "peer": true, + "engines": { + "node": ">=0.10.0" } }, - "minimatch": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", - "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "node_modules/expect-type": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.3.0.tgz", + "integrity": "sha512-knvyeauYhqjOYvQ66MznSMs83wmHrCycNEN6Ao+2AeYEfxUIkuiVxdEa1qlGEPK+We3n0THiDciYSsCcgW/DoA==", "dev": true, - "requires": { - "brace-expansion": "^1.1.7" + "license": "Apache-2.0", + "engines": { + "node": ">=12.0.0" } }, - "minimist": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", - "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", - "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", + "peer": true }, - "mkdirp": { - "version": "0.5.1", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", - "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", "dev": true, - "requires": { - "minimist": "0.0.8" + "license": "MIT", + "peer": 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", + "peer": true + }, + "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" }, - "dependencies": { - "minimist": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", - "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", - "dev": true + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true } } }, - "mocha": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/mocha/-/mocha-7.1.0.tgz", - "integrity": "sha512-MymHK8UkU0K15Q/zX7uflZgVoRWiTjy0fXE/QjKts6mowUvGxOdPhZ2qj3b0iZdUrNZlW9LAIMFHB4IW+2b3EQ==", - "dev": true, - "requires": { - "ansi-colors": "3.2.3", - "browser-stdout": "1.3.1", - "chokidar": "3.3.0", - "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": "3.0.0", - "minimatch": "3.0.4", - "mkdirp": "0.5.1", - "ms": "2.1.1", - "node-environment-flags": "1.0.6", - "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.3.0", - "yargs-parser": "13.1.1", - "yargs-unparser": "1.6.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/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, + "license": "MIT", + "peer": true, + "dependencies": { + "flat-cache": "^4.0.0" + }, + "engines": { + "node": ">=16.0.0" } }, - "ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" - }, - "node-environment-flags": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/node-environment-flags/-/node-environment-flags-1.0.6.tgz", - "integrity": "sha512-5Evy2epuL+6TM0lCQGpFIj6KwiEsGh1SrHUhTbNX+sLbBtjidPZFAnVK9y5yU1+h//RitLbRHTIMyxQPtxMdHw==", + "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": { - "object.getownpropertydescriptors": "^2.0.3", - "semver": "^5.7.0" + "license": "MIT", + "peer": true, + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "node-preload": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/node-preload/-/node-preload-0.2.1.tgz", - "integrity": "sha512-RM5oyBy45cLEoHqCeh+MNuFAxO0vTFBLskvQbOKnEE7YTTSN4tbN8QWDIPQ6L+WvKsB/qLEGpYe2ZZ9d4W9OIQ==", + "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": { - "process-on-spawn": "^1.0.0" + "license": "MIT", + "peer": true, + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.4" + }, + "engines": { + "node": ">=16" } }, - "normalize-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", - "dev": true - }, - "nyc": { - "version": "15.0.0", - "resolved": "https://registry.npmjs.org/nyc/-/nyc-15.0.0.tgz", - "integrity": "sha512-qcLBlNCKMDVuKb7d1fpxjPR8sHeMVX0CHarXAVzrVWoFrigCkYR8xcrjfXSPi5HXM7EU78L6ywO7w1c5rZNCNg==", - "dev": true, - "requires": { - "@istanbuljs/load-nyc-config": "^1.0.0", - "@istanbuljs/schema": "^0.1.2", - "caching-transform": "^4.0.0", - "convert-source-map": "^1.7.0", - "decamelize": "^1.2.0", - "find-cache-dir": "^3.2.0", - "find-up": "^4.1.0", - "foreground-child": "^2.0.0", - "glob": "^7.1.6", - "istanbul-lib-coverage": "^3.0.0", - "istanbul-lib-hook": "^3.0.0", - "istanbul-lib-instrument": "^4.0.0", - "istanbul-lib-processinfo": "^2.0.2", - "istanbul-lib-report": "^3.0.0", - "istanbul-lib-source-maps": "^4.0.0", - "istanbul-reports": "^3.0.0", - "js-yaml": "^3.13.1", - "make-dir": "^3.0.0", - "node-preload": "^0.2.0", - "p-map": "^3.0.0", - "process-on-spawn": "^1.0.0", - "resolve-from": "^5.0.0", - "rimraf": "^3.0.0", - "signal-exit": "^3.0.2", - "spawn-wrap": "^2.0.0", - "test-exclude": "^6.0.0", - "uuid": "^3.3.3", - "yargs": "^15.0.2" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "ansi-styles": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.2.1.tgz", - "integrity": "sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA==", - "dev": true, - "requires": { - "@types/color-name": "^1.1.1", - "color-convert": "^2.0.1" - } - }, - "cliui": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", - "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", - "dev": true, - "requires": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.0", - "wrap-ansi": "^6.2.0" - } - }, - "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": { - "color-name": "~1.1.4" - } - }, - "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 - }, - "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 - }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, - "glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", - "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" - } - }, - "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 - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.0" - } - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.0" - } - }, - "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 - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "dev": true, - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - }, - "wrap-ansi": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", - "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", - "dev": true, - "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - } - }, - "yargs": { - "version": "15.1.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.1.0.tgz", - "integrity": "sha512-T39FNN1b6hCW4SOIk1XyTOWxtXdcen0t+XYrysQmChzSipvhBO8Bj0nK1ozAasdk24dNWuMZvr4k24nz+8HHLg==", - "dev": true, - "requires": { - "cliui": "^6.0.0", - "decamelize": "^1.2.0", - "find-up": "^4.1.0", - "get-caller-file": "^2.0.1", - "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", - "string-width": "^4.2.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^16.1.0" - } - }, - "yargs-parser": { - "version": "16.1.0", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-16.1.0.tgz", - "integrity": "sha512-H/V41UNZQPkUMIT5h5hiwg4QKIY1RPvoBV4XcjUbRM8Bk2oKqqyZ0DIEbTFZB0XjbtSPG8SAa/0DxCQmiRgzKg==", - "dev": true, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } - } - } + "node_modules/flatted": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.4.2.tgz", + "integrity": "sha512-PjDse7RzhcPkIJwy5t7KPWQSZ9cAbzQXcafsetQoD7sOJRQlGikNbx7yZp2OotDnJyrDcbyRq3Ttb18iYOqkxA==", + "dev": true, + "license": "ISC", + "peer": 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 + "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, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } }, - "object-inspect": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz", - "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==", - "dev": true + "node_modules/glob": { + "version": "13.0.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-13.0.6.tgz", + "integrity": "sha512-Wjlyrolmm8uDpm/ogGyXZXb1Z+Ca2B8NbJwqBVg0axK9GbBeoS7yGV6vjXnYdGm6X53iehEuxxbyiKp8QmN4Vw==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "minimatch": "^10.2.2", + "minipass": "^7.1.3", + "path-scurry": "^2.0.2" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } }, - "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 + "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, + "license": "ISC", + "peer": true, + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } }, - "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/globals": { + "version": "17.6.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-17.6.0.tgz", + "integrity": "sha512-sepffkT8stwnIYbsMBpoCHJuJM5l98FUF2AnE07hfvE0m/qp3R586hw4jF4uadbhvg1ooIdzuu7CsfD2jzCaNA==", "dev": true, - "requires": { - "define-properties": "^1.1.2", - "function-bind": "^1.1.1", - "has-symbols": "^1.0.0", - "object-keys": "^1.0.11" + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "object.getownpropertydescriptors": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz", - "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==", + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1" + "license": "MIT", + "engines": { + "node": ">=8" } }, - "once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "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" + }, + "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, - "requires": { - "wrappy": "1" + "license": "MIT", + "peer": true, + "engines": { + "node": ">= 4" } }, - "p-limit": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.2.2.tgz", - "integrity": "sha512-WGR+xHecKTr7EbUEhyLSh5Dube9JtdiG78ufaeLxTgpudf/20KqyMioIUZJAezlTIi6evxuoUs9YXc11cU+yzQ==", + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", "dev": true, - "requires": { - "p-try": "^2.0.0" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=0.8.19" } }, - "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/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", "dev": true, - "requires": { - "p-limit": "^2.0.0" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=0.10.0" } }, - "p-map": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/p-map/-/p-map-3.0.0.tgz", - "integrity": "sha512-d3qXVTF/s+W+CdJ5A29wywV2n8CQQYahlgz2bFiA+4eVNJbHJodPZ+/gXwPGh0bOqA+j8S+6+ckmvLGPk1QpxQ==", + "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, - "requires": { - "aggregate-error": "^3.0.0" + "license": "MIT", + "peer": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.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 + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC", + "peer": true }, - "package-hash": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/package-hash/-/package-hash-4.0.0.tgz", - "integrity": "sha512-whdkPIooSu/bASggZ96BWVvZTRMOFxnyUG5PnTSGKoJE2gd5mbVNmR2Nj20QFzxYYgAXpoqC+AiXzl+UMRh7zQ==", + "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": { - "graceful-fs": "^4.1.15", - "hasha": "^5.0.0", - "lodash.flattendeep": "^4.4.0", - "release-zalgo": "^1.0.0" + "license": "BSD-3-Clause", + "engines": { + "node": ">=8" } }, - "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 + "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" + } }, - "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 + "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" + } }, - "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 + "node_modules/js-tokens": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-10.0.0.tgz", + "integrity": "sha512-lM/UBzQmfJRo9ABXbPWemivdCW8V2G8FHaHdypQaIy523snUjog0W71ayWXTjiR+ixeMyVHN2XcpnTd/liPg/Q==", + "dev": true, + "license": "MIT" + }, + "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", + "peer": 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/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, + "license": "MIT", + "peer": true }, - "pathval": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.0.tgz", - "integrity": "sha1-uULm1L3mUwBe9rcTYd74cn0GReA=", - "dev": true + "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", + "peer": true }, - "performance-now": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", - "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=", - "dev": true - }, - "picomatch": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.1.tgz", - "integrity": "sha512-ISBaA8xQNmwELC7eOjqFKMESB2VIqt4PPDD0nsS95b/9dZXvVKOlz9keMSnoGGKcOHXfTvDD6WMaRoSc9UuhRA==", - "dev": true - }, - "pkg-dir": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", - "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", - "dev": true, - "requires": { - "find-up": "^4.0.0" - }, - "dependencies": { - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.0" - } - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.0" - } - }, - "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 - } + "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", + "peer": true, + "dependencies": { + "json-buffer": "3.0.1" } }, - "process-on-spawn": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/process-on-spawn/-/process-on-spawn-1.0.0.tgz", - "integrity": "sha512-1WsPDsUSMmZH5LeMLegqkPDrsGgsWwk1Exipy2hvB0o/F0ASzbpIctSCcZIK1ykJvtTJULEH+20WOFjMvGnCTg==", + "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": { - "fromentries": "^1.2.0" + "license": "MIT", + "peer": true, + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" } }, - "psl": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/psl/-/psl-1.7.0.tgz", - "integrity": "sha512-5NsSEDv8zY70ScRnOTn7bK7eanl2MvFrOrS/R6x+dBt5g1ghnj9Zv90kO8GwT8gxcu2ANyFprnFYB85IogIJOQ==", - "dev": true - }, - "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 + "node_modules/lightningcss": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz", + "integrity": "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ==", + "dev": true, + "license": "MPL-2.0", + "dependencies": { + "detect-libc": "^2.0.3" + }, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + }, + "optionalDependencies": { + "lightningcss-android-arm64": "1.32.0", + "lightningcss-darwin-arm64": "1.32.0", + "lightningcss-darwin-x64": "1.32.0", + "lightningcss-freebsd-x64": "1.32.0", + "lightningcss-linux-arm-gnueabihf": "1.32.0", + "lightningcss-linux-arm64-gnu": "1.32.0", + "lightningcss-linux-arm64-musl": "1.32.0", + "lightningcss-linux-x64-gnu": "1.32.0", + "lightningcss-linux-x64-musl": "1.32.0", + "lightningcss-win32-arm64-msvc": "1.32.0", + "lightningcss-win32-x64-msvc": "1.32.0" + } + }, + "node_modules/lightningcss-android-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz", + "integrity": "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz", + "integrity": "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz", + "integrity": "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-freebsd-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz", + "integrity": "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm-gnueabihf": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz", + "integrity": "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz", + "integrity": "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz", + "integrity": "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg==", + "cpu": [ + "arm64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz", + "integrity": "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==", + "cpu": [ + "x64" + ], + "dev": true, + "libc": [ + "glibc" + ], + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz", + "integrity": "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==", + "cpu": [ + "x64" + ], + "dev": true, + "libc": [ + "musl" + ], + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-arm64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz", + "integrity": "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-x64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz", + "integrity": "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } }, - "readdirp": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.2.0.tgz", - "integrity": "sha512-crk4Qu3pmXwgxdSgGhgA/eXiJAPQiX4GMOZZMXnqKxHX7TaoL+3gQVo/WeuAiogr07DpnfjIMpXXa+PAIvwPGQ==", - "dev": true, - "requires": { - "picomatch": "^2.0.4" - } - }, - "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.2", - "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", - "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", - "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.3", - "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.5.0", - "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 + "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, + "license": "MIT", + "peer": true, + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } }, - "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 + "node_modules/lodash": { + "version": "4.18.1", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.18.1.tgz", + "integrity": "sha512-dMInicTPVE8d1e5otfwmmjlxkZoUpiVLwyeTdUsi/Caj/gfzzblBcCE5sRHV/AsjuCmxWrte2TNGSYuCeCq+0Q==", + "dev": true, + "license": "MIT" }, - "resolve": { - "version": "1.15.1", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.15.1.tgz", - "integrity": "sha512-84oo6ZTtoTUpjgNEr5SJyzQhzL72gaRodsSfyxC/AXRvwu0Yse9H8eF9IpGo7b8YetZhlI6v7ZQ6bKBFV/6S7w==", + "node_modules/lru-cache": { + "version": "11.5.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-11.5.1.tgz", + "integrity": "sha512-RPimw/7aMdv2oqRrxKwvZXcPfwBrn/JZ2xYcY9Hus/6LaS3VOAKVWKWgNLCFSiOm1ESXinjsDlidVU7JlnCN2A==", "dev": true, - "requires": { - "path-parse": "^1.0.6" + "license": "BlueOak-1.0.0", + "engines": { + "node": "20 || >=22" } }, - "resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "dev": true + "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" + } }, - "rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "node_modules/magicast": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/magicast/-/magicast-0.5.3.tgz", + "integrity": "sha512-pVKE4UdSQ7DvHzivsCIFx2BJn1mHG6KsyrFcaxFx6tONdneEuThrDx0Cj3AMg58KyN4pzYT+LHOotxDQDjNvkw==", "dev": true, - "requires": { - "glob": "^7.1.3" + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.29.3", + "@babel/types": "^7.29.0", + "source-map-js": "^1.2.1" } }, - "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 + "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" + } }, - "safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", - "dev": true + "node_modules/minimatch": { + "version": "10.2.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz", + "integrity": "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "brace-expansion": "^5.0.5" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true + "node_modules/minipass": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.3.tgz", + "integrity": "sha512-tEBHqDnIoM/1rXME1zgka9g6Q2lcoCkxHLuc7ODJ5BxbP5d4c2Z5cGgtXAku59200Cx7diuHTOYfSBD8n6mm8A==", + "dev": true, + "license": "BlueOak-1.0.0", + "engines": { + "node": ">=16 || 14 >=14.17" + } }, - "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 + "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" }, - "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==", + "node_modules/nanoid": { + "version": "3.3.12", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz", + "integrity": "sha512-ZB9RH/39qpq5Vu6Y+NmUaFhQR6pp+M2Xt76XBnEwDaGcVAqhlvxrl3B2bKS5D3NH3QR76v3aSrKaF/Kiy7lEtQ==", "dev": true, - "requires": { - "shebang-regex": "^3.0.0" + "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" } }, - "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 - }, - "signal-exit": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", - "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", - "dev": true + "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, + "license": "MIT" }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true + "node_modules/obug": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/obug/-/obug-2.1.2.tgz", + "integrity": "sha512-AWGB9WFcRXOQs48Z/udjI5ZcZMHXwX8XPByNpOydgcGsDLIzjGizhoMWJyKAWze7AVW/2W1i+/gPX4YtKe5cyg==", + "dev": true, + "funding": [ + "https://github.com/sponsors/sxzz", + "https://opencollective.com/debug" + ], + "license": "MIT", + "engines": { + "node": ">=12.20.0" + } }, - "spawn-wrap": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/spawn-wrap/-/spawn-wrap-2.0.0.tgz", - "integrity": "sha512-EeajNjfN9zMnULLwhZZQU3GWBoFNkbngTUPfaawT4RkMiviTxcX0qfhVbGey39mfctfDHkWtuecgQ8NJcyQWHg==", - "dev": true, - "requires": { - "foreground-child": "^2.0.0", - "is-windows": "^1.0.2", - "make-dir": "^3.0.0", - "rimraf": "^3.0.0", - "signal-exit": "^3.0.2", - "which": "^2.0.1" + "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", + "peer": true, + "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" + } + }, + "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", + "peer": true, "dependencies": { - "which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", - "dev": true, - "requires": { - "isexe": "^2.0.0" - } - } + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "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/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": { - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^4.0.0" + "license": "MIT", + "peer": true, + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "string.prototype.trimleft": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/string.prototype.trimleft/-/string.prototype.trimleft-2.1.1.tgz", - "integrity": "sha512-iu2AGd3PuP5Rp7x2kEZCrB2Nf41ehzh+goo8TV7z8/XDBbsvc6HQIlUl9RjkZ4oyrW1XM5UwlGl1oVEaDjg6Ag==", + "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/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": { - "define-properties": "^1.1.3", - "function-bind": "^1.1.1" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=8" } }, - "string.prototype.trimright": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/string.prototype.trimright/-/string.prototype.trimright-2.1.1.tgz", - "integrity": "sha512-qFvWL3/+QIgZXVmJBfpHmxLB7xsUXz6HsUmP8+5dRaC3Q7oKUv9Vo6aMCRZC1smrtyECFsIT30PqBJ1gTjAs+g==", + "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, - "requires": { - "define-properties": "^1.1.3", - "function-bind": "^1.1.1" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=8" } }, - "strip-ansi": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", - "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "node_modules/path-scurry": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-2.0.2.tgz", + "integrity": "sha512-3O/iVVsJAPsOnpwWIeD+d6z/7PmqApyQePUtCndjatj/9I5LylHvt5qluFaBT3I5h3r1ejfR056c+FCv+NnNXg==", "dev": true, - "requires": { - "ansi-regex": "^3.0.0" + "license": "BlueOak-1.0.0", + "dependencies": { + "lru-cache": "^11.0.0", + "minipass": "^7.1.2" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, - "strip-bom": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", - "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", - "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" }, - "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/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "dev": true, + "license": "ISC" }, - "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/picomatch": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", + "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", "dev": true, - "requires": { - "has-flag": "^3.0.0" + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" } }, - "test-exclude": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", - "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", - "dev": true, - "requires": { - "@istanbuljs/schema": "^0.1.2", - "glob": "^7.1.4", - "minimatch": "^3.0.4" - }, - "dependencies": { - "glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", - "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" - } + "node_modules/postcss": { + "version": "8.5.15", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.15.tgz", + "integrity": "sha512-FfR8sjd4em2T6fb3I2MwAJU7HWVMr9zba+enmQeeWFfCbm+UOC/0X4DS8XtpUTMwWMGbjKYP7xjfNekzyGmB3A==", + "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": { + "nanoid": "^3.3.12", + "picocolors": "^1.1.1", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" } }, - "to-fast-properties": { - "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 - }, - "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==", + "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, - "requires": { - "is-number": "^7.0.0" + "license": "MIT", + "peer": true, + "engines": { + "node": ">= 0.8.0" } }, - "tough-cookie": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", - "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", + "node_modules/prettier": { + "version": "3.8.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.8.3.tgz", + "integrity": "sha512-7igPTM53cGHMW8xWuVTydi2KO233VFiTNyF5hLJqpilHfmn8C8gPf+PS7dUT64YcXFbiMGZxS9pCSxL/Dxm/Jw==", "dev": true, - "requires": { - "psl": "^1.1.28", - "punycode": "^2.1.1" + "license": "MIT", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" } }, - "ts-mockito": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/ts-mockito/-/ts-mockito-2.5.0.tgz", - "integrity": "sha512-b3qUeMfghRq5k5jw3xNJcnU9RKhqKnRn0k9v9QkN+YpuawrFuMIiGwzFZCpdi5MHy26o7YPnK8gag2awURl3nA==", + "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, - "requires": { - "lodash": "^4.17.5" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=6" } }, - "tslib": { - "version": "1.11.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.11.1.tgz", - "integrity": "sha512-aZW88SY8kQbU7gpV19lN24LtXh/yD4ZZg6qieAJDDg+YBsJcSmLGK9QpnUjAKVG/xefmvJGd1WUmfpT/g6AJGA==", - "dev": true - }, - "tslint": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/tslint/-/tslint-6.0.0.tgz", - "integrity": "sha512-9nLya8GBtlFmmFMW7oXXwoXS1NkrccqTqAtwXzdPV9e2mqSEvCki6iHL/Fbzi5oqbugshzgGPk7KBb2qNP1DSA==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.0.0", - "builtin-modules": "^1.1.1", - "chalk": "^2.3.0", - "commander": "^2.12.1", - "diff": "^4.0.1", - "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.10.0", - "tsutils": "^2.29.0" - }, - "dependencies": { - "diff": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", - "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", - "dev": true - } + "node_modules/rimraf": { + "version": "6.1.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-6.1.3.tgz", + "integrity": "sha512-LKg+Cr2ZF61fkcaK1UdkH2yEBBKnYjTyWzTJT6KNPcSPaiT7HSdhtMXQuN5wkTX0Xu72KQ1l8S42rlmexS2hSA==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "glob": "^13.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" } }, - "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/rolldown": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.3.tgz", + "integrity": "sha512-i00lAJ2ks1BYr7rjNjKC7BcqAS7nVfiT3QX1SI5aY+AFHblCmaUf9OE9dbdzDvW6dJxbi2ZCZiy9v3CcwOiX3g==", "dev": true, - "requires": { - "immutable": "^3.8.2" + "license": "MIT", + "dependencies": { + "@oxc-project/types": "=0.133.0", + "@rolldown/pluginutils": "^1.0.0" + }, + "bin": { + "rolldown": "bin/cli.mjs" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "optionalDependencies": { + "@rolldown/binding-android-arm64": "1.0.3", + "@rolldown/binding-darwin-arm64": "1.0.3", + "@rolldown/binding-darwin-x64": "1.0.3", + "@rolldown/binding-freebsd-x64": "1.0.3", + "@rolldown/binding-linux-arm-gnueabihf": "1.0.3", + "@rolldown/binding-linux-arm64-gnu": "1.0.3", + "@rolldown/binding-linux-arm64-musl": "1.0.3", + "@rolldown/binding-linux-ppc64-gnu": "1.0.3", + "@rolldown/binding-linux-s390x-gnu": "1.0.3", + "@rolldown/binding-linux-x64-gnu": "1.0.3", + "@rolldown/binding-linux-x64-musl": "1.0.3", + "@rolldown/binding-openharmony-arm64": "1.0.3", + "@rolldown/binding-wasm32-wasi": "1.0.3", + "@rolldown/binding-win32-arm64-msvc": "1.0.3", + "@rolldown/binding-win32-x64-msvc": "1.0.3" + } + }, + "node_modules/semver": { + "version": "7.8.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.8.2.tgz", + "integrity": "sha512-c8jsqUZm3omBOI66G90z1Dyw5z622G8oLG+omfsHBJf3CWQTlOcwOjvOG6wtiNfW6anKm/eA39LMwMtMez2TiQ==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" } }, - "tsutils": { - "version": "2.29.0", - "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", - "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", + "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": { - "tslib": "^1.8.1" + "license": "MIT", + "peer": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" } }, - "tunnel-agent": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", - "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", + "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, - "requires": { - "safe-buffer": "^5.0.1" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=8" } }, - "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/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/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" + } }, - "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/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" }, - "type-fest": { - "version": "0.8.1", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", - "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", - "dev": true + "node_modules/std-env": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/std-env/-/std-env-4.1.0.tgz", + "integrity": "sha512-Rq7ybcX2RuC55r9oaPVEW7/xu3tj8u4GeBYHBWCychFtzMIr86A7e3PPEBPT37sHStKX3+TiX/Fr/ACmJLVlLQ==", + "dev": true, + "license": "MIT" }, - "typedarray-to-buffer": { - "version": "3.1.5", - "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", - "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", + "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, - "requires": { - "is-typedarray": "^1.0.0" + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" } }, - "typescript": { - "version": "3.8.2", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.8.2.tgz", - "integrity": "sha512-EgOVgL/4xfVrCMbhYKUQTdF37SQn4Iw73H5BgCrF1Abdun7Kwy/QZsE/ssAy0y4LxBbvua3PIbFsbRczWWnDdQ==", - "dev": true + "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" }, - "uWebSockets.js": { - "version": "github:uNetworking/uWebSockets.js#d867ae494981123760e2986cfaeaea24a4cf76b3", - "from": "github:uNetworking/uWebSockets.js#v17.2.0" + "node_modules/tinyexec": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-1.2.4.tgz", + "integrity": "sha512-SHf/r48b7vOrjve9PxJo3MN5v5yuyjHvdUcrQffT3WXMUfnGmHDVbC4k3sHJaJTgZCwpUplIaAo5ANtMyp3YHg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + } }, - "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/tinyglobby": { + "version": "0.2.17", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.17.tgz", + "integrity": "sha512-wXR/dYpcqKmfWpEdZjiKJOwCNFndD0DMnrW/cYjVGttEkBfVgcLFHoNrlj47mjOVic9yyNu65alsgF4NQyTa2g==", "dev": true, - "requires": { - "punycode": "^2.1.0" + "license": "MIT", + "dependencies": { + "fdir": "^6.5.0", + "picomatch": "^4.0.4" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" } }, - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true + "node_modules/tinyrainbow": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-3.1.0.tgz", + "integrity": "sha512-Bf+ILmBgretUrdJxzXM0SgXLZ3XfiaUuOj/IKQHuTXip+05Xn+uyEYdVg0kYDipTBcLrCVyUzAPz7QmArb0mmw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } }, - "verror": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", - "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", + "node_modules/ts-api-utils": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.5.0.tgz", + "integrity": "sha512-OJ/ibxhPlqrMM0UiNHJ/0CKQkoKF243/AEmplt3qpRgkW8VG7IfOS41h7V8TjITqdByHzrjcS/2si+y4lIh8NA==", "dev": true, - "requires": { - "assert-plus": "^1.0.0", - "core-util-is": "1.0.2", - "extsprintf": "^1.2.0" + "license": "MIT", + "engines": { + "node": ">=18.12" + }, + "peerDependencies": { + "typescript": ">=4.8.4" } }, - "which": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", - "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "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, - "requires": { - "isexe": "^2.0.0" + "license": "MIT", + "dependencies": { + "lodash": "^4.17.5" } }, - "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/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD", + "optional": true }, - "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/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, - "requires": { - "string-width": "^1.0.2 || 2" + "license": "MIT", + "peer": true, + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" } }, - "wrap-ansi": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", - "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", + "node_modules/typescript": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-6.0.3.tgz", + "integrity": "sha512-y2TvuxSZPDyQakkFRPZHKFm+KKVqIisdg9/CZwm9ftvKXLP8NRWj38/ODjNbr43SsoXqNuAisEf1GdCxqWcdBw==", "dev": true, - "requires": { - "ansi-styles": "^3.2.0", - "string-width": "^3.0.0", - "strip-ansi": "^5.0.0" + "license": "Apache-2.0", + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/typescript-eslint": { + "version": "8.60.1", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.60.1.tgz", + "integrity": "sha512-6m5hkkRAp8lKvhVpcprAIn5KkehQEh+47oHH2VGnExEh7dhNxXlg6GPAOIu6TxbVQxhebrJDvjl3020ooiWCMA==", + "dev": true, + "license": "MIT", "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 - }, - "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" - } - }, - "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" - } - } + "@typescript-eslint/eslint-plugin": "8.60.1", + "@typescript-eslint/parser": "8.60.1", + "@typescript-eslint/typescript-estree": "8.60.1", + "@typescript-eslint/utils": "8.60.1" + }, + "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 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" } }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", - "dev": true + "node_modules/undici-types": { + "version": "7.24.6", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.24.6.tgz", + "integrity": "sha512-WRNW+sJgj5OBN4/0JpHFqtqzhpbnV0GuB+OozA9gCL7a993SmU+1JBZCzLNxYsbMfIeDL+lTsphD5jN5N+n0zg==", + "dev": true, + "license": "MIT" }, - "write-file-atomic": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", - "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", + "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, - "requires": { - "imurmurhash": "^0.1.4", - "is-typedarray": "^1.0.0", - "signal-exit": "^3.0.2", - "typedarray-to-buffer": "^3.1.5" + "license": "BSD-2-Clause", + "peer": true, + "dependencies": { + "punycode": "^2.1.0" } }, - "ws": { - "version": "7.2.1", - "resolved": "https://registry.npmjs.org/ws/-/ws-7.2.1.tgz", - "integrity": "sha512-sucePNSafamSKoOqoNfBd8V0StlkzJKL2ZAhGQinCfNQ+oacw+Pk7lcdAElecBF2VkLNZRiIb5Oi1Q5lVUVt2A==", - "dev": true + "node_modules/uWebSockets.js": { + "version": "20.68.0", + "resolved": "git+ssh://git@github.com/uNetworking/uWebSockets.js.git#624987739d4da0acb628aaf2a10fc43e7ce27c5c", + "license": "Apache-2.0" }, - "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 - }, - "yargs": { - "version": "13.3.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.0.tgz", - "integrity": "sha512-2eehun/8ALW8TLoIl7MVaRUrg+yCnenu8B4kBlRxj3GJGDKU1Og7sMXPNm1BYyM1DOJmTZ4YeN/Nwxv+8XJsUA==", - "dev": true, - "requires": { - "cliui": "^5.0.0", - "find-up": "^3.0.0", - "get-caller-file": "^2.0.1", - "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.1.1" - }, - "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/vite": { + "version": "8.0.16", + "resolved": "https://registry.npmjs.org/vite/-/vite-8.0.16.tgz", + "integrity": "sha512-h9bXPmJichP5fLmVQo3PyaGSDE2n3aPuomeAlVRm0JLmt4rY6zmPKd59HYI4LNW8oTK7tlTsuC7l/m7awx9Jcw==", + "dev": true, + "license": "MIT", + "dependencies": { + "lightningcss": "^1.32.0", + "picomatch": "^4.0.4", + "postcss": "^8.5.15", + "rolldown": "1.0.3", + "tinyglobby": "^0.2.17" + }, + "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", + "@vitejs/devtools": "^0.1.18", + "esbuild": "^0.27.0 || ^0.28.0", + "jiti": ">=1.21.0", + "less": "^4.0.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 + }, + "@vitejs/devtools": { + "optional": true + }, + "esbuild": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { + "optional": true + }, + "sass": { + "optional": true + }, + "sass-embedded": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true + } + } + }, + "node_modules/vitest": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/vitest/-/vitest-4.1.8.tgz", + "integrity": "sha512-flY6ScbCIt9HThs+C5HS7jvGOB560DJtk/Z15IQROTA6zEy49Nh8T/dofWTQL+n3vswqn87sbJNiuqw1SDp5Ig==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/expect": "4.1.8", + "@vitest/mocker": "4.1.8", + "@vitest/pretty-format": "4.1.8", + "@vitest/runner": "4.1.8", + "@vitest/snapshot": "4.1.8", + "@vitest/spy": "4.1.8", + "@vitest/utils": "4.1.8", + "es-module-lexer": "^2.0.0", + "expect-type": "^1.3.0", + "magic-string": "^0.30.21", + "obug": "^2.1.1", + "pathe": "^2.0.3", + "picomatch": "^4.0.3", + "std-env": "^4.0.0-rc.1", + "tinybench": "^2.9.0", + "tinyexec": "^1.0.2", + "tinyglobby": "^0.2.15", + "tinyrainbow": "^3.1.0", + "vite": "^6.0.0 || ^7.0.0 || ^8.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": "*", + "@opentelemetry/api": "^1.9.0", + "@types/node": "^20.0.0 || ^22.0.0 || >=24.0.0", + "@vitest/browser-playwright": "4.1.8", + "@vitest/browser-preview": "4.1.8", + "@vitest/browser-webdriverio": "4.1.8", + "@vitest/coverage-istanbul": "4.1.8", + "@vitest/coverage-v8": "4.1.8", + "@vitest/ui": "4.1.8", + "happy-dom": "*", + "jsdom": "*", + "vite": "^6.0.0 || ^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "@edge-runtime/vm": { + "optional": true + }, + "@opentelemetry/api": { + "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" - } + "@types/node": { + "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" - } + "@vitest/browser-playwright": { + "optional": true + }, + "@vitest/browser-preview": { + "optional": true + }, + "@vitest/browser-webdriverio": { + "optional": true + }, + "@vitest/coverage-istanbul": { + "optional": true + }, + "@vitest/coverage-v8": { + "optional": true + }, + "@vitest/ui": { + "optional": true + }, + "happy-dom": { + "optional": true + }, + "jsdom": { + "optional": true + }, + "vite": { + "optional": false } } }, - "yargs-parser": { - "version": "13.1.1", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.1.tgz", - "integrity": "sha512-oVAVsHz6uFrg3XQheFII8ESO2ssAf9luWuAd6Wexsu4F3OtIW0o8IribPXYrD4WC24LWtPrJlGy87y5udK+dxQ==", + "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", + "peer": true, + "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, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" + "license": "MIT", + "peer": true, + "engines": { + "node": ">=0.10.0" } }, - "yargs-unparser": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-1.6.0.tgz", - "integrity": "sha512-W9tKgmSn0DpSatfri0nx52Joq5hVXgeLiqR/5G0sZNDoLZFOr/xjBUDcShCOGNsBnEMNo1KAMBkTej1Hm62HTw==", + "node_modules/ws": { + "version": "8.21.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.21.0.tgz", + "integrity": "sha512-Vsp28b7DRcimFQvrqu2Wek3z1iYxDCWqHYB8Qsnk/S4RfaCQzPGPyBNuVjJV3cd6UiKtUtp6sNM77gWvzcCH+g==", "dev": true, - "requires": { - "flat": "^4.1.0", - "lodash": "^4.17.15", - "yargs": "^13.3.0" + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "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", + "peer": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } } } diff --git a/package.json b/package.json index 3bfb2b1..650c9c7 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,38 @@ "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#v17.2.0" + "debug": "^4.4.3", + "uWebSockets.js": "github:uNetworking/uWebSockets.js#v20.68.0" }, "devDependencies": { - "@types/chai": "^4.2.9", - "@types/debug": "^4.1.5", - "@types/mocha": "^7.0.1", - "@types/node": "^13.7.6", - "@types/ws": "^7.2.2", - "chai": "^4.2.0", - "coveralls": "^3.0.9", - "mocha": "^7.1.0", - "nyc": "^15.0.0", - "rimraf": "^3.0.2", - "ts-mockito": "^2.5.0", - "tslint": "^6.0.0", - "tslint-sonarts": "^1.9.0", - "typescript": "^3.8.2", - "ws": "^7.2.1" + "@eslint/js": "^10.0.1", + "@types/debug": "^4.1.13", + "@types/node": "^25.9.2", + "@types/ws": "^8.18.1", + "@vitest/coverage-v8": "^4.1.8", + "globals": "^17.6.0", + "prettier": "^3.8.3", + "rimraf": "^6.1.3", + "ts-mockito": "^2.6.1", + "typescript": "^6.0.3", + "typescript-eslint": "^8.60.1", + "vitest": "^4.1.8", + "ws": "^8.21.0" } } 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..1c39043 --- /dev/null +++ b/src/fast-tracker.ts @@ -0,0 +1,871 @@ +/** + * 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; + completedCount: number; + peers: PeerContext[]; +} + +interface SwarmOnPeer { + swarm: Swarm; + swarmTime: number; + swarmIndex: number; // index of peer in swarm.peers for O(1) removals + isCompleted: boolean; + nextSwarmOnPeer?: SwarmOnPeer; +} + +interface PeerContext< + ConnectionContext, +> extends SwarmOnPeer { + peerId: string; + connection: ConnectionContext; + nextPeerOnConn?: PeerContext; +} + +type UnknownObject = Record; + +export type FastTrackerSettings = { + maxOffers: number; + announceInterval: number; +}; + +const MAX_POOL_SIZE = 100_000; + +// Pools are intentionally module-global: in worker-thread deployment (one tracker +// per thread) this allows objects from a destroyed tracker to be reused by the next. +// dispose() drains them if you need to reclaim memory. +const peerContextPool: PeerContext[] = []; +const swarmNodePool: SwarmOnPeer[] = []; + +const reusableAnnounceMessage = { + action: "announce" as const, + interval: 0, + info_hash: "", + complete: 0, + incomplete: 0, +}; + +const reusableOfferMessage = { + action: "announce" as const, + info_hash: "", + offer_id: undefined as unknown, + peer_id: "", + offer: { + type: "offer" as const, + sdp: undefined as unknown, + }, +}; + +function acquireSwarmNode( + swarm: Swarm, + time: number, + index: number, + isCompleted: boolean, + next?: SwarmOnPeer, +): SwarmOnPeer { + const node = swarmNodePool.pop() as + | SwarmOnPeer + | undefined; + if (node !== undefined) { + node.swarm = swarm; + node.swarmTime = time; + node.swarmIndex = index; + node.isCompleted = isCompleted; + node.nextSwarmOnPeer = next; + return node; + } + return { + swarm, + swarmTime: time, + swarmIndex: index, + isCompleted, + nextSwarmOnPeer: next, + }; +} + +function releaseSwarmNode( + node: SwarmOnPeer, +): void { + node.swarm = undefined as unknown as Swarm; + node.nextSwarmOnPeer = undefined; + if (swarmNodePool.length < MAX_POOL_SIZE * 2) { + swarmNodePool.push(node); + } +} + +function acquirePeerContext( + peerId: string, + connection: ConnectionContext, + swarm: Swarm, + swarmTime: number, + swarmIndex: number, + isCompleted: boolean, +): PeerContext { + const peer = peerContextPool.pop() as + | PeerContext + | undefined; + if (peer !== undefined) { + peer.peerId = peerId; + peer.connection = connection; + peer.swarm = swarm; + peer.swarmTime = swarmTime; + peer.swarmIndex = swarmIndex; + peer.isCompleted = isCompleted; + peer.nextSwarmOnPeer = undefined; + peer.nextPeerOnConn = undefined; + return peer; + } + return { + peerId, + connection, + swarm, + swarmTime, + swarmIndex, + isCompleted, + nextSwarmOnPeer: undefined, + nextPeerOnConn: undefined, + }; +} + +function releasePeerContext( + peer: PeerContext, +): void { + peer.peerId = ""; + peer.connection = undefined as unknown as ConnectionContext; + peer.swarm = undefined as unknown as Swarm; + peer.nextSwarmOnPeer = undefined; + peer.nextPeerOnConn = undefined; + if (peerContextPool.length < MAX_POOL_SIZE) { + peerContextPool.push(peer); + } +} + +function findSwarmOnPeer>( + peer: PeerContext, + swarm: Swarm, +): SwarmOnPeer | undefined { + let curr: SwarmOnPeer | undefined = peer; + while (curr !== undefined) { + if (curr.swarm === swarm) { + return curr; + } + curr = curr.nextSwarmOnPeer; + } + return undefined; +} + +function addSwarmToPeer>( + peer: PeerContext, + swarm: Swarm, + time: number, + index: number, + isCompleted: boolean, +): void { + const newNode = acquireSwarmNode( + swarm, + time, + index, + isCompleted, + peer.nextSwarmOnPeer, + ); + peer.nextSwarmOnPeer = newNode; +} + +// NOTE: Head-replacement logic (copying swarm/swarmTime/swarmIndex/isCompleted/nextSwarmOnPeer) +// is duplicated in startClearPeersInterval for in-place traversal mutation. +// If SwarmOnPeer fields change, update both locations. +function removeSwarmFromPeer>( + peer: PeerContext, + swarm: Swarm, +): void { + if (peer.swarm === swarm) { + const next = peer.nextSwarmOnPeer; + if (next !== undefined) { + peer.swarm = next.swarm; + peer.swarmTime = next.swarmTime; + peer.swarmIndex = next.swarmIndex; + peer.isCompleted = next.isCompleted; + peer.nextSwarmOnPeer = next.nextSwarmOnPeer; + releaseSwarmNode(next); + } else { + peer.swarmIndex = -1; + } + return; + } + + let prev: SwarmOnPeer = peer; + let curr = peer.nextSwarmOnPeer; + while (curr !== undefined) { + if (curr.swarm === swarm) { + prev.nextSwarmOnPeer = curr.nextSwarmOnPeer; + releaseSwarmNode(curr); + return; + } + prev = curr; + curr = curr.nextSwarmOnPeer; + } +} + +function addPeerToConnection>( + connectionPeers: Map>, + connection: ConnectionContext, + peer: PeerContext, +): void { + const head = connectionPeers.get(connection); + if (head !== undefined) { + peer.nextPeerOnConn = head; + } + connectionPeers.set(connection, peer); +} + +function removePeerFromConnection< + ConnectionContext extends Record, +>( + connectionPeers: Map>, + peer: PeerContext, +): void { + const head = connectionPeers.get(peer.connection); + if (head !== undefined) { + if (head === peer) { + if (peer.nextPeerOnConn !== undefined) { + connectionPeers.set(peer.connection, peer.nextPeerOnConn); + } else { + connectionPeers.delete(peer.connection); + } + peer.nextPeerOnConn = undefined; // break reference chain for GC + } else { + let prev = head; + let curr = head.nextPeerOnConn; + while (curr !== undefined) { + if (curr === peer) { + prev.nextPeerOnConn = curr.nextPeerOnConn; + curr.nextPeerOnConn = undefined; // break reference chain for GC + break; + } + prev = curr; + curr = curr.nextPeerOnConn; + } + } + } +} + +function peerHasSwarms>( + peer: PeerContext, +): boolean { + return peer.swarmIndex !== -1; +} + +export class FastTracker< + ConnectionContext extends Record, +> implements Tracker { + public readonly settings: FastTrackerSettings; + // Important: sendMessage MUST serialize or clone the `json` object synchronously + // before returning, because FastTracker reuses message objects for performance. + #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; + } + + readonly #connectionPeers = new Map< + ConnectionContext, + PeerContext + >(); + + #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, + completedCount: 0, + peers: [], + }; + + this.#swarms.set(infoHash, swarm); + } + + return swarm; + } + + private addPeerToSwarm( + swarm: Swarm, + peer: PeerContext, + isPeerCompleted: boolean, + ) { + swarm.peers.push(peer); + if (isPeerCompleted) { + swarm.completedCount++; + } + } + + private removePeerFromSwarm( + swarm: Swarm, + node: SwarmOnPeer, + ) { + if (node.isCompleted) { + swarm.completedCount--; + } + + const { swarmIndex } = node; + + // eslint-disable-next-line @typescript-eslint/no-non-null-assertion + const lastPeer = swarm.peers.pop()!; + if (swarmIndex < swarm.peers.length) { + swarm.peers[swarmIndex] = lastPeer; + const lastPeerNode = findSwarmOnPeer(lastPeer, swarm); + if (lastPeerNode !== undefined) { + lastPeerNode.swarmIndex = swarmIndex; + } + } + + 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, + node: SwarmOnPeer, + ) { + if (!node.isCompleted) { + node.isCompleted = true; + swarm.completedCount++; + } + } + + private startClearPeersInterval(): void { + this.#clearPeersInterval = setInterval(() => { + const now = performance.now(); + for (const peer of this.#peers.values()) { + let prev: SwarmOnPeer | undefined = undefined; + let curr: SwarmOnPeer | undefined = peer; + + while (curr !== undefined) { + const next: SwarmOnPeer | undefined = + curr.nextSwarmOnPeer; + if ( + now - curr.swarmTime > + this.settings.announceInterval * 2 * 1000 + ) { + this.removePeerFromSwarm(curr.swarm, curr); + + // Head-replacement: copy next node's fields into peer. + // NOTE: Mirrors removeSwarmFromPeer — keep both in sync if SwarmOnPeer fields change. + if (curr === peer) { + if (next !== undefined) { + peer.swarm = next.swarm; + peer.swarmTime = next.swarmTime; + peer.swarmIndex = next.swarmIndex; + peer.isCompleted = next.isCompleted; + peer.nextSwarmOnPeer = next.nextSwarmOnPeer; + releaseSwarmNode(next); + curr = peer; + continue; + } else { + peer.swarmIndex = -1; + } + } else if (prev !== undefined) { + prev.nextSwarmOnPeer = next; + releaseSwarmNode(curr); + } + } else { + prev = curr; + } + curr = next; + } + + if (!peerHasSwarms(peer)) { + this.removePeer(peer); + } + } + }, this.settings.announceInterval * 1000); + } + + private removePeer(peer: PeerContext) { + if (peer.swarmIndex !== -1) { + this.removePeerFromSwarm(peer.swarm, peer); + } + + let nextNode = peer.nextSwarmOnPeer; + peer.nextSwarmOnPeer = undefined; + while (nextNode !== undefined) { + const next: SwarmOnPeer | undefined = + nextNode.nextSwarmOnPeer; + if (nextNode.swarmIndex !== -1) { + this.removePeerFromSwarm(nextNode.swarm, nextNode); + } + releaseSwarmNode(nextNode); + nextNode = next; + } + + this.#peers.delete(peer.peerId); + + removePeerFromConnection(this.#connectionPeers, peer); + + this.#onRemovePeer?.(peer.peerId, peer.connection); + + releasePeerContext(peer); + } + + 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 + let peer: PeerContext | undefined = + this.#connectionPeers.get(connection); + if (peer !== undefined) { + this.#connectionPeers.delete(connection); + while (peer !== undefined) { + const next: PeerContext | undefined = + peer.nextPeerOnConn; + peer.nextPeerOnConn = undefined; // break reference chain for GC (pool may be full) + if (debugEnabled) { + debug( + "disconnect peer:", + Buffer.from(peer.peerId).toString("hex"), + "swarms:", + getPeerSwarmsInfoHashes(peer), + ); + } + this.removePeer(peer); + peer = next; + } + } + } + + private processAnnounce( + json: UnknownObject, + connection: ConnectionContext, + completed = false, + ): void { + const infoHash = json.info_hash as string; + const peerId = json.peer_id as string; + if (typeof peerId !== "string") { + throw new TrackerError("announce: peer_id field is missing or wrong"); + } + let swarm: Swarm | undefined; + const isPeerCompleted = completed || json.left === 0; + + let peer = this.#peers.get(peerId); + + if (peer !== undefined && peer.connection !== connection) { + // The peer previously was on a different connection + if (debugEnabled) { + debug( + "peer changed connection:", + Buffer.from(peer.peerId).toString("hex"), + ); + } + this.removePeer(peer); + peer = undefined; + } + + const now = performance.now(); + + if (peer === undefined) { + swarm = this.getOrCreateSwarm(infoHash); + + peer = acquirePeerContext( + peerId, + connection, + swarm, + now, + swarm.peers.length, + isPeerCompleted, + ); + + this.addPeerToSwarm(swarm, peer, isPeerCompleted); + + addPeerToConnection(this.#connectionPeers, connection, peer); + + this.#peers.set(peerId, peer); + } else { + swarm = this.getOrCreateSwarm(infoHash); + const node = findSwarmOnPeer(peer, swarm); + if (node === undefined) { + addSwarmToPeer(peer, swarm, now, swarm.peers.length, isPeerCompleted); + this.addPeerToSwarm(swarm, peer, isPeerCompleted); + } else { + node.swarmTime = now; + if (isPeerCompleted) { + this.setPeerCompletedInSwarm(swarm, node); + } + } + } + + const complete = swarm.completedCount; + + reusableAnnounceMessage.interval = this.settings.announceInterval; + reusableAnnounceMessage.info_hash = infoHash; + reusableAnnounceMessage.complete = complete; + reusableAnnounceMessage.incomplete = swarm.peers.length - complete; + + this.#sendMessage(reusableAnnounceMessage, 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 (!Array.isArray(offers)) { + 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 + let offerIdx = 0; + // eslint-disable-next-line @typescript-eslint/prefer-for-of + for (let i = 0; i < peers.length; i++) { + const toPeer = peers[i]; + if (toPeer !== peer) { + this.#sendMessage( + getSendOfferJson(offers[offerIdx++], 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 peerId = json.peer_id as string; + if (typeof toPeerId !== "string") { + throw new TrackerError("answer: to_peer_id field is missing or wrong"); + } + if (typeof peerId !== "string") { + throw new TrackerError("answer: peer_id field is missing or wrong"); + } + + const toPeer = this.#peers.get(toPeerId); + if (toPeer === undefined) { + throw new TrackerError("answer: to_peer_id is not in the swarm"); + } + + json.to_peer_id = undefined; + this.#sendMessage(json, toPeer.connection); + + if (debugEnabled) { + debug( + "answer: from peer", + Buffer.from(peerId).toString("hex"), + "to peer", + Buffer.from(toPeerId).toString("hex"), + ); + } + } + + private processStop(json: UnknownObject): void { + const peerId = json.peer_id as string; + if (typeof peerId !== "string") { + throw new TrackerError("stop: peer_id field is missing or wrong"); + } + const infoHash = json.info_hash as string; + + const swarm = this.#swarms.get(infoHash); + if (!swarm) return; + + const peer = this.#peers.get(peerId); + if (!peer) return; + + const node = findSwarmOnPeer(peer, swarm); + if (node === undefined) return; + + if (debugEnabled) { + debug( + "stop peer:", + Buffer.from(peer.peerId).toString("hex"), + "swarm:", + Buffer.from(infoHash).toString("hex"), + ); + } + + this.removePeerFromSwarm(swarm, node); + removeSwarmFromPeer(peer, swarm); + + if (!peerHasSwarms(peer)) { + 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.completedCount; + files[swarm.infoHash] = { + complete, + incomplete: swarm.peers.length - complete, + downloaded: complete, + }; + } + } else if (Array.isArray(infoHash)) { + for (const singleInfoHash of infoHash as unknown[]) { + const swarm = this.#swarms.get(singleInfoHash as string); + if (swarm !== undefined) { + const complete = swarm.completedCount; + 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.completedCount; + 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); + this.#swarms.clear(); + this.#peers.clear(); + this.#connectionPeers.clear(); + peerContextPool.length = 0; + swarmNodePool.length = 0; + reusableAnnounceMessage.info_hash = ""; + reusableOfferMessage.info_hash = ""; + reusableOfferMessage.offer_id = undefined; + reusableOfferMessage.peer_id = ""; + reusableOfferMessage.offer.sdp = undefined; + } +} + +function getPeerSwarmsInfoHashes( + peer: PeerContext>, +): string { + let result = ""; + let curr: SwarmOnPeer> | undefined = peer; + while (curr !== undefined) { + if (curr.swarmIndex !== -1) { + if (result !== "") { + result += ","; + } + result += Buffer.from(curr.swarm.infoHash).toString("hex"); + } + curr = curr.nextSwarmOnPeer; + } + return result; +} + +function getSendOfferJson( + offerItem: unknown, + fromPeerId: string, + infoHash: string, +) { + if (typeof offerItem !== "object" || offerItem === null) { + throw new TrackerError("announce: wrong offer item format"); + } + + const { offer } = offerItem as UnknownObject; + const offerId = (offerItem as UnknownObject).offer_id; + + if (typeof offer !== "object" || offer === null) { + throw new TrackerError("announce: wrong offer item field format"); + } + + reusableOfferMessage.info_hash = infoHash; + reusableOfferMessage.offer_id = offerId; + reusableOfferMessage.peer_id = fromPeerId; + reusableOfferMessage.offer.sdp = (offer as UnknownObject).sdp; + + return reusableOfferMessage; +} 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..29d8cd3 --- /dev/null +++ b/src/multi-worker-tracker/index.ts @@ -0,0 +1,241 @@ +/* +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?.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..f96cbdf --- /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]; + 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..b4ecc9a --- /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; + + 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..383c708 --- /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; + + 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..7f31b19 --- /dev/null +++ b/src/uws-tracker.ts @@ -0,0 +1,354 @@ +/** + * 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 Record< + string, + unknown + >; + } 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..f9c6282 100644 --- a/test/announce.test.ts +++ b/test/announce.test.ts @@ -14,454 +14,155 @@ * 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(2); + expect(tracker.swarms.get("swarm1")!.peers).to.include.members([ + peerContext0, + peerContext1, + ]); + 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..4b0ba94 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 type { Tracker } from "../../src/tracker.ts"; +import type { 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..c486cd7 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -1,24 +1,35 @@ { "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", + "rootDir": "./src", + "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 22cba58..0000000 --- a/tslint.json +++ /dev/null @@ -1,18 +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"]], - "array-type": [true, "array-simple"] - }, - "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"], + }, + }, +});