From 3f84b901ffa01b8f452ab4e4d7157734427b591d Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Sun, 18 Feb 2024 12:39:14 +0100 Subject: [PATCH 01/12] Upgrade dependencies --- water/package.json | 14 +- water/tsconfig.json | 2 +- water/yarn.lock | 3618 ++++++++++++++++++++----------------------- 3 files changed, 1653 insertions(+), 1981 deletions(-) diff --git a/water/package.json b/water/package.json index de4835e..171d522 100644 --- a/water/package.json +++ b/water/package.json @@ -14,14 +14,14 @@ "dist" ], "dependencies": { - "kafkajs": "^2.2.3" + "kafkajs": "^2.2.4" }, "devDependencies": { - "@types/node": "^18.11.18", - "@typescript-eslint/eslint-plugin": "^5.48.0", - "@typescript-eslint/parser": "^5.48.0", - "eslint": "^8.31.0", - "eslint-plugin-import": "^2.26.0", - "typescript": "^4.9.4" + "@types/node": "^20.11.19", + "@typescript-eslint/eslint-plugin": "^7.0.1", + "@typescript-eslint/parser": "^7.0.1", + "eslint": "^8.56.0", + "eslint-plugin-import": "^2.29.1", + "typescript": "^5.3.3" } } diff --git a/water/tsconfig.json b/water/tsconfig.json index c02576a..616c0e0 100644 --- a/water/tsconfig.json +++ b/water/tsconfig.json @@ -1,7 +1,7 @@ { "compilerOptions": { "target": "ES2022", - "module": "ES2022", + "module": "Node16", "lib": [ "ES2022" ], diff --git a/water/yarn.lock b/water/yarn.lock index d76cb00..0e717e3 100644 --- a/water/yarn.lock +++ b/water/yarn.lock @@ -1,1973 +1,1645 @@ -# This file is generated by running "yarn install" inside your project. -# Manual changes might be lost - proceed with caution! - -__metadata: - version: 6 - cacheKey: 8 - -"@beemobot/water@workspace:.": - version: 0.0.0-use.local - resolution: "@beemobot/water@workspace:." - dependencies: - "@types/node": ^18.11.18 - "@typescript-eslint/eslint-plugin": ^5.48.0 - "@typescript-eslint/parser": ^5.48.0 - eslint: ^8.31.0 - eslint-plugin-import: ^2.26.0 - kafkajs: ^2.2.3 - typescript: ^4.9.4 - languageName: unknown - linkType: soft - -"@eslint-community/eslint-utils@npm:^4.2.0": - version: 4.4.0 - resolution: "@eslint-community/eslint-utils@npm:4.4.0" - dependencies: - eslint-visitor-keys: ^3.3.0 - peerDependencies: - eslint: ^6.0.0 || ^7.0.0 || >=8.0.0 - checksum: cdfe3ae42b4f572cbfb46d20edafe6f36fc5fb52bf2d90875c58aefe226892b9677fef60820e2832caf864a326fe4fc225714c46e8389ccca04d5f9288aabd22 - languageName: node - linkType: hard - -"@eslint-community/regexpp@npm:^4.4.0": - version: 4.4.1 - resolution: "@eslint-community/regexpp@npm:4.4.1" - checksum: db97d8d08e784147b55ab0dda5892503c1a0eebad94d1c4646d89a94f02ca70b25f05d8e021fc05a075e7eb312e03e21f63d84f0b327719f8cf3bb64e66917cb - languageName: node - linkType: hard - -"@eslint/eslintrc@npm:^2.0.1": - version: 2.0.1 - resolution: "@eslint/eslintrc@npm:2.0.1" - dependencies: - ajv: ^6.12.4 - debug: ^4.3.2 - espree: ^9.5.0 - globals: ^13.19.0 - ignore: ^5.2.0 - import-fresh: ^3.2.1 - js-yaml: ^4.1.0 - minimatch: ^3.1.2 - strip-json-comments: ^3.1.1 - checksum: 56b9192a687a450db53a7b883daf9f0f447c43b3510189cf88808a7a2467c2a302a42a50f184cc6d5a9faf3d1df890a2ef0fd0d60b751f32a3e9dfea717c6b48 - languageName: node - linkType: hard - -"@eslint/js@npm:8.36.0": - version: 8.36.0 - resolution: "@eslint/js@npm:8.36.0" - checksum: b7d6b84b823c8c7784be390741196617565527b1f7c0977fde9455bfb57fd88f81c074a03dd878757d2c33fa29f24291e9ecbc1425710f067917324b55e1bf3a - languageName: node - linkType: hard - -"@humanwhocodes/config-array@npm:^0.11.8": - version: 0.11.8 - resolution: "@humanwhocodes/config-array@npm:0.11.8" - dependencies: - "@humanwhocodes/object-schema": ^1.2.1 - debug: ^4.1.1 - minimatch: ^3.0.5 - checksum: 0fd6b3c54f1674ce0a224df09b9c2f9846d20b9e54fabae1281ecfc04f2e6ad69bf19e1d6af6a28f88e8aa3990168b6cb9e1ef755868c3256a630605ec2cb1d3 - languageName: node - linkType: hard - -"@humanwhocodes/module-importer@npm:^1.0.1": - version: 1.0.1 - resolution: "@humanwhocodes/module-importer@npm:1.0.1" - checksum: 0fd22007db8034a2cdf2c764b140d37d9020bbfce8a49d3ec5c05290e77d4b0263b1b972b752df8c89e5eaa94073408f2b7d977aed131faf6cf396ebb5d7fb61 - languageName: node - linkType: hard - -"@humanwhocodes/object-schema@npm:^1.2.1": - version: 1.2.1 - resolution: "@humanwhocodes/object-schema@npm:1.2.1" - checksum: a824a1ec31591231e4bad5787641f59e9633827d0a2eaae131a288d33c9ef0290bd16fda8da6f7c0fcb014147865d12118df10db57f27f41e20da92369fcb3f1 - languageName: node - linkType: hard - -"@nodelib/fs.scandir@npm:2.1.5": - version: 2.1.5 - resolution: "@nodelib/fs.scandir@npm:2.1.5" - dependencies: - "@nodelib/fs.stat": 2.0.5 - run-parallel: ^1.1.9 - checksum: a970d595bd23c66c880e0ef1817791432dbb7acbb8d44b7e7d0e7a22f4521260d4a83f7f9fd61d44fda4610105577f8f58a60718105fb38352baed612fd79e59 - languageName: node - linkType: hard - -"@nodelib/fs.stat@npm:2.0.5, @nodelib/fs.stat@npm:^2.0.2": - version: 2.0.5 - resolution: "@nodelib/fs.stat@npm:2.0.5" - checksum: 012480b5ca9d97bff9261571dbbec7bbc6033f69cc92908bc1ecfad0792361a5a1994bc48674b9ef76419d056a03efadfce5a6cf6dbc0a36559571a7a483f6f0 - languageName: node - linkType: hard - -"@nodelib/fs.walk@npm:^1.2.3, @nodelib/fs.walk@npm:^1.2.8": - version: 1.2.8 - resolution: "@nodelib/fs.walk@npm:1.2.8" - dependencies: - "@nodelib/fs.scandir": 2.1.5 - fastq: ^1.6.0 - checksum: 190c643f156d8f8f277bf2a6078af1ffde1fd43f498f187c2db24d35b4b4b5785c02c7dc52e356497b9a1b65b13edc996de08de0b961c32844364da02986dc53 - languageName: node - linkType: hard - -"@types/json-schema@npm:^7.0.9": - version: 7.0.11 - resolution: "@types/json-schema@npm:7.0.11" - checksum: 527bddfe62db9012fccd7627794bd4c71beb77601861055d87e3ee464f2217c85fca7a4b56ae677478367bbd248dbde13553312b7d4dbc702a2f2bbf60c4018d - languageName: node - linkType: hard - -"@types/json5@npm:^0.0.29": - version: 0.0.29 - resolution: "@types/json5@npm:0.0.29" - checksum: e60b153664572116dfea673c5bda7778dbff150498f44f998e34b5886d8afc47f16799280e4b6e241c0472aef1bc36add771c569c68fc5125fc2ae519a3eb9ac - languageName: node - linkType: hard - -"@types/node@npm:^18.11.18": - version: 18.15.10 - resolution: "@types/node@npm:18.15.10" - checksum: 9aeae0b683eda82892def5315812bdee3f1a28c4898b7e70f8e2514564538b16c4dccbe8339c1266f8fc1d707a48f152689264a854f5ebc2eba5011e793612d9 - languageName: node - linkType: hard - -"@types/semver@npm:^7.3.12": - version: 7.3.13 - resolution: "@types/semver@npm:7.3.13" - checksum: 00c0724d54757c2f4bc60b5032fe91cda6410e48689633d5f35ece8a0a66445e3e57fa1d6e07eb780f792e82ac542948ec4d0b76eb3484297b79bd18b8cf1cb0 - languageName: node - linkType: hard - -"@typescript-eslint/eslint-plugin@npm:^5.48.0": - version: 5.56.0 - resolution: "@typescript-eslint/eslint-plugin@npm:5.56.0" - dependencies: - "@eslint-community/regexpp": ^4.4.0 - "@typescript-eslint/scope-manager": 5.56.0 - "@typescript-eslint/type-utils": 5.56.0 - "@typescript-eslint/utils": 5.56.0 - debug: ^4.3.4 - grapheme-splitter: ^1.0.4 - ignore: ^5.2.0 - natural-compare-lite: ^1.4.0 - semver: ^7.3.7 - tsutils: ^3.21.0 - peerDependencies: - "@typescript-eslint/parser": ^5.0.0 - eslint: ^6.0.0 || ^7.0.0 || ^8.0.0 - peerDependenciesMeta: - typescript: - optional: true - checksum: 2eed4a4ed8279950ad553252e8623e947ffdee39b0d677a13f6e4e2d863ea1cbc5d683ff189e55d0de6fd5a25afd72d3c3a9ab7ae417d5405a21ead907e1b154 - languageName: node - linkType: hard - -"@typescript-eslint/parser@npm:^5.48.0": - version: 5.56.0 - resolution: "@typescript-eslint/parser@npm:5.56.0" - dependencies: - "@typescript-eslint/scope-manager": 5.56.0 - "@typescript-eslint/types": 5.56.0 - "@typescript-eslint/typescript-estree": 5.56.0 - debug: ^4.3.4 - peerDependencies: - eslint: ^6.0.0 || ^7.0.0 || ^8.0.0 - peerDependenciesMeta: - typescript: - optional: true - checksum: eb25490290bd5e22f9c42603dedc0d2d8ee845553e3cf48ea377bd5dc22440d3463f8b84be637b6a2b37cd9ea56b21e4e43007a0a69998948d9c8965c03fe1aa - languageName: node - linkType: hard - -"@typescript-eslint/scope-manager@npm:5.56.0": - version: 5.56.0 - resolution: "@typescript-eslint/scope-manager@npm:5.56.0" - dependencies: - "@typescript-eslint/types": 5.56.0 - "@typescript-eslint/visitor-keys": 5.56.0 - checksum: bacac255ee52148cee6622be2811c0d7e25419058b89f1a11f4c1303faef4535a0a1237549f9556ec1d7a297c640ce4357183a1a8465d72e1393b7d8fb43874b - languageName: node - linkType: hard - -"@typescript-eslint/type-utils@npm:5.56.0": - version: 5.56.0 - resolution: "@typescript-eslint/type-utils@npm:5.56.0" - dependencies: - "@typescript-eslint/typescript-estree": 5.56.0 - "@typescript-eslint/utils": 5.56.0 - debug: ^4.3.4 - tsutils: ^3.21.0 - peerDependencies: - eslint: "*" - peerDependenciesMeta: - typescript: - optional: true - checksum: 3dd1fcfadad18790b900a3d90f6617904adb6b0e2bd1e1edb6ebf239e1399865ca9098647405385feb4252d8b2b4577883e6fd3ef8d00bdd521d6070972d486b - languageName: node - linkType: hard - -"@typescript-eslint/types@npm:5.56.0": - version: 5.56.0 - resolution: "@typescript-eslint/types@npm:5.56.0" - checksum: 82ca11553bbb1bbfcaf7e7760b03c0d898940238dc002552c21af3e58f7d482c64c3c6cf0666521aff2a1e7b4b58bb6e4d9a00b1e4998a16b5039f5d288d003a - languageName: node - linkType: hard - -"@typescript-eslint/typescript-estree@npm:5.56.0": - version: 5.56.0 - resolution: "@typescript-eslint/typescript-estree@npm:5.56.0" - dependencies: - "@typescript-eslint/types": 5.56.0 - "@typescript-eslint/visitor-keys": 5.56.0 - debug: ^4.3.4 - globby: ^11.1.0 - is-glob: ^4.0.3 - semver: ^7.3.7 - tsutils: ^3.21.0 - peerDependenciesMeta: - typescript: - optional: true - checksum: ec3e85201786aa9adddba7cb834a9f330a7f55c729ee9ccf847dbdc2f7437b760f3774152ccad6d0aa48d13fd78df766c880e3a7ca42e01a20aba0e1a1ed61c5 - languageName: node - linkType: hard - -"@typescript-eslint/utils@npm:5.56.0": - version: 5.56.0 - resolution: "@typescript-eslint/utils@npm:5.56.0" - dependencies: - "@eslint-community/eslint-utils": ^4.2.0 - "@types/json-schema": ^7.0.9 - "@types/semver": ^7.3.12 - "@typescript-eslint/scope-manager": 5.56.0 - "@typescript-eslint/types": 5.56.0 - "@typescript-eslint/typescript-estree": 5.56.0 - eslint-scope: ^5.1.1 - semver: ^7.3.7 - peerDependencies: - eslint: ^6.0.0 || ^7.0.0 || ^8.0.0 - checksum: 413e8d4bf7023ee5ba4f695b62e796a1f94930bb92fe5aa0cee58f63b9837116c23f618825a9c671f610e50f5630188b6059b4ed6b05a2a3336f01d8e977becb - languageName: node - linkType: hard - -"@typescript-eslint/visitor-keys@npm:5.56.0": - version: 5.56.0 - resolution: "@typescript-eslint/visitor-keys@npm:5.56.0" - dependencies: - "@typescript-eslint/types": 5.56.0 - eslint-visitor-keys: ^3.3.0 - checksum: 568fda40134e153d7befb59b55698f7919ba780d2d3431d8745feabf2e0fbb8aa7a02173b3c467dd20a0f6594e5248a1f82bb25d6c37827716d77452e86cad29 - languageName: node - linkType: hard - -"acorn-jsx@npm:^5.3.2": - version: 5.3.2 - resolution: "acorn-jsx@npm:5.3.2" - peerDependencies: - acorn: ^6.0.0 || ^7.0.0 || ^8.0.0 - checksum: c3d3b2a89c9a056b205b69530a37b972b404ee46ec8e5b341666f9513d3163e2a4f214a71f4dfc7370f5a9c07472d2fd1c11c91c3f03d093e37637d95da98950 - languageName: node - linkType: hard - -"acorn@npm:^8.8.0": - version: 8.8.2 - resolution: "acorn@npm:8.8.2" - bin: - acorn: bin/acorn - checksum: f790b99a1bf63ef160c967e23c46feea7787e531292bb827126334612c234ed489a0dc2c7ba33156416f0ffa8d25bf2b0fdb7f35c2ba60eb3e960572bece4001 - languageName: node - linkType: hard - -"ajv@npm:^6.10.0, ajv@npm:^6.12.4": - version: 6.12.6 - resolution: "ajv@npm:6.12.6" - 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 - checksum: 874972efe5c4202ab0a68379481fbd3d1b5d0a7bd6d3cc21d40d3536ebff3352a2a1fabb632d4fd2cc7fe4cbdcd5ed6782084c9bbf7f32a1536d18f9da5007d4 - languageName: node - linkType: hard - -"ansi-regex@npm:^5.0.1": - version: 5.0.1 - resolution: "ansi-regex@npm:5.0.1" - checksum: 2aa4bb54caf2d622f1afdad09441695af2a83aa3fe8b8afa581d205e57ed4261c183c4d3877cee25794443fde5876417d859c108078ab788d6af7e4fe52eb66b - languageName: node - linkType: hard - -"ansi-styles@npm:^4.1.0": - version: 4.3.0 - resolution: "ansi-styles@npm:4.3.0" - dependencies: - color-convert: ^2.0.1 - checksum: 513b44c3b2105dd14cc42a19271e80f386466c4be574bccf60b627432f9198571ebf4ab1e4c3ba17347658f4ee1711c163d574248c0c1cdc2d5917a0ad582ec4 - languageName: node - linkType: hard - -"argparse@npm:^2.0.1": - version: 2.0.1 - resolution: "argparse@npm:2.0.1" - checksum: 83644b56493e89a254bae05702abf3a1101b4fa4d0ca31df1c9985275a5a5bd47b3c27b7fa0b71098d41114d8ca000e6ed90cad764b306f8a503665e4d517ced - languageName: node - linkType: hard - -"array-buffer-byte-length@npm:^1.0.0": - version: 1.0.0 - resolution: "array-buffer-byte-length@npm:1.0.0" - dependencies: - call-bind: ^1.0.2 - is-array-buffer: ^3.0.1 - checksum: 044e101ce150f4804ad19c51d6c4d4cfa505c5b2577bd179256e4aa3f3f6a0a5e9874c78cd428ee566ac574c8a04d7ce21af9fe52e844abfdccb82b33035a7c3 - languageName: node - linkType: hard - -"array-includes@npm:^3.1.6": - version: 3.1.6 - resolution: "array-includes@npm:3.1.6" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.4 - es-abstract: ^1.20.4 - get-intrinsic: ^1.1.3 - is-string: ^1.0.7 - checksum: f22f8cd8ba8a6448d91eebdc69f04e4e55085d09232b5216ee2d476dab3ef59984e8d1889e662c6a0ed939dcb1b57fd05b2c0209c3370942fc41b752c82a2ca5 - languageName: node - linkType: hard - -"array-union@npm:^2.1.0": - version: 2.1.0 - resolution: "array-union@npm:2.1.0" - checksum: 5bee12395cba82da674931df6d0fea23c4aa4660cb3b338ced9f828782a65caa232573e6bf3968f23e0c5eb301764a382cef2f128b170a9dc59de0e36c39f98d - languageName: node - linkType: hard - -"array.prototype.flat@npm:^1.3.1": - version: 1.3.1 - resolution: "array.prototype.flat@npm:1.3.1" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.4 - es-abstract: ^1.20.4 - es-shim-unscopables: ^1.0.0 - checksum: 5a8415949df79bf6e01afd7e8839bbde5a3581300e8ad5d8449dea52639e9e59b26a467665622783697917b43bf39940a6e621877c7dd9b3d1c1f97484b9b88b - languageName: node - linkType: hard - -"array.prototype.flatmap@npm:^1.3.1": - version: 1.3.1 - resolution: "array.prototype.flatmap@npm:1.3.1" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.4 - es-abstract: ^1.20.4 - es-shim-unscopables: ^1.0.0 - checksum: 8c1c43a4995f12cf12523436da28515184c753807b3f0bc2ca6c075f71c470b099e2090cc67dba8e5280958fea401c1d0c59e1db0143272aef6cd1103921a987 - languageName: node - linkType: hard - -"available-typed-arrays@npm:^1.0.5": - version: 1.0.5 - resolution: "available-typed-arrays@npm:1.0.5" - checksum: 20eb47b3cefd7db027b9bbb993c658abd36d4edd3fe1060e83699a03ee275b0c9b216cc076ff3f2db29073225fb70e7613987af14269ac1fe2a19803ccc97f1a - languageName: node - linkType: hard - -"balanced-match@npm:^1.0.0": - version: 1.0.2 - resolution: "balanced-match@npm:1.0.2" - checksum: 9706c088a283058a8a99e0bf91b0a2f75497f185980d9ffa8b304de1d9e58ebda7c72c07ebf01dadedaac5b2907b2c6f566f660d62bd336c3468e960403b9d65 - languageName: node - linkType: hard - -"brace-expansion@npm:^1.1.7": - version: 1.1.11 - resolution: "brace-expansion@npm:1.1.11" - dependencies: - balanced-match: ^1.0.0 - concat-map: 0.0.1 - checksum: faf34a7bb0c3fcf4b59c7808bc5d2a96a40988addf2e7e09dfbb67a2251800e0d14cd2bfc1aa79174f2f5095c54ff27f46fb1289fe2d77dac755b5eb3434cc07 - languageName: node - linkType: hard - -"braces@npm:^3.0.2": - version: 3.0.2 - resolution: "braces@npm:3.0.2" - dependencies: - fill-range: ^7.0.1 - checksum: e2a8e769a863f3d4ee887b5fe21f63193a891c68b612ddb4b68d82d1b5f3ff9073af066c343e9867a393fe4c2555dcb33e89b937195feb9c1613d259edfcd459 - languageName: node - linkType: hard - -"call-bind@npm:^1.0.0, call-bind@npm:^1.0.2": - version: 1.0.2 - resolution: "call-bind@npm:1.0.2" - dependencies: - function-bind: ^1.1.1 - get-intrinsic: ^1.0.2 - checksum: f8e31de9d19988a4b80f3e704788c4a2d6b6f3d17cfec4f57dc29ced450c53a49270dc66bf0fbd693329ee948dd33e6c90a329519aef17474a4d961e8d6426b0 - languageName: node - linkType: hard - -"callsites@npm:^3.0.0": - version: 3.1.0 - resolution: "callsites@npm:3.1.0" - checksum: 072d17b6abb459c2ba96598918b55868af677154bec7e73d222ef95a8fdb9bbf7dae96a8421085cdad8cd190d86653b5b6dc55a4484f2e5b2e27d5e0c3fc15b3 - languageName: node - linkType: hard - -"chalk@npm:^4.0.0": - version: 4.1.2 - resolution: "chalk@npm:4.1.2" - dependencies: - ansi-styles: ^4.1.0 - supports-color: ^7.1.0 - checksum: fe75c9d5c76a7a98d45495b91b2172fa3b7a09e0cc9370e5c8feb1c567b85c4288e2b3fded7cfdd7359ac28d6b3844feb8b82b8686842e93d23c827c417e83fc - languageName: node - linkType: hard - -"color-convert@npm:^2.0.1": - version: 2.0.1 - resolution: "color-convert@npm:2.0.1" - dependencies: - color-name: ~1.1.4 - checksum: 79e6bdb9fd479a205c71d89574fccfb22bd9053bd98c6c4d870d65c132e5e904e6034978e55b43d69fcaa7433af2016ee203ce76eeba9cfa554b373e7f7db336 - languageName: node - linkType: hard - -"color-name@npm:~1.1.4": - version: 1.1.4 - resolution: "color-name@npm:1.1.4" - checksum: b0445859521eb4021cd0fb0cc1a75cecf67fceecae89b63f62b201cca8d345baf8b952c966862a9d9a2632987d4f6581f0ec8d957dfacece86f0a7919316f610 - languageName: node - linkType: hard - -"concat-map@npm:0.0.1": - version: 0.0.1 - resolution: "concat-map@npm:0.0.1" - checksum: 902a9f5d8967a3e2faf138d5cb784b9979bad2e6db5357c5b21c568df4ebe62bcb15108af1b2253744844eb964fc023fbd9afbbbb6ddd0bcc204c6fb5b7bf3af - languageName: node - linkType: hard - -"cross-spawn@npm:^7.0.2": - version: 7.0.3 - resolution: "cross-spawn@npm:7.0.3" - dependencies: - path-key: ^3.1.0 - shebang-command: ^2.0.0 - which: ^2.0.1 - checksum: 671cc7c7288c3a8406f3c69a3ae2fc85555c04169e9d611def9a675635472614f1c0ed0ef80955d5b6d4e724f6ced67f0ad1bb006c2ea643488fcfef994d7f52 - languageName: node - linkType: hard - -"debug@npm:^3.2.7": - version: 3.2.7 - resolution: "debug@npm:3.2.7" - dependencies: - ms: ^2.1.1 - checksum: b3d8c5940799914d30314b7c3304a43305fd0715581a919dacb8b3176d024a782062368405b47491516d2091d6462d4d11f2f4974a405048094f8bfebfa3071c - languageName: node - linkType: hard - -"debug@npm:^4.1.1, debug@npm:^4.3.2, debug@npm:^4.3.4": - version: 4.3.4 - resolution: "debug@npm:4.3.4" - dependencies: - ms: 2.1.2 - peerDependenciesMeta: - supports-color: - optional: true - checksum: 3dbad3f94ea64f34431a9cbf0bafb61853eda57bff2880036153438f50fb5a84f27683ba0d8e5426bf41a8c6ff03879488120cf5b3a761e77953169c0600a708 - languageName: node - linkType: hard - -"deep-is@npm:^0.1.3": - version: 0.1.4 - resolution: "deep-is@npm:0.1.4" - checksum: edb65dd0d7d1b9c40b2f50219aef30e116cedd6fc79290e740972c132c09106d2e80aa0bc8826673dd5a00222d4179c84b36a790eef63a4c4bca75a37ef90804 - languageName: node - linkType: hard - -"define-properties@npm:^1.1.3, define-properties@npm:^1.1.4": - version: 1.2.0 - resolution: "define-properties@npm:1.2.0" - dependencies: - has-property-descriptors: ^1.0.0 - object-keys: ^1.1.1 - checksum: e60aee6a19b102df4e2b1f301816804e81ab48bb91f00d0d935f269bf4b3f79c88b39e4f89eaa132890d23267335fd1140dfcd8d5ccd61031a0a2c41a54e33a6 - languageName: node - linkType: hard - -"dir-glob@npm:^3.0.1": - version: 3.0.1 - resolution: "dir-glob@npm:3.0.1" - dependencies: - path-type: ^4.0.0 - checksum: fa05e18324510d7283f55862f3161c6759a3f2f8dbce491a2fc14c8324c498286c54282c1f0e933cb930da8419b30679389499b919122952a4f8592362ef4615 - languageName: node - linkType: hard - -"doctrine@npm:^2.1.0": - version: 2.1.0 - resolution: "doctrine@npm:2.1.0" - dependencies: - esutils: ^2.0.2 - checksum: a45e277f7feaed309fe658ace1ff286c6e2002ac515af0aaf37145b8baa96e49899638c7cd47dccf84c3d32abfc113246625b3ac8f552d1046072adee13b0dc8 - languageName: node - linkType: hard - -"doctrine@npm:^3.0.0": - version: 3.0.0 - resolution: "doctrine@npm:3.0.0" - dependencies: - esutils: ^2.0.2 - checksum: fd7673ca77fe26cd5cba38d816bc72d641f500f1f9b25b83e8ce28827fe2da7ad583a8da26ab6af85f834138cf8dae9f69b0cd6ab925f52ddab1754db44d99ce - languageName: node - linkType: hard - -"es-abstract@npm:^1.19.0, es-abstract@npm:^1.20.4": - version: 1.21.2 - resolution: "es-abstract@npm:1.21.2" - dependencies: - array-buffer-byte-length: ^1.0.0 - available-typed-arrays: ^1.0.5 - call-bind: ^1.0.2 - es-set-tostringtag: ^2.0.1 - es-to-primitive: ^1.2.1 - function.prototype.name: ^1.1.5 - get-intrinsic: ^1.2.0 - get-symbol-description: ^1.0.0 - globalthis: ^1.0.3 - gopd: ^1.0.1 - has: ^1.0.3 - has-property-descriptors: ^1.0.0 - has-proto: ^1.0.1 - has-symbols: ^1.0.3 - internal-slot: ^1.0.5 - is-array-buffer: ^3.0.2 - is-callable: ^1.2.7 - is-negative-zero: ^2.0.2 - is-regex: ^1.1.4 - is-shared-array-buffer: ^1.0.2 - is-string: ^1.0.7 - is-typed-array: ^1.1.10 - is-weakref: ^1.0.2 - object-inspect: ^1.12.3 - object-keys: ^1.1.1 - object.assign: ^4.1.4 - regexp.prototype.flags: ^1.4.3 - safe-regex-test: ^1.0.0 - string.prototype.trim: ^1.2.7 - string.prototype.trimend: ^1.0.6 - string.prototype.trimstart: ^1.0.6 - typed-array-length: ^1.0.4 - unbox-primitive: ^1.0.2 - which-typed-array: ^1.1.9 - checksum: 037f55ee5e1cdf2e5edbab5524095a4f97144d95b94ea29e3611b77d852fd8c8a40e7ae7101fa6a759a9b9b1405f188c3c70928f2d3cd88d543a07fc0d5ad41a - languageName: node - linkType: hard - -"es-set-tostringtag@npm:^2.0.1": - version: 2.0.1 - resolution: "es-set-tostringtag@npm:2.0.1" - dependencies: - get-intrinsic: ^1.1.3 - has: ^1.0.3 - has-tostringtag: ^1.0.0 - checksum: ec416a12948cefb4b2a5932e62093a7cf36ddc3efd58d6c58ca7ae7064475ace556434b869b0bbeb0c365f1032a8ccd577211101234b69837ad83ad204fff884 - languageName: node - linkType: hard - -"es-shim-unscopables@npm:^1.0.0": - version: 1.0.0 - resolution: "es-shim-unscopables@npm:1.0.0" - dependencies: - has: ^1.0.3 - checksum: 83e95cadbb6ee44d3644dfad60dcad7929edbc42c85e66c3e99aefd68a3a5c5665f2686885cddb47dfeabfd77bd5ea5a7060f2092a955a729bbd8834f0d86fa1 - languageName: node - linkType: hard - -"es-to-primitive@npm:^1.2.1": - version: 1.2.1 - resolution: "es-to-primitive@npm:1.2.1" - dependencies: - is-callable: ^1.1.4 - is-date-object: ^1.0.1 - is-symbol: ^1.0.2 - checksum: 4ead6671a2c1402619bdd77f3503991232ca15e17e46222b0a41a5d81aebc8740a77822f5b3c965008e631153e9ef0580540007744521e72de8e33599fca2eed - languageName: node - linkType: hard - -"escape-string-regexp@npm:^4.0.0": - version: 4.0.0 - resolution: "escape-string-regexp@npm:4.0.0" - checksum: 98b48897d93060f2322108bf29db0feba7dd774be96cd069458d1453347b25ce8682ecc39859d4bca2203cc0ab19c237bcc71755eff49a0f8d90beadeeba5cc5 - languageName: node - linkType: hard - -"eslint-import-resolver-node@npm:^0.3.7": - version: 0.3.7 - resolution: "eslint-import-resolver-node@npm:0.3.7" - dependencies: - debug: ^3.2.7 - is-core-module: ^2.11.0 - resolve: ^1.22.1 - checksum: 3379aacf1d2c6952c1b9666c6fa5982c3023df695430b0d391c0029f6403a7775414873d90f397e98ba6245372b6c8960e16e74d9e4a3b0c0a4582f3bdbe3d6e - languageName: node - linkType: hard - -"eslint-module-utils@npm:^2.7.4": - version: 2.7.4 - resolution: "eslint-module-utils@npm:2.7.4" - dependencies: - debug: ^3.2.7 - peerDependenciesMeta: - eslint: - optional: true - checksum: 5da13645daff145a5c922896b258f8bba560722c3767254e458d894ff5fbb505d6dfd945bffa932a5b0ae06714da2379bd41011c4c20d2d59cc83e23895360f7 - languageName: node - linkType: hard - -"eslint-plugin-import@npm:^2.26.0": - version: 2.27.5 - resolution: "eslint-plugin-import@npm:2.27.5" - dependencies: - array-includes: ^3.1.6 - array.prototype.flat: ^1.3.1 - array.prototype.flatmap: ^1.3.1 - debug: ^3.2.7 - doctrine: ^2.1.0 - eslint-import-resolver-node: ^0.3.7 - eslint-module-utils: ^2.7.4 - has: ^1.0.3 - is-core-module: ^2.11.0 - is-glob: ^4.0.3 - minimatch: ^3.1.2 - object.values: ^1.1.6 - resolve: ^1.22.1 - semver: ^6.3.0 - tsconfig-paths: ^3.14.1 - peerDependencies: - eslint: ^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 - checksum: f500571a380167e25d72a4d925ef9a7aae8899eada57653e5f3051ec3d3c16d08271fcefe41a30a9a2f4fefc232f066253673ee4ea77b30dba65ae173dade85d - languageName: node - linkType: hard - -"eslint-scope@npm:^5.1.1": - version: 5.1.1 - resolution: "eslint-scope@npm:5.1.1" - dependencies: - esrecurse: ^4.3.0 - estraverse: ^4.1.1 - checksum: 47e4b6a3f0cc29c7feedee6c67b225a2da7e155802c6ea13bbef4ac6b9e10c66cd2dcb987867ef176292bf4e64eccc680a49e35e9e9c669f4a02bac17e86abdb - languageName: node - linkType: hard - -"eslint-scope@npm:^7.1.1": - version: 7.1.1 - resolution: "eslint-scope@npm:7.1.1" - dependencies: - esrecurse: ^4.3.0 - estraverse: ^5.2.0 - checksum: 9f6e974ab2db641ca8ab13508c405b7b859e72afe9f254e8131ff154d2f40c99ad4545ce326fd9fde3212ff29707102562a4834f1c48617b35d98c71a97fbf3e - languageName: node - linkType: hard - -"eslint-visitor-keys@npm:^3.3.0": - version: 3.3.0 - resolution: "eslint-visitor-keys@npm:3.3.0" - checksum: d59e68a7c5a6d0146526b0eec16ce87fbf97fe46b8281e0d41384224375c4e52f5ffb9e16d48f4ea50785cde93f766b0c898e31ab89978d88b0e1720fbfb7808 - languageName: node - linkType: hard - -"eslint@npm:^8.31.0": - version: 8.36.0 - resolution: "eslint@npm:8.36.0" - dependencies: - "@eslint-community/eslint-utils": ^4.2.0 - "@eslint-community/regexpp": ^4.4.0 - "@eslint/eslintrc": ^2.0.1 - "@eslint/js": 8.36.0 - "@humanwhocodes/config-array": ^0.11.8 - "@humanwhocodes/module-importer": ^1.0.1 - "@nodelib/fs.walk": ^1.2.8 - ajv: ^6.10.0 - chalk: ^4.0.0 - cross-spawn: ^7.0.2 - debug: ^4.3.2 - doctrine: ^3.0.0 - escape-string-regexp: ^4.0.0 - eslint-scope: ^7.1.1 - eslint-visitor-keys: ^3.3.0 - espree: ^9.5.0 - esquery: ^1.4.2 - esutils: ^2.0.2 - fast-deep-equal: ^3.1.3 - file-entry-cache: ^6.0.1 - find-up: ^5.0.0 - glob-parent: ^6.0.2 - globals: ^13.19.0 - grapheme-splitter: ^1.0.4 - ignore: ^5.2.0 - import-fresh: ^3.0.0 - imurmurhash: ^0.1.4 - is-glob: ^4.0.0 - is-path-inside: ^3.0.3 - js-sdsl: ^4.1.4 - js-yaml: ^4.1.0 - json-stable-stringify-without-jsonify: ^1.0.1 - levn: ^0.4.1 - lodash.merge: ^4.6.2 - minimatch: ^3.1.2 - natural-compare: ^1.4.0 - optionator: ^0.9.1 - strip-ansi: ^6.0.1 - strip-json-comments: ^3.1.0 - text-table: ^0.2.0 - bin: - eslint: bin/eslint.js - checksum: e9a961fc3b3de5cff5a1cb2c92eeffaa7e155a715489e30b3e1e76f186bd1255e0481e09564f2094733c0b1dbd3453499fb72ae7c043c83156e11e6d965b2304 - languageName: node - linkType: hard - -"espree@npm:^9.5.0": - version: 9.5.0 - resolution: "espree@npm:9.5.0" - dependencies: - acorn: ^8.8.0 - acorn-jsx: ^5.3.2 - eslint-visitor-keys: ^3.3.0 - checksum: a7f110aefb6407e0d3237aa635ab3cea87106ae63748dd23c67031afccc640d04c4209fca2daf16e2233c82efb505faead0fb84097478fd9cc6e8f8dd80bf99d - languageName: node - linkType: hard - -"esquery@npm:^1.4.2": - version: 1.5.0 - resolution: "esquery@npm:1.5.0" - dependencies: - estraverse: ^5.1.0 - checksum: aefb0d2596c230118656cd4ec7532d447333a410a48834d80ea648b1e7b5c9bc9ed8b5e33a89cb04e487b60d622f44cf5713bf4abed7c97343edefdc84a35900 - languageName: node - linkType: hard - -"esrecurse@npm:^4.3.0": - version: 4.3.0 - resolution: "esrecurse@npm:4.3.0" - dependencies: - estraverse: ^5.2.0 - checksum: ebc17b1a33c51cef46fdc28b958994b1dc43cd2e86237515cbc3b4e5d2be6a811b2315d0a1a4d9d340b6d2308b15322f5c8291059521cc5f4802f65e7ec32837 - languageName: node - linkType: hard - -"estraverse@npm:^4.1.1": - version: 4.3.0 - resolution: "estraverse@npm:4.3.0" - checksum: a6299491f9940bb246124a8d44b7b7a413a8336f5436f9837aaa9330209bd9ee8af7e91a654a3545aee9c54b3308e78ee360cef1d777d37cfef77d2fa33b5827 - languageName: node - linkType: hard - -"estraverse@npm:^5.1.0, estraverse@npm:^5.2.0": - version: 5.3.0 - resolution: "estraverse@npm:5.3.0" - checksum: 072780882dc8416ad144f8fe199628d2b3e7bbc9989d9ed43795d2c90309a2047e6bc5979d7e2322a341163d22cfad9e21f4110597fe487519697389497e4e2b - languageName: node - linkType: hard - -"esutils@npm:^2.0.2": - version: 2.0.3 - resolution: "esutils@npm:2.0.3" - checksum: 22b5b08f74737379a840b8ed2036a5fb35826c709ab000683b092d9054e5c2a82c27818f12604bfc2a9a76b90b6834ef081edbc1c7ae30d1627012e067c6ec87 - languageName: node - linkType: hard - -"fast-deep-equal@npm:^3.1.1, fast-deep-equal@npm:^3.1.3": - version: 3.1.3 - resolution: "fast-deep-equal@npm:3.1.3" - checksum: e21a9d8d84f53493b6aa15efc9cfd53dd5b714a1f23f67fb5dc8f574af80df889b3bce25dc081887c6d25457cce704e636395333abad896ccdec03abaf1f3f9d - languageName: node - linkType: hard - -"fast-glob@npm:^3.2.9": - version: 3.2.12 - resolution: "fast-glob@npm:3.2.12" - dependencies: - "@nodelib/fs.stat": ^2.0.2 - "@nodelib/fs.walk": ^1.2.3 - glob-parent: ^5.1.2 - merge2: ^1.3.0 - micromatch: ^4.0.4 - checksum: 0b1990f6ce831c7e28c4d505edcdaad8e27e88ab9fa65eedadb730438cfc7cde4910d6c975d6b7b8dc8a73da4773702ebcfcd6e3518e73938bb1383badfe01c2 - languageName: node - linkType: hard - -"fast-json-stable-stringify@npm:^2.0.0": - version: 2.1.0 - resolution: "fast-json-stable-stringify@npm:2.1.0" - checksum: b191531e36c607977e5b1c47811158733c34ccb3bfde92c44798929e9b4154884378536d26ad90dfecd32e1ffc09c545d23535ad91b3161a27ddbb8ebe0cbecb - languageName: node - linkType: hard - -"fast-levenshtein@npm:^2.0.6": - version: 2.0.6 - resolution: "fast-levenshtein@npm:2.0.6" - checksum: 92cfec0a8dfafd9c7a15fba8f2cc29cd0b62b85f056d99ce448bbcd9f708e18ab2764bda4dd5158364f4145a7c72788538994f0d1787b956ef0d1062b0f7c24c - languageName: node - linkType: hard - -"fastq@npm:^1.6.0": - version: 1.15.0 - resolution: "fastq@npm:1.15.0" - dependencies: - reusify: ^1.0.4 - checksum: 0170e6bfcd5d57a70412440b8ef600da6de3b2a6c5966aeaf0a852d542daff506a0ee92d6de7679d1de82e644bce69d7a574a6c93f0b03964b5337eed75ada1a - languageName: node - linkType: hard - -"file-entry-cache@npm:^6.0.1": - version: 6.0.1 - resolution: "file-entry-cache@npm:6.0.1" - dependencies: - flat-cache: ^3.0.4 - checksum: f49701feaa6314c8127c3c2f6173cfefff17612f5ed2daaafc6da13b5c91fd43e3b2a58fd0d63f9f94478a501b167615931e7200e31485e320f74a33885a9c74 - languageName: node - linkType: hard - -"fill-range@npm:^7.0.1": - version: 7.0.1 - resolution: "fill-range@npm:7.0.1" - dependencies: - to-regex-range: ^5.0.1 - checksum: cc283f4e65b504259e64fd969bcf4def4eb08d85565e906b7d36516e87819db52029a76b6363d0f02d0d532f0033c9603b9e2d943d56ee3b0d4f7ad3328ff917 - languageName: node - linkType: hard - -"find-up@npm:^5.0.0": - version: 5.0.0 - resolution: "find-up@npm:5.0.0" - dependencies: - locate-path: ^6.0.0 - path-exists: ^4.0.0 - checksum: 07955e357348f34660bde7920783204ff5a26ac2cafcaa28bace494027158a97b9f56faaf2d89a6106211a8174db650dd9f503f9c0d526b1202d5554a00b9095 - languageName: node - linkType: hard - -"flat-cache@npm:^3.0.4": - version: 3.0.4 - resolution: "flat-cache@npm:3.0.4" - dependencies: - flatted: ^3.1.0 - rimraf: ^3.0.2 - checksum: 4fdd10ecbcbf7d520f9040dd1340eb5dfe951e6f0ecf2252edeec03ee68d989ec8b9a20f4434270e71bcfd57800dc09b3344fca3966b2eb8f613072c7d9a2365 - languageName: node - linkType: hard - -"flatted@npm:^3.1.0": - version: 3.2.7 - resolution: "flatted@npm:3.2.7" - checksum: 427633049d55bdb80201c68f7eb1cbd533e03eac541f97d3aecab8c5526f12a20ccecaeede08b57503e772c769e7f8680b37e8d482d1e5f8d7e2194687f9ea35 - languageName: node - linkType: hard - -"for-each@npm:^0.3.3": - version: 0.3.3 - resolution: "for-each@npm:0.3.3" - dependencies: - is-callable: ^1.1.3 - checksum: 6c48ff2bc63362319c65e2edca4a8e1e3483a2fabc72fbe7feaf8c73db94fc7861bd53bc02c8a66a0c1dd709da6b04eec42e0abdd6b40ce47305ae92a25e5d28 - languageName: node - linkType: hard - -"fs.realpath@npm:^1.0.0": - version: 1.0.0 - resolution: "fs.realpath@npm:1.0.0" - checksum: 99ddea01a7e75aa276c250a04eedeffe5662bce66c65c07164ad6264f9de18fb21be9433ead460e54cff20e31721c811f4fb5d70591799df5f85dce6d6746fd0 - languageName: node - linkType: hard - -"function-bind@npm:^1.1.1": - version: 1.1.1 - resolution: "function-bind@npm:1.1.1" - checksum: b32fbaebb3f8ec4969f033073b43f5c8befbb58f1a79e12f1d7490358150359ebd92f49e72ff0144f65f2c48ea2a605bff2d07965f548f6474fd8efd95bf361a - languageName: node - linkType: hard - -"function.prototype.name@npm:^1.1.5": - version: 1.1.5 - resolution: "function.prototype.name@npm:1.1.5" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.3 - es-abstract: ^1.19.0 - functions-have-names: ^1.2.2 - checksum: acd21d733a9b649c2c442f067567743214af5fa248dbeee69d8278ce7df3329ea5abac572be9f7470b4ec1cd4d8f1040e3c5caccf98ebf2bf861a0deab735c27 - languageName: node - linkType: hard - -"functions-have-names@npm:^1.2.2": - version: 1.2.3 - resolution: "functions-have-names@npm:1.2.3" - checksum: c3f1f5ba20f4e962efb71344ce0a40722163e85bee2101ce25f88214e78182d2d2476aa85ef37950c579eb6cf6ee811c17b3101bb84004bb75655f3e33f3fdb5 - languageName: node - linkType: hard - -"get-intrinsic@npm:^1.0.2, get-intrinsic@npm:^1.1.1, get-intrinsic@npm:^1.1.3, get-intrinsic@npm:^1.2.0": - version: 1.2.0 - resolution: "get-intrinsic@npm:1.2.0" - dependencies: - function-bind: ^1.1.1 - has: ^1.0.3 - has-symbols: ^1.0.3 - checksum: 78fc0487b783f5c58cf2dccafc3ae656ee8d2d8062a8831ce4a95e7057af4587a1d4882246c033aca0a7b4965276f4802b45cc300338d1b77a73d3e3e3f4877d - languageName: node - linkType: hard - -"get-symbol-description@npm:^1.0.0": - version: 1.0.0 - resolution: "get-symbol-description@npm:1.0.0" - dependencies: - call-bind: ^1.0.2 - get-intrinsic: ^1.1.1 - checksum: 9ceff8fe968f9270a37a1f73bf3f1f7bda69ca80f4f80850670e0e7b9444ff99323f7ac52f96567f8b5f5fbe7ac717a0d81d3407c7313e82810c6199446a5247 - languageName: node - linkType: hard - -"glob-parent@npm:^5.1.2": - version: 5.1.2 - resolution: "glob-parent@npm:5.1.2" - dependencies: - is-glob: ^4.0.1 - checksum: f4f2bfe2425296e8a47e36864e4f42be38a996db40420fe434565e4480e3322f18eb37589617a98640c5dc8fdec1a387007ee18dbb1f3f5553409c34d17f425e - languageName: node - linkType: hard - -"glob-parent@npm:^6.0.2": - version: 6.0.2 - resolution: "glob-parent@npm:6.0.2" - dependencies: - is-glob: ^4.0.3 - checksum: c13ee97978bef4f55106b71e66428eb1512e71a7466ba49025fc2aec59a5bfb0954d5abd58fc5ee6c9b076eef4e1f6d3375c2e964b88466ca390da4419a786a8 - languageName: node - linkType: hard - -"glob@npm:^7.1.3": - version: 7.2.3 - resolution: "glob@npm:7.2.3" - dependencies: - fs.realpath: ^1.0.0 - inflight: ^1.0.4 - inherits: 2 - minimatch: ^3.1.1 - once: ^1.3.0 - path-is-absolute: ^1.0.0 - checksum: 29452e97b38fa704dabb1d1045350fb2467cf0277e155aa9ff7077e90ad81d1ea9d53d3ee63bd37c05b09a065e90f16aec4a65f5b8de401d1dac40bc5605d133 - languageName: node - linkType: hard - -"globals@npm:^13.19.0": - version: 13.20.0 - resolution: "globals@npm:13.20.0" - dependencies: - type-fest: ^0.20.2 - checksum: ad1ecf914bd051325faad281d02ea2c0b1df5d01bd94d368dcc5513340eac41d14b3c61af325768e3c7f8d44576e72780ec0b6f2d366121f8eec6e03c3a3b97a - languageName: node - linkType: hard - -"globalthis@npm:^1.0.3": - version: 1.0.3 - resolution: "globalthis@npm:1.0.3" - dependencies: - define-properties: ^1.1.3 - checksum: fbd7d760dc464c886d0196166d92e5ffb4c84d0730846d6621a39fbbc068aeeb9c8d1421ad330e94b7bca4bb4ea092f5f21f3d36077812af5d098b4dc006c998 - languageName: node - linkType: hard - -"globby@npm:^11.1.0": - version: 11.1.0 - resolution: "globby@npm:11.1.0" - dependencies: - array-union: ^2.1.0 - dir-glob: ^3.0.1 - fast-glob: ^3.2.9 - ignore: ^5.2.0 - merge2: ^1.4.1 - slash: ^3.0.0 - checksum: b4be8885e0cfa018fc783792942d53926c35c50b3aefd3fdcfb9d22c627639dc26bd2327a40a0b74b074100ce95bb7187bfeae2f236856aa3de183af7a02aea6 - languageName: node - linkType: hard - -"gopd@npm:^1.0.1": - version: 1.0.1 - resolution: "gopd@npm:1.0.1" - dependencies: - get-intrinsic: ^1.1.3 - checksum: a5ccfb8806e0917a94e0b3de2af2ea4979c1da920bc381667c260e00e7cafdbe844e2cb9c5bcfef4e5412e8bf73bab837285bc35c7ba73aaaf0134d4583393a6 - languageName: node - linkType: hard - -"grapheme-splitter@npm:^1.0.4": - version: 1.0.4 - resolution: "grapheme-splitter@npm:1.0.4" - checksum: 0c22ec54dee1b05cd480f78cf14f732cb5b108edc073572c4ec205df4cd63f30f8db8025afc5debc8835a8ddeacf648a1c7992fe3dcd6ad38f9a476d84906620 - languageName: node - linkType: hard - -"has-bigints@npm:^1.0.1, has-bigints@npm:^1.0.2": - version: 1.0.2 - resolution: "has-bigints@npm:1.0.2" - checksum: 390e31e7be7e5c6fe68b81babb73dfc35d413604d7ee5f56da101417027a4b4ce6a27e46eff97ad040c835b5d228676eae99a9b5c3bc0e23c8e81a49241ff45b - languageName: node - linkType: hard - -"has-flag@npm:^4.0.0": - version: 4.0.0 - resolution: "has-flag@npm:4.0.0" - checksum: 261a1357037ead75e338156b1f9452c016a37dcd3283a972a30d9e4a87441ba372c8b81f818cd0fbcd9c0354b4ae7e18b9e1afa1971164aef6d18c2b6095a8ad - languageName: node - linkType: hard - -"has-property-descriptors@npm:^1.0.0": - version: 1.0.0 - resolution: "has-property-descriptors@npm:1.0.0" - dependencies: - get-intrinsic: ^1.1.1 - checksum: a6d3f0a266d0294d972e354782e872e2fe1b6495b321e6ef678c9b7a06a40408a6891817350c62e752adced73a94ac903c54734fee05bf65b1905ee1368194bb - languageName: node - linkType: hard - -"has-proto@npm:^1.0.1": - version: 1.0.1 - resolution: "has-proto@npm:1.0.1" - checksum: febc5b5b531de8022806ad7407935e2135f1cc9e64636c3916c6842bd7995994ca3b29871ecd7954bd35f9e2986c17b3b227880484d22259e2f8e6ce63fd383e - languageName: node - linkType: hard - -"has-symbols@npm:^1.0.2, has-symbols@npm:^1.0.3": - version: 1.0.3 - resolution: "has-symbols@npm:1.0.3" - checksum: a054c40c631c0d5741a8285010a0777ea0c068f99ed43e5d6eb12972da223f8af553a455132fdb0801bdcfa0e0f443c0c03a68d8555aa529b3144b446c3f2410 - languageName: node - linkType: hard - -"has-tostringtag@npm:^1.0.0": - version: 1.0.0 - resolution: "has-tostringtag@npm:1.0.0" - dependencies: - has-symbols: ^1.0.2 - checksum: cc12eb28cb6ae22369ebaad3a8ab0799ed61270991be88f208d508076a1e99abe4198c965935ce85ea90b60c94ddda73693b0920b58e7ead048b4a391b502c1c - languageName: node - linkType: hard - -"has@npm:^1.0.3": - version: 1.0.3 - resolution: "has@npm:1.0.3" - dependencies: - function-bind: ^1.1.1 - checksum: b9ad53d53be4af90ce5d1c38331e712522417d017d5ef1ebd0507e07c2fbad8686fffb8e12ddecd4c39ca9b9b47431afbb975b8abf7f3c3b82c98e9aad052792 - languageName: node - linkType: hard - -"ignore@npm:^5.2.0": - version: 5.2.4 - resolution: "ignore@npm:5.2.4" - checksum: 3d4c309c6006e2621659311783eaea7ebcd41fe4ca1d78c91c473157ad6666a57a2df790fe0d07a12300d9aac2888204d7be8d59f9aaf665b1c7fcdb432517ef - languageName: node - linkType: hard - -"import-fresh@npm:^3.0.0, import-fresh@npm:^3.2.1": - version: 3.3.0 - resolution: "import-fresh@npm:3.3.0" - dependencies: - parent-module: ^1.0.0 - resolve-from: ^4.0.0 - checksum: 2cacfad06e652b1edc50be650f7ec3be08c5e5a6f6d12d035c440a42a8cc028e60a5b99ca08a77ab4d6b1346da7d971915828f33cdab730d3d42f08242d09baa - languageName: node - linkType: hard - -"imurmurhash@npm:^0.1.4": - version: 0.1.4 - resolution: "imurmurhash@npm:0.1.4" - checksum: 7cae75c8cd9a50f57dadd77482359f659eaebac0319dd9368bcd1714f55e65badd6929ca58569da2b6494ef13fdd5598cd700b1eba23f8b79c5f19d195a3ecf7 - languageName: node - linkType: hard - -"inflight@npm:^1.0.4": - version: 1.0.6 - resolution: "inflight@npm:1.0.6" - dependencies: - once: ^1.3.0 - wrappy: 1 - checksum: f4f76aa072ce19fae87ce1ef7d221e709afb59d445e05d47fba710e85470923a75de35bfae47da6de1b18afc3ce83d70facf44cfb0aff89f0a3f45c0a0244dfd - languageName: node - linkType: hard - -"inherits@npm:2": - version: 2.0.4 - resolution: "inherits@npm:2.0.4" - checksum: 4a48a733847879d6cf6691860a6b1e3f0f4754176e4d71494c41f3475553768b10f84b5ce1d40fbd0e34e6bfbb864ee35858ad4dd2cf31e02fc4a154b724d7f1 - languageName: node - linkType: hard - -"internal-slot@npm:^1.0.5": - version: 1.0.5 - resolution: "internal-slot@npm:1.0.5" - dependencies: - get-intrinsic: ^1.2.0 - has: ^1.0.3 - side-channel: ^1.0.4 - checksum: 97e84046bf9e7574d0956bd98d7162313ce7057883b6db6c5c7b5e5f05688864b0978ba07610c726d15d66544ffe4b1050107d93f8a39ebc59b15d8b429b497a - languageName: node - linkType: hard - -"is-array-buffer@npm:^3.0.1, is-array-buffer@npm:^3.0.2": - version: 3.0.2 - resolution: "is-array-buffer@npm:3.0.2" - dependencies: - call-bind: ^1.0.2 - get-intrinsic: ^1.2.0 - is-typed-array: ^1.1.10 - checksum: dcac9dda66ff17df9cabdc58214172bf41082f956eab30bb0d86bc0fab1e44b690fc8e1f855cf2481245caf4e8a5a006a982a71ddccec84032ed41f9d8da8c14 - languageName: node - linkType: hard - -"is-bigint@npm:^1.0.1": - version: 1.0.4 - resolution: "is-bigint@npm:1.0.4" - dependencies: - has-bigints: ^1.0.1 - checksum: c56edfe09b1154f8668e53ebe8252b6f185ee852a50f9b41e8d921cb2bed425652049fbe438723f6cb48a63ca1aa051e948e7e401e093477c99c84eba244f666 - languageName: node - linkType: hard - -"is-boolean-object@npm:^1.1.0": - version: 1.1.2 - resolution: "is-boolean-object@npm:1.1.2" - dependencies: - call-bind: ^1.0.2 - has-tostringtag: ^1.0.0 - checksum: c03b23dbaacadc18940defb12c1c0e3aaece7553ef58b162a0f6bba0c2a7e1551b59f365b91e00d2dbac0522392d576ef322628cb1d036a0fe51eb466db67222 - languageName: node - linkType: hard - -"is-callable@npm:^1.1.3, is-callable@npm:^1.1.4, is-callable@npm:^1.2.7": - version: 1.2.7 - resolution: "is-callable@npm:1.2.7" - checksum: 61fd57d03b0d984e2ed3720fb1c7a897827ea174bd44402878e059542ea8c4aeedee0ea0985998aa5cc2736b2fa6e271c08587addb5b3959ac52cf665173d1ac - languageName: node - linkType: hard - -"is-core-module@npm:^2.11.0, is-core-module@npm:^2.9.0": - version: 2.11.0 - resolution: "is-core-module@npm:2.11.0" - dependencies: - has: ^1.0.3 - checksum: f96fd490c6b48eb4f6d10ba815c6ef13f410b0ba6f7eb8577af51697de523e5f2cd9de1c441b51d27251bf0e4aebc936545e33a5d26d5d51f28d25698d4a8bab - languageName: node - linkType: hard - -"is-date-object@npm:^1.0.1": - version: 1.0.5 - resolution: "is-date-object@npm:1.0.5" - dependencies: - has-tostringtag: ^1.0.0 - checksum: baa9077cdf15eb7b58c79398604ca57379b2fc4cf9aa7a9b9e295278648f628c9b201400c01c5e0f7afae56507d741185730307cbe7cad3b9f90a77e5ee342fc - languageName: node - linkType: hard - -"is-extglob@npm:^2.1.1": - version: 2.1.1 - resolution: "is-extglob@npm:2.1.1" - checksum: df033653d06d0eb567461e58a7a8c9f940bd8c22274b94bf7671ab36df5719791aae15eef6d83bbb5e23283967f2f984b8914559d4449efda578c775c4be6f85 - languageName: node - linkType: hard - -"is-glob@npm:^4.0.0, is-glob@npm:^4.0.1, is-glob@npm:^4.0.3": - version: 4.0.3 - resolution: "is-glob@npm:4.0.3" - dependencies: - is-extglob: ^2.1.1 - checksum: d381c1319fcb69d341cc6e6c7cd588e17cd94722d9a32dbd60660b993c4fb7d0f19438674e68dfec686d09b7c73139c9166b47597f846af387450224a8101ab4 - languageName: node - linkType: hard - -"is-negative-zero@npm:^2.0.2": - version: 2.0.2 - resolution: "is-negative-zero@npm:2.0.2" - checksum: f3232194c47a549da60c3d509c9a09be442507616b69454716692e37ae9f37c4dea264fb208ad0c9f3efd15a796a46b79df07c7e53c6227c32170608b809149a - languageName: node - linkType: hard - -"is-number-object@npm:^1.0.4": - version: 1.0.7 - resolution: "is-number-object@npm:1.0.7" - dependencies: - has-tostringtag: ^1.0.0 - checksum: d1e8d01bb0a7134c74649c4e62da0c6118a0bfc6771ea3c560914d52a627873e6920dd0fd0ebc0e12ad2ff4687eac4c308f7e80320b973b2c8a2c8f97a7524f7 - languageName: node - linkType: hard - -"is-number@npm:^7.0.0": - version: 7.0.0 - resolution: "is-number@npm:7.0.0" - checksum: 456ac6f8e0f3111ed34668a624e45315201dff921e5ac181f8ec24923b99e9f32ca1a194912dc79d539c97d33dba17dc635202ff0b2cf98326f608323276d27a - languageName: node - linkType: hard - -"is-path-inside@npm:^3.0.3": - version: 3.0.3 - resolution: "is-path-inside@npm:3.0.3" - checksum: abd50f06186a052b349c15e55b182326f1936c89a78bf6c8f2b707412517c097ce04bc49a0ca221787bc44e1049f51f09a2ffb63d22899051988d3a618ba13e9 - languageName: node - linkType: hard - -"is-regex@npm:^1.1.4": - version: 1.1.4 - resolution: "is-regex@npm:1.1.4" - dependencies: - call-bind: ^1.0.2 - has-tostringtag: ^1.0.0 - checksum: 362399b33535bc8f386d96c45c9feb04cf7f8b41c182f54174c1a45c9abbbe5e31290bbad09a458583ff6bf3b2048672cdb1881b13289569a7c548370856a652 - languageName: node - linkType: hard - -"is-shared-array-buffer@npm:^1.0.2": - version: 1.0.2 - resolution: "is-shared-array-buffer@npm:1.0.2" - dependencies: - call-bind: ^1.0.2 - checksum: 9508929cf14fdc1afc9d61d723c6e8d34f5e117f0bffda4d97e7a5d88c3a8681f633a74f8e3ad1fe92d5113f9b921dc5ca44356492079612f9a247efbce7032a - languageName: node - linkType: hard - -"is-string@npm:^1.0.5, is-string@npm:^1.0.7": - version: 1.0.7 - resolution: "is-string@npm:1.0.7" - dependencies: - has-tostringtag: ^1.0.0 - checksum: 323b3d04622f78d45077cf89aab783b2f49d24dc641aa89b5ad1a72114cfeff2585efc8c12ef42466dff32bde93d839ad321b26884cf75e5a7892a938b089989 - languageName: node - linkType: hard - -"is-symbol@npm:^1.0.2, is-symbol@npm:^1.0.3": - version: 1.0.4 - resolution: "is-symbol@npm:1.0.4" - dependencies: - has-symbols: ^1.0.2 - checksum: 92805812ef590738d9de49d677cd17dfd486794773fb6fa0032d16452af46e9b91bb43ffe82c983570f015b37136f4b53b28b8523bfb10b0ece7a66c31a54510 - languageName: node - linkType: hard - -"is-typed-array@npm:^1.1.10, is-typed-array@npm:^1.1.9": - version: 1.1.10 - resolution: "is-typed-array@npm:1.1.10" - dependencies: - available-typed-arrays: ^1.0.5 - call-bind: ^1.0.2 - for-each: ^0.3.3 - gopd: ^1.0.1 - has-tostringtag: ^1.0.0 - checksum: aac6ecb59d4c56a1cdeb69b1f129154ef462bbffe434cb8a8235ca89b42f258b7ae94073c41b3cb7bce37f6a1733ad4499f07882d5d5093a7ba84dfc4ebb8017 - languageName: node - linkType: hard - -"is-weakref@npm:^1.0.2": - version: 1.0.2 - resolution: "is-weakref@npm:1.0.2" - dependencies: - call-bind: ^1.0.2 - checksum: 95bd9a57cdcb58c63b1c401c60a474b0f45b94719c30f548c891860f051bc2231575c290a6b420c6bc6e7ed99459d424c652bd5bf9a1d5259505dc35b4bf83de - languageName: node - linkType: hard - -"isexe@npm:^2.0.0": - version: 2.0.0 - resolution: "isexe@npm:2.0.0" - checksum: 26bf6c5480dda5161c820c5b5c751ae1e766c587b1f951ea3fcfc973bafb7831ae5b54a31a69bd670220e42e99ec154475025a468eae58ea262f813fdc8d1c62 - languageName: node - linkType: hard - -"js-sdsl@npm:^4.1.4": - version: 4.4.0 - resolution: "js-sdsl@npm:4.4.0" - checksum: 7bb08a2d746ab7ff742720339aa006c631afe05e77d11eda988c1c35fae8e03e492e4e347e883e786e3ce6170685d4780c125619111f0730c11fdb41b04059c7 - languageName: node - linkType: hard - -"js-yaml@npm:^4.1.0": - version: 4.1.0 - resolution: "js-yaml@npm:4.1.0" - dependencies: - argparse: ^2.0.1 - bin: - js-yaml: bin/js-yaml.js - checksum: c7830dfd456c3ef2c6e355cc5a92e6700ceafa1d14bba54497b34a99f0376cecbb3e9ac14d3e5849b426d5a5140709a66237a8c991c675431271c4ce5504151a - languageName: node - linkType: hard - -"json-schema-traverse@npm:^0.4.1": - version: 0.4.1 - resolution: "json-schema-traverse@npm:0.4.1" - checksum: 7486074d3ba247769fda17d5181b345c9fb7d12e0da98b22d1d71a5db9698d8b4bd900a3ec1a4ffdd60846fc2556274a5c894d0c48795f14cb03aeae7b55260b - languageName: node - linkType: hard - -"json-stable-stringify-without-jsonify@npm:^1.0.1": - version: 1.0.1 - resolution: "json-stable-stringify-without-jsonify@npm:1.0.1" - checksum: cff44156ddce9c67c44386ad5cddf91925fe06b1d217f2da9c4910d01f358c6e3989c4d5a02683c7a5667f9727ff05831f7aa8ae66c8ff691c556f0884d49215 - languageName: node - linkType: hard - -"json5@npm:^1.0.2": - version: 1.0.2 - resolution: "json5@npm:1.0.2" - dependencies: - minimist: ^1.2.0 - bin: - json5: lib/cli.js - checksum: 866458a8c58a95a49bef3adba929c625e82532bcff1fe93f01d29cb02cac7c3fe1f4b79951b7792c2da9de0b32871a8401a6e3c5b36778ad852bf5b8a61165d7 - languageName: node - linkType: hard - -"kafkajs@npm:^2.2.3": - version: 2.2.4 - resolution: "kafkajs@npm:2.2.4" - checksum: 83e9e8bc50a09b142f4ff79f6a2bd88ecc21b83bcefe6621ab1716118d624886befb7371731274f67812ce35dd50b53140ff3b49a06e5d9169fe6b164d72fea5 - languageName: node - linkType: hard - -"levn@npm:^0.4.1": - version: 0.4.1 - resolution: "levn@npm:0.4.1" - dependencies: - prelude-ls: ^1.2.1 - type-check: ~0.4.0 - checksum: 12c5021c859bd0f5248561bf139121f0358285ec545ebf48bb3d346820d5c61a4309535c7f387ed7d84361cf821e124ce346c6b7cef8ee09a67c1473b46d0fc4 - languageName: node - linkType: hard - -"locate-path@npm:^6.0.0": - version: 6.0.0 - resolution: "locate-path@npm:6.0.0" - dependencies: - p-locate: ^5.0.0 - checksum: 72eb661788a0368c099a184c59d2fee760b3831c9c1c33955e8a19ae4a21b4116e53fa736dc086cdeb9fce9f7cc508f2f92d2d3aae516f133e16a2bb59a39f5a - languageName: node - linkType: hard - -"lodash.merge@npm:^4.6.2": - version: 4.6.2 - resolution: "lodash.merge@npm:4.6.2" - checksum: ad580b4bdbb7ca1f7abf7e1bce63a9a0b98e370cf40194b03380a46b4ed799c9573029599caebc1b14e3f24b111aef72b96674a56cfa105e0f5ac70546cdc005 - languageName: node - linkType: hard - -"lru-cache@npm:^6.0.0": - version: 6.0.0 - resolution: "lru-cache@npm:6.0.0" - dependencies: - yallist: ^4.0.0 - checksum: f97f499f898f23e4585742138a22f22526254fdba6d75d41a1c2526b3b6cc5747ef59c5612ba7375f42aca4f8461950e925ba08c991ead0651b4918b7c978297 - languageName: node - linkType: hard - -"merge2@npm:^1.3.0, merge2@npm:^1.4.1": - version: 1.4.1 - resolution: "merge2@npm:1.4.1" - checksum: 7268db63ed5169466540b6fb947aec313200bcf6d40c5ab722c22e242f651994619bcd85601602972d3c85bd2cc45a358a4c61937e9f11a061919a1da569b0c2 - languageName: node - linkType: hard - -"micromatch@npm:^4.0.4": - version: 4.0.5 - resolution: "micromatch@npm:4.0.5" - dependencies: - braces: ^3.0.2 - picomatch: ^2.3.1 - checksum: 02a17b671c06e8fefeeb6ef996119c1e597c942e632a21ef589154f23898c9c6a9858526246abb14f8bca6e77734aa9dcf65476fca47cedfb80d9577d52843fc - languageName: node - linkType: hard - -"minimatch@npm:^3.0.5, minimatch@npm:^3.1.1, minimatch@npm:^3.1.2": - version: 3.1.2 - resolution: "minimatch@npm:3.1.2" - dependencies: - brace-expansion: ^1.1.7 - checksum: c154e566406683e7bcb746e000b84d74465b3a832c45d59912b9b55cd50dee66e5c4b1e5566dba26154040e51672f9aa450a9aef0c97cfc7336b78b7afb9540a - languageName: node - linkType: hard - -"minimist@npm:^1.2.0, minimist@npm:^1.2.6": - version: 1.2.8 - resolution: "minimist@npm:1.2.8" - checksum: 75a6d645fb122dad29c06a7597bddea977258957ed88d7a6df59b5cd3fe4a527e253e9bbf2e783e4b73657f9098b96a5fe96ab8a113655d4109108577ecf85b0 - languageName: node - linkType: hard - -"ms@npm:2.1.2": - version: 2.1.2 - resolution: "ms@npm:2.1.2" - checksum: 673cdb2c3133eb050c745908d8ce632ed2c02d85640e2edb3ace856a2266a813b30c613569bf3354fdf4ea7d1a1494add3bfa95e2713baa27d0c2c71fc44f58f - languageName: node - linkType: hard - -"ms@npm:^2.1.1": - version: 2.1.3 - resolution: "ms@npm:2.1.3" - checksum: aa92de608021b242401676e35cfa5aa42dd70cbdc082b916da7fb925c542173e36bce97ea3e804923fe92c0ad991434e4a38327e15a1b5b5f945d66df615ae6d - languageName: node - linkType: hard - -"natural-compare-lite@npm:^1.4.0": - version: 1.4.0 - resolution: "natural-compare-lite@npm:1.4.0" - checksum: 5222ac3986a2b78dd6069ac62cbb52a7bf8ffc90d972ab76dfe7b01892485d229530ed20d0c62e79a6b363a663b273db3bde195a1358ce9e5f779d4453887225 - languageName: node - linkType: hard - -"natural-compare@npm:^1.4.0": - version: 1.4.0 - resolution: "natural-compare@npm:1.4.0" - checksum: 23ad088b08f898fc9b53011d7bb78ec48e79de7627e01ab5518e806033861bef68d5b0cd0e2205c2f36690ac9571ff6bcb05eb777ced2eeda8d4ac5b44592c3d - languageName: node - linkType: hard - -"object-inspect@npm:^1.12.3, object-inspect@npm:^1.9.0": - version: 1.12.3 - resolution: "object-inspect@npm:1.12.3" - checksum: dabfd824d97a5f407e6d5d24810d888859f6be394d8b733a77442b277e0808860555176719c5905e765e3743a7cada6b8b0a3b85e5331c530fd418cc8ae991db - languageName: node - linkType: hard - -"object-keys@npm:^1.1.1": - version: 1.1.1 - resolution: "object-keys@npm:1.1.1" - checksum: b363c5e7644b1e1b04aa507e88dcb8e3a2f52b6ffd0ea801e4c7a62d5aa559affe21c55a07fd4b1fd55fc03a33c610d73426664b20032405d7b92a1414c34d6a - languageName: node - linkType: hard - -"object.assign@npm:^4.1.4": - version: 4.1.4 - resolution: "object.assign@npm:4.1.4" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.4 - has-symbols: ^1.0.3 - object-keys: ^1.1.1 - checksum: 76cab513a5999acbfe0ff355f15a6a125e71805fcf53de4e9d4e082e1989bdb81d1e329291e1e4e0ae7719f0e4ef80e88fb2d367ae60500d79d25a6224ac8864 - languageName: node - linkType: hard - -"object.values@npm:^1.1.6": - version: 1.1.6 - resolution: "object.values@npm:1.1.6" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.4 - es-abstract: ^1.20.4 - checksum: f6fff9fd817c24cfd8107f50fb33061d81cd11bacc4e3dbb3852e9ff7692fde4dbce823d4333ea27cd9637ef1b6690df5fbb61f1ed314fa2959598dc3ae23d8e - languageName: node - linkType: hard - -"once@npm:^1.3.0": - version: 1.4.0 - resolution: "once@npm:1.4.0" - dependencies: - wrappy: 1 - checksum: cd0a88501333edd640d95f0d2700fbde6bff20b3d4d9bdc521bdd31af0656b5706570d6c6afe532045a20bb8dc0849f8332d6f2a416e0ba6d3d3b98806c7db68 - languageName: node - linkType: hard - -"optionator@npm:^0.9.1": - version: 0.9.1 - resolution: "optionator@npm:0.9.1" - 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.3 - checksum: dbc6fa065604b24ea57d734261914e697bd73b69eff7f18e967e8912aa2a40a19a9f599a507fa805be6c13c24c4eae8c71306c239d517d42d4c041c942f508a0 - languageName: node - linkType: hard - -"p-limit@npm:^3.0.2": - version: 3.1.0 - resolution: "p-limit@npm:3.1.0" - dependencies: - yocto-queue: ^0.1.0 - checksum: 7c3690c4dbf62ef625671e20b7bdf1cbc9534e83352a2780f165b0d3ceba21907e77ad63401708145ca4e25bfc51636588d89a8c0aeb715e6c37d1c066430360 - languageName: node - linkType: hard - -"p-locate@npm:^5.0.0": - version: 5.0.0 - resolution: "p-locate@npm:5.0.0" - dependencies: - p-limit: ^3.0.2 - checksum: 1623088f36cf1cbca58e9b61c4e62bf0c60a07af5ae1ca99a720837356b5b6c5ba3eb1b2127e47a06865fee59dd0453cad7cc844cda9d5a62ac1a5a51b7c86d3 - languageName: node - linkType: hard - -"parent-module@npm:^1.0.0": - version: 1.0.1 - resolution: "parent-module@npm:1.0.1" - dependencies: - callsites: ^3.0.0 - checksum: 6ba8b255145cae9470cf5551eb74be2d22281587af787a2626683a6c20fbb464978784661478dd2a3f1dad74d1e802d403e1b03c1a31fab310259eec8ac560ff - languageName: node - linkType: hard - -"path-exists@npm:^4.0.0": - version: 4.0.0 - resolution: "path-exists@npm:4.0.0" - checksum: 505807199dfb7c50737b057dd8d351b82c033029ab94cb10a657609e00c1bc53b951cfdbccab8de04c5584d5eff31128ce6afd3db79281874a5ef2adbba55ed1 - languageName: node - linkType: hard - -"path-is-absolute@npm:^1.0.0": - version: 1.0.1 - resolution: "path-is-absolute@npm:1.0.1" - checksum: 060840f92cf8effa293bcc1bea81281bd7d363731d214cbe5c227df207c34cd727430f70c6037b5159c8a870b9157cba65e775446b0ab06fd5ecc7e54615a3b8 - languageName: node - linkType: hard - -"path-key@npm:^3.1.0": - version: 3.1.1 - resolution: "path-key@npm:3.1.1" - checksum: 55cd7a9dd4b343412a8386a743f9c746ef196e57c823d90ca3ab917f90ab9f13dd0ded27252ba49dbdfcab2b091d998bc446f6220cd3cea65db407502a740020 - languageName: node - linkType: hard - -"path-parse@npm:^1.0.7": - version: 1.0.7 - resolution: "path-parse@npm:1.0.7" - checksum: 49abf3d81115642938a8700ec580da6e830dde670be21893c62f4e10bd7dd4c3742ddc603fe24f898cba7eb0c6bc1777f8d9ac14185d34540c6d4d80cd9cae8a - languageName: node - linkType: hard - -"path-type@npm:^4.0.0": - version: 4.0.0 - resolution: "path-type@npm:4.0.0" - checksum: 5b1e2daa247062061325b8fdbfd1fb56dde0a448fb1455453276ea18c60685bdad23a445dc148cf87bc216be1573357509b7d4060494a6fd768c7efad833ee45 - languageName: node - linkType: hard - -"picomatch@npm:^2.3.1": - version: 2.3.1 - resolution: "picomatch@npm:2.3.1" - checksum: 050c865ce81119c4822c45d3c84f1ced46f93a0126febae20737bd05ca20589c564d6e9226977df859ed5e03dc73f02584a2b0faad36e896936238238b0446cf - languageName: node - linkType: hard - -"prelude-ls@npm:^1.2.1": - version: 1.2.1 - resolution: "prelude-ls@npm:1.2.1" - checksum: cd192ec0d0a8e4c6da3bb80e4f62afe336df3f76271ac6deb0e6a36187133b6073a19e9727a1ff108cd8b9982e4768850d413baa71214dd80c7979617dca827a - languageName: node - linkType: hard - -"punycode@npm:^2.1.0": - version: 2.3.0 - resolution: "punycode@npm:2.3.0" - checksum: 39f760e09a2a3bbfe8f5287cf733ecdad69d6af2fe6f97ca95f24b8921858b91e9ea3c9eeec6e08cede96181b3bb33f95c6ffd8c77e63986508aa2e8159fa200 - languageName: node - linkType: hard - -"queue-microtask@npm:^1.2.2": - version: 1.2.3 - resolution: "queue-microtask@npm:1.2.3" - checksum: b676f8c040cdc5b12723ad2f91414d267605b26419d5c821ff03befa817ddd10e238d22b25d604920340fd73efd8ba795465a0377c4adf45a4a41e4234e42dc4 - languageName: node - linkType: hard - -"regexp.prototype.flags@npm:^1.4.3": - version: 1.4.3 - resolution: "regexp.prototype.flags@npm:1.4.3" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.3 - functions-have-names: ^1.2.2 - checksum: 51228bae732592adb3ededd5e15426be25f289e9c4ef15212f4da73f4ec3919b6140806374b8894036a86020d054a8d2657d3fee6bb9b4d35d8939c20030b7a6 - languageName: node - linkType: hard - -"resolve-from@npm:^4.0.0": - version: 4.0.0 - resolution: "resolve-from@npm:4.0.0" - checksum: f4ba0b8494846a5066328ad33ef8ac173801a51739eb4d63408c847da9a2e1c1de1e6cbbf72699211f3d13f8fc1325648b169bd15eb7da35688e30a5fb0e4a7f - languageName: node - linkType: hard - -"resolve@npm:^1.22.1": - version: 1.22.1 - resolution: "resolve@npm:1.22.1" - dependencies: - is-core-module: ^2.9.0 - path-parse: ^1.0.7 - supports-preserve-symlinks-flag: ^1.0.0 - bin: - resolve: bin/resolve - checksum: 07af5fc1e81aa1d866cbc9e9460fbb67318a10fa3c4deadc35c3ad8a898ee9a71a86a65e4755ac3195e0ea0cfbe201eb323ebe655ce90526fd61917313a34e4e - languageName: node - linkType: hard - -"resolve@patch:resolve@^1.22.1#~builtin": - version: 1.22.1 - resolution: "resolve@patch:resolve@npm%3A1.22.1#~builtin::version=1.22.1&hash=c3c19d" - dependencies: - is-core-module: ^2.9.0 - path-parse: ^1.0.7 - supports-preserve-symlinks-flag: ^1.0.0 - bin: - resolve: bin/resolve - checksum: 5656f4d0bedcf8eb52685c1abdf8fbe73a1603bb1160a24d716e27a57f6cecbe2432ff9c89c2bd57542c3a7b9d14b1882b73bfe2e9d7849c9a4c0b8b39f02b8b - languageName: node - linkType: hard - -"reusify@npm:^1.0.4": - version: 1.0.4 - resolution: "reusify@npm:1.0.4" - checksum: c3076ebcc22a6bc252cb0b9c77561795256c22b757f40c0d8110b1300723f15ec0fc8685e8d4ea6d7666f36c79ccc793b1939c748bf36f18f542744a4e379fcc - languageName: node - linkType: hard - -"rimraf@npm:^3.0.2": - version: 3.0.2 - resolution: "rimraf@npm:3.0.2" - dependencies: - glob: ^7.1.3 - bin: - rimraf: bin.js - checksum: 87f4164e396f0171b0a3386cc1877a817f572148ee13a7e113b238e48e8a9f2f31d009a92ec38a591ff1567d9662c6b67fd8818a2dbbaed74bc26a87a2a4a9a0 - languageName: node - linkType: hard - -"run-parallel@npm:^1.1.9": - version: 1.2.0 - resolution: "run-parallel@npm:1.2.0" - dependencies: - queue-microtask: ^1.2.2 - checksum: cb4f97ad25a75ebc11a8ef4e33bb962f8af8516bb2001082ceabd8902e15b98f4b84b4f8a9b222e5d57fc3bd1379c483886ed4619367a7680dad65316993021d - languageName: node - linkType: hard - -"safe-regex-test@npm:^1.0.0": - version: 1.0.0 - resolution: "safe-regex-test@npm:1.0.0" - dependencies: - call-bind: ^1.0.2 - get-intrinsic: ^1.1.3 - is-regex: ^1.1.4 - checksum: bc566d8beb8b43c01b94e67de3f070fd2781685e835959bbbaaec91cc53381145ca91f69bd837ce6ec244817afa0a5e974fc4e40a2957f0aca68ac3add1ddd34 - languageName: node - linkType: hard - -"semver@npm:^6.3.0": - version: 6.3.0 - resolution: "semver@npm:6.3.0" - bin: - semver: ./bin/semver.js - checksum: 1b26ecf6db9e8292dd90df4e781d91875c0dcc1b1909e70f5d12959a23c7eebb8f01ea581c00783bbee72ceeaad9505797c381756326073850dc36ed284b21b9 - languageName: node - linkType: hard - -"semver@npm:^7.3.7": - version: 7.3.8 - resolution: "semver@npm:7.3.8" - dependencies: - lru-cache: ^6.0.0 - bin: - semver: bin/semver.js - checksum: ba9c7cbbf2b7884696523450a61fee1a09930d888b7a8d7579025ad93d459b2d1949ee5bbfeb188b2be5f4ac163544c5e98491ad6152df34154feebc2cc337c1 - languageName: node - linkType: hard - -"shebang-command@npm:^2.0.0": - version: 2.0.0 - resolution: "shebang-command@npm:2.0.0" - dependencies: - shebang-regex: ^3.0.0 - checksum: 6b52fe87271c12968f6a054e60f6bde5f0f3d2db483a1e5c3e12d657c488a15474121a1d55cd958f6df026a54374ec38a4a963988c213b7570e1d51575cea7fa - languageName: node - linkType: hard - -"shebang-regex@npm:^3.0.0": - version: 3.0.0 - resolution: "shebang-regex@npm:3.0.0" - checksum: 1a2bcae50de99034fcd92ad4212d8e01eedf52c7ec7830eedcf886622804fe36884278f2be8be0ea5fde3fd1c23911643a4e0f726c8685b61871c8908af01222 - languageName: node - linkType: hard - -"side-channel@npm:^1.0.4": - version: 1.0.4 - resolution: "side-channel@npm:1.0.4" - dependencies: - call-bind: ^1.0.0 - get-intrinsic: ^1.0.2 - object-inspect: ^1.9.0 - checksum: 351e41b947079c10bd0858364f32bb3a7379514c399edb64ab3dce683933483fc63fb5e4efe0a15a2e8a7e3c436b6a91736ddb8d8c6591b0460a24bb4a1ee245 - languageName: node - linkType: hard - -"slash@npm:^3.0.0": - version: 3.0.0 - resolution: "slash@npm:3.0.0" - checksum: 94a93fff615f25a999ad4b83c9d5e257a7280c90a32a7cb8b4a87996e4babf322e469c42b7f649fd5796edd8687652f3fb452a86dc97a816f01113183393f11c - languageName: node - linkType: hard - -"string.prototype.trim@npm:^1.2.7": - version: 1.2.7 - resolution: "string.prototype.trim@npm:1.2.7" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.4 - es-abstract: ^1.20.4 - checksum: 05b7b2d6af63648e70e44c4a8d10d8cc457536df78b55b9d6230918bde75c5987f6b8604438c4c8652eb55e4fc9725d2912789eb4ec457d6995f3495af190c09 - languageName: node - linkType: hard - -"string.prototype.trimend@npm:^1.0.6": - version: 1.0.6 - resolution: "string.prototype.trimend@npm:1.0.6" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.4 - es-abstract: ^1.20.4 - checksum: 0fdc34645a639bd35179b5a08227a353b88dc089adf438f46be8a7c197fc3f22f8514c1c9be4629b3cd29c281582730a8cbbad6466c60f76b5f99cf2addb132e - languageName: node - linkType: hard - -"string.prototype.trimstart@npm:^1.0.6": - version: 1.0.6 - resolution: "string.prototype.trimstart@npm:1.0.6" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.4 - es-abstract: ^1.20.4 - checksum: 89080feef416621e6ef1279588994305477a7a91648d9436490d56010a1f7adc39167cddac7ce0b9884b8cdbef086987c4dcb2960209f2af8bac0d23ceff4f41 - languageName: node - linkType: hard - -"strip-ansi@npm:^6.0.1": - version: 6.0.1 - resolution: "strip-ansi@npm:6.0.1" - dependencies: - ansi-regex: ^5.0.1 - checksum: f3cd25890aef3ba6e1a74e20896c21a46f482e93df4a06567cebf2b57edabb15133f1f94e57434e0a958d61186087b1008e89c94875d019910a213181a14fc8c - languageName: node - linkType: hard - -"strip-bom@npm:^3.0.0": - version: 3.0.0 - resolution: "strip-bom@npm:3.0.0" - checksum: 8d50ff27b7ebe5ecc78f1fe1e00fcdff7af014e73cf724b46fb81ef889eeb1015fc5184b64e81a2efe002180f3ba431bdd77e300da5c6685d702780fbf0c8d5b - languageName: node - linkType: hard - -"strip-json-comments@npm:^3.1.0, strip-json-comments@npm:^3.1.1": - version: 3.1.1 - resolution: "strip-json-comments@npm:3.1.1" - checksum: 492f73e27268f9b1c122733f28ecb0e7e8d8a531a6662efbd08e22cccb3f9475e90a1b82cab06a392f6afae6d2de636f977e231296400d0ec5304ba70f166443 - languageName: node - linkType: hard - -"supports-color@npm:^7.1.0": - version: 7.2.0 - resolution: "supports-color@npm:7.2.0" - dependencies: - has-flag: ^4.0.0 - checksum: 3dda818de06ebbe5b9653e07842d9479f3555ebc77e9a0280caf5a14fb877ffee9ed57007c3b78f5a6324b8dbeec648d9e97a24e2ed9fdb81ddc69ea07100f4a - languageName: node - linkType: hard - -"supports-preserve-symlinks-flag@npm:^1.0.0": - version: 1.0.0 - resolution: "supports-preserve-symlinks-flag@npm:1.0.0" - checksum: 53b1e247e68e05db7b3808b99b892bd36fb096e6fba213a06da7fab22045e97597db425c724f2bbd6c99a3c295e1e73f3e4de78592289f38431049e1277ca0ae - languageName: node - linkType: hard - -"text-table@npm:^0.2.0": - version: 0.2.0 - resolution: "text-table@npm:0.2.0" - checksum: b6937a38c80c7f84d9c11dd75e49d5c44f71d95e810a3250bd1f1797fc7117c57698204adf676b71497acc205d769d65c16ae8fa10afad832ae1322630aef10a - languageName: node - linkType: hard - -"to-regex-range@npm:^5.0.1": - version: 5.0.1 - resolution: "to-regex-range@npm:5.0.1" - dependencies: - is-number: ^7.0.0 - checksum: f76fa01b3d5be85db6a2a143e24df9f60dd047d151062d0ba3df62953f2f697b16fe5dad9b0ac6191c7efc7b1d9dcaa4b768174b7b29da89d4428e64bc0a20ed - languageName: node - linkType: hard - -"tsconfig-paths@npm:^3.14.1": - version: 3.14.2 - resolution: "tsconfig-paths@npm:3.14.2" - dependencies: - "@types/json5": ^0.0.29 - json5: ^1.0.2 - minimist: ^1.2.6 - strip-bom: ^3.0.0 - checksum: a6162eaa1aed680537f93621b82399c7856afd10ec299867b13a0675e981acac4e0ec00896860480efc59fc10fd0b16fdc928c0b885865b52be62cadac692447 - languageName: node - linkType: hard - -"tslib@npm:^1.8.1": - version: 1.14.1 - resolution: "tslib@npm:1.14.1" - checksum: dbe628ef87f66691d5d2959b3e41b9ca0045c3ee3c7c7b906cc1e328b39f199bb1ad9e671c39025bd56122ac57dfbf7385a94843b1cc07c60a4db74795829acd - languageName: node - linkType: hard - -"tsutils@npm:^3.21.0": - version: 3.21.0 - resolution: "tsutils@npm:3.21.0" - dependencies: - tslib: ^1.8.1 - peerDependencies: - typescript: ">=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta" - checksum: 1843f4c1b2e0f975e08c4c21caa4af4f7f65a12ac1b81b3b8489366826259323feb3fc7a243123453d2d1a02314205a7634e048d4a8009921da19f99755cdc48 - languageName: node - linkType: hard - -"type-check@npm:^0.4.0, type-check@npm:~0.4.0": - version: 0.4.0 - resolution: "type-check@npm:0.4.0" - dependencies: - prelude-ls: ^1.2.1 - checksum: ec688ebfc9c45d0c30412e41ca9c0cdbd704580eb3a9ccf07b9b576094d7b86a012baebc95681999dd38f4f444afd28504cb3a89f2ef16b31d4ab61a0739025a - languageName: node - linkType: hard - -"type-fest@npm:^0.20.2": - version: 0.20.2 - resolution: "type-fest@npm:0.20.2" - checksum: 4fb3272df21ad1c552486f8a2f8e115c09a521ad7a8db3d56d53718d0c907b62c6e9141ba5f584af3f6830d0872c521357e512381f24f7c44acae583ad517d73 - languageName: node - linkType: hard - -"typed-array-length@npm:^1.0.4": - version: 1.0.4 - resolution: "typed-array-length@npm:1.0.4" - dependencies: - call-bind: ^1.0.2 - for-each: ^0.3.3 - is-typed-array: ^1.1.9 - checksum: 2228febc93c7feff142b8c96a58d4a0d7623ecde6c7a24b2b98eb3170e99f7c7eff8c114f9b283085cd59dcd2bd43aadf20e25bba4b034a53c5bb292f71f8956 - languageName: node - linkType: hard - -"typescript@npm:^4.9.4": - version: 4.9.5 - resolution: "typescript@npm:4.9.5" - bin: - tsc: bin/tsc - tsserver: bin/tsserver - checksum: ee000bc26848147ad423b581bd250075662a354d84f0e06eb76d3b892328d8d4440b7487b5a83e851b12b255f55d71835b008a66cbf8f255a11e4400159237db - languageName: node - linkType: hard - -"typescript@patch:typescript@^4.9.4#~builtin": - version: 4.9.5 - resolution: "typescript@patch:typescript@npm%3A4.9.5#~builtin::version=4.9.5&hash=ad5954" - bin: - tsc: bin/tsc - tsserver: bin/tsserver - checksum: 8f6260acc86b56bfdda6004bc53f32ea548f543e8baef7071c8e34d29d292f3e375c8416556c8de10b24deef6933cd1c16a8233dc84a3dd43a13a13265d0faab - languageName: node - linkType: hard - -"unbox-primitive@npm:^1.0.2": - version: 1.0.2 - resolution: "unbox-primitive@npm:1.0.2" - dependencies: - call-bind: ^1.0.2 - has-bigints: ^1.0.2 - has-symbols: ^1.0.3 - which-boxed-primitive: ^1.0.2 - checksum: b7a1cf5862b5e4b5deb091672ffa579aa274f648410009c81cca63fed3b62b610c4f3b773f912ce545bb4e31edc3138975b5bc777fc6e4817dca51affb6380e9 - languageName: node - linkType: hard - -"uri-js@npm:^4.2.2": - version: 4.4.1 - resolution: "uri-js@npm:4.4.1" - dependencies: - punycode: ^2.1.0 - checksum: 7167432de6817fe8e9e0c9684f1d2de2bb688c94388f7569f7dbdb1587c9f4ca2a77962f134ec90be0cc4d004c939ff0d05acc9f34a0db39a3c797dada262633 - languageName: node - linkType: hard - -"which-boxed-primitive@npm:^1.0.2": - version: 1.0.2 - resolution: "which-boxed-primitive@npm:1.0.2" - dependencies: - is-bigint: ^1.0.1 - is-boolean-object: ^1.1.0 - is-number-object: ^1.0.4 - is-string: ^1.0.5 - is-symbol: ^1.0.3 - checksum: 53ce774c7379071729533922adcca47220228405e1895f26673bbd71bdf7fb09bee38c1d6399395927c6289476b5ae0629863427fd151491b71c4b6cb04f3a5e - languageName: node - linkType: hard - -"which-typed-array@npm:^1.1.9": - version: 1.1.9 - resolution: "which-typed-array@npm:1.1.9" - dependencies: - available-typed-arrays: ^1.0.5 - call-bind: ^1.0.2 - for-each: ^0.3.3 - gopd: ^1.0.1 - has-tostringtag: ^1.0.0 - is-typed-array: ^1.1.10 - checksum: fe0178ca44c57699ca2c0e657b64eaa8d2db2372a4e2851184f568f98c478ae3dc3fdb5f7e46c384487046b0cf9e23241423242b277e03e8ba3dabc7c84c98ef - languageName: node - linkType: hard - -"which@npm:^2.0.1": - version: 2.0.2 - resolution: "which@npm:2.0.2" - dependencies: - isexe: ^2.0.0 - bin: - node-which: ./bin/node-which - checksum: 1a5c563d3c1b52d5f893c8b61afe11abc3bab4afac492e8da5bde69d550de701cf9806235f20a47b5c8fa8a1d6a9135841de2596535e998027a54589000e66d1 - languageName: node - linkType: hard - -"word-wrap@npm:^1.2.3": - version: 1.2.3 - resolution: "word-wrap@npm:1.2.3" - checksum: 30b48f91fcf12106ed3186ae4fa86a6a1842416df425be7b60485de14bec665a54a68e4b5156647dec3a70f25e84d270ca8bc8cd23182ed095f5c7206a938c1f - languageName: node - linkType: hard - -"wrappy@npm:1": - version: 1.0.2 - resolution: "wrappy@npm:1.0.2" - checksum: 159da4805f7e84a3d003d8841557196034155008f817172d4e986bd591f74aa82aa7db55929a54222309e01079a65a92a9e6414da5a6aa4b01ee44a511ac3ee5 - languageName: node - linkType: hard - -"yallist@npm:^4.0.0": - version: 4.0.0 - resolution: "yallist@npm:4.0.0" - checksum: 343617202af32df2a15a3be36a5a8c0c8545208f3d3dfbc6bb7c3e3b7e8c6f8e7485432e4f3b88da3031a6e20afa7c711eded32ddfb122896ac5d914e75848d5 - languageName: node - linkType: hard - -"yocto-queue@npm:^0.1.0": - version: 0.1.0 - resolution: "yocto-queue@npm:0.1.0" - checksum: f77b3d8d00310def622123df93d4ee654fc6a0096182af8bd60679ddcdfb3474c56c6c7190817c84a2785648cdee9d721c0154eb45698c62176c322fb46fc700 - languageName: node - linkType: hard +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@aashutoshrathi/word-wrap@^1.2.3": + version "1.2.6" + resolved "https://registry.yarnpkg.com/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz#bd9154aec9983f77b3a034ecaa015c2e4201f6cf" + integrity sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA== + +"@eslint-community/eslint-utils@^4.2.0", "@eslint-community/eslint-utils@^4.4.0": + version "4.4.0" + resolved "https://registry.yarnpkg.com/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz#a23514e8fb9af1269d5f7788aa556798d61c6b59" + integrity sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA== + dependencies: + eslint-visitor-keys "^3.3.0" + +"@eslint-community/regexpp@^4.5.1", "@eslint-community/regexpp@^4.6.1": + version "4.10.0" + resolved "https://registry.yarnpkg.com/@eslint-community/regexpp/-/regexpp-4.10.0.tgz#548f6de556857c8bb73bbee70c35dc82a2e74d63" + integrity sha512-Cu96Sd2By9mCNTx2iyKOmq10v22jUVQv0lQnlGNy16oE9589yE+QADPbrMGCkA51cKZSg3Pu/aTJVTGfL/qjUA== + +"@eslint/eslintrc@^2.1.4": + version "2.1.4" + resolved "https://registry.yarnpkg.com/@eslint/eslintrc/-/eslintrc-2.1.4.tgz#388a269f0f25c1b6adc317b5a2c55714894c70ad" + integrity sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ== + dependencies: + ajv "^6.12.4" + debug "^4.3.2" + espree "^9.6.0" + globals "^13.19.0" + ignore "^5.2.0" + import-fresh "^3.2.1" + js-yaml "^4.1.0" + minimatch "^3.1.2" + strip-json-comments "^3.1.1" + +"@eslint/js@8.56.0": + version "8.56.0" + resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.56.0.tgz#ef20350fec605a7f7035a01764731b2de0f3782b" + integrity sha512-gMsVel9D7f2HLkBma9VbtzZRehRogVRfbr++f06nL2vnCGCNlzOD+/MUov/F4p8myyAHspEhVobgjpX64q5m6A== + +"@humanwhocodes/config-array@^0.11.13": + version "0.11.14" + resolved "https://registry.yarnpkg.com/@humanwhocodes/config-array/-/config-array-0.11.14.tgz#d78e481a039f7566ecc9660b4ea7fe6b1fec442b" + integrity sha512-3T8LkOmg45BV5FICb15QQMsyUSWrQ8AygVfC7ZG32zOalnqrilm018ZVCw0eapXux8FtA33q8PSRSstjee3jSg== + dependencies: + "@humanwhocodes/object-schema" "^2.0.2" + debug "^4.3.1" + minimatch "^3.0.5" + +"@humanwhocodes/module-importer@^1.0.1": + version "1.0.1" + resolved "https://registry.yarnpkg.com/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz#af5b2691a22b44be847b0ca81641c5fb6ad0172c" + integrity sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA== + +"@humanwhocodes/object-schema@^2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-2.0.2.tgz#d9fae00a2d5cb40f92cfe64b47ad749fbc38f917" + integrity sha512-6EwiSjwWYP7pTckG6I5eyFANjPhmPjUX9JRLUSfNPC7FX7zK9gyZAfUEaECL6ALTpGX5AjnBq3C9XmVWPitNpw== + +"@nodelib/fs.scandir@2.1.5": + version "2.1.5" + resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" + integrity sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g== + dependencies: + "@nodelib/fs.stat" "2.0.5" + run-parallel "^1.1.9" + +"@nodelib/fs.stat@2.0.5", "@nodelib/fs.stat@^2.0.2": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz#5bd262af94e9d25bd1e71b05deed44876a222e8b" + integrity sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A== + +"@nodelib/fs.walk@^1.2.3", "@nodelib/fs.walk@^1.2.8": + version "1.2.8" + resolved "https://registry.yarnpkg.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz#e95737e8bb6746ddedf69c556953494f196fe69a" + integrity sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg== + dependencies: + "@nodelib/fs.scandir" "2.1.5" + fastq "^1.6.0" + +"@types/json-schema@^7.0.12": + version "7.0.15" + resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.15.tgz#596a1747233694d50f6ad8a7869fcb6f56cf5841" + integrity sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA== + +"@types/json5@^0.0.29": + version "0.0.29" + resolved "https://registry.yarnpkg.com/@types/json5/-/json5-0.0.29.tgz#ee28707ae94e11d2b827bcbe5270bcea7f3e71ee" + integrity sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ== + +"@types/node@^20.11.19": + version "20.11.19" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.11.19.tgz#b466de054e9cb5b3831bee38938de64ac7f81195" + integrity sha512-7xMnVEcZFu0DikYjWOlRq7NTPETrm7teqUT2WkQjrTIkEgUyyGdWsj/Zg8bEJt5TNklzbPD1X3fqfsHw3SpapQ== + dependencies: + undici-types "~5.26.4" + +"@types/semver@^7.5.0": + version "7.5.7" + resolved "https://registry.yarnpkg.com/@types/semver/-/semver-7.5.7.tgz#326f5fdda70d13580777bcaa1bc6fa772a5aef0e" + integrity sha512-/wdoPq1QqkSj9/QOeKkFquEuPzQbHTWAMPH/PaUMB+JuR31lXhlWXRZ52IpfDYVlDOUBvX09uBrPwxGT1hjNBg== + +"@typescript-eslint/eslint-plugin@^7.0.1": + version "7.0.1" + resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.0.1.tgz#407daffe09d964d57aceaf3ac51846359fbe61b0" + integrity sha512-OLvgeBv3vXlnnJGIAgCLYKjgMEU+wBGj07MQ/nxAaON+3mLzX7mJbhRYrVGiVvFiXtwFlkcBa/TtmglHy0UbzQ== + dependencies: + "@eslint-community/regexpp" "^4.5.1" + "@typescript-eslint/scope-manager" "7.0.1" + "@typescript-eslint/type-utils" "7.0.1" + "@typescript-eslint/utils" "7.0.1" + "@typescript-eslint/visitor-keys" "7.0.1" + debug "^4.3.4" + graphemer "^1.4.0" + ignore "^5.2.4" + natural-compare "^1.4.0" + semver "^7.5.4" + ts-api-utils "^1.0.1" + +"@typescript-eslint/parser@^7.0.1": + version "7.0.1" + resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-7.0.1.tgz#e9c61d9a5e32242477d92756d36086dc40322eed" + integrity sha512-8GcRRZNzaHxKzBPU3tKtFNing571/GwPBeCvmAUw0yBtfE2XVd0zFKJIMSWkHJcPQi0ekxjIts6L/rrZq5cxGQ== + dependencies: + "@typescript-eslint/scope-manager" "7.0.1" + "@typescript-eslint/types" "7.0.1" + "@typescript-eslint/typescript-estree" "7.0.1" + "@typescript-eslint/visitor-keys" "7.0.1" + debug "^4.3.4" + +"@typescript-eslint/scope-manager@7.0.1": + version "7.0.1" + resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-7.0.1.tgz#611ec8e78c70439b152a805e1b10aaac36de7c00" + integrity sha512-v7/T7As10g3bcWOOPAcbnMDuvctHzCFYCG/8R4bK4iYzdFqsZTbXGln0cZNVcwQcwewsYU2BJLay8j0/4zOk4w== + dependencies: + "@typescript-eslint/types" "7.0.1" + "@typescript-eslint/visitor-keys" "7.0.1" + +"@typescript-eslint/type-utils@7.0.1": + version "7.0.1" + resolved "https://registry.yarnpkg.com/@typescript-eslint/type-utils/-/type-utils-7.0.1.tgz#0fba92c1f81cad561d7b3adc812aa1cc0e35cdae" + integrity sha512-YtT9UcstTG5Yqy4xtLiClm1ZpM/pWVGFnkAa90UfdkkZsR1eP2mR/1jbHeYp8Ay1l1JHPyGvoUYR6o3On5Nhmw== + dependencies: + "@typescript-eslint/typescript-estree" "7.0.1" + "@typescript-eslint/utils" "7.0.1" + debug "^4.3.4" + ts-api-utils "^1.0.1" + +"@typescript-eslint/types@7.0.1": + version "7.0.1" + resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-7.0.1.tgz#dcfabce192db5b8bf77ea3c82cfaabe6e6a3c901" + integrity sha512-uJDfmirz4FHib6ENju/7cz9SdMSkeVvJDK3VcMFvf/hAShg8C74FW+06MaQPODHfDJp/z/zHfgawIJRjlu0RLg== + +"@typescript-eslint/typescript-estree@7.0.1": + version "7.0.1" + resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-7.0.1.tgz#1d52ac03da541693fa5bcdc13ad655def5046faf" + integrity sha512-SO9wHb6ph0/FN5OJxH4MiPscGah5wjOd0RRpaLvuBv9g8565Fgu0uMySFEPqwPHiQU90yzJ2FjRYKGrAhS1xig== + dependencies: + "@typescript-eslint/types" "7.0.1" + "@typescript-eslint/visitor-keys" "7.0.1" + debug "^4.3.4" + globby "^11.1.0" + is-glob "^4.0.3" + minimatch "9.0.3" + semver "^7.5.4" + ts-api-utils "^1.0.1" + +"@typescript-eslint/utils@7.0.1": + version "7.0.1" + resolved "https://registry.yarnpkg.com/@typescript-eslint/utils/-/utils-7.0.1.tgz#b8ceac0ba5fef362b4a03a33c0e1fedeea3734ed" + integrity sha512-oe4his30JgPbnv+9Vef1h48jm0S6ft4mNwi9wj7bX10joGn07QRfqIqFHoMiajrtoU88cIhXf8ahwgrcbNLgPA== + dependencies: + "@eslint-community/eslint-utils" "^4.4.0" + "@types/json-schema" "^7.0.12" + "@types/semver" "^7.5.0" + "@typescript-eslint/scope-manager" "7.0.1" + "@typescript-eslint/types" "7.0.1" + "@typescript-eslint/typescript-estree" "7.0.1" + semver "^7.5.4" + +"@typescript-eslint/visitor-keys@7.0.1": + version "7.0.1" + resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-7.0.1.tgz#864680ac5a8010ec4814f8a818e57595f79f464e" + integrity sha512-hwAgrOyk++RTXrP4KzCg7zB2U0xt7RUU0ZdMSCsqF3eKUwkdXUMyTb0qdCuji7VIbcpG62kKTU9M1J1c9UpFBw== + dependencies: + "@typescript-eslint/types" "7.0.1" + eslint-visitor-keys "^3.4.1" + +"@ungap/structured-clone@^1.2.0": + version "1.2.0" + resolved "https://registry.yarnpkg.com/@ungap/structured-clone/-/structured-clone-1.2.0.tgz#756641adb587851b5ccb3e095daf27ae581c8406" + integrity sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ== + +acorn-jsx@^5.3.2: + version "5.3.2" + resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937" + integrity sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ== + +acorn@^8.9.0: + version "8.11.3" + resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.11.3.tgz#71e0b14e13a4ec160724b38fb7b0f233b1b81d7a" + integrity sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg== + +ajv@^6.12.4: + version "6.12.6" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" + integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== + 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" + +ansi-regex@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304" + integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ== + +ansi-styles@^4.1.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" + integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== + dependencies: + color-convert "^2.0.1" + +argparse@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38" + integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q== + +array-buffer-byte-length@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/array-buffer-byte-length/-/array-buffer-byte-length-1.0.1.tgz#1e5583ec16763540a27ae52eed99ff899223568f" + integrity sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg== + dependencies: + call-bind "^1.0.5" + is-array-buffer "^3.0.4" + +array-includes@^3.1.7: + version "3.1.7" + resolved "https://registry.yarnpkg.com/array-includes/-/array-includes-3.1.7.tgz#8cd2e01b26f7a3086cbc87271593fe921c62abda" + integrity sha512-dlcsNBIiWhPkHdOEEKnehA+RNUWDc4UqFtnIXU4uuYDPtA4LDkr7qip2p0VvFAEXNDr0yWZ9PJyIRiGjRLQzwQ== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + get-intrinsic "^1.2.1" + is-string "^1.0.7" + +array-union@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/array-union/-/array-union-2.1.0.tgz#b798420adbeb1de828d84acd8a2e23d3efe85e8d" + integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw== + +array.prototype.filter@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/array.prototype.filter/-/array.prototype.filter-1.0.3.tgz#423771edeb417ff5914111fff4277ea0624c0d0e" + integrity sha512-VizNcj/RGJiUyQBgzwxzE5oHdeuXY5hSbbmKMlphj1cy1Vl7Pn2asCGbSrru6hSQjmCzqTBPVWAF/whmEOVHbw== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + es-array-method-boxes-properly "^1.0.0" + is-string "^1.0.7" + +array.prototype.findlastindex@^1.2.3: + version "1.2.4" + resolved "https://registry.yarnpkg.com/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.4.tgz#d1c50f0b3a9da191981ff8942a0aedd82794404f" + integrity sha512-hzvSHUshSpCflDR1QMUBLHGHP1VIEBegT4pix9H/Z92Xw3ySoy6c2qh7lJWTJnRJ8JCZ9bJNCgTyYaJGcJu6xQ== + dependencies: + call-bind "^1.0.5" + define-properties "^1.2.1" + es-abstract "^1.22.3" + es-errors "^1.3.0" + es-shim-unscopables "^1.0.2" + +array.prototype.flat@^1.3.2: + version "1.3.2" + resolved "https://registry.yarnpkg.com/array.prototype.flat/-/array.prototype.flat-1.3.2.tgz#1476217df8cff17d72ee8f3ba06738db5b387d18" + integrity sha512-djYB+Zx2vLewY8RWlNCUdHjDXs2XOgm602S9E7P/UpHgfeHL00cRiIF+IN/G/aUJ7kGPb6yO/ErDI5V2s8iycA== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + es-shim-unscopables "^1.0.0" + +array.prototype.flatmap@^1.3.2: + version "1.3.2" + resolved "https://registry.yarnpkg.com/array.prototype.flatmap/-/array.prototype.flatmap-1.3.2.tgz#c9a7c6831db8e719d6ce639190146c24bbd3e527" + integrity sha512-Ewyx0c9PmpcsByhSW4r+9zDU7sGjFc86qf/kKtuSCRdhfbk0SNLLkaT5qvcHnRGgc5NP/ly/y+qkXkqONX54CQ== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + es-shim-unscopables "^1.0.0" + +arraybuffer.prototype.slice@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.3.tgz#097972f4255e41bc3425e37dc3f6421cf9aefde6" + integrity sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A== + dependencies: + array-buffer-byte-length "^1.0.1" + call-bind "^1.0.5" + define-properties "^1.2.1" + es-abstract "^1.22.3" + es-errors "^1.2.1" + get-intrinsic "^1.2.3" + is-array-buffer "^3.0.4" + is-shared-array-buffer "^1.0.2" + +available-typed-arrays@^1.0.6: + version "1.0.6" + resolved "https://registry.yarnpkg.com/available-typed-arrays/-/available-typed-arrays-1.0.6.tgz#ac812d8ce5a6b976d738e1c45f08d0b00bc7d725" + integrity sha512-j1QzY8iPNPG4o4xmO3ptzpRxTciqD3MgEHtifP/YnJpIo58Xu+ne4BejlbkuaLfXn/nz6HFiw29bLpj2PNMdGg== + +balanced-match@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" + integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== + +brace-expansion@^1.1.7: + version "1.1.11" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" + integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== + dependencies: + balanced-match "^1.0.0" + concat-map "0.0.1" + +brace-expansion@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-2.0.1.tgz#1edc459e0f0c548486ecf9fc99f2221364b9a0ae" + integrity sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA== + dependencies: + balanced-match "^1.0.0" + +braces@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107" + integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A== + dependencies: + fill-range "^7.0.1" + +call-bind@^1.0.2, call-bind@^1.0.5, call-bind@^1.0.6, call-bind@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.7.tgz#06016599c40c56498c18769d2730be242b6fa3b9" + integrity sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w== + dependencies: + es-define-property "^1.0.0" + es-errors "^1.3.0" + function-bind "^1.1.2" + get-intrinsic "^1.2.4" + set-function-length "^1.2.1" + +callsites@^3.0.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73" + integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ== + +chalk@^4.0.0: + version "4.1.2" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" + integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== + dependencies: + ansi-styles "^4.1.0" + supports-color "^7.1.0" + +color-convert@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" + integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== + dependencies: + color-name "~1.1.4" + +color-name@~1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" + integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== + +concat-map@0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" + integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg== + +cross-spawn@^7.0.2: + version "7.0.3" + resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" + integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w== + dependencies: + path-key "^3.1.0" + shebang-command "^2.0.0" + which "^2.0.1" + +debug@^3.2.7: + version "3.2.7" + resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.7.tgz#72580b7e9145fb39b6676f9c5e5fb100b934179a" + integrity sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ== + dependencies: + ms "^2.1.1" + +debug@^4.3.1, debug@^4.3.2, debug@^4.3.4: + version "4.3.4" + resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" + integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== + dependencies: + ms "2.1.2" + +deep-is@^0.1.3: + version "0.1.4" + resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831" + integrity sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ== + +define-data-property@^1.0.1, define-data-property@^1.1.2: + version "1.1.4" + resolved "https://registry.yarnpkg.com/define-data-property/-/define-data-property-1.1.4.tgz#894dc141bb7d3060ae4366f6a0107e68fbe48c5e" + integrity sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A== + dependencies: + es-define-property "^1.0.0" + es-errors "^1.3.0" + gopd "^1.0.1" + +define-properties@^1.1.3, define-properties@^1.2.0, define-properties@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.2.1.tgz#10781cc616eb951a80a034bafcaa7377f6af2b6c" + integrity sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg== + dependencies: + define-data-property "^1.0.1" + has-property-descriptors "^1.0.0" + object-keys "^1.1.1" + +dir-glob@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-3.0.1.tgz#56dbf73d992a4a93ba1584f4534063fd2e41717f" + integrity sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA== + dependencies: + path-type "^4.0.0" + +doctrine@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-2.1.0.tgz#5cd01fc101621b42c4cd7f5d1a66243716d3f39d" + integrity sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw== + dependencies: + esutils "^2.0.2" + +doctrine@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-3.0.0.tgz#addebead72a6574db783639dc87a121773973961" + integrity sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w== + dependencies: + esutils "^2.0.2" + +es-abstract@^1.22.1, es-abstract@^1.22.3: + version "1.22.4" + resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.22.4.tgz#26eb2e7538c3271141f5754d31aabfdb215f27bf" + integrity sha512-vZYJlk2u6qHYxBOTjAeg7qUxHdNfih64Uu2J8QqWgXZ2cri0ZpJAkzDUK/q593+mvKwlxyaxr6F1Q+3LKoQRgg== + dependencies: + array-buffer-byte-length "^1.0.1" + arraybuffer.prototype.slice "^1.0.3" + available-typed-arrays "^1.0.6" + call-bind "^1.0.7" + es-define-property "^1.0.0" + es-errors "^1.3.0" + es-set-tostringtag "^2.0.2" + es-to-primitive "^1.2.1" + function.prototype.name "^1.1.6" + get-intrinsic "^1.2.4" + get-symbol-description "^1.0.2" + globalthis "^1.0.3" + gopd "^1.0.1" + has-property-descriptors "^1.0.2" + has-proto "^1.0.1" + has-symbols "^1.0.3" + hasown "^2.0.1" + internal-slot "^1.0.7" + is-array-buffer "^3.0.4" + is-callable "^1.2.7" + is-negative-zero "^2.0.2" + is-regex "^1.1.4" + is-shared-array-buffer "^1.0.2" + is-string "^1.0.7" + is-typed-array "^1.1.13" + is-weakref "^1.0.2" + object-inspect "^1.13.1" + object-keys "^1.1.1" + object.assign "^4.1.5" + regexp.prototype.flags "^1.5.2" + safe-array-concat "^1.1.0" + safe-regex-test "^1.0.3" + string.prototype.trim "^1.2.8" + string.prototype.trimend "^1.0.7" + string.prototype.trimstart "^1.0.7" + typed-array-buffer "^1.0.1" + typed-array-byte-length "^1.0.0" + typed-array-byte-offset "^1.0.0" + typed-array-length "^1.0.4" + unbox-primitive "^1.0.2" + which-typed-array "^1.1.14" + +es-array-method-boxes-properly@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/es-array-method-boxes-properly/-/es-array-method-boxes-properly-1.0.0.tgz#873f3e84418de4ee19c5be752990b2e44718d09e" + integrity sha512-wd6JXUmyHmt8T5a2xreUwKcGPq6f1f+WwIJkijUqiGcJz1qqnZgP6XIK+QyIWU5lT7imeNxUll48bziG+TSYcA== + +es-define-property@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/es-define-property/-/es-define-property-1.0.0.tgz#c7faefbdff8b2696cf5f46921edfb77cc4ba3845" + integrity sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ== + dependencies: + get-intrinsic "^1.2.4" + +es-errors@^1.0.0, es-errors@^1.2.1, es-errors@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/es-errors/-/es-errors-1.3.0.tgz#05f75a25dab98e4fb1dcd5e1472c0546d5057c8f" + integrity sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw== + +es-set-tostringtag@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/es-set-tostringtag/-/es-set-tostringtag-2.0.2.tgz#11f7cc9f63376930a5f20be4915834f4bc74f9c9" + integrity sha512-BuDyupZt65P9D2D2vA/zqcI3G5xRsklm5N3xCwuiy+/vKy8i0ifdsQP1sLgO4tZDSCaQUSnmC48khknGMV3D2Q== + dependencies: + get-intrinsic "^1.2.2" + has-tostringtag "^1.0.0" + hasown "^2.0.0" + +es-shim-unscopables@^1.0.0, es-shim-unscopables@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/es-shim-unscopables/-/es-shim-unscopables-1.0.2.tgz#1f6942e71ecc7835ed1c8a83006d8771a63a3763" + integrity sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw== + dependencies: + hasown "^2.0.0" + +es-to-primitive@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/es-to-primitive/-/es-to-primitive-1.2.1.tgz#e55cd4c9cdc188bcefb03b366c736323fc5c898a" + integrity sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA== + dependencies: + is-callable "^1.1.4" + is-date-object "^1.0.1" + is-symbol "^1.0.2" + +escape-string-regexp@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" + integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA== + +eslint-import-resolver-node@^0.3.9: + version "0.3.9" + resolved "https://registry.yarnpkg.com/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz#d4eaac52b8a2e7c3cd1903eb00f7e053356118ac" + integrity sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g== + dependencies: + debug "^3.2.7" + is-core-module "^2.13.0" + resolve "^1.22.4" + +eslint-module-utils@^2.8.0: + version "2.8.0" + resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.8.0.tgz#e439fee65fc33f6bba630ff621efc38ec0375c49" + integrity sha512-aWajIYfsqCKRDgUfjEXNN/JlrzauMuSEy5sbd7WXbtW3EH6A6MpwEh42c7qD+MqQo9QMJ6fWLAeIJynx0g6OAw== + dependencies: + debug "^3.2.7" + +eslint-plugin-import@^2.29.1: + version "2.29.1" + resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.29.1.tgz#d45b37b5ef5901d639c15270d74d46d161150643" + integrity sha512-BbPC0cuExzhiMo4Ff1BTVwHpjjv28C5R+btTOGaCRC7UEz801up0JadwkeSk5Ued6TG34uaczuVuH6qyy5YUxw== + dependencies: + array-includes "^3.1.7" + array.prototype.findlastindex "^1.2.3" + array.prototype.flat "^1.3.2" + array.prototype.flatmap "^1.3.2" + debug "^3.2.7" + doctrine "^2.1.0" + eslint-import-resolver-node "^0.3.9" + eslint-module-utils "^2.8.0" + hasown "^2.0.0" + is-core-module "^2.13.1" + is-glob "^4.0.3" + minimatch "^3.1.2" + object.fromentries "^2.0.7" + object.groupby "^1.0.1" + object.values "^1.1.7" + semver "^6.3.1" + tsconfig-paths "^3.15.0" + +eslint-scope@^7.2.2: + version "7.2.2" + resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-7.2.2.tgz#deb4f92563390f32006894af62a22dba1c46423f" + integrity sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg== + dependencies: + esrecurse "^4.3.0" + estraverse "^5.2.0" + +eslint-visitor-keys@^3.3.0, eslint-visitor-keys@^3.4.1, eslint-visitor-keys@^3.4.3: + version "3.4.3" + resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz#0cd72fe8550e3c2eae156a96a4dddcd1c8ac5800" + integrity sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag== + +eslint@^8.56.0: + version "8.56.0" + resolved "https://registry.yarnpkg.com/eslint/-/eslint-8.56.0.tgz#4957ce8da409dc0809f99ab07a1b94832ab74b15" + integrity sha512-Go19xM6T9puCOWntie1/P997aXxFsOi37JIHRWI514Hc6ZnaHGKY9xFhrU65RT6CcBEzZoGG1e6Nq+DT04ZtZQ== + dependencies: + "@eslint-community/eslint-utils" "^4.2.0" + "@eslint-community/regexpp" "^4.6.1" + "@eslint/eslintrc" "^2.1.4" + "@eslint/js" "8.56.0" + "@humanwhocodes/config-array" "^0.11.13" + "@humanwhocodes/module-importer" "^1.0.1" + "@nodelib/fs.walk" "^1.2.8" + "@ungap/structured-clone" "^1.2.0" + ajv "^6.12.4" + chalk "^4.0.0" + cross-spawn "^7.0.2" + debug "^4.3.2" + doctrine "^3.0.0" + escape-string-regexp "^4.0.0" + eslint-scope "^7.2.2" + eslint-visitor-keys "^3.4.3" + espree "^9.6.1" + esquery "^1.4.2" + esutils "^2.0.2" + fast-deep-equal "^3.1.3" + file-entry-cache "^6.0.1" + find-up "^5.0.0" + glob-parent "^6.0.2" + globals "^13.19.0" + graphemer "^1.4.0" + ignore "^5.2.0" + imurmurhash "^0.1.4" + is-glob "^4.0.0" + is-path-inside "^3.0.3" + js-yaml "^4.1.0" + json-stable-stringify-without-jsonify "^1.0.1" + levn "^0.4.1" + lodash.merge "^4.6.2" + minimatch "^3.1.2" + natural-compare "^1.4.0" + optionator "^0.9.3" + strip-ansi "^6.0.1" + text-table "^0.2.0" + +espree@^9.6.0, espree@^9.6.1: + version "9.6.1" + resolved "https://registry.yarnpkg.com/espree/-/espree-9.6.1.tgz#a2a17b8e434690a5432f2f8018ce71d331a48c6f" + integrity sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ== + dependencies: + acorn "^8.9.0" + acorn-jsx "^5.3.2" + eslint-visitor-keys "^3.4.1" + +esquery@^1.4.2: + version "1.5.0" + resolved "https://registry.yarnpkg.com/esquery/-/esquery-1.5.0.tgz#6ce17738de8577694edd7361c57182ac8cb0db0b" + integrity sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg== + dependencies: + estraverse "^5.1.0" + +esrecurse@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/esrecurse/-/esrecurse-4.3.0.tgz#7ad7964d679abb28bee72cec63758b1c5d2c9921" + integrity sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag== + dependencies: + estraverse "^5.2.0" + +estraverse@^5.1.0, estraverse@^5.2.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-5.3.0.tgz#2eea5290702f26ab8fe5370370ff86c965d21123" + integrity sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA== + +esutils@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64" + integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== + +fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" + integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== + +fast-glob@^3.2.9: + version "3.3.2" + resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-3.3.2.tgz#a904501e57cfdd2ffcded45e99a54fef55e46129" + integrity sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow== + dependencies: + "@nodelib/fs.stat" "^2.0.2" + "@nodelib/fs.walk" "^1.2.3" + glob-parent "^5.1.2" + merge2 "^1.3.0" + micromatch "^4.0.4" + +fast-json-stable-stringify@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633" + integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw== + +fast-levenshtein@^2.0.6: + version "2.0.6" + resolved "https://registry.yarnpkg.com/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz#3d8a5c66883a16a30ca8643e851f19baa7797917" + integrity sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw== + +fastq@^1.6.0: + version "1.17.1" + resolved "https://registry.yarnpkg.com/fastq/-/fastq-1.17.1.tgz#2a523f07a4e7b1e81a42b91b8bf2254107753b47" + integrity sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w== + dependencies: + reusify "^1.0.4" + +file-entry-cache@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/file-entry-cache/-/file-entry-cache-6.0.1.tgz#211b2dd9659cb0394b073e7323ac3c933d522027" + integrity sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg== + dependencies: + flat-cache "^3.0.4" + +fill-range@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40" + integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ== + dependencies: + to-regex-range "^5.0.1" + +find-up@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc" + integrity sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng== + dependencies: + locate-path "^6.0.0" + path-exists "^4.0.0" + +flat-cache@^3.0.4: + version "3.2.0" + resolved "https://registry.yarnpkg.com/flat-cache/-/flat-cache-3.2.0.tgz#2c0c2d5040c99b1632771a9d105725c0115363ee" + integrity sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw== + dependencies: + flatted "^3.2.9" + keyv "^4.5.3" + rimraf "^3.0.2" + +flatted@^3.2.9: + version "3.2.9" + resolved "https://registry.yarnpkg.com/flatted/-/flatted-3.2.9.tgz#7eb4c67ca1ba34232ca9d2d93e9886e611ad7daf" + integrity sha512-36yxDn5H7OFZQla0/jFJmbIKTdZAQHngCedGxiMmpNfEZM0sdEeT+WczLQrjK6D7o2aiyLYDnkw0R3JK0Qv1RQ== + +for-each@^0.3.3: + version "0.3.3" + resolved "https://registry.yarnpkg.com/for-each/-/for-each-0.3.3.tgz#69b447e88a0a5d32c3e7084f3f1710034b21376e" + integrity sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw== + dependencies: + is-callable "^1.1.3" + +fs.realpath@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" + integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw== + +function-bind@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.2.tgz#2c02d864d97f3ea6c8830c464cbd11ab6eab7a1c" + integrity sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA== + +function.prototype.name@^1.1.6: + version "1.1.6" + resolved "https://registry.yarnpkg.com/function.prototype.name/-/function.prototype.name-1.1.6.tgz#cdf315b7d90ee77a4c6ee216c3c3362da07533fd" + integrity sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + functions-have-names "^1.2.3" + +functions-have-names@^1.2.3: + version "1.2.3" + resolved "https://registry.yarnpkg.com/functions-have-names/-/functions-have-names-1.2.3.tgz#0404fe4ee2ba2f607f0e0ec3c80bae994133b834" + integrity sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ== + +get-intrinsic@^1.1.3, get-intrinsic@^1.2.1, get-intrinsic@^1.2.2, get-intrinsic@^1.2.3, get-intrinsic@^1.2.4: + version "1.2.4" + resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.2.4.tgz#e385f5a4b5227d449c3eabbad05494ef0abbeadd" + integrity sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ== + dependencies: + es-errors "^1.3.0" + function-bind "^1.1.2" + has-proto "^1.0.1" + has-symbols "^1.0.3" + hasown "^2.0.0" + +get-symbol-description@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/get-symbol-description/-/get-symbol-description-1.0.2.tgz#533744d5aa20aca4e079c8e5daf7fd44202821f5" + integrity sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg== + dependencies: + call-bind "^1.0.5" + es-errors "^1.3.0" + get-intrinsic "^1.2.4" + +glob-parent@^5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4" + integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow== + dependencies: + is-glob "^4.0.1" + +glob-parent@^6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-6.0.2.tgz#6d237d99083950c79290f24c7642a3de9a28f9e3" + integrity sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A== + dependencies: + is-glob "^4.0.3" + +glob@^7.1.3: + version "7.2.3" + resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b" + integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q== + dependencies: + fs.realpath "^1.0.0" + inflight "^1.0.4" + inherits "2" + minimatch "^3.1.1" + once "^1.3.0" + path-is-absolute "^1.0.0" + +globals@^13.19.0: + version "13.24.0" + resolved "https://registry.yarnpkg.com/globals/-/globals-13.24.0.tgz#8432a19d78ce0c1e833949c36adb345400bb1171" + integrity sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ== + dependencies: + type-fest "^0.20.2" + +globalthis@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.3.tgz#5852882a52b80dc301b0660273e1ed082f0b6ccf" + integrity sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA== + dependencies: + define-properties "^1.1.3" + +globby@^11.1.0: + version "11.1.0" + resolved "https://registry.yarnpkg.com/globby/-/globby-11.1.0.tgz#bd4be98bb042f83d796f7e3811991fbe82a0d34b" + integrity sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g== + dependencies: + array-union "^2.1.0" + dir-glob "^3.0.1" + fast-glob "^3.2.9" + ignore "^5.2.0" + merge2 "^1.4.1" + slash "^3.0.0" + +gopd@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.0.1.tgz#29ff76de69dac7489b7c0918a5788e56477c332c" + integrity sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA== + dependencies: + get-intrinsic "^1.1.3" + +graphemer@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/graphemer/-/graphemer-1.4.0.tgz#fb2f1d55e0e3a1849aeffc90c4fa0dd53a0e66c6" + integrity sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag== + +has-bigints@^1.0.1, has-bigints@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/has-bigints/-/has-bigints-1.0.2.tgz#0871bd3e3d51626f6ca0966668ba35d5602d6eaa" + integrity sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ== + +has-flag@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" + integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== + +has-property-descriptors@^1.0.0, has-property-descriptors@^1.0.1, has-property-descriptors@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz#963ed7d071dc7bf5f084c5bfbe0d1b6222586854" + integrity sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg== + dependencies: + es-define-property "^1.0.0" + +has-proto@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/has-proto/-/has-proto-1.0.1.tgz#1885c1305538958aff469fef37937c22795408e0" + integrity sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg== + +has-symbols@^1.0.2, has-symbols@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.3.tgz#bb7b2c4349251dce87b125f7bdf874aa7c8b39f8" + integrity sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A== + +has-tostringtag@^1.0.0, has-tostringtag@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/has-tostringtag/-/has-tostringtag-1.0.2.tgz#2cdc42d40bef2e5b4eeab7c01a73c54ce7ab5abc" + integrity sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw== + dependencies: + has-symbols "^1.0.3" + +hasown@^2.0.0, hasown@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.1.tgz#26f48f039de2c0f8d3356c223fb8d50253519faa" + integrity sha512-1/th4MHjnwncwXsIW6QMzlvYL9kG5e/CpVvLRZe4XPa8TOUNbCELqmvhDmnkNsAjwaG4+I8gJJL0JBvTTLO9qA== + dependencies: + function-bind "^1.1.2" + +ignore@^5.2.0, ignore@^5.2.4: + version "5.3.1" + resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.3.1.tgz#5073e554cd42c5b33b394375f538b8593e34d4ef" + integrity sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw== + +import-fresh@^3.2.1: + version "3.3.0" + resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-3.3.0.tgz#37162c25fcb9ebaa2e6e53d5b4d88ce17d9e0c2b" + integrity sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw== + dependencies: + parent-module "^1.0.0" + resolve-from "^4.0.0" + +imurmurhash@^0.1.4: + version "0.1.4" + resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" + integrity sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA== + +inflight@^1.0.4: + version "1.0.6" + resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" + integrity sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA== + dependencies: + once "^1.3.0" + wrappy "1" + +inherits@2: + version "2.0.4" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" + integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== + +internal-slot@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/internal-slot/-/internal-slot-1.0.7.tgz#c06dcca3ed874249881007b0a5523b172a190802" + integrity sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g== + dependencies: + es-errors "^1.3.0" + hasown "^2.0.0" + side-channel "^1.0.4" + +is-array-buffer@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/is-array-buffer/-/is-array-buffer-3.0.4.tgz#7a1f92b3d61edd2bc65d24f130530ea93d7fae98" + integrity sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw== + dependencies: + call-bind "^1.0.2" + get-intrinsic "^1.2.1" + +is-bigint@^1.0.1: + version "1.0.4" + resolved "https://registry.yarnpkg.com/is-bigint/-/is-bigint-1.0.4.tgz#08147a1875bc2b32005d41ccd8291dffc6691df3" + integrity sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg== + dependencies: + has-bigints "^1.0.1" + +is-boolean-object@^1.1.0: + version "1.1.2" + resolved "https://registry.yarnpkg.com/is-boolean-object/-/is-boolean-object-1.1.2.tgz#5c6dc200246dd9321ae4b885a114bb1f75f63719" + integrity sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA== + dependencies: + call-bind "^1.0.2" + has-tostringtag "^1.0.0" + +is-callable@^1.1.3, is-callable@^1.1.4, is-callable@^1.2.7: + version "1.2.7" + resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.7.tgz#3bc2a85ea742d9e36205dcacdd72ca1fdc51b055" + integrity sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA== + +is-core-module@^2.13.0, is-core-module@^2.13.1: + version "2.13.1" + resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.13.1.tgz#ad0d7532c6fea9da1ebdc82742d74525c6273384" + integrity sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw== + dependencies: + hasown "^2.0.0" + +is-date-object@^1.0.1: + version "1.0.5" + resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.5.tgz#0841d5536e724c25597bf6ea62e1bd38298df31f" + integrity sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ== + dependencies: + has-tostringtag "^1.0.0" + +is-extglob@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" + integrity sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ== + +is-glob@^4.0.0, is-glob@^4.0.1, is-glob@^4.0.3: + version "4.0.3" + resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084" + integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg== + dependencies: + is-extglob "^2.1.1" + +is-negative-zero@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/is-negative-zero/-/is-negative-zero-2.0.2.tgz#7bf6f03a28003b8b3965de3ac26f664d765f3150" + integrity sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA== + +is-number-object@^1.0.4: + version "1.0.7" + resolved "https://registry.yarnpkg.com/is-number-object/-/is-number-object-1.0.7.tgz#59d50ada4c45251784e9904f5246c742f07a42fc" + integrity sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ== + dependencies: + has-tostringtag "^1.0.0" + +is-number@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" + integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== + +is-path-inside@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.3.tgz#d231362e53a07ff2b0e0ea7fed049161ffd16283" + integrity sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ== + +is-regex@^1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.1.4.tgz#eef5663cd59fa4c0ae339505323df6854bb15958" + integrity sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg== + dependencies: + call-bind "^1.0.2" + has-tostringtag "^1.0.0" + +is-shared-array-buffer@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz#8f259c573b60b6a32d4058a1a07430c0a7344c79" + integrity sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA== + dependencies: + call-bind "^1.0.2" + +is-string@^1.0.5, is-string@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/is-string/-/is-string-1.0.7.tgz#0dd12bf2006f255bb58f695110eff7491eebc0fd" + integrity sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg== + dependencies: + has-tostringtag "^1.0.0" + +is-symbol@^1.0.2, is-symbol@^1.0.3: + version "1.0.4" + resolved "https://registry.yarnpkg.com/is-symbol/-/is-symbol-1.0.4.tgz#a6dac93b635b063ca6872236de88910a57af139c" + integrity sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg== + dependencies: + has-symbols "^1.0.2" + +is-typed-array@^1.1.10, is-typed-array@^1.1.13, is-typed-array@^1.1.9: + version "1.1.13" + resolved "https://registry.yarnpkg.com/is-typed-array/-/is-typed-array-1.1.13.tgz#d6c5ca56df62334959322d7d7dd1cca50debe229" + integrity sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw== + dependencies: + which-typed-array "^1.1.14" + +is-weakref@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-weakref/-/is-weakref-1.0.2.tgz#9529f383a9338205e89765e0392efc2f100f06f2" + integrity sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ== + dependencies: + call-bind "^1.0.2" + +isarray@^2.0.5: + version "2.0.5" + resolved "https://registry.yarnpkg.com/isarray/-/isarray-2.0.5.tgz#8af1e4c1221244cc62459faf38940d4e644a5723" + integrity sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw== + +isexe@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" + integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw== + +js-yaml@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602" + integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA== + dependencies: + argparse "^2.0.1" + +json-buffer@3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.1.tgz#9338802a30d3b6605fbe0613e094008ca8c05a13" + integrity sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ== + +json-schema-traverse@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" + integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg== + +json-stable-stringify-without-jsonify@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz#9db7b59496ad3f3cfef30a75142d2d930ad72651" + integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw== + +json5@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.2.tgz#63d98d60f21b313b77c4d6da18bfa69d80e1d593" + integrity sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA== + dependencies: + minimist "^1.2.0" + +kafkajs@^2.2.4: + version "2.2.4" + resolved "https://registry.yarnpkg.com/kafkajs/-/kafkajs-2.2.4.tgz#59e6e16459d87fdf8b64be73970ed5aa42370a5b" + integrity sha512-j/YeapB1vfPT2iOIUn/vxdyKEuhuY2PxMBvf5JWux6iSaukAccrMtXEY/Lb7OvavDhOWME589bpLrEdnVHjfjA== + +keyv@^4.5.3: + version "4.5.4" + resolved "https://registry.yarnpkg.com/keyv/-/keyv-4.5.4.tgz#a879a99e29452f942439f2a405e3af8b31d4de93" + integrity sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw== + dependencies: + json-buffer "3.0.1" + +levn@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/levn/-/levn-0.4.1.tgz#ae4562c007473b932a6200d403268dd2fffc6ade" + integrity sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ== + dependencies: + prelude-ls "^1.2.1" + type-check "~0.4.0" + +locate-path@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286" + integrity sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw== + dependencies: + p-locate "^5.0.0" + +lodash.merge@^4.6.2: + version "4.6.2" + resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a" + integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ== + +lru-cache@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" + integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA== + dependencies: + yallist "^4.0.0" + +merge2@^1.3.0, merge2@^1.4.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.4.1.tgz#4368892f885e907455a6fd7dc55c0c9d404990ae" + integrity sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg== + +micromatch@^4.0.4: + version "4.0.5" + resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.5.tgz#bc8999a7cbbf77cdc89f132f6e467051b49090c6" + integrity sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA== + dependencies: + braces "^3.0.2" + picomatch "^2.3.1" + +minimatch@9.0.3: + version "9.0.3" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-9.0.3.tgz#a6e00c3de44c3a542bfaae70abfc22420a6da825" + integrity sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg== + dependencies: + brace-expansion "^2.0.1" + +minimatch@^3.0.5, minimatch@^3.1.1, minimatch@^3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" + integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw== + dependencies: + brace-expansion "^1.1.7" + +minimist@^1.2.0, minimist@^1.2.6: + version "1.2.8" + resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c" + integrity sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA== + +ms@2.1.2: + version "2.1.2" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" + integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== + +ms@^2.1.1: + version "2.1.3" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" + integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== + +natural-compare@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" + integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== + +object-inspect@^1.13.1: + version "1.13.1" + resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.13.1.tgz#b96c6109324ccfef6b12216a956ca4dc2ff94bc2" + integrity sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ== + +object-keys@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e" + integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA== + +object.assign@^4.1.5: + version "4.1.5" + resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.5.tgz#3a833f9ab7fdb80fc9e8d2300c803d216d8fdbb0" + integrity sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ== + dependencies: + call-bind "^1.0.5" + define-properties "^1.2.1" + has-symbols "^1.0.3" + object-keys "^1.1.1" + +object.fromentries@^2.0.7: + version "2.0.7" + resolved "https://registry.yarnpkg.com/object.fromentries/-/object.fromentries-2.0.7.tgz#71e95f441e9a0ea6baf682ecaaf37fa2a8d7e616" + integrity sha512-UPbPHML6sL8PI/mOqPwsH4G6iyXcCGzLin8KvEPenOZN5lpCNBZZQ+V62vdjB1mQHrmqGQt5/OJzemUA+KJmEA== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + +object.groupby@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/object.groupby/-/object.groupby-1.0.2.tgz#494800ff5bab78fd0eff2835ec859066e00192ec" + integrity sha512-bzBq58S+x+uo0VjurFT0UktpKHOZmv4/xePiOA1nbB9pMqpGK7rUPNgf+1YC+7mE+0HzhTMqNUuCqvKhj6FnBw== + dependencies: + array.prototype.filter "^1.0.3" + call-bind "^1.0.5" + define-properties "^1.2.1" + es-abstract "^1.22.3" + es-errors "^1.0.0" + +object.values@^1.1.7: + version "1.1.7" + resolved "https://registry.yarnpkg.com/object.values/-/object.values-1.1.7.tgz#617ed13272e7e1071b43973aa1655d9291b8442a" + integrity sha512-aU6xnDFYT3x17e/f0IiiwlGPTy2jzMySGfUB4fq6z7CV8l85CWHDk5ErhyhpfDHhrOMwGFhSQkhMGHaIotA6Ng== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + +once@^1.3.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" + integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w== + dependencies: + wrappy "1" + +optionator@^0.9.3: + version "0.9.3" + resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.9.3.tgz#007397d44ed1872fdc6ed31360190f81814e2c64" + integrity sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg== + dependencies: + "@aashutoshrathi/word-wrap" "^1.2.3" + deep-is "^0.1.3" + fast-levenshtein "^2.0.6" + levn "^0.4.1" + prelude-ls "^1.2.1" + type-check "^0.4.0" + +p-limit@^3.0.2: + version "3.1.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b" + integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ== + dependencies: + yocto-queue "^0.1.0" + +p-locate@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834" + integrity sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw== + dependencies: + p-limit "^3.0.2" + +parent-module@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/parent-module/-/parent-module-1.0.1.tgz#691d2709e78c79fae3a156622452d00762caaaa2" + integrity sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g== + dependencies: + callsites "^3.0.0" + +path-exists@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3" + integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w== + +path-is-absolute@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" + integrity sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg== + +path-key@^3.1.0: + version "3.1.1" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375" + integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q== + +path-parse@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.7.tgz#fbc114b60ca42b30d9daf5858e4bd68bbedb6735" + integrity sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw== + +path-type@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-type/-/path-type-4.0.0.tgz#84ed01c0a7ba380afe09d90a8c180dcd9d03043b" + integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw== + +picomatch@^2.3.1: + version "2.3.1" + resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" + integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== + +prelude-ls@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396" + integrity sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g== + +punycode@^2.1.0: + version "2.3.1" + resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.1.tgz#027422e2faec0b25e1549c3e1bd8309b9133b6e5" + integrity sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg== + +queue-microtask@^1.2.2: + version "1.2.3" + resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" + integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== + +regexp.prototype.flags@^1.5.2: + version "1.5.2" + resolved "https://registry.yarnpkg.com/regexp.prototype.flags/-/regexp.prototype.flags-1.5.2.tgz#138f644a3350f981a858c44f6bb1a61ff59be334" + integrity sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw== + dependencies: + call-bind "^1.0.6" + define-properties "^1.2.1" + es-errors "^1.3.0" + set-function-name "^2.0.1" + +resolve-from@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-4.0.0.tgz#4abcd852ad32dd7baabfe9b40e00a36db5f392e6" + integrity sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g== + +resolve@^1.22.4: + version "1.22.8" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.8.tgz#b6c87a9f2aa06dfab52e3d70ac8cde321fa5a48d" + integrity sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw== + dependencies: + is-core-module "^2.13.0" + path-parse "^1.0.7" + supports-preserve-symlinks-flag "^1.0.0" + +reusify@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76" + integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw== + +rimraf@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz#f1a5402ba6220ad52cc1282bac1ae3aa49fd061a" + integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA== + dependencies: + glob "^7.1.3" + +run-parallel@^1.1.9: + version "1.2.0" + resolved "https://registry.yarnpkg.com/run-parallel/-/run-parallel-1.2.0.tgz#66d1368da7bdf921eb9d95bd1a9229e7f21a43ee" + integrity sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA== + dependencies: + queue-microtask "^1.2.2" + +safe-array-concat@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/safe-array-concat/-/safe-array-concat-1.1.0.tgz#8d0cae9cb806d6d1c06e08ab13d847293ebe0692" + integrity sha512-ZdQ0Jeb9Ofti4hbt5lX3T2JcAamT9hfzYU1MNB+z/jaEbB6wfFfPIR/zEORmZqobkCCJhSjodobH6WHNmJ97dg== + dependencies: + call-bind "^1.0.5" + get-intrinsic "^1.2.2" + has-symbols "^1.0.3" + isarray "^2.0.5" + +safe-regex-test@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/safe-regex-test/-/safe-regex-test-1.0.3.tgz#a5b4c0f06e0ab50ea2c395c14d8371232924c377" + integrity sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw== + dependencies: + call-bind "^1.0.6" + es-errors "^1.3.0" + is-regex "^1.1.4" + +semver@^6.3.1: + version "6.3.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" + integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== + +semver@^7.5.4: + version "7.6.0" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.0.tgz#1a46a4db4bffcccd97b743b5005c8325f23d4e2d" + integrity sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg== + dependencies: + lru-cache "^6.0.0" + +set-function-length@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/set-function-length/-/set-function-length-1.2.1.tgz#47cc5945f2c771e2cf261c6737cf9684a2a5e425" + integrity sha512-j4t6ccc+VsKwYHso+kElc5neZpjtq9EnRICFZtWyBsLojhmeF/ZBd/elqm22WJh/BziDe/SBiOeAt0m2mfLD0g== + dependencies: + define-data-property "^1.1.2" + es-errors "^1.3.0" + function-bind "^1.1.2" + get-intrinsic "^1.2.3" + gopd "^1.0.1" + has-property-descriptors "^1.0.1" + +set-function-name@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/set-function-name/-/set-function-name-2.0.1.tgz#12ce38b7954310b9f61faa12701620a0c882793a" + integrity sha512-tMNCiqYVkXIZgc2Hnoy2IvC/f8ezc5koaRFkCjrpWzGpCd3qbZXPzVy9MAZzK1ch/X0jvSkojys3oqJN0qCmdA== + dependencies: + define-data-property "^1.0.1" + functions-have-names "^1.2.3" + has-property-descriptors "^1.0.0" + +shebang-command@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" + integrity sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA== + dependencies: + shebang-regex "^3.0.0" + +shebang-regex@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172" + integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A== + +side-channel@^1.0.4: + version "1.0.5" + resolved "https://registry.yarnpkg.com/side-channel/-/side-channel-1.0.5.tgz#9a84546599b48909fb6af1211708d23b1946221b" + integrity sha512-QcgiIWV4WV7qWExbN5llt6frQB/lBven9pqliLXfGPB+K9ZYXxDozp0wLkHS24kWCm+6YXH/f0HhnObZnZOBnQ== + dependencies: + call-bind "^1.0.6" + es-errors "^1.3.0" + get-intrinsic "^1.2.4" + object-inspect "^1.13.1" + +slash@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634" + integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== + +string.prototype.trim@^1.2.8: + version "1.2.8" + resolved "https://registry.yarnpkg.com/string.prototype.trim/-/string.prototype.trim-1.2.8.tgz#f9ac6f8af4bd55ddfa8895e6aea92a96395393bd" + integrity sha512-lfjY4HcixfQXOfaqCvcBuOIapyaroTXhbkfJN3gcB1OtyupngWK4sEET9Knd0cXd28kTUqu/kHoV4HKSJdnjiQ== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + +string.prototype.trimend@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/string.prototype.trimend/-/string.prototype.trimend-1.0.7.tgz#1bb3afc5008661d73e2dc015cd4853732d6c471e" + integrity sha512-Ni79DqeB72ZFq1uH/L6zJ+DKZTkOtPIHovb3YZHQViE+HDouuU4mBrLOLDn5Dde3RF8qw5qVETEjhu9locMLvA== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + +string.prototype.trimstart@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/string.prototype.trimstart/-/string.prototype.trimstart-1.0.7.tgz#d4cdb44b83a4737ffbac2d406e405d43d0184298" + integrity sha512-NGhtDFu3jCEm7B4Fy0DpLewdJQOZcQ0rGbwQ/+stjnrp2i+rlKeCvos9hOIeCmqwratM47OBxY7uFZzjxHXmrg== + dependencies: + call-bind "^1.0.2" + define-properties "^1.2.0" + es-abstract "^1.22.1" + +strip-ansi@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" + integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== + dependencies: + ansi-regex "^5.0.1" + +strip-bom@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3" + integrity sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA== + +strip-json-comments@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006" + integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig== + +supports-color@^7.1.0: + version "7.2.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" + integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== + dependencies: + has-flag "^4.0.0" + +supports-preserve-symlinks-flag@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09" + integrity sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w== + +text-table@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" + integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw== + +to-regex-range@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" + integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ== + dependencies: + is-number "^7.0.0" + +ts-api-utils@^1.0.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-1.2.1.tgz#f716c7e027494629485b21c0df6180f4d08f5e8b" + integrity sha512-RIYA36cJn2WiH9Hy77hdF9r7oEwxAtB/TS9/S4Qd90Ap4z5FSiin5zEiTL44OII1Y3IIlEvxwxFUVgrHSZ/UpA== + +tsconfig-paths@^3.15.0: + version "3.15.0" + resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz#5299ec605e55b1abb23ec939ef15edaf483070d4" + integrity sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg== + dependencies: + "@types/json5" "^0.0.29" + json5 "^1.0.2" + minimist "^1.2.6" + strip-bom "^3.0.0" + +type-check@^0.4.0, type-check@~0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/type-check/-/type-check-0.4.0.tgz#07b8203bfa7056c0657050e3ccd2c37730bab8f1" + integrity sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew== + dependencies: + prelude-ls "^1.2.1" + +type-fest@^0.20.2: + version "0.20.2" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.20.2.tgz#1bf207f4b28f91583666cb5fbd327887301cd5f4" + integrity sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ== + +typed-array-buffer@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/typed-array-buffer/-/typed-array-buffer-1.0.1.tgz#0608ffe6bca71bf15a45bff0ca2604107a1325f5" + integrity sha512-RSqu1UEuSlrBhHTWC8O9FnPjOduNs4M7rJ4pRKoEjtx1zUNOPN2sSXHLDX+Y2WPbHIxbvg4JFo2DNAEfPIKWoQ== + dependencies: + call-bind "^1.0.6" + es-errors "^1.3.0" + is-typed-array "^1.1.13" + +typed-array-byte-length@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/typed-array-byte-length/-/typed-array-byte-length-1.0.0.tgz#d787a24a995711611fb2b87a4052799517b230d0" + integrity sha512-Or/+kvLxNpeQ9DtSydonMxCx+9ZXOswtwJn17SNLvhptaXYDJvkFFP5zbfU/uLmvnBJlI4yrnXRxpdWH/M5tNA== + dependencies: + call-bind "^1.0.2" + for-each "^0.3.3" + has-proto "^1.0.1" + is-typed-array "^1.1.10" + +typed-array-byte-offset@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/typed-array-byte-offset/-/typed-array-byte-offset-1.0.1.tgz#5e2bcc1d93e1a332d50e8b363a48604a134692f8" + integrity sha512-tcqKMrTRXjqvHN9S3553NPCaGL0VPgFI92lXszmrE8DMhiDPLBYLlvo8Uu4WZAAX/aGqp/T1sbA4ph8EWjDF9Q== + dependencies: + available-typed-arrays "^1.0.6" + call-bind "^1.0.7" + for-each "^0.3.3" + gopd "^1.0.1" + has-proto "^1.0.1" + is-typed-array "^1.1.13" + +typed-array-length@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/typed-array-length/-/typed-array-length-1.0.4.tgz#89d83785e5c4098bec72e08b319651f0eac9c1bb" + integrity sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng== + dependencies: + call-bind "^1.0.2" + for-each "^0.3.3" + is-typed-array "^1.1.9" + +typescript@^5.3.3: + version "5.3.3" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.3.3.tgz#b3ce6ba258e72e6305ba66f5c9b452aaee3ffe37" + integrity sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw== + +unbox-primitive@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/unbox-primitive/-/unbox-primitive-1.0.2.tgz#29032021057d5e6cdbd08c5129c226dff8ed6f9e" + integrity sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw== + dependencies: + call-bind "^1.0.2" + has-bigints "^1.0.2" + has-symbols "^1.0.3" + which-boxed-primitive "^1.0.2" + +undici-types@~5.26.4: + version "5.26.5" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" + integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== + +uri-js@^4.2.2: + version "4.4.1" + resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" + integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== + dependencies: + punycode "^2.1.0" + +which-boxed-primitive@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz#13757bc89b209b049fe5d86430e21cf40a89a8e6" + integrity sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg== + dependencies: + is-bigint "^1.0.1" + is-boolean-object "^1.1.0" + is-number-object "^1.0.4" + is-string "^1.0.5" + is-symbol "^1.0.3" + +which-typed-array@^1.1.14: + version "1.1.14" + resolved "https://registry.yarnpkg.com/which-typed-array/-/which-typed-array-1.1.14.tgz#1f78a111aee1e131ca66164d8bdc3ab062c95a06" + integrity sha512-VnXFiIW8yNn9kIHN88xvZ4yOWchftKDsRJ8fEPacX/wl1lOvBrhsJ/OeJCXq7B0AaijRuqgzSKalJoPk+D8MPg== + dependencies: + available-typed-arrays "^1.0.6" + call-bind "^1.0.5" + for-each "^0.3.3" + gopd "^1.0.1" + has-tostringtag "^1.0.1" + +which@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" + integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA== + dependencies: + isexe "^2.0.0" + +wrappy@1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" + integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ== + +yallist@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" + integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== + +yocto-queue@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b" + integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q== From 24b1b123e71dde74cca820f13dfd1cdf9c943a1c Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Sun, 18 Feb 2024 12:41:53 +0100 Subject: [PATCH 02/12] Add prettier --- water/.prettierignore | 5 + water/.prettierrc.yml | 10 + water/package.json | 52 ++-- water/src/broker/BrokerClient.ts | 231 +++++++++--------- water/src/broker/BrokerMessage.ts | 23 +- water/src/broker/IBrokerConnection.ts | 27 +- water/src/broker/IBrokerMessageHeaders.ts | 18 +- water/src/broker/KafkaConnection.ts | 204 ---------------- water/src/broker/KafkaMessageHeaders.ts | 45 ---- water/src/broker/kafka/KafkaConnection.ts | 205 ++++++++++++++++ water/src/broker/kafka/KafkaMessageHeaders.ts | 42 ++++ water/src/index.ts | 4 +- water/src/logging/Logger.ts | 102 ++++---- water/src/util/CountDownLatch.ts | 20 +- water/tsconfig.json | 52 ++-- water/yarn.lock | 5 + 16 files changed, 513 insertions(+), 532 deletions(-) create mode 100644 water/.prettierignore create mode 100644 water/.prettierrc.yml delete mode 100644 water/src/broker/KafkaConnection.ts delete mode 100644 water/src/broker/KafkaMessageHeaders.ts create mode 100644 water/src/broker/kafka/KafkaConnection.ts create mode 100644 water/src/broker/kafka/KafkaMessageHeaders.ts diff --git a/water/.prettierignore b/water/.prettierignore new file mode 100644 index 0000000..e55f819 --- /dev/null +++ b/water/.prettierignore @@ -0,0 +1,5 @@ +/* +!/src +!eslintrc.cjs +!package.json +!tsconfig.json diff --git a/water/.prettierrc.yml b/water/.prettierrc.yml new file mode 100644 index 0000000..1d7a376 --- /dev/null +++ b/water/.prettierrc.yml @@ -0,0 +1,10 @@ +useTabs: true +singleQuote: false +jsxSingleQuote: false +bracketSameLine: true +quoteProps: as-needed +trailingComma: es5 +semi: true +printWidth: 120 +arrowParens: avoid +endOfLine: auto diff --git a/water/package.json b/water/package.json index 171d522..89f6022 100644 --- a/water/package.json +++ b/water/package.json @@ -1,27 +1,29 @@ { - "name": "@beemobot/water", - "version": "1.0.0", - "author": "Beemo Devs (https://beemo.gg)", - "license": "MIT", - "type": "module", - "main": "./dist/index.js", - "types": "./dist/index.d.ts", - "scripts": { - "build": "rm -rf dist/ && tsc", - "lint": "eslint ." - }, - "files": [ - "dist" - ], - "dependencies": { - "kafkajs": "^2.2.4" - }, - "devDependencies": { - "@types/node": "^20.11.19", - "@typescript-eslint/eslint-plugin": "^7.0.1", - "@typescript-eslint/parser": "^7.0.1", - "eslint": "^8.56.0", - "eslint-plugin-import": "^2.29.1", - "typescript": "^5.3.3" - } + "name": "@beemobot/water", + "version": "1.0.0", + "author": "Beemo Devs (https://beemo.gg)", + "license": "MIT", + "type": "module", + "main": "./dist/index.js", + "types": "./dist/index.d.ts", + "scripts": { + "build": "rm -rf dist/ && tsc", + "format": "prettier --write .", + "lint": "eslint ." + }, + "files": [ + "dist" + ], + "dependencies": { + "kafkajs": "^2.2.4" + }, + "devDependencies": { + "@types/node": "^20.11.19", + "@typescript-eslint/eslint-plugin": "^7.0.1", + "@typescript-eslint/parser": "^7.0.1", + "eslint": "^8.56.0", + "eslint-plugin-import": "^2.29.1", + "prettier": "^3.2.5", + "typescript": "^5.3.3" + } } diff --git a/water/src/broker/BrokerClient.ts b/water/src/broker/BrokerClient.ts index 3330146..21fa474 100644 --- a/water/src/broker/BrokerClient.ts +++ b/water/src/broker/BrokerClient.ts @@ -4,134 +4,121 @@ import type { IBrokerConnection } from "./IBrokerConnection.js"; import type { IBrokerMessageHeaders } from "./IBrokerMessageHeaders.js"; import { BrokerMessage } from "./BrokerMessage.js"; - const TAG = "BrokerClient"; export type BrokerEventListener = (msg: BrokerMessage) => void | Promise; export type BrokerMessageListener = (clusterId: string, msg: BrokerMessage) => void | Promise; export interface ClusterResult { - responses: Map>; - timeout: boolean; + responses: Map>; + timeout: boolean; } export class BrokerClient { - - private keyListeners: Map>> = new Map(); - - public constructor( - private readonly connection: IBrokerConnection, - private readonly topicName: string, - ) { - Logger.debug(TAG, `Initializing BrokerClient with topic '${topicName}'`); - connection.on(topicName, this.onTopicMessage.bind(this)); - } - - protected async send( - key: string, - obj: T | null, - headers: IBrokerMessageHeaders = this.connection.createHeaders(), - ): Promise { - return await this.connection.send(this.topicName, key, this.stringify(obj), headers); - } - - protected async sendClusterRequest( - key: string, - obj: T | null, - timeout: number = 0, - targetClusters: Set = new Set(), - expectedResponses: number | null = null, - messageCallback: BrokerMessageListener | null = null, - ): Promise> { - const responseKey = this.toResponseKey(key); - - const responses: Map> = new Map(); - const latch = new CountDownLatch(expectedResponses ?? targetClusters.size); - let requestId = ""; - - const cb: BrokerEventListener = (msg: BrokerMessage) => { - if (msg.headers.requestId !== requestId) { - return; - } - messageCallback?.(msg.headers.sourceCluster, msg) - ?.catch(e => Logger.error(TAG, "Uncaught error in sendClusterRequest message callback", e)); - - responses.set(msg.headers.sourceCluster, msg); - }; - - this.on(responseKey, cb); - let timeoutReached = false; - try { - requestId = await this.send( - key, - obj, - this.connection.createHeaders(targetClusters), - ); - - timeoutReached = !(await latch.await(timeout)); - } finally { - this.off(responseKey, cb); - } - - return { - responses, - timeout: timeoutReached, - }; - } - - protected on(key: string, cb: BrokerEventListener): void { - let listeners = this.keyListeners.get(key); - if (!listeners) { - listeners = new Set(); - this.keyListeners.set(key, listeners); - } - listeners.add(cb); - } - - protected off(key: string, cb: BrokerEventListener): void { - const listeners = this.keyListeners.get(key); - if (listeners) { - listeners.delete(cb); - if (!listeners.size) { - this.keyListeners.delete(key); - } - } - } - - // For internal use only. Do not call externally! - public async _respond(msg: BrokerMessage, data: T | null): Promise { - const newHeaders = this.connection.createHeaders( - new Set([msg.headers.sourceCluster]), - msg.headers.requestId, - ); - await this.send( - this.toResponseKey(msg.key), - data, - newHeaders, - ); - } - - private parse(json: string): T | null { - return JSON.parse(json) as T | null; - } - - private stringify(obj: T | null): string { - return JSON.stringify(obj); - } - - private async onTopicMessage(key: string, value: string, headers: IBrokerMessageHeaders): Promise { - const obj = this.parse(value); - const msg = new BrokerMessage(this, key, obj, headers); - const listeners = this.keyListeners.get(key); - if (!listeners || !listeners.size) { - return; - } - for (const listener of listeners) { - listener(msg)?.catch(e => Logger.error(TAG, "Uncaught error in BrokerClient listener", e)); - } - } - - private toResponseKey(key: string): string { - return `${key}-response`; - } - + private keyListeners: Map>> = new Map(); + + public constructor( + private readonly connection: IBrokerConnection, + private readonly topicName: string + ) { + Logger.debug(TAG, `Initializing BrokerClient with topic '${topicName}'`); + connection.on(topicName, this.onTopicMessage.bind(this)); + } + + protected async send( + key: string, + obj: T | null, + headers: IBrokerMessageHeaders = this.connection.createHeaders() + ): Promise { + return await this.connection.send(this.topicName, key, this.stringify(obj), headers); + } + + protected async sendClusterRequest( + key: string, + obj: T | null, + timeout: number = 0, + targetClusters: Set = new Set(), + expectedResponses: number | null = null, + messageCallback: BrokerMessageListener | null = null + ): Promise> { + const responseKey = this.toResponseKey(key); + + const responses: Map> = new Map(); + const latch = new CountDownLatch(expectedResponses ?? targetClusters.size); + let requestId = ""; + + const cb: BrokerEventListener = (msg: BrokerMessage) => { + if (msg.headers.requestId !== requestId) { + return; + } + messageCallback?.(msg.headers.sourceCluster, msg)?.catch(e => + Logger.error(TAG, "Uncaught error in sendClusterRequest message callback", e) + ); + + responses.set(msg.headers.sourceCluster, msg); + }; + + this.on(responseKey, cb); + let timeoutReached = false; + try { + requestId = await this.send(key, obj, this.connection.createHeaders(targetClusters)); + + timeoutReached = !(await latch.await(timeout)); + } finally { + this.off(responseKey, cb); + } + + return { + responses, + timeout: timeoutReached, + }; + } + + protected on(key: string, cb: BrokerEventListener): void { + let listeners = this.keyListeners.get(key); + if (!listeners) { + listeners = new Set(); + this.keyListeners.set(key, listeners); + } + listeners.add(cb); + } + + protected off(key: string, cb: BrokerEventListener): void { + const listeners = this.keyListeners.get(key); + if (listeners) { + listeners.delete(cb); + if (!listeners.size) { + this.keyListeners.delete(key); + } + } + } + + // For internal use only. Do not call externally! + public async _respond(msg: BrokerMessage, data: T | null): Promise { + const newHeaders = this.connection.createHeaders(new Set([msg.headers.sourceCluster]), msg.headers.requestId); + await this.send(this.toResponseKey(msg.key), data, newHeaders); + } + + private parse(json: string): T | null { + return JSON.parse(json) as T | null; + } + + private stringify(obj: T | null): string { + return JSON.stringify(obj); + } + + private async onTopicMessage(key: string, value: string, headers: IBrokerMessageHeaders): Promise { + const obj = this.parse(value); + const msg = new BrokerMessage(this, key, obj, headers); + const listeners = this.keyListeners.get(key); + if (!listeners || !listeners.size) { + return; + } + for (const listener of listeners) { + listener(msg)?.catch(e => Logger.error(TAG, "Uncaught error in BrokerClient listener", e)); + } + } + + private toResponseKey(key: string): string { + return `${key}-response`; + } } diff --git a/water/src/broker/BrokerMessage.ts b/water/src/broker/BrokerMessage.ts index 90d153f..6b520c1 100644 --- a/water/src/broker/BrokerMessage.ts +++ b/water/src/broker/BrokerMessage.ts @@ -1,18 +1,15 @@ import type { BrokerClient } from "./BrokerClient.js"; import type { IBrokerMessageHeaders } from "./IBrokerMessageHeaders.js"; - export class BrokerMessage { - - public constructor( - public readonly client: BrokerClient, - public readonly key: string, - public readonly value: T | null, - public readonly headers: IBrokerMessageHeaders, - ) { } - - public async respond(data: T | null): Promise { - await this.client._respond(this, data); - } - + public constructor( + public readonly client: BrokerClient, + public readonly key: string, + public readonly value: T | null, + public readonly headers: IBrokerMessageHeaders + ) {} + + public async respond(data: T | null): Promise { + await this.client._respond(this, data); + } } diff --git a/water/src/broker/IBrokerConnection.ts b/water/src/broker/IBrokerConnection.ts index 7b105f1..dacb0b5 100644 --- a/water/src/broker/IBrokerConnection.ts +++ b/water/src/broker/IBrokerConnection.ts @@ -1,29 +1,18 @@ import type { IBrokerMessageHeaders } from "./IBrokerMessageHeaders.js"; - export type BrokerTopicListener = (key: string, value: string, headers: IBrokerMessageHeaders) => void | Promise; export interface IBrokerConnection { + get clientId(): string; + get clusterId(): string; - get clientId(): string; - get clusterId(): string; - - start(): Promise; - destroy(): Promise; - - on(topic: string, cb: BrokerTopicListener): void; - off(topic: string, cb: BrokerTopicListener): void; + start(): Promise; + destroy(): Promise; - send( - topic: string, - key: string, - value: string, - headers: IBrokerMessageHeaders, - ): Promise; + on(topic: string, cb: BrokerTopicListener): void; + off(topic: string, cb: BrokerTopicListener): void; - createHeaders( - targetClusters?: Set, - requestId?: string, - ): IBrokerMessageHeaders; + send(topic: string, key: string, value: string, headers: IBrokerMessageHeaders): Promise; + createHeaders(targetClusters?: Set, requestId?: string): IBrokerMessageHeaders; } diff --git a/water/src/broker/IBrokerMessageHeaders.ts b/water/src/broker/IBrokerMessageHeaders.ts index 8df843f..1c1350a 100644 --- a/water/src/broker/IBrokerMessageHeaders.ts +++ b/water/src/broker/IBrokerMessageHeaders.ts @@ -1,15 +1,13 @@ export const BROKER_HEADERS = { - CLIENT_ID: "client-id", - REQUEST_ID: "request-id", - SOURCE_CLUSTER: "source-cluster", - TARGET_CLUSTERS: "target-clusters", + CLIENT_ID: "client-id", + REQUEST_ID: "request-id", + SOURCE_CLUSTER: "source-cluster", + TARGET_CLUSTERS: "target-clusters", }; export interface IBrokerMessageHeaders { - - get clientId(): string; - get sourceCluster(): string; - get targetClusters(): Set; - get requestId(): string; - + get clientId(): string; + get sourceCluster(): string; + get targetClusters(): Set; + get requestId(): string; } diff --git a/water/src/broker/KafkaConnection.ts b/water/src/broker/KafkaConnection.ts deleted file mode 100644 index a737221..0000000 --- a/water/src/broker/KafkaConnection.ts +++ /dev/null @@ -1,204 +0,0 @@ -import type { Consumer, KafkaMessage, Producer } from "kafkajs"; -import { Kafka, logLevel } from "kafkajs"; -import { Logger } from "../logging/Logger.js"; -import type { IBrokerConnection, BrokerTopicListener } from "./IBrokerConnection.js"; -import type { IBrokerMessageHeaders } from "./IBrokerMessageHeaders.js"; -import { KafkaMessageHeaders } from "./KafkaMessageHeaders.js"; - - -const TAG = "KafkaConnection"; - -export class KafkaConnection implements IBrokerConnection { - - private readonly kafka: Kafka; - private readonly producer: Producer; - private readonly consumer: Consumer; - - private readonly subscribedTopics: Set = new Set(); - - private readonly topicListeners: Map> = new Map(); - private isRunning: boolean = false; - - public constructor( - kafkaHost: string, - public readonly clientId: string, - consumerGroupId: string, - public readonly clusterId: string, - ) { - this.kafka = new Kafka({ - brokers: [kafkaHost], - clientId: consumerGroupId, - logCreator: () => ({ namespace, level, log }) => { - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const { message, timestamp, ...other } = log; - switch (level) { - case logLevel.DEBUG: - Logger.debug(TAG, `[${namespace}] ${message}`, other); - break; - case logLevel.INFO: - Logger.info(TAG, `[${namespace}] ${message}`, other); - break; - case logLevel.WARN: - Logger.warn(TAG, `[${namespace}] ${message}`, other); - break; - case logLevel.ERROR: - case logLevel.NOTHING: - Logger.error(TAG, `[${namespace}] ${message}`, other); - break; - } - }, - }); - - this.producer = this.kafka.producer({ - retry: { - retries: 2, - }, - idempotent: true, - }); - this.consumer = this.kafka.consumer({ - groupId: consumerGroupId, - }); - } - - public async send( - topic: string, - key: string, - value: string, - headers: KafkaMessageHeaders, - ): Promise { - await this.producer.send({ - topic, - messages: [{ - key, - value, - headers: headers.serialize(), - }], - }); - // If the record is meant for ourselves (amongst other clusters), - // immediately dispatch it to the listeners. - if (!headers.targetClusters.size || headers.targetClusters.has(this.clusterId)) { - await this.invokeCallbacks(topic, key, value, headers); - } - return headers.requestId; - } - - public createHeaders(targetClusters?: Set, requestId?: string): IBrokerMessageHeaders { - return new KafkaMessageHeaders(this.clientId, this.clusterId, targetClusters, requestId); - } - - public on(topic: string, cb: BrokerTopicListener): void { - let listeners = this.topicListeners.get(topic); - if (!listeners) { - if (this.isRunning) { - throw new Error("Cannot subscribe to new topic after KafkaConnection has started"); - } - listeners = new Set(); - this.subscribedTopics.add(topic); - this.topicListeners.set(topic, listeners); - } - listeners.add(cb); - } - - public off(topic: string, cb: BrokerTopicListener): void { - const listeners = this.topicListeners.get(topic); - if (listeners) { - listeners.delete(cb); - if (!listeners.size) { - this.subscribedTopics.delete(topic); - this.topicListeners.delete(topic); - } - } - } - - public async start(): Promise { - if (this.isRunning) { - throw new Error("KafkaConnection is already running!"); - } - Logger.debug(TAG, "Starting Kafka Connection"); - await this.createTopics(); - await this.createProducer(); - await this.createConsumer(); - this.isRunning = true; - } - - public async destroy(): Promise { - await this.consumer.disconnect(); - await this.producer.disconnect(); - this.subscribedTopics.clear(); - this.topicListeners.clear(); - this.isRunning = false; - } - - private async createTopics(): Promise { - const targetTopics = [...this.subscribedTopics]; - Logger.debug(TAG, `Creating missing topics, target topics: ${targetTopics}`); - const admin = this.kafka.admin(); - const existingTopics = await admin.listTopics(); - const missingTopics = targetTopics.filter(t => !existingTopics.includes(t)); - Logger.debug(TAG, `Missing topics: ${missingTopics}`); - if (missingTopics.length) { - Logger.info(TAG, "Creating missing topics"); - await admin.createTopics({ - topics: missingTopics.map(t => ({ - topic: t, - numPartitions: 1, - replicationFactor: 1, - })), - }); - } - Logger.debug(TAG, "Created all missing topics"); - } - - private async createProducer(): Promise { - Logger.debug(TAG, "Creating Producer"); - await this.producer.connect(); - } - - private async createConsumer(): Promise { - Logger.debug(TAG, "Creating Consumer"); - await this.consumer.connect(); - await this.consumer.subscribe({ - topics: [...this.subscribedTopics], - }); - // TODO Runtime error handling (such as for missing topics) - await this.consumer.run({ - eachMessage: async ({ topic, message: record }) => { - void this.handleIncomingRecord(topic, record); - }, - }); - } - - private async handleIncomingRecord(topic: string, record: KafkaMessage): Promise { - try { - const key = record.key?.toString(); - const value = record.value?.toString(); - if (!key || !value || !record.headers) { - return; - } - - const headers = KafkaMessageHeaders.from(record.headers); - if (headers.targetClusters.size && !headers.targetClusters.has(this.clusterId)) { - // If there is a target cluster restriction and this record wasn't meant for us, - // discard it immediately without notifying any listeners. - return; - } - if (headers.sourceCluster === this.clusterId) { - // If this record was sent by ourselves, discard it too, as we already dispatch events - // to our listeners in `send()` to avoid the round trip through Kafka. - return; - } - - await this.invokeCallbacks(topic, key, value, headers); - } catch (e) { - Logger.error(TAG, "Uncaught error in internal KafkaConnection record handler", e); - } - } - - private async invokeCallbacks(topic: string, key: string, value: string, headers: IBrokerMessageHeaders): Promise { - const listeners = this.topicListeners.get(topic) ?? []; - for (const listener of listeners) { - listener(key, value, headers)?.catch(e => Logger.error(TAG, "Uncaught error in KafkaConnection listener", e)); - } - } - -} diff --git a/water/src/broker/KafkaMessageHeaders.ts b/water/src/broker/KafkaMessageHeaders.ts deleted file mode 100644 index 29c0e7c..0000000 --- a/water/src/broker/KafkaMessageHeaders.ts +++ /dev/null @@ -1,45 +0,0 @@ -import { randomUUID } from "node:crypto"; -import type { IHeaders } from "kafkajs"; -import type { IBrokerMessageHeaders } from "./IBrokerMessageHeaders.js"; -import { BROKER_HEADERS } from "./IBrokerMessageHeaders.js"; - - -export class KafkaMessageHeaders implements IBrokerMessageHeaders { - - public static readonly INVALID_CLUSTER_ID = `${-Math.pow(2, 32)}`; - - public constructor( - public readonly clientId: string, - public readonly sourceCluster: string, - public readonly targetClusters: Set = new Set(), // Empty = No target restriction - public readonly requestId: string = randomUUID(), - ) { } - - public serialize(): IHeaders { - return { - [BROKER_HEADERS.CLIENT_ID]: this.clientId, - [BROKER_HEADERS.TARGET_CLUSTERS]: [...this.targetClusters].join(","), - [BROKER_HEADERS.REQUEST_ID]: this.requestId, - [BROKER_HEADERS.SOURCE_CLUSTER]: this.sourceCluster.toString(), - }; - } - - public static from(headers: IHeaders): KafkaMessageHeaders { - const transformed = KafkaMessageHeaders.headersBufferToString(headers); - return new KafkaMessageHeaders( - transformed[BROKER_HEADERS.CLIENT_ID] ?? "default", - transformed[BROKER_HEADERS.SOURCE_CLUSTER] || KafkaMessageHeaders.INVALID_CLUSTER_ID, - new Set((transformed[BROKER_HEADERS.TARGET_CLUSTERS] ?? "") - .split(",") - .filter(c => c)), - transformed[BROKER_HEADERS.REQUEST_ID] ?? "", - ); - } - - private static headersBufferToString(headers: IHeaders): Record { - return Object.fromEntries( - Object.entries(headers).filter(([_, v]) => v).map(([k, v]) => [k, v!.toString()]) - ); - } - -} diff --git a/water/src/broker/kafka/KafkaConnection.ts b/water/src/broker/kafka/KafkaConnection.ts new file mode 100644 index 0000000..dce42b5 --- /dev/null +++ b/water/src/broker/kafka/KafkaConnection.ts @@ -0,0 +1,205 @@ +import type { Consumer, KafkaMessage, Producer } from "kafkajs"; +import { Kafka, logLevel } from "kafkajs"; +import { Logger } from "../../logging/Logger.js"; +import type { IBrokerConnection, BrokerTopicListener } from "../IBrokerConnection.js"; +import type { IBrokerMessageHeaders } from "../IBrokerMessageHeaders.js"; +import { KafkaMessageHeaders } from "./KafkaMessageHeaders.js"; + +const TAG = "KafkaConnection"; + +export class KafkaConnection implements IBrokerConnection { + private readonly kafka: Kafka; + private readonly producer: Producer; + private readonly consumer: Consumer; + + private readonly subscribedTopics: Set = new Set(); + + private readonly topicListeners: Map> = new Map(); + private isRunning: boolean = false; + + public constructor( + kafkaHost: string, + public readonly clientId: string, + consumerGroupId: string, + public readonly clusterId: string + ) { + this.kafka = new Kafka({ + brokers: [kafkaHost], + clientId: consumerGroupId, + logCreator: + () => + ({ namespace, level, log }) => { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + const { message, timestamp, ...other } = log; + switch (level) { + case logLevel.DEBUG: + Logger.debug(TAG, `[${namespace}] ${message}`, other); + break; + case logLevel.INFO: + Logger.info(TAG, `[${namespace}] ${message}`, other); + break; + case logLevel.WARN: + Logger.warn(TAG, `[${namespace}] ${message}`, other); + break; + case logLevel.ERROR: + case logLevel.NOTHING: + Logger.error(TAG, `[${namespace}] ${message}`, other); + break; + } + }, + }); + + this.producer = this.kafka.producer({ + retry: { + retries: 2, + }, + idempotent: true, + }); + this.consumer = this.kafka.consumer({ + groupId: consumerGroupId, + }); + } + + public async send(topic: string, key: string, value: string, headers: KafkaMessageHeaders): Promise { + await this.producer.send({ + topic, + messages: [ + { + key, + value, + headers: headers.serialize(), + }, + ], + }); + // If the record is meant for ourselves (amongst other clusters), + // immediately dispatch it to the listeners. + if (!headers.targetClusters.size || headers.targetClusters.has(this.clusterId)) { + await this.invokeCallbacks(topic, key, value, headers); + } + return headers.requestId; + } + + public createHeaders(targetClusters?: Set, requestId?: string): IBrokerMessageHeaders { + return new KafkaMessageHeaders(this.clientId, this.clusterId, targetClusters, requestId); + } + + public on(topic: string, cb: BrokerTopicListener): void { + let listeners = this.topicListeners.get(topic); + if (!listeners) { + if (this.isRunning) { + throw new Error("Cannot subscribe to new topic after KafkaConnection has started"); + } + listeners = new Set(); + this.subscribedTopics.add(topic); + this.topicListeners.set(topic, listeners); + } + listeners.add(cb); + } + + public off(topic: string, cb: BrokerTopicListener): void { + const listeners = this.topicListeners.get(topic); + if (listeners) { + listeners.delete(cb); + if (!listeners.size) { + this.subscribedTopics.delete(topic); + this.topicListeners.delete(topic); + } + } + } + + public async start(): Promise { + if (this.isRunning) { + throw new Error("KafkaConnection is already running!"); + } + Logger.debug(TAG, "Starting Kafka Connection"); + await this.createTopics(); + await this.createProducer(); + await this.createConsumer(); + this.isRunning = true; + } + + public async destroy(): Promise { + await this.consumer.disconnect(); + await this.producer.disconnect(); + this.subscribedTopics.clear(); + this.topicListeners.clear(); + this.isRunning = false; + } + + private async createTopics(): Promise { + const targetTopics = [...this.subscribedTopics]; + Logger.debug(TAG, `Creating missing topics, target topics: ${targetTopics}`); + const admin = this.kafka.admin(); + const existingTopics = await admin.listTopics(); + const missingTopics = targetTopics.filter(t => !existingTopics.includes(t)); + Logger.debug(TAG, `Missing topics: ${missingTopics}`); + if (missingTopics.length) { + Logger.info(TAG, "Creating missing topics"); + await admin.createTopics({ + topics: missingTopics.map(t => ({ + topic: t, + numPartitions: 1, + replicationFactor: 1, + })), + }); + } + Logger.debug(TAG, "Created all missing topics"); + } + + private async createProducer(): Promise { + Logger.debug(TAG, "Creating Producer"); + await this.producer.connect(); + } + + private async createConsumer(): Promise { + Logger.debug(TAG, "Creating Consumer"); + await this.consumer.connect(); + await this.consumer.subscribe({ + topics: [...this.subscribedTopics], + }); + // TODO Runtime error handling (such as for missing topics) + await this.consumer.run({ + eachMessage: async ({ topic, message: record }) => { + void this.handleIncomingRecord(topic, record); + }, + }); + } + + private async handleIncomingRecord(topic: string, record: KafkaMessage): Promise { + try { + const key = record.key?.toString(); + const value = record.value?.toString(); + if (!key || !value || !record.headers) { + return; + } + + const headers = KafkaMessageHeaders.from(record.headers); + if (headers.targetClusters.size && !headers.targetClusters.has(this.clusterId)) { + // If there is a target cluster restriction and this record wasn't meant for us, + // discard it immediately without notifying any listeners. + return; + } + if (headers.sourceCluster === this.clusterId) { + // If this record was sent by ourselves, discard it too, as we already dispatch events + // to our listeners in `send()` to avoid the round trip through Kafka. + return; + } + + await this.invokeCallbacks(topic, key, value, headers); + } catch (e) { + Logger.error(TAG, "Uncaught error in internal KafkaConnection record handler", e); + } + } + + private async invokeCallbacks( + topic: string, + key: string, + value: string, + headers: IBrokerMessageHeaders + ): Promise { + const listeners = this.topicListeners.get(topic) ?? []; + for (const listener of listeners) { + listener(key, value, headers)?.catch(e => Logger.error(TAG, "Uncaught error in KafkaConnection listener", e)); + } + } +} diff --git a/water/src/broker/kafka/KafkaMessageHeaders.ts b/water/src/broker/kafka/KafkaMessageHeaders.ts new file mode 100644 index 0000000..ee05cac --- /dev/null +++ b/water/src/broker/kafka/KafkaMessageHeaders.ts @@ -0,0 +1,42 @@ +import { randomUUID } from "node:crypto"; +import type { IHeaders } from "kafkajs"; +import type { IBrokerMessageHeaders } from "../IBrokerMessageHeaders.js"; +import { BROKER_HEADERS } from "../IBrokerMessageHeaders.js"; + +export class KafkaMessageHeaders implements IBrokerMessageHeaders { + public static readonly INVALID_CLUSTER_ID = `${-Math.pow(2, 32)}`; + + public constructor( + public readonly clientId: string, + public readonly sourceCluster: string, + public readonly targetClusters: Set = new Set(), // Empty = No target restriction + public readonly requestId: string = randomUUID() + ) {} + + public serialize(): IHeaders { + return { + [BROKER_HEADERS.CLIENT_ID]: this.clientId, + [BROKER_HEADERS.TARGET_CLUSTERS]: [...this.targetClusters].join(","), + [BROKER_HEADERS.REQUEST_ID]: this.requestId, + [BROKER_HEADERS.SOURCE_CLUSTER]: this.sourceCluster.toString(), + }; + } + + public static from(headers: IHeaders): KafkaMessageHeaders { + const transformed = KafkaMessageHeaders.headersBufferToString(headers); + return new KafkaMessageHeaders( + transformed[BROKER_HEADERS.CLIENT_ID] ?? "default", + transformed[BROKER_HEADERS.SOURCE_CLUSTER] || KafkaMessageHeaders.INVALID_CLUSTER_ID, + new Set((transformed[BROKER_HEADERS.TARGET_CLUSTERS] ?? "").split(",").filter(c => c)), + transformed[BROKER_HEADERS.REQUEST_ID] ?? "" + ); + } + + private static headersBufferToString(headers: IHeaders): Record { + return Object.fromEntries( + Object.entries(headers) + .filter(([_, v]) => v) + .map(([k, v]) => [k, v!.toString()]) + ); + } +} diff --git a/water/src/index.ts b/water/src/index.ts index 2fc373c..4312229 100644 --- a/water/src/index.ts +++ b/water/src/index.ts @@ -2,8 +2,8 @@ export * from "./broker/BrokerClient.js"; export * from "./broker/BrokerMessage.js"; export * from "./broker/IBrokerConnection.js"; export * from "./broker/IBrokerMessageHeaders.js"; -export * from "./broker/KafkaConnection.js"; -export * from "./broker/KafkaMessageHeaders.js"; +export * from "./broker/kafka/KafkaConnection.js"; +export * from "./broker/kafka/KafkaMessageHeaders.js"; export * from "./logging/Logger.js"; diff --git a/water/src/logging/Logger.ts b/water/src/logging/Logger.ts index 78e8968..67974e6 100644 --- a/water/src/logging/Logger.ts +++ b/water/src/logging/Logger.ts @@ -1,63 +1,61 @@ import { inspect } from "node:util"; enum LogLevels { - VERBOSE, - DEBUG, - INFO, - WARN, - ERROR, - WTF, + VERBOSE, + DEBUG, + INFO, + WARN, + ERROR, + WTF, } const LevelNames = ["verbose", "debug", "info", "warn", "error", "assert"]; const LevelColors = [ - ["\u001b[97m", "\u001b[39m"], - ["\u001b[94m", "\u001b[39m"], - ["\u001b[92m", "\u001b[39m"], - ["\u001b[93m", "\u001b[39m"], - ["\u001b[91m", "\u001b[39m"], - ["\u001b[95m", "\u001b[39m"], + ["\u001b[97m", "\u001b[39m"], + ["\u001b[94m", "\u001b[39m"], + ["\u001b[92m", "\u001b[39m"], + ["\u001b[93m", "\u001b[39m"], + ["\u001b[91m", "\u001b[39m"], + ["\u001b[95m", "\u001b[39m"], ]; export class Logger { - - public static verbose(tag: string, message: string, error?: any): void { - Logger.log(LogLevels.VERBOSE, tag, message, error); - } - - public static debug(tag: string, message: string, error?: any): void { - Logger.log(LogLevels.DEBUG, tag, message, error); - } - - public static info(tag: string, message: string, error?: any): void { - Logger.log(LogLevels.INFO, tag, message, error); - } - - public static warn(tag: string, message: string, error?: any): void { - Logger.log(LogLevels.WARN, tag, message, error); - } - - public static error(tag: string, message: string, error?: any): void { - Logger.log(LogLevels.ERROR, tag, message, error); - } - - public static wtf(tag: string, message: string, error?: any): void { - Logger.log(LogLevels.WTF, tag, message, error); - process.exit(1); - } - - private static log(level: LogLevels, tag: string, message: string, error?: any): void { - const coloredLevel = Logger.getColoredLevelName(level); - const stringifiedError = `${error ? `${inspect(error, false, 4)}` : ""}`; - const stringifiedDate = `[${new Date().toISOString()}]`; - const consoleLogString = `${stringifiedDate} ${coloredLevel}/${tag}: ${message} ${stringifiedError}`.trim(); - // eslint-disable-next-line no-console - console.log(consoleLogString); - } - - private static getColoredLevelName(level: LogLevels): string { - const c = LevelColors[level] ?? []; - return `${c[0]}${LevelNames[level]}${c[1]}`; - } - + public static verbose(tag: string, message: string, error?: any): void { + Logger.log(LogLevels.VERBOSE, tag, message, error); + } + + public static debug(tag: string, message: string, error?: any): void { + Logger.log(LogLevels.DEBUG, tag, message, error); + } + + public static info(tag: string, message: string, error?: any): void { + Logger.log(LogLevels.INFO, tag, message, error); + } + + public static warn(tag: string, message: string, error?: any): void { + Logger.log(LogLevels.WARN, tag, message, error); + } + + public static error(tag: string, message: string, error?: any): void { + Logger.log(LogLevels.ERROR, tag, message, error); + } + + public static wtf(tag: string, message: string, error?: any): void { + Logger.log(LogLevels.WTF, tag, message, error); + process.exit(1); + } + + private static log(level: LogLevels, tag: string, message: string, error?: any): void { + const coloredLevel = Logger.getColoredLevelName(level); + const stringifiedError = `${error ? `${inspect(error, false, 4)}` : ""}`; + const stringifiedDate = `[${new Date().toISOString()}]`; + const consoleLogString = `${stringifiedDate} ${coloredLevel}/${tag}: ${message} ${stringifiedError}`.trim(); + // eslint-disable-next-line no-console + console.log(consoleLogString); + } + + private static getColoredLevelName(level: LogLevels): string { + const c = LevelColors[level] ?? []; + return `${c[0]}${LevelNames[level]}${c[1]}`; + } } diff --git a/water/src/util/CountDownLatch.ts b/water/src/util/CountDownLatch.ts index fb0a4dc..b3e9f9c 100644 --- a/water/src/util/CountDownLatch.ts +++ b/water/src/util/CountDownLatch.ts @@ -1,7 +1,6 @@ type PromiseResolver = (timeout: boolean) => void; export class CountDownLatch { - private counter: number; private resolvers = new Map(); @@ -42,20 +41,17 @@ export class CountDownLatch { } let resolver: PromiseResolver; - const promise = new Promise(r => { - resolver = r; - this.resolvers.set(r, null); - }); - if (!waitTime) { - return promise; - } - const timeout = setTimeout(() => { + const timeout = waitTime ? setTimeout(() => { this.resolvers.delete(resolver); resolver(false); - }, waitTime); - timeout.unref(); + }, waitTime) : null; + timeout?.unref(); + + const promise = new Promise(r => { + resolver = r; + this.resolvers.set(r, timeout); + }); return promise; } - } diff --git a/water/tsconfig.json b/water/tsconfig.json index 616c0e0..0e873c2 100644 --- a/water/tsconfig.json +++ b/water/tsconfig.json @@ -1,30 +1,26 @@ { - "compilerOptions": { - "target": "ES2022", - "module": "Node16", - "lib": [ - "ES2022" - ], - "moduleResolution": "Node16", - "allowJs": false, - "allowSyntheticDefaultImports": true, - "alwaysStrict": true, - "declaration": true, - "esModuleInterop": false, - "forceConsistentCasingInFileNames": true, - "noImplicitAny": true, - "noImplicitOverride": true, - "noImplicitReturns": true, - "noUncheckedIndexedAccess": true, - "resolveJsonModule": true, - "preserveConstEnums": true, - "skipLibCheck": true, - "sourceMap": true, - "strict": true, - "useDefineForClassFields": true, - "outDir": "dist" - }, - "include": [ - "src/**/*" - ] + "compilerOptions": { + "target": "ES2022", + "module": "Node16", + "lib": ["ES2022"], + "moduleResolution": "Node16", + "allowJs": false, + "allowSyntheticDefaultImports": true, + "alwaysStrict": true, + "declaration": true, + "esModuleInterop": false, + "forceConsistentCasingInFileNames": true, + "noImplicitAny": true, + "noImplicitOverride": true, + "noImplicitReturns": true, + "noUncheckedIndexedAccess": true, + "resolveJsonModule": true, + "preserveConstEnums": true, + "skipLibCheck": true, + "sourceMap": true, + "strict": true, + "useDefineForClassFields": true, + "outDir": "dist" + }, + "include": ["src/**/*"] } diff --git a/water/yarn.lock b/water/yarn.lock index 0e717e3..da380fe 100644 --- a/water/yarn.lock +++ b/water/yarn.lock @@ -1306,6 +1306,11 @@ prelude-ls@^1.2.1: resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396" integrity sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g== +prettier@^3.2.5: + version "3.2.5" + resolved "https://registry.yarnpkg.com/prettier/-/prettier-3.2.5.tgz#e52bc3090586e824964a8813b09aba6233b28368" + integrity sha512-3/GWa9aOC0YeD7LUfvOG2NiDyhOWRvt1k+rcKhOuYnMY24iiCphgneUfJDyFXd6rZCAnuLBv6UeAULtrhT/F4A== + punycode@^2.1.0: version "2.3.1" resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.1.tgz#027422e2faec0b25e1549c3e1bd8309b9133b6e5" From 95bb7b5867ea4e924e014cd5360a3bd9777fc705 Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Sun, 18 Feb 2024 14:56:29 +0100 Subject: [PATCH 03/12] Start porting new broker implementation from latte --- water/src/CommonConfig.ts | 7 ++ water/src/broker/BrokerClient.ts | 4 +- water/src/broker/BrokerConnection.ts | 111 ++++++++++++++++++ water/src/broker/BrokerMessage.ts | 2 +- water/src/broker/BrokerMessageHeaders.ts | 105 +++++++++++++++++ water/src/broker/Exceptions.ts | 23 ++++ water/src/broker/IBrokerConnection.ts | 18 --- water/src/broker/IBrokerMessageHeaders.ts | 13 -- water/src/broker/kafka/KafkaConnection.ts | 10 +- water/src/broker/kafka/KafkaMessageHeaders.ts | 4 +- water/src/index.ts | 6 +- 11 files changed, 260 insertions(+), 43 deletions(-) create mode 100644 water/src/CommonConfig.ts create mode 100644 water/src/broker/BrokerConnection.ts create mode 100644 water/src/broker/BrokerMessageHeaders.ts create mode 100644 water/src/broker/Exceptions.ts delete mode 100644 water/src/broker/IBrokerConnection.ts delete mode 100644 water/src/broker/IBrokerMessageHeaders.ts diff --git a/water/src/CommonConfig.ts b/water/src/CommonConfig.ts new file mode 100644 index 0000000..ffe8736 --- /dev/null +++ b/water/src/CommonConfig.ts @@ -0,0 +1,7 @@ +export enum BrokerServices { + TEA = "tea", // Bot + VANILLA = "vanilla", // Bot cluster coordinator + MILK = "milk", // Raid logs + SUGAR = "sugar", // Premium management + COFFEE = "coffee" // Raid bans +} diff --git a/water/src/broker/BrokerClient.ts b/water/src/broker/BrokerClient.ts index 21fa474..723f951 100644 --- a/water/src/broker/BrokerClient.ts +++ b/water/src/broker/BrokerClient.ts @@ -1,7 +1,7 @@ import { Logger } from "../logging/Logger.js"; import { CountDownLatch } from "../util/CountDownLatch.js"; -import type { IBrokerConnection } from "./IBrokerConnection.js"; -import type { IBrokerMessageHeaders } from "./IBrokerMessageHeaders.js"; +import type { IBrokerConnection } from "./BrokerConnection.js"; +import type { IBrokerMessageHeaders } from "./BrokerMessageHeaders.js"; import { BrokerMessage } from "./BrokerMessage.js"; const TAG = "BrokerClient"; diff --git a/water/src/broker/BrokerConnection.ts b/water/src/broker/BrokerConnection.ts new file mode 100644 index 0000000..f075984 --- /dev/null +++ b/water/src/broker/BrokerConnection.ts @@ -0,0 +1,111 @@ +import { Logger } from "../index.js"; +import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; + + +export type TopicListener = (topic: string, key: string, value: string, headers: BrokerMessageHeaders) => void; + +export type MessageId = string; + +const TAG = "BrokerConnection"; + +export abstract class BrokerConnection { + + public abstract serviceName: string; + public abstract instanceId: string; + public abstract supportsTopicHotSwap: boolean; + + protected topicListeners: Map> = new Map(); + + public abstract start(): Promise; + public async destroy(): Promise { + this.topicListeners.clear(); + } + + public abstract abstractSend(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise; + + public send(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { + Logger.verbose(TAG, `Sending message ${headers.messageId} with key '${key}' in topic '${topic}'`); + return this.abstractSend(topic, key, value, headers); + } + + protected abstract createTopic(topic: string): void; + protected abstract removeTopic(topic: string): void; + + public on(topic: string, cb: TopicListener): void { + let listeners = this.topicListeners.get(topic); + if (!listeners) { + listeners = new Set(); + Logger.debug(TAG, `Creating new topic '${topic}'`); + this.createTopic(topic); + this.topicListeners.set(topic, listeners); + } + listeners.add(cb); + } + + public off(topic: string, cb: TopicListener): void { + const listeners = this.topicListeners.get(topic); + if (listeners) { + listeners.delete(cb); + if (listeners.size === 0) { + Logger.debug(TAG, `Removing topic '${topic}'`); + this.removeTopic(topic); + this.topicListeners.delete(topic); + } + } + } + + // To be called by implementers + protected dispatchIncomingMessage(topic: string, key: string, value: string, headers: BrokerMessageHeaders): void { + if ( + (headers.targetServices.size >= 1 && !headers.targetServices.has(this.serviceName)) || + (headers.targetInstances.size >= 1 && !headers.targetInstances.has(this.instanceId)) + ) { + // If there is a target cluster restriction and this message wasn't meant for us, + // discard it immediately without notifying any listeners. + return; + } + if (headers.sourceInstance === this.instanceId && headers.sourceService === this.serviceName) { + // If this message was sent by ourselves, discard it too, as we already dispatch events + // to our listeners in `send()` to avoid the round trip through an external service. + return; + } + this.invokeLocalCallbacks(topic, key, value, headers); + } + + protected shouldDispatchExternallyAfterShortCircuit(topic: string, key: string, value: string, headers: BrokerMessageHeaders): boolean { + const targetServices = headers.targetServices; + const targetInstances = headers.targetInstances; + const isThisConnectionTargeted = (targetServices.size === 0 || targetServices.has(this.serviceName)) && + (targetInstances.size === 0 || targetInstances.has(this.instanceId)); + + // If the message is meant for ourselves (amongst other clusters), + // immediately dispatch it to the listeners. + if (isThisConnectionTargeted) { + this.invokeLocalCallbacks(topic, key, value, headers); + } + + // Return whether implementers should dispatch this message to external services + return ( + // For all services/instances + targetServices.size === 0 || targetInstances.size === 0 || + // Not for us, so it must be for somebody else + !isThisConnectionTargeted || + // For us, so check if it is also for someone else + targetServices.size > 1 || targetInstances.size > 1 + ); + } + + private invokeLocalCallbacks(topic: string, key: string, value: string, headers: BrokerMessageHeaders): void { + const listeners = this.topicListeners.get(topic); + if (listeners) { + for (const listener of listeners) { + try { + listener(topic, key, value, headers); + } catch (e) { + Logger.error(TAG, `Uncaught error in BrokerConnection listener for key '${key}' in topic '${topic}'`, e); + } + } + } + } + +} diff --git a/water/src/broker/BrokerMessage.ts b/water/src/broker/BrokerMessage.ts index 6b520c1..54d803d 100644 --- a/water/src/broker/BrokerMessage.ts +++ b/water/src/broker/BrokerMessage.ts @@ -1,5 +1,5 @@ import type { BrokerClient } from "./BrokerClient.js"; -import type { IBrokerMessageHeaders } from "./IBrokerMessageHeaders.js"; +import type { IBrokerMessageHeaders } from "./BrokerMessageHeaders.js"; export class BrokerMessage { public constructor( diff --git a/water/src/broker/BrokerMessageHeaders.ts b/water/src/broker/BrokerMessageHeaders.ts new file mode 100644 index 0000000..809f35c --- /dev/null +++ b/water/src/broker/BrokerMessageHeaders.ts @@ -0,0 +1,105 @@ +import { randomUUID } from "node:crypto"; +import { BrokerConnection } from "./BrokerConnection.js"; +import type { MessageId } from "./BrokerConnection.js"; + +export const BROKER_HEADER_NAMES = { + SOURCE_SERVICE: "source-service", + SOURCE_INSTANCE: "source-instance", + TARGET_SERVICES: "target-services", + TARGET_INSTANCES: "target-instances", + MESSAGE_ID: "message-id", +}; + +export class BrokerMessageHeaders { + + public readonly headers: Record; + public readonly sourceService: string; + public readonly sourceInstance: string; + public readonly targetServices: Set; + public readonly targetInstances: Set; + public readonly messageId: MessageId; + + public constructor( + sourceService: string, + sourceInstance: string, + targetServices: Set, + targetInstances: Set, + ); + public constructor( + connection: BrokerConnection, + targetServices: Set, + targetInstances: Set, + ); + public constructor(headers: Record); + public constructor( + sourceServiceOrHeaders: string | BrokerConnection | Record, + sourceInstanceOrTargetServices?: string | Set, + targetServicesOrTargetInstances?: Set, + targetInstances?: Set, + ) { + // I hate that JS doesn't have native method overloading + if (typeof sourceServiceOrHeaders === "string") { + this.sourceService = sourceServiceOrHeaders; + this.sourceInstance = sourceInstanceOrTargetServices as string; + this.targetServices = targetServicesOrTargetInstances as Set; + this.targetInstances = targetInstances as Set; + this.messageId = randomUUID(); + this.headers = BrokerMessageHeaders.createHeadersMap( + this.sourceService, + this.sourceInstance, + this.targetServices, + this.targetInstances, + this.messageId, + ); + } else if (sourceServiceOrHeaders instanceof BrokerConnection) { + const connection: BrokerConnection = sourceServiceOrHeaders; + this.sourceService = connection.serviceName; + this.sourceInstance = connection.instanceId; + this.targetServices = sourceInstanceOrTargetServices as Set; + this.targetInstances = targetServicesOrTargetInstances as Set; + this.messageId = randomUUID(); + this.headers = BrokerMessageHeaders.createHeadersMap( + this.sourceService, + this.sourceInstance, + this.targetServices, + this.targetInstances, + this.messageId, + ); + } else { + this.headers = sourceServiceOrHeaders; + this.sourceService = BrokerMessageHeaders.getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_SERVICE); + this.sourceInstance = BrokerMessageHeaders.getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_INSTANCE); + this.targetServices = new Set((this.headers[BROKER_HEADER_NAMES.TARGET_SERVICES] ?? "").split(",").filter(s => s)); + this.targetInstances = new Set((this.headers[BROKER_HEADER_NAMES.TARGET_INSTANCES] ?? "").split(",").filter(s => s)); + this.messageId = BrokerMessageHeaders.getOrThrow(this.headers, BROKER_HEADER_NAMES.MESSAGE_ID); + } + } + + protected static createHeadersMap( + sourceService: string, + sourceInstance: string, + targetServices: Set, + targetInstances: Set, + messageId: MessageId | null, + extra: Record = {}, + ): Record { + const headers: Record = { + [BROKER_HEADER_NAMES.SOURCE_SERVICE]: sourceService, + [BROKER_HEADER_NAMES.SOURCE_INSTANCE]: sourceInstance, + [BROKER_HEADER_NAMES.TARGET_SERVICES]: [...targetServices].join(","), + [BROKER_HEADER_NAMES.TARGET_INSTANCES]: [...targetInstances].join(","), + [BROKER_HEADER_NAMES.MESSAGE_ID]: messageId ?? randomUUID(), + ...extra, + }; + return headers; + } + + private static getOrThrow(headers: Record, key: string): string { + const value = headers[key]; + if (value === undefined) { + throw new Error(`Missing broker message header '${key}'`); + } + return value; + } + +} diff --git a/water/src/broker/Exceptions.ts b/water/src/broker/Exceptions.ts new file mode 100644 index 0000000..43b005f --- /dev/null +++ b/water/src/broker/Exceptions.ts @@ -0,0 +1,23 @@ +export class BrokerException extends Error { + public constructor(message?: string) { + super(message); + } +} + +export class RpcRequestTimeout extends BrokerException { + public constructor(message: string) { + super(message); + } +} + +export class IgnoreRpcRequest extends BrokerException { + public constructor() { + super("Ignoring RPC request"); + } +} + +export class RpcException extends BrokerException { + public constructor(public readonly status: RpcStatus) { + super(`RPC failed with status ${status}`); + } +} diff --git a/water/src/broker/IBrokerConnection.ts b/water/src/broker/IBrokerConnection.ts deleted file mode 100644 index dacb0b5..0000000 --- a/water/src/broker/IBrokerConnection.ts +++ /dev/null @@ -1,18 +0,0 @@ -import type { IBrokerMessageHeaders } from "./IBrokerMessageHeaders.js"; - -export type BrokerTopicListener = (key: string, value: string, headers: IBrokerMessageHeaders) => void | Promise; - -export interface IBrokerConnection { - get clientId(): string; - get clusterId(): string; - - start(): Promise; - destroy(): Promise; - - on(topic: string, cb: BrokerTopicListener): void; - off(topic: string, cb: BrokerTopicListener): void; - - send(topic: string, key: string, value: string, headers: IBrokerMessageHeaders): Promise; - - createHeaders(targetClusters?: Set, requestId?: string): IBrokerMessageHeaders; -} diff --git a/water/src/broker/IBrokerMessageHeaders.ts b/water/src/broker/IBrokerMessageHeaders.ts deleted file mode 100644 index 1c1350a..0000000 --- a/water/src/broker/IBrokerMessageHeaders.ts +++ /dev/null @@ -1,13 +0,0 @@ -export const BROKER_HEADERS = { - CLIENT_ID: "client-id", - REQUEST_ID: "request-id", - SOURCE_CLUSTER: "source-cluster", - TARGET_CLUSTERS: "target-clusters", -}; - -export interface IBrokerMessageHeaders { - get clientId(): string; - get sourceCluster(): string; - get targetClusters(): Set; - get requestId(): string; -} diff --git a/water/src/broker/kafka/KafkaConnection.ts b/water/src/broker/kafka/KafkaConnection.ts index dce42b5..6d95600 100644 --- a/water/src/broker/kafka/KafkaConnection.ts +++ b/water/src/broker/kafka/KafkaConnection.ts @@ -1,8 +1,8 @@ import type { Consumer, KafkaMessage, Producer } from "kafkajs"; import { Kafka, logLevel } from "kafkajs"; import { Logger } from "../../logging/Logger.js"; -import type { IBrokerConnection, BrokerTopicListener } from "../IBrokerConnection.js"; -import type { IBrokerMessageHeaders } from "../IBrokerMessageHeaders.js"; +import type { IBrokerConnection, TopicListener } from "../BrokerConnection.js"; +import type { IBrokerMessageHeaders } from "../BrokerMessageHeaders.js"; import { KafkaMessageHeaders } from "./KafkaMessageHeaders.js"; const TAG = "KafkaConnection"; @@ -14,7 +14,7 @@ export class KafkaConnection implements IBrokerConnection { private readonly subscribedTopics: Set = new Set(); - private readonly topicListeners: Map> = new Map(); + private readonly topicListeners: Map> = new Map(); private isRunning: boolean = false; public constructor( @@ -83,7 +83,7 @@ export class KafkaConnection implements IBrokerConnection { return new KafkaMessageHeaders(this.clientId, this.clusterId, targetClusters, requestId); } - public on(topic: string, cb: BrokerTopicListener): void { + public on(topic: string, cb: TopicListener): void { let listeners = this.topicListeners.get(topic); if (!listeners) { if (this.isRunning) { @@ -96,7 +96,7 @@ export class KafkaConnection implements IBrokerConnection { listeners.add(cb); } - public off(topic: string, cb: BrokerTopicListener): void { + public off(topic: string, cb: TopicListener): void { const listeners = this.topicListeners.get(topic); if (listeners) { listeners.delete(cb); diff --git a/water/src/broker/kafka/KafkaMessageHeaders.ts b/water/src/broker/kafka/KafkaMessageHeaders.ts index ee05cac..c8ad9bf 100644 --- a/water/src/broker/kafka/KafkaMessageHeaders.ts +++ b/water/src/broker/kafka/KafkaMessageHeaders.ts @@ -1,7 +1,7 @@ import { randomUUID } from "node:crypto"; import type { IHeaders } from "kafkajs"; -import type { IBrokerMessageHeaders } from "../IBrokerMessageHeaders.js"; -import { BROKER_HEADERS } from "../IBrokerMessageHeaders.js"; +import type { IBrokerMessageHeaders } from "../BrokerMessageHeaders.js"; +import { BROKER_HEADERS } from "../BrokerMessageHeaders.js"; export class KafkaMessageHeaders implements IBrokerMessageHeaders { public static readonly INVALID_CLUSTER_ID = `${-Math.pow(2, 32)}`; diff --git a/water/src/index.ts b/water/src/index.ts index 4312229..4667efa 100644 --- a/water/src/index.ts +++ b/water/src/index.ts @@ -1,7 +1,9 @@ +export * from "./CommonConfig.js"; + export * from "./broker/BrokerClient.js"; export * from "./broker/BrokerMessage.js"; -export * from "./broker/IBrokerConnection.js"; -export * from "./broker/IBrokerMessageHeaders.js"; +export * from "./broker/BrokerConnection.js"; +export * from "./broker/BrokerMessageHeaders.js"; export * from "./broker/kafka/KafkaConnection.js"; export * from "./broker/kafka/KafkaMessageHeaders.js"; From 3187900440a4133353953a644ad891d6400c3f9f Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Sun, 14 Jul 2024 18:12:08 +0200 Subject: [PATCH 04/12] Upgrade dependencies --- water/package.json | 12 +-- water/yarn.lock | 234 ++++++++++++++++++++------------------------- 2 files changed, 111 insertions(+), 135 deletions(-) diff --git a/water/package.json b/water/package.json index 89f6022..8774dd7 100644 --- a/water/package.json +++ b/water/package.json @@ -18,12 +18,12 @@ "kafkajs": "^2.2.4" }, "devDependencies": { - "@types/node": "^20.11.19", - "@typescript-eslint/eslint-plugin": "^7.0.1", - "@typescript-eslint/parser": "^7.0.1", - "eslint": "^8.56.0", + "@types/node": "^20.14.10", + "@typescript-eslint/eslint-plugin": "^7.16.0", + "@typescript-eslint/parser": "^7.16.0", + "eslint": "^8.57.0", "eslint-plugin-import": "^2.29.1", - "prettier": "^3.2.5", - "typescript": "^5.3.3" + "prettier": "^3.3.3", + "typescript": "^5.5.3" } } diff --git a/water/yarn.lock b/water/yarn.lock index da380fe..da355f4 100644 --- a/water/yarn.lock +++ b/water/yarn.lock @@ -14,7 +14,12 @@ dependencies: eslint-visitor-keys "^3.3.0" -"@eslint-community/regexpp@^4.5.1", "@eslint-community/regexpp@^4.6.1": +"@eslint-community/regexpp@^4.10.0": + version "4.11.0" + resolved "https://registry.yarnpkg.com/@eslint-community/regexpp/-/regexpp-4.11.0.tgz#b0ffd0312b4a3fd2d6f77237e7248a5ad3a680ae" + integrity sha512-G/M/tIiMrTAxEWRfLfQJMmGNX28IxBg4PBz8XqQhqUHLFI6TL2htpIB1iQCj144V5ee/JaKyT9/WZ0MGZWfA7A== + +"@eslint-community/regexpp@^4.6.1": version "4.10.0" resolved "https://registry.yarnpkg.com/@eslint-community/regexpp/-/regexpp-4.10.0.tgz#548f6de556857c8bb73bbee70c35dc82a2e74d63" integrity sha512-Cu96Sd2By9mCNTx2iyKOmq10v22jUVQv0lQnlGNy16oE9589yE+QADPbrMGCkA51cKZSg3Pu/aTJVTGfL/qjUA== @@ -34,12 +39,12 @@ minimatch "^3.1.2" strip-json-comments "^3.1.1" -"@eslint/js@8.56.0": - version "8.56.0" - resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.56.0.tgz#ef20350fec605a7f7035a01764731b2de0f3782b" - integrity sha512-gMsVel9D7f2HLkBma9VbtzZRehRogVRfbr++f06nL2vnCGCNlzOD+/MUov/F4p8myyAHspEhVobgjpX64q5m6A== +"@eslint/js@8.57.0": + version "8.57.0" + resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.57.0.tgz#a5417ae8427873f1dd08b70b3574b453e67b5f7f" + integrity sha512-Ys+3g2TaW7gADOJzPt83SJtCDhMjndcDMFVQ/Tj9iA1BfJzFKD9mAUXT3OenpuPHbI6P/myECxRJrofUsDx/5g== -"@humanwhocodes/config-array@^0.11.13": +"@humanwhocodes/config-array@^0.11.14": version "0.11.14" resolved "https://registry.yarnpkg.com/@humanwhocodes/config-array/-/config-array-0.11.14.tgz#d78e481a039f7566ecc9660b4ea7fe6b1fec442b" integrity sha512-3T8LkOmg45BV5FICb15QQMsyUSWrQ8AygVfC7ZG32zOalnqrilm018ZVCw0eapXux8FtA33q8PSRSstjee3jSg== @@ -79,113 +84,98 @@ "@nodelib/fs.scandir" "2.1.5" fastq "^1.6.0" -"@types/json-schema@^7.0.12": - version "7.0.15" - resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.15.tgz#596a1747233694d50f6ad8a7869fcb6f56cf5841" - integrity sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA== - "@types/json5@^0.0.29": version "0.0.29" resolved "https://registry.yarnpkg.com/@types/json5/-/json5-0.0.29.tgz#ee28707ae94e11d2b827bcbe5270bcea7f3e71ee" integrity sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ== -"@types/node@^20.11.19": - version "20.11.19" - resolved "https://registry.yarnpkg.com/@types/node/-/node-20.11.19.tgz#b466de054e9cb5b3831bee38938de64ac7f81195" - integrity sha512-7xMnVEcZFu0DikYjWOlRq7NTPETrm7teqUT2WkQjrTIkEgUyyGdWsj/Zg8bEJt5TNklzbPD1X3fqfsHw3SpapQ== +"@types/node@^20.14.10": + version "20.14.10" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.14.10.tgz#a1a218290f1b6428682e3af044785e5874db469a" + integrity sha512-MdiXf+nDuMvY0gJKxyfZ7/6UFsETO7mGKF54MVD/ekJS6HdFtpZFBgrh6Pseu64XTb2MLyFPlbW6hj8HYRQNOQ== dependencies: undici-types "~5.26.4" -"@types/semver@^7.5.0": - version "7.5.7" - resolved "https://registry.yarnpkg.com/@types/semver/-/semver-7.5.7.tgz#326f5fdda70d13580777bcaa1bc6fa772a5aef0e" - integrity sha512-/wdoPq1QqkSj9/QOeKkFquEuPzQbHTWAMPH/PaUMB+JuR31lXhlWXRZ52IpfDYVlDOUBvX09uBrPwxGT1hjNBg== - -"@typescript-eslint/eslint-plugin@^7.0.1": - version "7.0.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.0.1.tgz#407daffe09d964d57aceaf3ac51846359fbe61b0" - integrity sha512-OLvgeBv3vXlnnJGIAgCLYKjgMEU+wBGj07MQ/nxAaON+3mLzX7mJbhRYrVGiVvFiXtwFlkcBa/TtmglHy0UbzQ== - dependencies: - "@eslint-community/regexpp" "^4.5.1" - "@typescript-eslint/scope-manager" "7.0.1" - "@typescript-eslint/type-utils" "7.0.1" - "@typescript-eslint/utils" "7.0.1" - "@typescript-eslint/visitor-keys" "7.0.1" - debug "^4.3.4" +"@typescript-eslint/eslint-plugin@^7.16.0": + version "7.16.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.16.0.tgz#b3563927341eca15124a18c6f94215f779f5c02a" + integrity sha512-py1miT6iQpJcs1BiJjm54AMzeuMPBSPuKPlnT8HlfudbcS5rYeX5jajpLf3mrdRh9dA/Ec2FVUY0ifeVNDIhZw== + dependencies: + "@eslint-community/regexpp" "^4.10.0" + "@typescript-eslint/scope-manager" "7.16.0" + "@typescript-eslint/type-utils" "7.16.0" + "@typescript-eslint/utils" "7.16.0" + "@typescript-eslint/visitor-keys" "7.16.0" graphemer "^1.4.0" - ignore "^5.2.4" + ignore "^5.3.1" natural-compare "^1.4.0" - semver "^7.5.4" - ts-api-utils "^1.0.1" + ts-api-utils "^1.3.0" -"@typescript-eslint/parser@^7.0.1": - version "7.0.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-7.0.1.tgz#e9c61d9a5e32242477d92756d36086dc40322eed" - integrity sha512-8GcRRZNzaHxKzBPU3tKtFNing571/GwPBeCvmAUw0yBtfE2XVd0zFKJIMSWkHJcPQi0ekxjIts6L/rrZq5cxGQ== +"@typescript-eslint/parser@^7.16.0": + version "7.16.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-7.16.0.tgz#53fae8112f8c912024aea7b499cf7374487af6d8" + integrity sha512-ar9E+k7CU8rWi2e5ErzQiC93KKEFAXA2Kky0scAlPcxYblLt8+XZuHUZwlyfXILyQa95P6lQg+eZgh/dDs3+Vw== dependencies: - "@typescript-eslint/scope-manager" "7.0.1" - "@typescript-eslint/types" "7.0.1" - "@typescript-eslint/typescript-estree" "7.0.1" - "@typescript-eslint/visitor-keys" "7.0.1" + "@typescript-eslint/scope-manager" "7.16.0" + "@typescript-eslint/types" "7.16.0" + "@typescript-eslint/typescript-estree" "7.16.0" + "@typescript-eslint/visitor-keys" "7.16.0" debug "^4.3.4" -"@typescript-eslint/scope-manager@7.0.1": - version "7.0.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-7.0.1.tgz#611ec8e78c70439b152a805e1b10aaac36de7c00" - integrity sha512-v7/T7As10g3bcWOOPAcbnMDuvctHzCFYCG/8R4bK4iYzdFqsZTbXGln0cZNVcwQcwewsYU2BJLay8j0/4zOk4w== +"@typescript-eslint/scope-manager@7.16.0": + version "7.16.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-7.16.0.tgz#eb0757af5720c9c53c8010d7a0355ae27e17b7e5" + integrity sha512-8gVv3kW6n01Q6TrI1cmTZ9YMFi3ucDT7i7aI5lEikk2ebk1AEjrwX8MDTdaX5D7fPXMBLvnsaa0IFTAu+jcfOw== dependencies: - "@typescript-eslint/types" "7.0.1" - "@typescript-eslint/visitor-keys" "7.0.1" + "@typescript-eslint/types" "7.16.0" + "@typescript-eslint/visitor-keys" "7.16.0" -"@typescript-eslint/type-utils@7.0.1": - version "7.0.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/type-utils/-/type-utils-7.0.1.tgz#0fba92c1f81cad561d7b3adc812aa1cc0e35cdae" - integrity sha512-YtT9UcstTG5Yqy4xtLiClm1ZpM/pWVGFnkAa90UfdkkZsR1eP2mR/1jbHeYp8Ay1l1JHPyGvoUYR6o3On5Nhmw== +"@typescript-eslint/type-utils@7.16.0": + version "7.16.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/type-utils/-/type-utils-7.16.0.tgz#ec52b1932b8fb44a15a3e20208e0bd49d0b6bd00" + integrity sha512-j0fuUswUjDHfqV/UdW6mLtOQQseORqfdmoBNDFOqs9rvNVR2e+cmu6zJu/Ku4SDuqiJko6YnhwcL8x45r8Oqxg== dependencies: - "@typescript-eslint/typescript-estree" "7.0.1" - "@typescript-eslint/utils" "7.0.1" + "@typescript-eslint/typescript-estree" "7.16.0" + "@typescript-eslint/utils" "7.16.0" debug "^4.3.4" - ts-api-utils "^1.0.1" + ts-api-utils "^1.3.0" -"@typescript-eslint/types@7.0.1": - version "7.0.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-7.0.1.tgz#dcfabce192db5b8bf77ea3c82cfaabe6e6a3c901" - integrity sha512-uJDfmirz4FHib6ENju/7cz9SdMSkeVvJDK3VcMFvf/hAShg8C74FW+06MaQPODHfDJp/z/zHfgawIJRjlu0RLg== +"@typescript-eslint/types@7.16.0": + version "7.16.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-7.16.0.tgz#60a19d7e7a6b1caa2c06fac860829d162a036ed2" + integrity sha512-fecuH15Y+TzlUutvUl9Cc2XJxqdLr7+93SQIbcZfd4XRGGKoxyljK27b+kxKamjRkU7FYC6RrbSCg0ALcZn/xw== -"@typescript-eslint/typescript-estree@7.0.1": - version "7.0.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-7.0.1.tgz#1d52ac03da541693fa5bcdc13ad655def5046faf" - integrity sha512-SO9wHb6ph0/FN5OJxH4MiPscGah5wjOd0RRpaLvuBv9g8565Fgu0uMySFEPqwPHiQU90yzJ2FjRYKGrAhS1xig== +"@typescript-eslint/typescript-estree@7.16.0": + version "7.16.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-7.16.0.tgz#98ac779d526fab2a781e5619c9250f3e33867c09" + integrity sha512-a5NTvk51ZndFuOLCh5OaJBELYc2O3Zqxfl3Js78VFE1zE46J2AaVuW+rEbVkQznjkmlzWsUI15BG5tQMixzZLw== dependencies: - "@typescript-eslint/types" "7.0.1" - "@typescript-eslint/visitor-keys" "7.0.1" + "@typescript-eslint/types" "7.16.0" + "@typescript-eslint/visitor-keys" "7.16.0" debug "^4.3.4" globby "^11.1.0" is-glob "^4.0.3" - minimatch "9.0.3" - semver "^7.5.4" - ts-api-utils "^1.0.1" + minimatch "^9.0.4" + semver "^7.6.0" + ts-api-utils "^1.3.0" -"@typescript-eslint/utils@7.0.1": - version "7.0.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/utils/-/utils-7.0.1.tgz#b8ceac0ba5fef362b4a03a33c0e1fedeea3734ed" - integrity sha512-oe4his30JgPbnv+9Vef1h48jm0S6ft4mNwi9wj7bX10joGn07QRfqIqFHoMiajrtoU88cIhXf8ahwgrcbNLgPA== +"@typescript-eslint/utils@7.16.0": + version "7.16.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/utils/-/utils-7.16.0.tgz#b38dc0ce1778e8182e227c98d91d3418449aa17f" + integrity sha512-PqP4kP3hb4r7Jav+NiRCntlVzhxBNWq6ZQ+zQwII1y/G/1gdIPeYDCKr2+dH6049yJQsWZiHU6RlwvIFBXXGNA== dependencies: "@eslint-community/eslint-utils" "^4.4.0" - "@types/json-schema" "^7.0.12" - "@types/semver" "^7.5.0" - "@typescript-eslint/scope-manager" "7.0.1" - "@typescript-eslint/types" "7.0.1" - "@typescript-eslint/typescript-estree" "7.0.1" - semver "^7.5.4" - -"@typescript-eslint/visitor-keys@7.0.1": - version "7.0.1" - resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-7.0.1.tgz#864680ac5a8010ec4814f8a818e57595f79f464e" - integrity sha512-hwAgrOyk++RTXrP4KzCg7zB2U0xt7RUU0ZdMSCsqF3eKUwkdXUMyTb0qdCuji7VIbcpG62kKTU9M1J1c9UpFBw== + "@typescript-eslint/scope-manager" "7.16.0" + "@typescript-eslint/types" "7.16.0" + "@typescript-eslint/typescript-estree" "7.16.0" + +"@typescript-eslint/visitor-keys@7.16.0": + version "7.16.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-7.16.0.tgz#a1d99fa7a3787962d6e0efd436575ef840e23b06" + integrity sha512-rMo01uPy9C7XxG7AFsxa8zLnWXTF8N3PYclekWSrurvhwiw1eW88mrKiAYe6s53AUY57nTRz8dJsuuXdkAhzCg== dependencies: - "@typescript-eslint/types" "7.0.1" - eslint-visitor-keys "^3.4.1" + "@typescript-eslint/types" "7.16.0" + eslint-visitor-keys "^3.4.3" "@ungap/structured-clone@^1.2.0": version "1.2.0" @@ -595,16 +585,16 @@ eslint-visitor-keys@^3.3.0, eslint-visitor-keys@^3.4.1, eslint-visitor-keys@^3.4 resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz#0cd72fe8550e3c2eae156a96a4dddcd1c8ac5800" integrity sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag== -eslint@^8.56.0: - version "8.56.0" - resolved "https://registry.yarnpkg.com/eslint/-/eslint-8.56.0.tgz#4957ce8da409dc0809f99ab07a1b94832ab74b15" - integrity sha512-Go19xM6T9puCOWntie1/P997aXxFsOi37JIHRWI514Hc6ZnaHGKY9xFhrU65RT6CcBEzZoGG1e6Nq+DT04ZtZQ== +eslint@^8.57.0: + version "8.57.0" + resolved "https://registry.yarnpkg.com/eslint/-/eslint-8.57.0.tgz#c786a6fd0e0b68941aaf624596fb987089195668" + integrity sha512-dZ6+mexnaTIbSBZWgou51U6OmzIhYM2VcNdtiTtI7qPNZm35Akpr0f6vtw3w1Kmn5PYo+tZVfh13WrhpS6oLqQ== dependencies: "@eslint-community/eslint-utils" "^4.2.0" "@eslint-community/regexpp" "^4.6.1" "@eslint/eslintrc" "^2.1.4" - "@eslint/js" "8.56.0" - "@humanwhocodes/config-array" "^0.11.13" + "@eslint/js" "8.57.0" + "@humanwhocodes/config-array" "^0.11.14" "@humanwhocodes/module-importer" "^1.0.1" "@nodelib/fs.walk" "^1.2.8" "@ungap/structured-clone" "^1.2.0" @@ -898,7 +888,7 @@ hasown@^2.0.0, hasown@^2.0.1: dependencies: function-bind "^1.1.2" -ignore@^5.2.0, ignore@^5.2.4: +ignore@^5.2.0, ignore@^5.3.1: version "5.3.1" resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.3.1.tgz#5073e554cd42c5b33b394375f538b8593e34d4ef" integrity sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw== @@ -1128,13 +1118,6 @@ lodash.merge@^4.6.2: resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a" integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ== -lru-cache@^6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" - integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA== - dependencies: - yallist "^4.0.0" - merge2@^1.3.0, merge2@^1.4.1: version "1.4.1" resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.4.1.tgz#4368892f885e907455a6fd7dc55c0c9d404990ae" @@ -1148,13 +1131,6 @@ micromatch@^4.0.4: braces "^3.0.2" picomatch "^2.3.1" -minimatch@9.0.3: - version "9.0.3" - resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-9.0.3.tgz#a6e00c3de44c3a542bfaae70abfc22420a6da825" - integrity sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg== - dependencies: - brace-expansion "^2.0.1" - minimatch@^3.0.5, minimatch@^3.1.1, minimatch@^3.1.2: version "3.1.2" resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" @@ -1162,6 +1138,13 @@ minimatch@^3.0.5, minimatch@^3.1.1, minimatch@^3.1.2: dependencies: brace-expansion "^1.1.7" +minimatch@^9.0.4: + version "9.0.5" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-9.0.5.tgz#d74f9dd6b57d83d8e98cfb82133b03978bc929e5" + integrity sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow== + dependencies: + brace-expansion "^2.0.1" + minimist@^1.2.0, minimist@^1.2.6: version "1.2.8" resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c" @@ -1306,10 +1289,10 @@ prelude-ls@^1.2.1: resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396" integrity sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g== -prettier@^3.2.5: - version "3.2.5" - resolved "https://registry.yarnpkg.com/prettier/-/prettier-3.2.5.tgz#e52bc3090586e824964a8813b09aba6233b28368" - integrity sha512-3/GWa9aOC0YeD7LUfvOG2NiDyhOWRvt1k+rcKhOuYnMY24iiCphgneUfJDyFXd6rZCAnuLBv6UeAULtrhT/F4A== +prettier@^3.3.3: + version "3.3.3" + resolved "https://registry.yarnpkg.com/prettier/-/prettier-3.3.3.tgz#30c54fe0be0d8d12e6ae61dbb10109ea00d53105" + integrity sha512-i2tDNA0O5IrMO757lfrdQZCc2jPNDVntV0m/+4whiDfWaTKfMNgR7Qz0NAeGz/nRqF4m5/6CLzbP4/liHt12Ew== punycode@^2.1.0: version "2.3.1" @@ -1388,12 +1371,10 @@ semver@^6.3.1: resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== -semver@^7.5.4: - version "7.6.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.0.tgz#1a46a4db4bffcccd97b743b5005c8325f23d4e2d" - integrity sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg== - dependencies: - lru-cache "^6.0.0" +semver@^7.6.0: + version "7.6.2" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.2.tgz#1e3b34759f896e8f14d6134732ce798aeb0c6e13" + integrity sha512-FNAIBWCx9qcRhoHcgcJ0gvU7SN1lYU2ZXuSfl04bSC5OpvDHFyJCjdNHomPXxjQlCBU67YW64PzY7/VIEH7F2w== set-function-length@^1.2.1: version "1.2.1" @@ -1511,10 +1492,10 @@ to-regex-range@^5.0.1: dependencies: is-number "^7.0.0" -ts-api-utils@^1.0.1: - version "1.2.1" - resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-1.2.1.tgz#f716c7e027494629485b21c0df6180f4d08f5e8b" - integrity sha512-RIYA36cJn2WiH9Hy77hdF9r7oEwxAtB/TS9/S4Qd90Ap4z5FSiin5zEiTL44OII1Y3IIlEvxwxFUVgrHSZ/UpA== +ts-api-utils@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-1.3.0.tgz#4b490e27129f1e8e686b45cc4ab63714dc60eea1" + integrity sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ== tsconfig-paths@^3.15.0: version "3.15.0" @@ -1578,10 +1559,10 @@ typed-array-length@^1.0.4: for-each "^0.3.3" is-typed-array "^1.1.9" -typescript@^5.3.3: - version "5.3.3" - resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.3.3.tgz#b3ce6ba258e72e6305ba66f5c9b452aaee3ffe37" - integrity sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw== +typescript@^5.5.3: + version "5.5.3" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.5.3.tgz#e1b0a3c394190838a0b168e771b0ad56a0af0faa" + integrity sha512-/hreyEujaB0w76zKo6717l3L0o/qEUtRgdvUBvlkhoWeOVMjMuHNHk0BRBzikzuGDqNmPQbg5ifMEqsHLiIUcQ== unbox-primitive@^1.0.2: version "1.0.2" @@ -1639,11 +1620,6 @@ wrappy@1: resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ== -yallist@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" - integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== - yocto-queue@^0.1.0: version "0.1.0" resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b" From 3cade2a4b4e988cbf63b3430fbf694a9bcb92efe Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Sun, 14 Jul 2024 22:52:24 +0200 Subject: [PATCH 05/12] Continue Water Broker rewrite --- water/package.json | 2 +- water/src/broker/BrokerClient.ts | 332 +++++++++++------- water/src/broker/BrokerConnection.ts | 40 ++- water/src/broker/BrokerMessage.ts | 43 ++- water/src/broker/BrokerMessageHeaders.ts | 89 ++--- water/src/broker/Exceptions.ts | 2 + water/src/broker/LocalConnection.ts | 38 ++ water/src/broker/Subclients.ts | 129 +++++++ water/src/broker/kafka/KafkaConnection.ts | 205 ----------- water/src/broker/kafka/KafkaMessageHeaders.ts | 42 --- water/src/broker/rpc/RpcClient.ts | 125 +++++++ water/src/broker/rpc/RpcMessage.ts | 42 +++ water/src/broker/rpc/RpcMessageHeaders.ts | 81 +++++ water/src/broker/rpc/RpcStatus.ts | 16 + water/yarn.lock | 10 +- 15 files changed, 745 insertions(+), 451 deletions(-) create mode 100644 water/src/broker/LocalConnection.ts create mode 100644 water/src/broker/Subclients.ts delete mode 100644 water/src/broker/kafka/KafkaConnection.ts delete mode 100644 water/src/broker/kafka/KafkaMessageHeaders.ts create mode 100644 water/src/broker/rpc/RpcClient.ts create mode 100644 water/src/broker/rpc/RpcMessage.ts create mode 100644 water/src/broker/rpc/RpcMessageHeaders.ts create mode 100644 water/src/broker/rpc/RpcStatus.ts diff --git a/water/package.json b/water/package.json index 8774dd7..fac73f9 100644 --- a/water/package.json +++ b/water/package.json @@ -15,7 +15,7 @@ "dist" ], "dependencies": { - "kafkajs": "^2.2.4" + "valibot": "^0.36.0" }, "devDependencies": { "@types/node": "^20.14.10", diff --git a/water/src/broker/BrokerClient.ts b/water/src/broker/BrokerClient.ts index 723f951..f9aa463 100644 --- a/water/src/broker/BrokerClient.ts +++ b/water/src/broker/BrokerClient.ts @@ -1,124 +1,216 @@ import { Logger } from "../logging/Logger.js"; -import { CountDownLatch } from "../util/CountDownLatch.js"; -import type { IBrokerConnection } from "./BrokerConnection.js"; -import type { IBrokerMessageHeaders } from "./BrokerMessageHeaders.js"; -import { BrokerMessage } from "./BrokerMessage.js"; - -const TAG = "BrokerClient"; - -export type BrokerEventListener = (msg: BrokerMessage) => void | Promise; -export type BrokerMessageListener = (clusterId: string, msg: BrokerMessage) => void | Promise; -export interface ClusterResult { - responses: Map>; - timeout: boolean; +import type { BrokerConnection, TopicListener } from "./BrokerConnection.js"; +import type { BrokerMessage } from "./BrokerMessage.js"; +import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; +import { RpcClient } from "./rpc/RpcClient.js"; +import type { RpcRequestMessage, RpcResponse } from "./rpc/RpcMessage.js"; +import type { BaseSubclient } from "./Subclients.js"; +import { BrokerClientOptions, ConsumerSubclient, ProducerSubclient } from "./Subclients.js"; + +export abstract class BrokerClient { + + private static readonly TAG = "BrokerClient"; + + private readonly topics: Map = new Map(); + + public constructor(public readonly connection: BrokerConnection) { } + + public consumer( + topic: string, + key: string, + schema: T, // TODO + options: BrokerClientOptions = new BrokerClientOptions(), + callback: (msg: BrokerMessage) => Promise, + ): ConsumerSubclient { + Logger.debug(BrokerClient.TAG, `Creating consumer for key '${key}' in topic '${topic}'`); + const client = new ConsumerSubclient(this.connection, this, topic, key, options, schema, callback); + this.registerSubclient(client); + return client; + } + + public producer( + topic: string, + key: string, + schema: T, // TODO + options: BrokerClientOptions = new BrokerClientOptions(), + ): ProducerSubclient { + Logger.debug(BrokerClient.TAG, `Creating producer for key '${key}' in topic '${topic}'`); + const client = new ProducerSubclient(this.connection, this, topic, key, options, schema); + this.registerSubclient(client); + return client; + } + + public rpc( + topic: string, + key: string, + requestSchema: RequestT, // TODO + responseSchema: ResponseT, // TODO + options: BrokerClientOptions = new BrokerClientOptions(), + callback: (msg: RpcRequestMessage) => Promise>, + ): RpcClient { + return new RpcClient( + this, + topic, + key, + options, + requestSchema, + responseSchema, + callback, + ); + } + + private registerSubclient(subclient: BaseSubclient): void { + const topic = subclient.topic; + if (!this.topics.has(topic)) { + this.topics.set(topic, new TopicMetadata(this.connection, topic)); + } + const metadata = this.topics.get(topic)!; + metadata.registerSubclient(subclient); + } + + // Private API + public deregisterSubclient(subclient: BaseSubclient): void { + const topic = subclient.topic; + if (this.topics.has(topic)) { + const metadata = this.topics.get(topic)!; + metadata.deregisterSubclient(subclient); + if (metadata.isEmpty) { + this.topics.delete(topic); + } + } + } + + public destroy(): void { + Logger.debug(BrokerClient.TAG, `Destroying BrokerClient with active topics: ${[...this.topics.keys()]}`); + while (this.topics.size > 0) { + const [topic, metadata] = this.topics.entries().next().value as [string, TopicMetadata]; + metadata.destroy(); + this.topics.delete(topic); + } + } + +} + +class TopicMetadata { + + private static readonly TAG = "TopicMetadata"; + + private readonly keys: Map = new Map(); + private isDestroyed: boolean = false; + private connectionListener: TopicListener | null = null; + + public constructor( + private readonly connection: BrokerConnection, + private readonly topic: string, + ) { } + + public get isEmpty(): boolean { + return this.keys.size === 0; + } + + public registerSubclient(subclient: BaseSubclient): void { + if (subclient.topic !== this.topic) { + throw new Error(`Attempting to register subclient with topic '${subclient.topic}' in TopicMetadata of '${this.topic}'`); + } + Logger.debug(TopicMetadata.TAG, `Adding ${this.constructor.name} for key '${subclient.key}' in topic '${this.topic}'`); + const metadata = this.getOrCreateKeyMetadata(subclient.key); + if (subclient instanceof ConsumerSubclient) { + if (metadata.consumers.size === 0 && this.connectionListener === null) { + Logger.debug(TopicMetadata.TAG, `Creating new connection listener for topic '${this.topic}'`); + this.connectionListener = (topic, key, value, headers) => this.onTopicMessage(topic, key, value, headers); + this.connection.on(this.topic, this.connectionListener); + } + metadata.consumers.add(subclient); + } else if (subclient instanceof ProducerSubclient) { + metadata.producers.add(subclient); + } + } + + public deregisterSubclient(subclient: BaseSubclient): void { + Logger.debug(TopicMetadata.TAG, `Removing ${this.constructor.name} for key '${subclient.key}' in topic '${this.topic}'`); + const metadata = this.getExistingKeyMetadata(subclient.key); + if (metadata) { + metadata.producers.delete(subclient as ProducerSubclient); + metadata.consumers.delete(subclient as ConsumerSubclient); + this.maybeCleanupKeyMetadata(metadata); + } + } + + private maybeCleanupKeyMetadata(metadata: KeyMetadata): void { + if (metadata.isEmpty) { + this.keys.delete(metadata.key); + } + if (this.isEmpty && this.connectionListener !== null) { + Logger.debug(TopicMetadata.TAG, `Removing connection listener for topic '${this.topic}' after key cleanup`); + this.connection.off(this.topic, this.connectionListener); + this.connectionListener = null; + } + } + + private getOrCreateKeyMetadata(key: string): KeyMetadata { + if (!this.keys.has(key)) { + this.keys.set(key, new KeyMetadata(key)); + } + return this.keys.get(key)!; + } + + private getExistingKeyMetadata(key: string): KeyMetadata | null { + return this.keys.get(key) ?? null; + } + + private onTopicMessage( + topic: string, + key: string, + value: string, + headers: BrokerMessageHeaders, + ): void { + const metadata = this.getExistingKeyMetadata(key); + if (!metadata) { + return; + } + for (const consumer of metadata.consumers) { + consumer + .onIncomingMessage(value, headers) + .catch(e => Logger.error(TopicMetadata.TAG, + `Uncaught error in BrokerClient listener for key '${key}' in topic '${topic}'`, e)); + } + } + + public destroy(): void { + if (this.isDestroyed) { + return; + } + while (this.keys.size > 0) { + const [key, metadata] = this.keys.entries().next().value as [string, KeyMetadata]; + metadata.destroy(); + this.keys.delete(key); + } + if (this.connectionListener !== null) { + Logger.debug(TopicMetadata.TAG, `Removing connection listener for topic '${this.topic}' during destroy`); + this.connection.off(this.topic, this.connectionListener); + this.connectionListener = null; + } + } + } -export class BrokerClient { - private keyListeners: Map>> = new Map(); - - public constructor( - private readonly connection: IBrokerConnection, - private readonly topicName: string - ) { - Logger.debug(TAG, `Initializing BrokerClient with topic '${topicName}'`); - connection.on(topicName, this.onTopicMessage.bind(this)); - } - - protected async send( - key: string, - obj: T | null, - headers: IBrokerMessageHeaders = this.connection.createHeaders() - ): Promise { - return await this.connection.send(this.topicName, key, this.stringify(obj), headers); - } - - protected async sendClusterRequest( - key: string, - obj: T | null, - timeout: number = 0, - targetClusters: Set = new Set(), - expectedResponses: number | null = null, - messageCallback: BrokerMessageListener | null = null - ): Promise> { - const responseKey = this.toResponseKey(key); - - const responses: Map> = new Map(); - const latch = new CountDownLatch(expectedResponses ?? targetClusters.size); - let requestId = ""; - - const cb: BrokerEventListener = (msg: BrokerMessage) => { - if (msg.headers.requestId !== requestId) { - return; - } - messageCallback?.(msg.headers.sourceCluster, msg)?.catch(e => - Logger.error(TAG, "Uncaught error in sendClusterRequest message callback", e) - ); - - responses.set(msg.headers.sourceCluster, msg); - }; - - this.on(responseKey, cb); - let timeoutReached = false; - try { - requestId = await this.send(key, obj, this.connection.createHeaders(targetClusters)); - - timeoutReached = !(await latch.await(timeout)); - } finally { - this.off(responseKey, cb); - } - - return { - responses, - timeout: timeoutReached, - }; - } - - protected on(key: string, cb: BrokerEventListener): void { - let listeners = this.keyListeners.get(key); - if (!listeners) { - listeners = new Set(); - this.keyListeners.set(key, listeners); - } - listeners.add(cb); - } - - protected off(key: string, cb: BrokerEventListener): void { - const listeners = this.keyListeners.get(key); - if (listeners) { - listeners.delete(cb); - if (!listeners.size) { - this.keyListeners.delete(key); - } - } - } - - // For internal use only. Do not call externally! - public async _respond(msg: BrokerMessage, data: T | null): Promise { - const newHeaders = this.connection.createHeaders(new Set([msg.headers.sourceCluster]), msg.headers.requestId); - await this.send(this.toResponseKey(msg.key), data, newHeaders); - } - - private parse(json: string): T | null { - return JSON.parse(json) as T | null; - } - - private stringify(obj: T | null): string { - return JSON.stringify(obj); - } - - private async onTopicMessage(key: string, value: string, headers: IBrokerMessageHeaders): Promise { - const obj = this.parse(value); - const msg = new BrokerMessage(this, key, obj, headers); - const listeners = this.keyListeners.get(key); - if (!listeners || !listeners.size) { - return; - } - for (const listener of listeners) { - listener(msg)?.catch(e => Logger.error(TAG, "Uncaught error in BrokerClient listener", e)); - } - } - - private toResponseKey(key: string): string { - return `${key}-response`; - } +class KeyMetadata { + + public readonly producers: Set> = new Set(); + public readonly consumers: Set> = new Set(); + + public constructor( + public readonly key: string, + ) { } + + public get isEmpty(): boolean { + return this.producers.size === 0 && this.consumers.size === 0; + } + + public destroy(): void { + this.producers.forEach(producer => producer.destroy()); + this.consumers.forEach(consumer => consumer.destroy()); + this.producers.clear(); + this.consumers.clear(); + } } diff --git a/water/src/broker/BrokerConnection.ts b/water/src/broker/BrokerConnection.ts index f075984..35ed543 100644 --- a/water/src/broker/BrokerConnection.ts +++ b/water/src/broker/BrokerConnection.ts @@ -10,17 +10,31 @@ const TAG = "BrokerConnection"; export abstract class BrokerConnection { - public abstract serviceName: string; - public abstract instanceId: string; - public abstract supportsTopicHotSwap: boolean; - - protected topicListeners: Map> = new Map(); + public abstract readonly serviceName: string; + public abstract readonly instanceId: string; + public abstract readonly supportsTopicHotSwap: boolean; + public abstract readonly deferInitialTopicCreation: boolean; + + protected readonly topicListeners: Map> = new Map(); + private readonly deferredTopicsToCreate: Set = new Set(); + private hasStarted: boolean = false; + + public async start(): Promise { + await this.abstractStart(); + this.hasStarted = true; + for (const topic of this.deferredTopicsToCreate) { + this.createTopic(topic); + } + this.deferredTopicsToCreate.clear(); + } - public abstract start(): Promise; public async destroy(): Promise { + Logger.debug(TAG, "Destroying BrokerConnection"); this.topicListeners.clear(); } + + public abstract abstractStart(): Promise; public abstract abstractSend(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise; public send(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { @@ -35,8 +49,13 @@ export abstract class BrokerConnection { let listeners = this.topicListeners.get(topic); if (!listeners) { listeners = new Set(); - Logger.debug(TAG, `Creating new topic '${topic}'`); - this.createTopic(topic); + if (!this.hasStarted && this.deferInitialTopicCreation) { + Logger.debug(TAG, `Deferring creation of topic '${topic}' to after connected`); + this.deferredTopicsToCreate.add(topic); + } else { + Logger.debug(TAG, `Creating new topic '${topic}'`); + this.createTopic(topic); + } this.topicListeners.set(topic, listeners); } listeners.add(cb); @@ -48,6 +67,7 @@ export abstract class BrokerConnection { listeners.delete(cb); if (listeners.size === 0) { Logger.debug(TAG, `Removing topic '${topic}'`); + this.deferredTopicsToCreate.delete(topic); this.removeTopic(topic); this.topicListeners.delete(topic); } @@ -57,8 +77,8 @@ export abstract class BrokerConnection { // To be called by implementers protected dispatchIncomingMessage(topic: string, key: string, value: string, headers: BrokerMessageHeaders): void { if ( - (headers.targetServices.size >= 1 && !headers.targetServices.has(this.serviceName)) || - (headers.targetInstances.size >= 1 && !headers.targetInstances.has(this.instanceId)) + (headers.targetServices.size > 0 && !headers.targetServices.has(this.serviceName)) || + (headers.targetInstances.size > 0 && !headers.targetInstances.has(this.instanceId)) ) { // If there is a target cluster restriction and this message wasn't meant for us, // discard it immediately without notifying any listeners. diff --git a/water/src/broker/BrokerMessage.ts b/water/src/broker/BrokerMessage.ts index 54d803d..5819a05 100644 --- a/water/src/broker/BrokerMessage.ts +++ b/water/src/broker/BrokerMessage.ts @@ -1,15 +1,30 @@ -import type { BrokerClient } from "./BrokerClient.js"; -import type { IBrokerMessageHeaders } from "./BrokerMessageHeaders.js"; - -export class BrokerMessage { - public constructor( - public readonly client: BrokerClient, - public readonly key: string, - public readonly value: T | null, - public readonly headers: IBrokerMessageHeaders - ) {} - - public async respond(data: T | null): Promise { - await this.client._respond(this, data); - } +import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; +import { RpcRequestMessage, RpcResponseMessage } from "./rpc/RpcMessage.js"; +import { RpcMessageHeaders } from "./rpc/RpcMessageHeaders.js"; +import type { RpcStatus } from "./rpc/RpcStatus.js"; + +export class BrokerMessage { + public constructor( + public readonly topic: string, + public readonly key: string, + public readonly value: T, + public readonly headers: H + ) { } + + public get messageId(): string { + return this.headers.messageId; + } + + // Private API + public toRpcRequestMessage( + updateSender: (status: RpcStatus, data: ResponseT) => Promise, + ): RpcRequestMessage { + return new RpcRequestMessage(this.topic, this.key, this.value, this.headers, updateSender); + } + + // Private API + public toRpcResponseMessage(): RpcResponseMessage { + return new RpcResponseMessage(this.topic, this.key, this.value, new RpcMessageHeaders(this.headers)); + } + } diff --git a/water/src/broker/BrokerMessageHeaders.ts b/water/src/broker/BrokerMessageHeaders.ts index 809f35c..65835c6 100644 --- a/water/src/broker/BrokerMessageHeaders.ts +++ b/water/src/broker/BrokerMessageHeaders.ts @@ -2,7 +2,7 @@ import { randomUUID } from "node:crypto"; import { BrokerConnection } from "./BrokerConnection.js"; import type { MessageId } from "./BrokerConnection.js"; -export const BROKER_HEADER_NAMES = { +const BROKER_HEADER_NAMES = { SOURCE_SERVICE: "source-service", SOURCE_INSTANCE: "source-instance", TARGET_SERVICES: "target-services", @@ -13,65 +13,46 @@ export const BROKER_HEADER_NAMES = { export class BrokerMessageHeaders { public readonly headers: Record; - public readonly sourceService: string; - public readonly sourceInstance: string; - public readonly targetServices: Set; - public readonly targetInstances: Set; - public readonly messageId: MessageId; - public constructor( - sourceService: string, - sourceInstance: string, - targetServices: Set, - targetInstances: Set, - ); + public get sourceService(): string { + return getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_SERVICE); + } + public get sourceInstance(): string { + return getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_INSTANCE); + } + public get targetServices(): Set { + return new Set((this.headers[BROKER_HEADER_NAMES.TARGET_SERVICES] ?? "").split(",").filter(s => s)); + } + public get targetInstances(): Set { + return new Set((this.headers[BROKER_HEADER_NAMES.TARGET_INSTANCES] ?? "").split(",").filter(s => s)); + } + public get messageId(): MessageId { + return getOrThrow(this.headers, BROKER_HEADER_NAMES.MESSAGE_ID); + } + + public constructor(headers: Record); public constructor( connection: BrokerConnection, targetServices: Set, targetInstances: Set, ); - public constructor(headers: Record); public constructor( - sourceServiceOrHeaders: string | BrokerConnection | Record, - sourceInstanceOrTargetServices?: string | Set, - targetServicesOrTargetInstances?: Set, + headersOrConnection: BrokerConnection | Record, + targetServices?: Set, targetInstances?: Set, ) { // I hate that JS doesn't have native method overloading - if (typeof sourceServiceOrHeaders === "string") { - this.sourceService = sourceServiceOrHeaders; - this.sourceInstance = sourceInstanceOrTargetServices as string; - this.targetServices = targetServicesOrTargetInstances as Set; - this.targetInstances = targetInstances as Set; - this.messageId = randomUUID(); - this.headers = BrokerMessageHeaders.createHeadersMap( - this.sourceService, - this.sourceInstance, - this.targetServices, - this.targetInstances, - this.messageId, - ); - } else if (sourceServiceOrHeaders instanceof BrokerConnection) { - const connection: BrokerConnection = sourceServiceOrHeaders; - this.sourceService = connection.serviceName; - this.sourceInstance = connection.instanceId; - this.targetServices = sourceInstanceOrTargetServices as Set; - this.targetInstances = targetServicesOrTargetInstances as Set; - this.messageId = randomUUID(); + if (headersOrConnection instanceof BrokerConnection) { + const connection = headersOrConnection; this.headers = BrokerMessageHeaders.createHeadersMap( - this.sourceService, - this.sourceInstance, - this.targetServices, - this.targetInstances, - this.messageId, + connection.serviceName, + connection.instanceId, + targetServices!, + targetInstances!, + null, ); } else { - this.headers = sourceServiceOrHeaders; - this.sourceService = BrokerMessageHeaders.getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_SERVICE); - this.sourceInstance = BrokerMessageHeaders.getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_INSTANCE); - this.targetServices = new Set((this.headers[BROKER_HEADER_NAMES.TARGET_SERVICES] ?? "").split(",").filter(s => s)); - this.targetInstances = new Set((this.headers[BROKER_HEADER_NAMES.TARGET_INSTANCES] ?? "").split(",").filter(s => s)); - this.messageId = BrokerMessageHeaders.getOrThrow(this.headers, BROKER_HEADER_NAMES.MESSAGE_ID); + this.headers = headersOrConnection; } } @@ -94,12 +75,12 @@ export class BrokerMessageHeaders { return headers; } - private static getOrThrow(headers: Record, key: string): string { - const value = headers[key]; - if (value === undefined) { - throw new Error(`Missing broker message header '${key}'`); - } - return value; - } +} +export function getOrThrow(headers: Record, key: string): string { + const value = headers[key]; + if (value === undefined) { + throw new Error(`Missing broker message header '${key}'`); + } + return value; } diff --git a/water/src/broker/Exceptions.ts b/water/src/broker/Exceptions.ts index 43b005f..a475f03 100644 --- a/water/src/broker/Exceptions.ts +++ b/water/src/broker/Exceptions.ts @@ -1,3 +1,5 @@ +import type { RpcStatus } from "./rpc/RpcStatus.js"; + export class BrokerException extends Error { public constructor(message?: string) { super(message); diff --git a/water/src/broker/LocalConnection.ts b/water/src/broker/LocalConnection.ts new file mode 100644 index 0000000..192d8b2 --- /dev/null +++ b/water/src/broker/LocalConnection.ts @@ -0,0 +1,38 @@ +import type { MessageId } from "./BrokerConnection.js"; +import { BrokerConnection } from "./BrokerConnection.js"; +import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; + +export class LocalConnection extends BrokerConnection { + + public override readonly supportsTopicHotSwap: boolean = true; + public override readonly deferInitialTopicCreation: boolean = false; + + public constructor(public override readonly serviceName: string, public override readonly instanceId: string) { + super(); + } + + public override async abstractSend(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { + if (this.shouldDispatchExternallyAfterShortCircuit(topic, key, value, headers)) { + const isForExternalService = headers.targetServices.size > 0 && [...headers.targetServices][0] !== this.serviceName; + const isForExternalInstance = headers.targetInstances.size > 0 && [...headers.targetInstances][0] !== this.instanceId; + if (isForExternalService || isForExternalInstance) { + throw new Error("Attempting to send message to other services/instances in a LocalConnection " + + `(services=${[...headers.targetServices]}; instances=${[...headers.targetInstances]})`); + } + } + return headers.messageId; + } + + public override async abstractStart(): Promise { + // Nothing to start :) + } + + public override createTopic(topic: string): void { + // noop + } + + public override removeTopic(topic: string): void { + // noop + } + +} diff --git a/water/src/broker/Subclients.ts b/water/src/broker/Subclients.ts new file mode 100644 index 0000000..db0d59f --- /dev/null +++ b/water/src/broker/Subclients.ts @@ -0,0 +1,129 @@ +import { Logger } from "../logging/Logger.js"; +import type { BrokerClient } from "./BrokerClient.js"; +import type { BrokerConnection, MessageId } from "./BrokerConnection.js"; +import { BrokerMessage } from "./BrokerMessage.js"; +import { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; + +export class BrokerClientOptions { + // No options exist in the JS universe at the moment +} + +export abstract class BaseSubclient { + + private isDestroyed: boolean = false; + + public constructor( + protected readonly connection: BrokerConnection, + protected readonly client: BrokerClient, + public readonly topic: string, + public readonly key: string, + protected readonly options: BrokerClientOptions, + ) { } + + protected abstract doDestroy(): void; + + public destroy(): void { + if (!this.isDestroyed) { + this.isDestroyed = true; + this.doDestroy(); + } + } +} + +export class ProducerSubclient extends BaseSubclient { + + private static readonly TAG = "ProducerSubclient"; + + public constructor( + connection: BrokerConnection, + client: BrokerClient, + topic: string, + key: string, + options: BrokerClientOptions, + private readonly schema: T, // TODO eeeeeh + ) { + super(connection, client, topic, key, options); + } + + protected doDestroy(): void { + this.client.deregisterSubclient(this); + } + + public async send( + data: T, + services: Set = new Set(), + instances: Set = new Set(), + ): Promise { + const msg = new BrokerMessage( + this.topic, + this.key, + data, + new BrokerMessageHeaders( + this.connection, + services, + instances, + ), + ); + return this.internalSend(msg); + } + + // Private API + public async internalSend(msg: BrokerMessage): Promise { + const stringifiedData = this.stringifyOutgoing(msg.value); + Logger.verbose(ProducerSubclient.TAG, `Sending message ${msg.messageId} with key '${msg.key}' in topic '${msg.topic}'`); + return this.connection.send(msg.topic, msg.key, stringifiedData, msg.headers); + } + + private stringifyOutgoing(data: T | null): string { + return JSON.stringify(data); + } + +} + +export class ConsumerSubclient extends BaseSubclient { + + private static readonly TAG = "ConsumerSubclient"; + + public constructor( + connection: BrokerConnection, + client: BrokerClient, + topic: string, + key: string, + options: BrokerClientOptions, + private readonly schema: T, // TODO eeeeeh + private readonly callback: (msg: BrokerMessage) => Promise, + ) { + super(connection, client, topic, key, options); + } + + protected doDestroy(): void { + this.client.deregisterSubclient(this); + } + + // Private API + public async onIncomingMessage( + value: string, + headers: BrokerMessageHeaders, + ): Promise { + const data = this.parseIncoming(value); + const msg = new BrokerMessage( + this.topic, + this.key, + data, + headers, + ); + Logger.verbose(ConsumerSubclient.TAG, `Received message ${msg.messageId} with key '${msg.key}' in topic '${msg.topic}'`); + try { + await this.callback(msg); + } catch (e) { + Logger.error(ConsumerSubclient.TAG, "Uncaught consumer callback error while processing message " + + `${headers.messageId} with key '${this.key}' in topic '${this.topic}'`, e); + } + } + + private parseIncoming(data: string): T { + // TODO schema validation + return JSON.parse(data); + } + +} diff --git a/water/src/broker/kafka/KafkaConnection.ts b/water/src/broker/kafka/KafkaConnection.ts deleted file mode 100644 index 6d95600..0000000 --- a/water/src/broker/kafka/KafkaConnection.ts +++ /dev/null @@ -1,205 +0,0 @@ -import type { Consumer, KafkaMessage, Producer } from "kafkajs"; -import { Kafka, logLevel } from "kafkajs"; -import { Logger } from "../../logging/Logger.js"; -import type { IBrokerConnection, TopicListener } from "../BrokerConnection.js"; -import type { IBrokerMessageHeaders } from "../BrokerMessageHeaders.js"; -import { KafkaMessageHeaders } from "./KafkaMessageHeaders.js"; - -const TAG = "KafkaConnection"; - -export class KafkaConnection implements IBrokerConnection { - private readonly kafka: Kafka; - private readonly producer: Producer; - private readonly consumer: Consumer; - - private readonly subscribedTopics: Set = new Set(); - - private readonly topicListeners: Map> = new Map(); - private isRunning: boolean = false; - - public constructor( - kafkaHost: string, - public readonly clientId: string, - consumerGroupId: string, - public readonly clusterId: string - ) { - this.kafka = new Kafka({ - brokers: [kafkaHost], - clientId: consumerGroupId, - logCreator: - () => - ({ namespace, level, log }) => { - // eslint-disable-next-line @typescript-eslint/no-unused-vars - const { message, timestamp, ...other } = log; - switch (level) { - case logLevel.DEBUG: - Logger.debug(TAG, `[${namespace}] ${message}`, other); - break; - case logLevel.INFO: - Logger.info(TAG, `[${namespace}] ${message}`, other); - break; - case logLevel.WARN: - Logger.warn(TAG, `[${namespace}] ${message}`, other); - break; - case logLevel.ERROR: - case logLevel.NOTHING: - Logger.error(TAG, `[${namespace}] ${message}`, other); - break; - } - }, - }); - - this.producer = this.kafka.producer({ - retry: { - retries: 2, - }, - idempotent: true, - }); - this.consumer = this.kafka.consumer({ - groupId: consumerGroupId, - }); - } - - public async send(topic: string, key: string, value: string, headers: KafkaMessageHeaders): Promise { - await this.producer.send({ - topic, - messages: [ - { - key, - value, - headers: headers.serialize(), - }, - ], - }); - // If the record is meant for ourselves (amongst other clusters), - // immediately dispatch it to the listeners. - if (!headers.targetClusters.size || headers.targetClusters.has(this.clusterId)) { - await this.invokeCallbacks(topic, key, value, headers); - } - return headers.requestId; - } - - public createHeaders(targetClusters?: Set, requestId?: string): IBrokerMessageHeaders { - return new KafkaMessageHeaders(this.clientId, this.clusterId, targetClusters, requestId); - } - - public on(topic: string, cb: TopicListener): void { - let listeners = this.topicListeners.get(topic); - if (!listeners) { - if (this.isRunning) { - throw new Error("Cannot subscribe to new topic after KafkaConnection has started"); - } - listeners = new Set(); - this.subscribedTopics.add(topic); - this.topicListeners.set(topic, listeners); - } - listeners.add(cb); - } - - public off(topic: string, cb: TopicListener): void { - const listeners = this.topicListeners.get(topic); - if (listeners) { - listeners.delete(cb); - if (!listeners.size) { - this.subscribedTopics.delete(topic); - this.topicListeners.delete(topic); - } - } - } - - public async start(): Promise { - if (this.isRunning) { - throw new Error("KafkaConnection is already running!"); - } - Logger.debug(TAG, "Starting Kafka Connection"); - await this.createTopics(); - await this.createProducer(); - await this.createConsumer(); - this.isRunning = true; - } - - public async destroy(): Promise { - await this.consumer.disconnect(); - await this.producer.disconnect(); - this.subscribedTopics.clear(); - this.topicListeners.clear(); - this.isRunning = false; - } - - private async createTopics(): Promise { - const targetTopics = [...this.subscribedTopics]; - Logger.debug(TAG, `Creating missing topics, target topics: ${targetTopics}`); - const admin = this.kafka.admin(); - const existingTopics = await admin.listTopics(); - const missingTopics = targetTopics.filter(t => !existingTopics.includes(t)); - Logger.debug(TAG, `Missing topics: ${missingTopics}`); - if (missingTopics.length) { - Logger.info(TAG, "Creating missing topics"); - await admin.createTopics({ - topics: missingTopics.map(t => ({ - topic: t, - numPartitions: 1, - replicationFactor: 1, - })), - }); - } - Logger.debug(TAG, "Created all missing topics"); - } - - private async createProducer(): Promise { - Logger.debug(TAG, "Creating Producer"); - await this.producer.connect(); - } - - private async createConsumer(): Promise { - Logger.debug(TAG, "Creating Consumer"); - await this.consumer.connect(); - await this.consumer.subscribe({ - topics: [...this.subscribedTopics], - }); - // TODO Runtime error handling (such as for missing topics) - await this.consumer.run({ - eachMessage: async ({ topic, message: record }) => { - void this.handleIncomingRecord(topic, record); - }, - }); - } - - private async handleIncomingRecord(topic: string, record: KafkaMessage): Promise { - try { - const key = record.key?.toString(); - const value = record.value?.toString(); - if (!key || !value || !record.headers) { - return; - } - - const headers = KafkaMessageHeaders.from(record.headers); - if (headers.targetClusters.size && !headers.targetClusters.has(this.clusterId)) { - // If there is a target cluster restriction and this record wasn't meant for us, - // discard it immediately without notifying any listeners. - return; - } - if (headers.sourceCluster === this.clusterId) { - // If this record was sent by ourselves, discard it too, as we already dispatch events - // to our listeners in `send()` to avoid the round trip through Kafka. - return; - } - - await this.invokeCallbacks(topic, key, value, headers); - } catch (e) { - Logger.error(TAG, "Uncaught error in internal KafkaConnection record handler", e); - } - } - - private async invokeCallbacks( - topic: string, - key: string, - value: string, - headers: IBrokerMessageHeaders - ): Promise { - const listeners = this.topicListeners.get(topic) ?? []; - for (const listener of listeners) { - listener(key, value, headers)?.catch(e => Logger.error(TAG, "Uncaught error in KafkaConnection listener", e)); - } - } -} diff --git a/water/src/broker/kafka/KafkaMessageHeaders.ts b/water/src/broker/kafka/KafkaMessageHeaders.ts deleted file mode 100644 index c8ad9bf..0000000 --- a/water/src/broker/kafka/KafkaMessageHeaders.ts +++ /dev/null @@ -1,42 +0,0 @@ -import { randomUUID } from "node:crypto"; -import type { IHeaders } from "kafkajs"; -import type { IBrokerMessageHeaders } from "../BrokerMessageHeaders.js"; -import { BROKER_HEADERS } from "../BrokerMessageHeaders.js"; - -export class KafkaMessageHeaders implements IBrokerMessageHeaders { - public static readonly INVALID_CLUSTER_ID = `${-Math.pow(2, 32)}`; - - public constructor( - public readonly clientId: string, - public readonly sourceCluster: string, - public readonly targetClusters: Set = new Set(), // Empty = No target restriction - public readonly requestId: string = randomUUID() - ) {} - - public serialize(): IHeaders { - return { - [BROKER_HEADERS.CLIENT_ID]: this.clientId, - [BROKER_HEADERS.TARGET_CLUSTERS]: [...this.targetClusters].join(","), - [BROKER_HEADERS.REQUEST_ID]: this.requestId, - [BROKER_HEADERS.SOURCE_CLUSTER]: this.sourceCluster.toString(), - }; - } - - public static from(headers: IHeaders): KafkaMessageHeaders { - const transformed = KafkaMessageHeaders.headersBufferToString(headers); - return new KafkaMessageHeaders( - transformed[BROKER_HEADERS.CLIENT_ID] ?? "default", - transformed[BROKER_HEADERS.SOURCE_CLUSTER] || KafkaMessageHeaders.INVALID_CLUSTER_ID, - new Set((transformed[BROKER_HEADERS.TARGET_CLUSTERS] ?? "").split(",").filter(c => c)), - transformed[BROKER_HEADERS.REQUEST_ID] ?? "" - ); - } - - private static headersBufferToString(headers: IHeaders): Record { - return Object.fromEntries( - Object.entries(headers) - .filter(([_, v]) => v) - .map(([k, v]) => [k, v!.toString()]) - ); - } -} diff --git a/water/src/broker/rpc/RpcClient.ts b/water/src/broker/rpc/RpcClient.ts new file mode 100644 index 0000000..1db0b80 --- /dev/null +++ b/water/src/broker/rpc/RpcClient.ts @@ -0,0 +1,125 @@ +import { Logger } from "../../logging/Logger.js"; +import type { BrokerClient } from "../BrokerClient.js"; +import { IgnoreRpcRequest, RpcException } from "../Exceptions.js"; +import type { BrokerClientOptions, ConsumerSubclient, ProducerSubclient } from "../Subclients.js"; +import { BaseSubclient } from "../Subclients.js"; +import type { RpcRequestMessage, RpcResponse } from "./RpcMessage.js"; +import { RpcResponseMessage } from "./RpcMessage.js"; +import { RpcMessageHeaders } from "./RpcMessageHeaders.js"; +import type { RpcStatus } from "./RpcStatus.js"; + +export class RpcClient extends BaseSubclient { + + private static readonly TAG = "RpcClient"; + + private readonly requestProducer: ProducerSubclient; + private readonly requestConsumer: ConsumerSubclient; + private readonly responseProducer: ProducerSubclient; + private readonly responseConsumer: ConsumerSubclient; + + public constructor( + client: BrokerClient, + topic: string, + key: string, + options: BrokerClientOptions, + private readonly requestSchema: RequestT, // TODO eeeeeh + private readonly responseSchema: ResponseT, // TODO eeeeeh + private readonly callback: (msg: RpcRequestMessage) => Promise>, + ) { + super(client.connection, client, topic, key, options); + + this.requestProducer = this.client.producer( + topic, + key, + requestSchema, + options, + ); + this.requestConsumer = this.client.consumer( + topic, + key, + requestSchema, + options, + async msg => { + const sendResponse = async ( + response: ResponseT | null, + status: RpcStatus, + isException: boolean, + isUpdate: boolean, + ) => { + const responseMsg = new RpcResponseMessage( + this.toResponseTopic(topic), + this.toResponseKey(key), + response, + new RpcMessageHeaders( + this.connection, + new Set([msg.headers.sourceService]), + new Set([msg.headers.sourceInstance]), + msg.headers.messageId, + status, + isException, + isUpdate, + ), + ); + await this.responseProducer.internalSend(responseMsg); + }; + + const rpcMessage = msg.toRpcRequestMessage(async (status, data) => { + await sendResponse(data, status, false, true); + }); + try { + const [status, response] = await this.callback(rpcMessage); + await sendResponse(response, status, false, false); + } catch (e) { + if (e instanceof IgnoreRpcRequest) { + return; + } else if (e instanceof RpcException) { + await sendResponse(null, e.status, true, false); + return; + } + Logger.error(RpcClient.TAG, "Uncaught RPC callback error while processing message " + + `${msg.headers.messageId} with key '$key' in topic '$topic'`, e); + } + }, + ); + this.responseProducer = this.client.producer( + this.toResponseTopic(topic), + this.toResponseKey(key), + responseSchema, + options, + ); + this.responseConsumer = this.client.consumer( + this.toResponseTopic(topic), + this.toResponseKey(key), + responseSchema, + options, + async msg => { + // TODO + }, + ); + } + + public async call( + request: RequestT, + services: Set = new Set(), + instances: Set = new Set(), + timeout: number = 10 * 1000, + ): Promise> { + return this.stream(request, services, instances, timeout, 1); + } + + protected override doDestroy(): void { + this.requestProducer.destroy(); + this.requestConsumer.destroy(); + this.responseProducer.destroy(); + this.responseConsumer.destroy(); + } + + private toResponseTopic(topic: string): string { + return this.connection.supportsTopicHotSwap ? `${topic}.responses` : topic; + } + + private toResponseKey(key: string): string { + return `${key}.response`; + } + +} diff --git a/water/src/broker/rpc/RpcMessage.ts b/water/src/broker/rpc/RpcMessage.ts new file mode 100644 index 0000000..bf9b300 --- /dev/null +++ b/water/src/broker/rpc/RpcMessage.ts @@ -0,0 +1,42 @@ +import { BrokerMessage } from "../BrokerMessage.js"; +import type { BrokerMessageHeaders } from "../BrokerMessageHeaders.js"; +import type { RpcMessageHeaders } from "./RpcMessageHeaders.js"; +import type { RpcStatus } from "./RpcStatus.js"; + +export type RpcResponse = [RpcStatus, ResponseT]; + +export class RpcRequestMessage + extends BrokerMessage { + + public constructor( + topic: string, + key: string, + value: RequestT, + headers: H, + private readonly updateSender: (stats: RpcStatus, data: ResponseT) => Promise, + ) { + super(topic, key, value, headers); + } + + public async sendUpdate(status: RpcStatus, data: ResponseT): Promise { + await this.updateSender(status, data); + } + +} + +export class RpcResponseMessage extends BrokerMessage { + + public constructor( + topic: string, + key: string, + value: ResponseT, + headers: RpcMessageHeaders, + ) { + super(topic, key, value, headers); + } + + public get status(): RpcStatus { + return this.headers.status; + } + +} diff --git a/water/src/broker/rpc/RpcMessageHeaders.ts b/water/src/broker/rpc/RpcMessageHeaders.ts new file mode 100644 index 0000000..ba4fafd --- /dev/null +++ b/water/src/broker/rpc/RpcMessageHeaders.ts @@ -0,0 +1,81 @@ +import type { MessageId } from "../BrokerConnection.js"; +import { BrokerConnection } from "../BrokerConnection.js"; +import { BrokerMessageHeaders, getOrThrow } from "../BrokerMessageHeaders.js"; +import { RpcStatus } from "./RpcStatus.js"; + +const RPC_HEADER_NAMES = { + IN_REPLY_TO: "rpc-in-reply-to", + STATUS: "rpc-response-status", + IS_EXCEPTION: "rpc-is-exception", + IS_UPDATE: "rpc-is-update", +}; + +export class RpcMessageHeaders extends BrokerMessageHeaders { + + public get inReplyTo(): MessageId { + return getOrThrow(this.headers, RPC_HEADER_NAMES.IN_REPLY_TO); + } + public get status(): RpcStatus { + return new RpcStatus(safeToInt(getOrThrow(this.headers, RPC_HEADER_NAMES.STATUS))); + } + public get isException(): boolean { + return this.headers[RPC_HEADER_NAMES.IS_EXCEPTION]?.toLowerCase() === "true"; + } + public get isUpdate(): boolean { + return this.headers[RPC_HEADER_NAMES.IS_UPDATE]?.toLowerCase() === "true"; + } + + public constructor( + base: Record | BrokerMessageHeaders, + ); + public constructor( + connection: BrokerConnection, + targetServices: Set, + targetInstances: Set, + inReplyTo: MessageId, + status: RpcStatus, + isException: boolean, + isUpdate: boolean, + ); + public constructor( + baseOrConnection: Record | BrokerMessageHeaders | BrokerConnection, + targetServices?: Set, + targetInstances?: Set, + inReplyTo?: MessageId, + status?: RpcStatus, + isException?: boolean, + isUpdate?: boolean, + ) { + if (baseOrConnection instanceof BrokerMessageHeaders) { + super(baseOrConnection.headers); + } else if (baseOrConnection instanceof BrokerConnection) { + const connection = baseOrConnection; + super( + BrokerMessageHeaders.createHeadersMap( + connection.serviceName, + connection.instanceId, + targetServices!, + targetInstances!, + null, + { + [RPC_HEADER_NAMES.IN_REPLY_TO]: inReplyTo!, + [RPC_HEADER_NAMES.STATUS]: status!.code.toString(), + [RPC_HEADER_NAMES.IS_EXCEPTION]: isException!.toString(), + [RPC_HEADER_NAMES.IS_UPDATE]: isUpdate!.toString(), + }, + ), + ); + } else { + super(baseOrConnection); + } + } + +} + +function safeToInt(value: string): number { + const parsed = parseInt(value, 10); + if (isNaN(parsed)) { + throw new Error(`Failed to parse integer from value: ${value}`); + } + return parsed; +} diff --git a/water/src/broker/rpc/RpcStatus.ts b/water/src/broker/rpc/RpcStatus.ts new file mode 100644 index 0000000..19803a5 --- /dev/null +++ b/water/src/broker/rpc/RpcStatus.ts @@ -0,0 +1,16 @@ +export class RpcStatus { + + public static readonly OK = new RpcStatus(0); + public static readonly UNKNOWN = new RpcStatus(999_999); + + public constructor(public readonly code: number) { } + + public equals(other: RpcStatus): boolean { + return this.code === other.code; + } + + public toString(): string { + return `RpcStatus(${this.code})`; + } + +} diff --git a/water/yarn.lock b/water/yarn.lock index da355f4..907d849 100644 --- a/water/yarn.lock +++ b/water/yarn.lock @@ -1086,11 +1086,6 @@ json5@^1.0.2: dependencies: minimist "^1.2.0" -kafkajs@^2.2.4: - version "2.2.4" - resolved "https://registry.yarnpkg.com/kafkajs/-/kafkajs-2.2.4.tgz#59e6e16459d87fdf8b64be73970ed5aa42370a5b" - integrity sha512-j/YeapB1vfPT2iOIUn/vxdyKEuhuY2PxMBvf5JWux6iSaukAccrMtXEY/Lb7OvavDhOWME589bpLrEdnVHjfjA== - keyv@^4.5.3: version "4.5.4" resolved "https://registry.yarnpkg.com/keyv/-/keyv-4.5.4.tgz#a879a99e29452f942439f2a405e3af8b31d4de93" @@ -1586,6 +1581,11 @@ uri-js@^4.2.2: dependencies: punycode "^2.1.0" +valibot@^0.36.0: + version "0.36.0" + resolved "https://registry.yarnpkg.com/valibot/-/valibot-0.36.0.tgz#74e746694b1abcc1879e4393db551d4ce1e10578" + integrity sha512-CjF1XN4sUce8sBK9TixrDqFM7RwNkuXdJu174/AwmQUB62QbCQADg5lLe8ldBalFgtj1uKj+pKwDJiNo4Mn+eQ== + which-boxed-primitive@^1.0.2: version "1.0.2" resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz#13757bc89b209b049fe5d86430e21cf40a89a8e6" From 3d757dd036744981dada9d4225154df865df281f Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Sun, 14 Jul 2024 22:53:06 +0200 Subject: [PATCH 06/12] Latte broker code improvements --- .../gg/beemo/latte/broker/BrokerClient.kt | 38 ++++++++-------- .../gg/beemo/latte/broker/BrokerConnection.kt | 1 + .../latte/broker/BrokerMessageHeaders.kt | 31 +++---------- .../gg/beemo/latte/broker/rpc/RpcClient.kt | 19 +++++--- .../latte/broker/rpc/RpcMessageHeaders.kt | 44 +++++-------------- 5 files changed, 49 insertions(+), 84 deletions(-) diff --git a/latte/src/main/java/gg/beemo/latte/broker/BrokerClient.kt b/latte/src/main/java/gg/beemo/latte/broker/BrokerClient.kt index 43e6660..5c53c59 100644 --- a/latte/src/main/java/gg/beemo/latte/broker/BrokerClient.kt +++ b/latte/src/main/java/gg/beemo/latte/broker/BrokerClient.kt @@ -114,11 +114,6 @@ abstract class BrokerClient( } } - internal fun toResponseTopic(topic: String): String = - if (connection.supportsTopicHotSwap) "$topic.responses" else topic - - internal fun toResponseKey(key: String): String = "$key.response" - } private class TopicMetadata( @@ -127,21 +122,6 @@ private class TopicMetadata( private val topic: String, ) { - private class KeyMetadata(val key: String) { - val producers: MutableSet> = Collections.synchronizedSet(HashSet()) - val consumers: MutableSet> = Collections.synchronizedSet(HashSet()) - - val isEmpty: Boolean - get() = producers.isEmpty() && consumers.isEmpty() - - fun destroy() { - producers.forEach(ProducerSubclient<*>::destroy) - consumers.forEach(ConsumerSubclient<*>::destroy) - producers.clear() - consumers.clear() - } - } - private val log by Log private val _keys: MutableMap = Collections.synchronizedMap(HashMap()) private val isBeingDestroyed = AtomicBoolean(false) @@ -158,6 +138,9 @@ private class TopicMetadata( subclient.key, subclient.topic ) + check(subclient.topic == topic) { + "Attempting to register subclient with topic '${subclient.topic}' in TopicMetadata of '$topic'" + } val metadata = getOrCreateKeyMetadata(subclient.key) when (subclient) { is ConsumerSubclient<*> -> { @@ -255,6 +238,21 @@ private class TopicMetadata( } +private class KeyMetadata(val key: String) { + val producers: MutableSet> = Collections.synchronizedSet(HashSet()) + val consumers: MutableSet> = Collections.synchronizedSet(HashSet()) + + val isEmpty: Boolean + get() = producers.isEmpty() && consumers.isEmpty() + + fun destroy() { + producers.forEach(ProducerSubclient<*>::destroy) + consumers.forEach(ConsumerSubclient<*>::destroy) + producers.clear() + consumers.clear() + } +} + @PublishedApi internal inline fun isTypeNullable(): Boolean { return null is T || T::class.java == Unit::class.java || T::class.java == Void::class.java diff --git a/latte/src/main/java/gg/beemo/latte/broker/BrokerConnection.kt b/latte/src/main/java/gg/beemo/latte/broker/BrokerConnection.kt index 2651652..91271ed 100644 --- a/latte/src/main/java/gg/beemo/latte/broker/BrokerConnection.kt +++ b/latte/src/main/java/gg/beemo/latte/broker/BrokerConnection.kt @@ -85,6 +85,7 @@ abstract class BrokerConnection { listeners.remove(cb) if (listeners.size == 0) { log.debug("Removing topic '{}'", topic) + deferredTopicsToCreate.remove(topic) removeTopic(topic) null } else { diff --git a/latte/src/main/java/gg/beemo/latte/broker/BrokerMessageHeaders.kt b/latte/src/main/java/gg/beemo/latte/broker/BrokerMessageHeaders.kt index f9937ac..559da34 100644 --- a/latte/src/main/java/gg/beemo/latte/broker/BrokerMessageHeaders.kt +++ b/latte/src/main/java/gg/beemo/latte/broker/BrokerMessageHeaders.kt @@ -21,29 +21,17 @@ open class BrokerMessageHeaders(val headers: Map) { } constructor( - sourceService: String, - sourceInstance: String, + connection: BrokerConnection, targetServices: Set, targetInstances: Set, ) : this( createHeadersMap( - sourceService, - sourceInstance, + connection.serviceName, + connection.instanceId, targetServices, targetInstances, null, - ) - ) - - constructor( - connection: BrokerConnection, - targetServices: Set, - targetInstances: Set, - ) : this( - connection.serviceName, - connection.instanceId, - targetServices, - targetInstances, + ), ) companion object { @@ -54,6 +42,7 @@ open class BrokerMessageHeaders(val headers: Map) { private const val HEADER_TARGET_INSTANCES = "target-instances" private const val HEADER_MESSAGE_ID = "message-id" + // Needs to be JvmStatic to be used in subclasses @JvmStatic protected fun createHeadersMap( sourceService: String, @@ -66,23 +55,17 @@ open class BrokerMessageHeaders(val headers: Map) { val headers = HashMap() headers[HEADER_SOURCE_SERVICE] = sourceService headers[HEADER_SOURCE_INSTANCE] = sourceInstance - headers[HEADER_TARGET_SERVICES] = joinToString(targetServices) - headers[HEADER_TARGET_INSTANCES] = joinToString(targetInstances) + headers[HEADER_TARGET_SERVICES] = targetServices.joinToString(",") + headers[HEADER_TARGET_INSTANCES] = targetInstances.joinToString(",") headers[HEADER_MESSAGE_ID] = messageId ?: UUID.randomUUID().toString() headers.putAll(extra) return headers } - @JvmStatic protected fun splitToSet(value: String): Set { return value.split(",").filter { it.isNotEmpty() }.toSet() } - @JvmStatic - protected fun joinToString(value: Set): String { - return value.joinToString(",") - } - } } diff --git a/latte/src/main/java/gg/beemo/latte/broker/rpc/RpcClient.kt b/latte/src/main/java/gg/beemo/latte/broker/rpc/RpcClient.kt index 4ca0b28..0f5a2c7 100644 --- a/latte/src/main/java/gg/beemo/latte/broker/rpc/RpcClient.kt +++ b/latte/src/main/java/gg/beemo/latte/broker/rpc/RpcClient.kt @@ -41,8 +41,8 @@ class RpcClient( private val requestConsumer = client.consumer(topic, key, options, requestType, requestIsNullable) { msg -> suspend fun sendResponse(response: ResponseT?, status: RpcStatus, isException: Boolean, isUpdate: Boolean) { val responseMsg = RpcResponseMessage( - client.toResponseTopic(topic), - client.toResponseKey(key), + toResponseTopic(topic), + toResponseKey(key), response, RpcMessageHeaders( connection, @@ -71,7 +71,7 @@ class RpcClient( return@consumer } catch (ex: Exception) { log.error( - "Uncaught RPC callbac#k error while processing message ${msg.headers.messageId} " + + "Uncaught RPC callback error while processing message ${msg.headers.messageId} " + "with key '$key' in topic '$topic'", ex, ) @@ -79,16 +79,16 @@ class RpcClient( } } private val responseProducer = client.producer( - client.toResponseTopic(topic), - client.toResponseKey(key), + toResponseTopic(topic), + toResponseKey(key), options, responseType, responseIsNullable, ) private val responseFlow = MutableSharedFlow>() private val responseConsumer = client.consumer( - client.toResponseTopic(topic), - client.toResponseKey(key), + toResponseTopic(topic), + toResponseKey(key), options, responseType, responseIsNullable, @@ -160,6 +160,11 @@ class RpcClient( } + private fun toResponseTopic(topic: String): String = + if (connection.supportsTopicHotSwap) "$topic.responses" else topic + + private fun toResponseKey(key: String): String = "$key.response" + override fun doDestroy() { requestProducer.destroy() requestConsumer.destroy() diff --git a/latte/src/main/java/gg/beemo/latte/broker/rpc/RpcMessageHeaders.kt b/latte/src/main/java/gg/beemo/latte/broker/rpc/RpcMessageHeaders.kt index 4545143..de4c112 100644 --- a/latte/src/main/java/gg/beemo/latte/broker/rpc/RpcMessageHeaders.kt +++ b/latte/src/main/java/gg/beemo/latte/broker/rpc/RpcMessageHeaders.kt @@ -23,8 +23,7 @@ class RpcMessageHeaders(headers: Map) : BrokerMessageHeaders(hea constructor(base: BrokerMessageHeaders) : this(base.headers) constructor( - sourceService: String, - sourceInstance: String, + connection: BrokerConnection, targetServices: Set, targetInstances: Set, inReplyTo: MessageId, @@ -33,46 +32,25 @@ class RpcMessageHeaders(headers: Map) : BrokerMessageHeaders(hea isUpdate: Boolean, ) : this( createHeadersMap( - sourceService, - sourceInstance, + connection.serviceName, + connection.instanceId, targetServices, targetInstances, null, - extra = mapOf( - HEADER_IN_REPLY_TO to inReplyTo, - HEADER_STATUS to status.code.toString(), - HEADER_IS_EXCEPTION to isException.toString(), - HEADER_IS_UPDATE to isUpdate.toString(), - ) - ) - ) - - constructor( - connection: BrokerConnection, - targetServices: Set, - targetInstances: Set, - inReplyTo: MessageId, - status: RpcStatus, - isException: Boolean, - isUpdate: Boolean, - ) : this( - connection.serviceName, - connection.instanceId, - targetServices, - targetInstances, - inReplyTo, - status, - isException, - isUpdate, + extra = + mapOf( + HEADER_IN_REPLY_TO to inReplyTo, + HEADER_STATUS to status.code.toString(), + HEADER_IS_EXCEPTION to isException.toString(), + HEADER_IS_UPDATE to isUpdate.toString(), + ), + ), ) companion object { - private const val HEADER_IN_REPLY_TO = "rpc-in-reply-to" private const val HEADER_STATUS = "rpc-response-status" private const val HEADER_IS_EXCEPTION = "rpc-is-exception" private const val HEADER_IS_UPDATE = "rpc-is-update" - } - } From ed1dcf677fa3dfb39f33329cd02cbfd2eab83573 Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Tue, 16 Jul 2024 16:39:29 +0200 Subject: [PATCH 07/12] Finish Water broker system rewrite Hopefully. Now only missing the RabbitConnection. --- water/src/broker/BrokerClient.ts | 28 +++++---- water/src/broker/Subclients.ts | 14 ++--- water/src/broker/rpc/RpcClient.ts | 95 +++++++++++++++++++++++++------ water/src/index.ts | 12 +++- water/src/util/OnlineEmitter.ts | 43 ++++++++++++++ 5 files changed, 154 insertions(+), 38 deletions(-) create mode 100644 water/src/util/OnlineEmitter.ts diff --git a/water/src/broker/BrokerClient.ts b/water/src/broker/BrokerClient.ts index f9aa463..2c08bf9 100644 --- a/water/src/broker/BrokerClient.ts +++ b/water/src/broker/BrokerClient.ts @@ -1,3 +1,4 @@ +import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; import { Logger } from "../logging/Logger.js"; import type { BrokerConnection, TopicListener } from "./BrokerConnection.js"; import type { BrokerMessage } from "./BrokerMessage.js"; @@ -15,13 +16,13 @@ export abstract class BrokerClient { public constructor(public readonly connection: BrokerConnection) { } - public consumer( + public consumer>>( topic: string, key: string, - schema: T, // TODO + schema: TSchema, options: BrokerClientOptions = new BrokerClientOptions(), - callback: (msg: BrokerMessage) => Promise, - ): ConsumerSubclient { + callback: (msg: BrokerMessage>) => Promise, + ): ConsumerSubclient { Logger.debug(BrokerClient.TAG, `Creating consumer for key '${key}' in topic '${topic}'`); const client = new ConsumerSubclient(this.connection, this, topic, key, options, schema, callback); this.registerSubclient(client); @@ -31,23 +32,28 @@ export abstract class BrokerClient { public producer( topic: string, key: string, - schema: T, // TODO options: BrokerClientOptions = new BrokerClientOptions(), ): ProducerSubclient { Logger.debug(BrokerClient.TAG, `Creating producer for key '${key}' in topic '${topic}'`); - const client = new ProducerSubclient(this.connection, this, topic, key, options, schema); + const client = new ProducerSubclient(this.connection, this, topic, key, options); this.registerSubclient(client); return client; } - public rpc( + public rpc< + RequestTSchema extends BaseSchema>, + ResponseTSchema extends BaseSchema>, + >( topic: string, key: string, - requestSchema: RequestT, // TODO - responseSchema: ResponseT, // TODO + requestSchema: RequestTSchema, + responseSchema: ResponseTSchema, options: BrokerClientOptions = new BrokerClientOptions(), - callback: (msg: RpcRequestMessage) => Promise>, - ): RpcClient { + callback: (msg: RpcRequestMessage< + InferOutput, + InferOutput + >) => Promise>, + ): RpcClient { return new RpcClient( this, topic, diff --git a/water/src/broker/Subclients.ts b/water/src/broker/Subclients.ts index db0d59f..28dcadd 100644 --- a/water/src/broker/Subclients.ts +++ b/water/src/broker/Subclients.ts @@ -1,3 +1,5 @@ +import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; +import { parse } from "valibot"; import { Logger } from "../logging/Logger.js"; import type { BrokerClient } from "./BrokerClient.js"; import type { BrokerConnection, MessageId } from "./BrokerConnection.js"; @@ -40,7 +42,6 @@ export class ProducerSubclient extends BaseSubclient { topic: string, key: string, options: BrokerClientOptions, - private readonly schema: T, // TODO eeeeeh ) { super(connection, client, topic, key, options); } @@ -80,7 +81,7 @@ export class ProducerSubclient extends BaseSubclient { } -export class ConsumerSubclient extends BaseSubclient { +export class ConsumerSubclient>> extends BaseSubclient { private static readonly TAG = "ConsumerSubclient"; @@ -90,8 +91,8 @@ export class ConsumerSubclient extends BaseSubclient { topic: string, key: string, options: BrokerClientOptions, - private readonly schema: T, // TODO eeeeeh - private readonly callback: (msg: BrokerMessage) => Promise, + private readonly schema: TSchema, + private readonly callback: (msg: BrokerMessage>) => Promise, ) { super(connection, client, topic, key, options); } @@ -121,9 +122,8 @@ export class ConsumerSubclient extends BaseSubclient { } } - private parseIncoming(data: string): T { - // TODO schema validation - return JSON.parse(data); + private parseIncoming(data: string): InferOutput { + return parse(this.schema, JSON.parse(data)); } } diff --git a/water/src/broker/rpc/RpcClient.ts b/water/src/broker/rpc/RpcClient.ts index 1db0b80..5c27cfc 100644 --- a/water/src/broker/rpc/RpcClient.ts +++ b/water/src/broker/rpc/RpcClient.ts @@ -1,37 +1,47 @@ +import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; import { Logger } from "../../logging/Logger.js"; import type { BrokerClient } from "../BrokerClient.js"; -import { IgnoreRpcRequest, RpcException } from "../Exceptions.js"; +import { IgnoreRpcRequest, RpcException, RpcRequestTimeout } from "../Exceptions.js"; import type { BrokerClientOptions, ConsumerSubclient, ProducerSubclient } from "../Subclients.js"; import { BaseSubclient } from "../Subclients.js"; +import { OnlineEmitter } from "../../util/OnlineEmitter.js"; +import type { BrokerMessage } from "../BrokerMessage.js"; +import { CountDownLatch } from "../../util/CountDownLatch.js"; import type { RpcRequestMessage, RpcResponse } from "./RpcMessage.js"; import { RpcResponseMessage } from "./RpcMessage.js"; import { RpcMessageHeaders } from "./RpcMessageHeaders.js"; import type { RpcStatus } from "./RpcStatus.js"; -export class RpcClient extends BaseSubclient { +export class RpcClient< + RequestTSchema extends BaseSchema>, + ResponseTSchema extends BaseSchema>, +> extends BaseSubclient { private static readonly TAG = "RpcClient"; - private readonly requestProducer: ProducerSubclient; - private readonly requestConsumer: ConsumerSubclient; - private readonly responseProducer: ProducerSubclient; - private readonly responseConsumer: ConsumerSubclient; + private readonly requestProducer: ProducerSubclient>; + private readonly requestConsumer: ConsumerSubclient; + private readonly responseProducer: ProducerSubclient>; + private readonly responseConsumer: ConsumerSubclient; + private readonly responses: OnlineEmitter>> = new OnlineEmitter(); public constructor( client: BrokerClient, topic: string, key: string, options: BrokerClientOptions, - private readonly requestSchema: RequestT, // TODO eeeeeh - private readonly responseSchema: ResponseT, // TODO eeeeeh - private readonly callback: (msg: RpcRequestMessage) => Promise>, + requestSchema: RequestTSchema, + responseSchema: ResponseTSchema, + private readonly callback: (msg: RpcRequestMessage< + InferOutput, + InferOutput + >) => Promise>, ) { super(client.connection, client, topic, key, options); this.requestProducer = this.client.producer( topic, key, - requestSchema, options, ); this.requestConsumer = this.client.consumer( @@ -41,7 +51,7 @@ export class RpcClient extends BaseSubclient { options, async msg => { const sendResponse = async ( - response: ResponseT | null, + response: InferOutput | null, status: RpcStatus, isException: boolean, isUpdate: boolean, @@ -63,7 +73,7 @@ export class RpcClient extends BaseSubclient { await this.responseProducer.internalSend(responseMsg); }; - const rpcMessage = msg.toRpcRequestMessage(async (status, data) => { + const rpcMessage = msg.toRpcRequestMessage>(async (status, data) => { await sendResponse(data, status, false, true); }); try { @@ -84,7 +94,6 @@ export class RpcClient extends BaseSubclient { this.responseProducer = this.client.producer( this.toResponseTopic(topic), this.toResponseKey(key), - responseSchema, options, ); this.responseConsumer = this.client.consumer( @@ -93,18 +102,70 @@ export class RpcClient extends BaseSubclient { responseSchema, options, async msg => { - // TODO + this.responses.emit(msg); }, ); } public async call( - request: RequestT, + request: InferOutput, + services: Set = new Set(), + instances: Set = new Set(), + timeout: number = 10 * 1000, + ): Promise>> { + const generator = this.stream(request, services, instances, timeout, 1); + const msg = (await generator.next()).value; + if (!msg) { + throw new Error("Unexpected end of single-response stream"); + } + return msg; + } + + public async *stream( + request: InferOutput, services: Set = new Set(), instances: Set = new Set(), timeout: number = 10 * 1000, - ): Promise> { - return this.stream(request, services, instances, timeout, 1); + maxResponses: number = Infinity, + ): AsyncGenerator>, void> { + if (timeout === Infinity && maxResponses === Infinity) { + throw new Error("Must specify either a timeout or a max number of responses"); + } + if (maxResponses !== Infinity && maxResponses <= 0) { + throw new Error("maxResponses must be at least 1"); + } + let responseCounter = 0; + const timeoutLatch = maxResponses ? new CountDownLatch(maxResponses) : null; + const timeoutPromise = timeoutLatch?.await(timeout) ?? new Promise(() => { }); + + const messageId = await this.requestProducer.send(request, services, instances); + + while (true) { + const result = await Promise.race([ + this.responses.awaitValue(), + timeoutPromise, + ]); + if (typeof result === "boolean") { + if (result) { + return; + } else { + throw new RpcRequestTimeout(`RPC request timed out after ${timeout} ms`); + } + } + const msg = result.toRpcResponseMessage(); + if (msg.headers.inReplyTo !== messageId) { + return; + } + if (msg.headers.isException) { + throw new RpcException(msg.headers.status); + } + yield msg; + timeoutLatch?.countDown(); + responseCounter++; + if (responseCounter >= maxResponses) { + return; + } + } } protected override doDestroy(): void { diff --git a/water/src/index.ts b/water/src/index.ts index 4667efa..71977fb 100644 --- a/water/src/index.ts +++ b/water/src/index.ts @@ -1,12 +1,18 @@ export * from "./CommonConfig.js"; export * from "./broker/BrokerClient.js"; -export * from "./broker/BrokerMessage.js"; export * from "./broker/BrokerConnection.js"; +export * from "./broker/BrokerMessage.js"; export * from "./broker/BrokerMessageHeaders.js"; -export * from "./broker/kafka/KafkaConnection.js"; -export * from "./broker/kafka/KafkaMessageHeaders.js"; +export * from "./broker/Exceptions.js"; +export * from "./broker/LocalConnection.js"; +export * from "./broker/Subclients.js"; +export * from "./broker/rpc/RpcClient.js"; +export * from "./broker/rpc/RpcMessage.js"; +export * from "./broker/rpc/RpcMessageHeaders.js"; +export * from "./broker/rpc/RpcStatus.js"; export * from "./logging/Logger.js"; export * from "./util/CountDownLatch.js"; +export * from "./util/OnlineEmitter.js"; diff --git a/water/src/util/OnlineEmitter.ts b/water/src/util/OnlineEmitter.ts new file mode 100644 index 0000000..6b24709 --- /dev/null +++ b/water/src/util/OnlineEmitter.ts @@ -0,0 +1,43 @@ +export class OnlineEmitter { + + private readonly listeners: Set<(data: T) => void> = new Set(); + + public emit(data: T): void { + for (const listener of this.listeners) { + listener(data); + } + } + + public addListener(listener: (data: T) => void): void { + this.listeners.add(listener); + } + + + public removeListener(listener: (data: T) => void): void { + this.listeners.delete(listener); + } + + public async awaitValue(): Promise { + return new Promise(resolve => { + const listener = (data: T) => { + this.removeListener(listener); + resolve(data); + }; + this.addListener(listener); + }); + } + + public [Symbol.asyncIterator](): AsyncIterableIterator { + const awaitValue = this.awaitValue.bind(this); + return { + async next(): Promise> { + const value = await awaitValue(); + return { value, done: false }; + }, + [Symbol.asyncIterator](): AsyncIterableIterator { + return this; + }, + }; + } + +} From 435d1fc9068c83e7a8c90e3d2181b909ebca6076 Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Tue, 16 Jul 2024 22:00:30 +0200 Subject: [PATCH 08/12] Add Water RabbitConnection --- .../latte/broker/rabbitmq/RabbitConnection.kt | 1 + water/package.json | 1 + water/src/broker/BrokerConnection.ts | 3 +- water/src/broker/rabbitmq/RabbitConnection.ts | 144 ++++++++++++++++++ water/src/index.ts | 1 + water/yarn.lock | 5 + 6 files changed, 154 insertions(+), 1 deletion(-) create mode 100644 water/src/broker/rabbitmq/RabbitConnection.ts diff --git a/latte/src/main/java/gg/beemo/latte/broker/rabbitmq/RabbitConnection.kt b/latte/src/main/java/gg/beemo/latte/broker/rabbitmq/RabbitConnection.kt index a882965..9098aca 100644 --- a/latte/src/main/java/gg/beemo/latte/broker/rabbitmq/RabbitConnection.kt +++ b/latte/src/main/java/gg/beemo/latte/broker/rabbitmq/RabbitConnection.kt @@ -66,6 +66,7 @@ class RabbitConnection( // https://www.rabbitmq.com/docs/publishers#message-properties deliveryMode(2) // Persistent headers(headers.headers) // lol + messageId(headers.messageId) }.build() channelData.channel.basicPublish(topic, key, properties, value.toByteArray()) } diff --git a/water/package.json b/water/package.json index fac73f9..68a9134 100644 --- a/water/package.json +++ b/water/package.json @@ -15,6 +15,7 @@ "dist" ], "dependencies": { + "rabbitmq-client": "^4.6.0", "valibot": "^0.36.0" }, "devDependencies": { diff --git a/water/src/broker/BrokerConnection.ts b/water/src/broker/BrokerConnection.ts index 35ed543..0d97e21 100644 --- a/water/src/broker/BrokerConnection.ts +++ b/water/src/broker/BrokerConnection.ts @@ -33,8 +33,9 @@ export abstract class BrokerConnection { this.topicListeners.clear(); } - + // Internal API public abstract abstractStart(): Promise; + // Internal API public abstract abstractSend(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise; public send(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { diff --git a/water/src/broker/rabbitmq/RabbitConnection.ts b/water/src/broker/rabbitmq/RabbitConnection.ts new file mode 100644 index 0000000..3fb44b9 --- /dev/null +++ b/water/src/broker/rabbitmq/RabbitConnection.ts @@ -0,0 +1,144 @@ +import type { Consumer, Publisher } from "rabbitmq-client"; +import { Connection } from "rabbitmq-client"; +import type { MessageId } from "../BrokerConnection.js"; +import { BrokerConnection } from "../BrokerConnection.js"; +import { BrokerMessageHeaders } from "../BrokerMessageHeaders.js"; +import { Logger } from "../../logging/Logger.js"; + +export class RabbitConnection extends BrokerConnection { + + private static readonly TAG = "RabbitConnection"; + + public override supportsTopicHotSwap: boolean = true; + public override deferInitialTopicCreation: boolean = true; + + private connection: Connection | null = null; + private readonly consumers: Map = new Map(); + private readonly publishers: Map = new Map(); + + public constructor( + private readonly rabbitHosts: string[], + public override readonly serviceName: string, + public override readonly instanceId: string, + private readonly useTls: boolean = false, + private readonly username: string = "guest", + private readonly password: string = "guest", + ) { + super(); + } + + public override async abstractStart(): Promise { + this.connection = new Connection({ + hosts: this.rabbitHosts, + username: this.username, + password: this.password, + tls: this.useTls, + connectionName: `${this.serviceName}: ${this.instanceId}`, + }); + this.connection.on("connection.blocked", reason => { + Logger.warn(RabbitConnection.TAG, `RabbitMQ server blocked connection for reason: ${reason}`); + }); + this.connection.on("connection.unblocked", () => { + Logger.info(RabbitConnection.TAG, "RabbitMQ server unblocked connection"); + }); + await new Promise((resolve, reject) => { + // Use `on` instead of `once` to re-use the same logging code for future connection updates + this.connection!.on("connection", () => { + Logger.info(RabbitConnection.TAG, "RabbitMQ connection (re)established"); + resolve(); + }); + this.connection!.on("error", error => { + Logger.error(RabbitConnection.TAG, "Error in RabbitMQ connection", error); + reject(error); + }); + }); + } + + public override async destroy(): Promise { + await super.destroy(); + await this.connection?.close(); + } + + public override async abstractSend(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { + if (this.shouldDispatchExternallyAfterShortCircuit(topic, key, value, headers)) { + let publisher = this.publishers.get(topic); + if (!publisher) { + publisher = this + .ensureConnection() + .createPublisher(this.commonPublisherConsumerArguments(topic)); + this.publishers.set(topic, publisher); + } + await publisher.send({ + exchange: topic, + routingKey: key, + headers: headers.headers, + messageId: headers.messageId, + durable: true, + }, value); + } + return headers.messageId; + } + + protected override createTopic(topic: string): void { + if (this.consumers.has(topic)) { + return; + } + const queueName = this.createQueueName(topic); + this.consumers.set(topic, this.ensureConnection().createConsumer({ + queue: queueName, + ...this.commonPublisherConsumerArguments(topic), + }, async (msg, reply) => { + const key = msg.routingKey; + const value = msg.body instanceof Buffer ? msg.body.toString("utf8") : String(msg.body); + const headers = new BrokerMessageHeaders(msg.headers ?? {}); + this.dispatchIncomingMessage(topic, key, value, headers); + })); + } + + protected override removeTopic(topic: string): void { + const consumer = this.consumers.get(topic); + if (consumer) { + consumer.close().catch(e => { + Logger.error(RabbitConnection.TAG, `Error closing consumer for topic ${topic}`, e); + }); + this.consumers.delete(topic); + } + } + + private ensureConnection(): Connection { + const connection = this.connection; + if (!connection) { + throw new Error("Connection not open"); + } + return connection; + } + + private createQueueName(topic: string): string { + return `${this.serviceName}.${this.instanceId}.${topic}`; + } + + private commonPublisherConsumerArguments(topic: string) { + const exchangeName = topic; + const queueName = this.createQueueName(topic); + const routingKey = "#"; + return { + queues: [{ + queue: queueName, + durable: true, + exclusive: false, + autoDelete: false, + }], + exchanges: [{ + exchange: exchangeName, + type: "topic", + durable: true, + }], + queueBindings: [{ + queue: queueName, + exchange: exchangeName, + routingKey, + }], + }; + } + +} diff --git a/water/src/index.ts b/water/src/index.ts index 71977fb..f2b410f 100644 --- a/water/src/index.ts +++ b/water/src/index.ts @@ -11,6 +11,7 @@ export * from "./broker/rpc/RpcClient.js"; export * from "./broker/rpc/RpcMessage.js"; export * from "./broker/rpc/RpcMessageHeaders.js"; export * from "./broker/rpc/RpcStatus.js"; +export * from "./broker/rabbitmq/RabbitConnection.js"; export * from "./logging/Logger.js"; diff --git a/water/yarn.lock b/water/yarn.lock index 907d849..7c23e06 100644 --- a/water/yarn.lock +++ b/water/yarn.lock @@ -1299,6 +1299,11 @@ queue-microtask@^1.2.2: resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== +rabbitmq-client@^4.6.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/rabbitmq-client/-/rabbitmq-client-4.6.0.tgz#7f8d777ade6079e894476c98c388a7560c692c64" + integrity sha512-RHt6Vz+nCzsMlSJ5HbMgxR1SMej1LtRVYCmbIWXfUhgeej3HshByQdVOirF195V3RVCRI3UHEec9PzgDQQtS8Q== + regexp.prototype.flags@^1.5.2: version "1.5.2" resolved "https://registry.yarnpkg.com/regexp.prototype.flags/-/regexp.prototype.flags-1.5.2.tgz#138f644a3350f981a858c44f6bb1a61ff59be334" From d23ee30587844e79e15f505f8bd8eca32d437ae3 Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Tue, 16 Jul 2024 22:06:08 +0200 Subject: [PATCH 09/12] Cleanup publishers as well --- water/src/broker/rabbitmq/RabbitConnection.ts | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/water/src/broker/rabbitmq/RabbitConnection.ts b/water/src/broker/rabbitmq/RabbitConnection.ts index 3fb44b9..370bdf1 100644 --- a/water/src/broker/rabbitmq/RabbitConnection.ts +++ b/water/src/broker/rabbitmq/RabbitConnection.ts @@ -103,6 +103,13 @@ export class RabbitConnection extends BrokerConnection { }); this.consumers.delete(topic); } + const publisher = this.publishers.get(topic); + if (publisher) { + publisher.close().catch(e => { + Logger.error(RabbitConnection.TAG, `Error closing publisher for topic ${topic}`, e); + }); + this.publishers.delete(topic); + } } private ensureConnection(): Connection { From 39223258391ca3f0532a159bfb667f054acb5135 Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Wed, 17 Jul 2024 19:54:12 +0200 Subject: [PATCH 10/12] Fix circular dependencies --- water/src/broker/BrokerClient.ts | 4 +--- water/src/broker/BrokerMessage.ts | 3 +-- water/src/broker/BrokerMessageHeaders.ts | 2 +- water/src/broker/Exceptions.ts | 2 +- water/src/broker/LocalConnection.ts | 2 +- water/src/broker/Subclients.ts | 4 +--- water/src/broker/rabbitmq/RabbitConnection.ts | 4 +--- 7 files changed, 7 insertions(+), 14 deletions(-) diff --git a/water/src/broker/BrokerClient.ts b/water/src/broker/BrokerClient.ts index 2c08bf9..4f5ade1 100644 --- a/water/src/broker/BrokerClient.ts +++ b/water/src/broker/BrokerClient.ts @@ -1,12 +1,10 @@ import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; -import { Logger } from "../logging/Logger.js"; +import { BrokerClientOptions, ConsumerSubclient, Logger, ProducerSubclient, RpcClient } from "../index.js"; import type { BrokerConnection, TopicListener } from "./BrokerConnection.js"; import type { BrokerMessage } from "./BrokerMessage.js"; import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; -import { RpcClient } from "./rpc/RpcClient.js"; import type { RpcRequestMessage, RpcResponse } from "./rpc/RpcMessage.js"; import type { BaseSubclient } from "./Subclients.js"; -import { BrokerClientOptions, ConsumerSubclient, ProducerSubclient } from "./Subclients.js"; export abstract class BrokerClient { diff --git a/water/src/broker/BrokerMessage.ts b/water/src/broker/BrokerMessage.ts index 5819a05..0b4bdb9 100644 --- a/water/src/broker/BrokerMessage.ts +++ b/water/src/broker/BrokerMessage.ts @@ -1,6 +1,5 @@ +import { RpcMessageHeaders, RpcRequestMessage, RpcResponseMessage } from "../index.js"; import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; -import { RpcRequestMessage, RpcResponseMessage } from "./rpc/RpcMessage.js"; -import { RpcMessageHeaders } from "./rpc/RpcMessageHeaders.js"; import type { RpcStatus } from "./rpc/RpcStatus.js"; export class BrokerMessage { diff --git a/water/src/broker/BrokerMessageHeaders.ts b/water/src/broker/BrokerMessageHeaders.ts index 65835c6..9fa6d22 100644 --- a/water/src/broker/BrokerMessageHeaders.ts +++ b/water/src/broker/BrokerMessageHeaders.ts @@ -1,5 +1,5 @@ import { randomUUID } from "node:crypto"; -import { BrokerConnection } from "./BrokerConnection.js"; +import { BrokerConnection } from "../index.js"; import type { MessageId } from "./BrokerConnection.js"; const BROKER_HEADER_NAMES = { diff --git a/water/src/broker/Exceptions.ts b/water/src/broker/Exceptions.ts index a475f03..8a2568e 100644 --- a/water/src/broker/Exceptions.ts +++ b/water/src/broker/Exceptions.ts @@ -1,4 +1,4 @@ -import type { RpcStatus } from "./rpc/RpcStatus.js"; +import type { RpcStatus } from "../index.js"; export class BrokerException extends Error { public constructor(message?: string) { diff --git a/water/src/broker/LocalConnection.ts b/water/src/broker/LocalConnection.ts index 192d8b2..2b369e8 100644 --- a/water/src/broker/LocalConnection.ts +++ b/water/src/broker/LocalConnection.ts @@ -1,5 +1,5 @@ +import { BrokerConnection } from "../index.js"; import type { MessageId } from "./BrokerConnection.js"; -import { BrokerConnection } from "./BrokerConnection.js"; import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; export class LocalConnection extends BrokerConnection { diff --git a/water/src/broker/Subclients.ts b/water/src/broker/Subclients.ts index 28dcadd..40d538e 100644 --- a/water/src/broker/Subclients.ts +++ b/water/src/broker/Subclients.ts @@ -1,10 +1,8 @@ import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; import { parse } from "valibot"; -import { Logger } from "../logging/Logger.js"; +import { BrokerMessage, BrokerMessageHeaders, Logger } from "../index.js"; import type { BrokerClient } from "./BrokerClient.js"; import type { BrokerConnection, MessageId } from "./BrokerConnection.js"; -import { BrokerMessage } from "./BrokerMessage.js"; -import { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; export class BrokerClientOptions { // No options exist in the JS universe at the moment diff --git a/water/src/broker/rabbitmq/RabbitConnection.ts b/water/src/broker/rabbitmq/RabbitConnection.ts index 370bdf1..8541899 100644 --- a/water/src/broker/rabbitmq/RabbitConnection.ts +++ b/water/src/broker/rabbitmq/RabbitConnection.ts @@ -1,9 +1,7 @@ import type { Consumer, Publisher } from "rabbitmq-client"; import { Connection } from "rabbitmq-client"; import type { MessageId } from "../BrokerConnection.js"; -import { BrokerConnection } from "../BrokerConnection.js"; -import { BrokerMessageHeaders } from "../BrokerMessageHeaders.js"; -import { Logger } from "../../logging/Logger.js"; +import { BrokerConnection, BrokerMessageHeaders, Logger } from "../../index.js"; export class RabbitConnection extends BrokerConnection { From ef9153d883354c228c6e1ab9ef5c547082b7fa30 Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Wed, 17 Jul 2024 20:10:02 +0200 Subject: [PATCH 11/12] Properly use prettier --- .gitignore | 1 - water/.gitignore | 1 - water/.prettierrc.yml | 2 +- water/.vscode/settings.json | 16 + water/src/CommonConfig.ts | 10 +- water/src/broker/BrokerClient.ts | 410 +++++++++--------- water/src/broker/BrokerConnection.ts | 256 +++++------ water/src/broker/BrokerMessage.ts | 44 +- water/src/broker/BrokerMessageHeaders.ts | 139 +++--- water/src/broker/Exceptions.ts | 26 +- water/src/broker/LocalConnection.ts | 79 ++-- water/src/broker/Subclients.ts | 218 +++++----- water/src/broker/rabbitmq/RabbitConnection.ts | 282 ++++++------ water/src/broker/rpc/RpcClient.ts | 337 +++++++------- water/src/broker/rpc/RpcMessage.ts | 60 ++- water/src/broker/rpc/RpcMessageHeaders.ts | 137 +++--- water/src/broker/rpc/RpcStatus.ts | 20 +- water/src/util/CountDownLatch.ts | 108 ++--- water/src/util/OnlineEmitter.ts | 79 ++-- water/src/util/internal.ts | 7 + water/tsconfig.json | 6 +- 21 files changed, 1127 insertions(+), 1111 deletions(-) create mode 100644 water/.vscode/settings.json create mode 100644 water/src/util/internal.ts diff --git a/.gitignore b/.gitignore index 95d12f2..59294ff 100644 --- a/.gitignore +++ b/.gitignore @@ -1,7 +1,6 @@ .gradle/ build/ .idea/ -.vscode/ node_modules/ dist/ yarn-error.log diff --git a/water/.gitignore b/water/.gitignore index 0070105..e9816e4 100644 --- a/water/.gitignore +++ b/water/.gitignore @@ -1,5 +1,4 @@ .idea/ -.vscode/ .pnp.* .yarn/* !.yarn/patches diff --git a/water/.prettierrc.yml b/water/.prettierrc.yml index 1d7a376..3df0c25 100644 --- a/water/.prettierrc.yml +++ b/water/.prettierrc.yml @@ -3,7 +3,7 @@ singleQuote: false jsxSingleQuote: false bracketSameLine: true quoteProps: as-needed -trailingComma: es5 +trailingComma: all semi: true printWidth: 120 arrowParens: avoid diff --git a/water/.vscode/settings.json b/water/.vscode/settings.json new file mode 100644 index 0000000..b4d55df --- /dev/null +++ b/water/.vscode/settings.json @@ -0,0 +1,16 @@ +{ + "[typescript]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "[json]": { + "editor.defaultFormatter": "esbenp.prettier-vscode" + }, + "editor.insertSpaces": false, + "editor.detectIndentation": false, + "editor.formatOnSave": true, + "editor.tabSize": 4, + "prettier.requireConfig": true, + "prettier.configPath": ".prettierrc.yml", + "typescript.tsdk": "node_modules\\typescript\\lib", + "typescript.enablePromptUseWorkspaceTsdk": true +} diff --git a/water/src/CommonConfig.ts b/water/src/CommonConfig.ts index ffe8736..e3ac00e 100644 --- a/water/src/CommonConfig.ts +++ b/water/src/CommonConfig.ts @@ -1,7 +1,7 @@ export enum BrokerServices { - TEA = "tea", // Bot - VANILLA = "vanilla", // Bot cluster coordinator - MILK = "milk", // Raid logs - SUGAR = "sugar", // Premium management - COFFEE = "coffee" // Raid bans + TEA = "tea", // Bot + VANILLA = "vanilla", // Bot cluster coordinator + MILK = "milk", // Raid logs + SUGAR = "sugar", // Premium management + COFFEE = "coffee", // Raid bans } diff --git a/water/src/broker/BrokerClient.ts b/water/src/broker/BrokerClient.ts index 4f5ade1..4e1abde 100644 --- a/water/src/broker/BrokerClient.ts +++ b/water/src/broker/BrokerClient.ts @@ -1,220 +1,216 @@ import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; -import { BrokerClientOptions, ConsumerSubclient, Logger, ProducerSubclient, RpcClient } from "../index.js"; -import type { BrokerConnection, TopicListener } from "./BrokerConnection.js"; -import type { BrokerMessage } from "./BrokerMessage.js"; -import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; -import type { RpcRequestMessage, RpcResponse } from "./rpc/RpcMessage.js"; -import type { BaseSubclient } from "./Subclients.js"; +import type { + BaseSubclient, + BrokerConnection, + BrokerMessage, + BrokerMessageHeaders, + RpcRequestMessage, + RpcResponse, + TopicListener, +} from "@"; +import { BrokerClientOptions, ConsumerSubclient, Logger, ProducerSubclient, RpcClient } from "@"; + +const TAG = "BrokerClient"; export abstract class BrokerClient { - - private static readonly TAG = "BrokerClient"; - - private readonly topics: Map = new Map(); - - public constructor(public readonly connection: BrokerConnection) { } - - public consumer>>( - topic: string, - key: string, - schema: TSchema, - options: BrokerClientOptions = new BrokerClientOptions(), - callback: (msg: BrokerMessage>) => Promise, - ): ConsumerSubclient { - Logger.debug(BrokerClient.TAG, `Creating consumer for key '${key}' in topic '${topic}'`); - const client = new ConsumerSubclient(this.connection, this, topic, key, options, schema, callback); - this.registerSubclient(client); - return client; - } - - public producer( - topic: string, - key: string, - options: BrokerClientOptions = new BrokerClientOptions(), - ): ProducerSubclient { - Logger.debug(BrokerClient.TAG, `Creating producer for key '${key}' in topic '${topic}'`); - const client = new ProducerSubclient(this.connection, this, topic, key, options); - this.registerSubclient(client); - return client; - } - - public rpc< - RequestTSchema extends BaseSchema>, - ResponseTSchema extends BaseSchema>, - >( - topic: string, - key: string, - requestSchema: RequestTSchema, - responseSchema: ResponseTSchema, - options: BrokerClientOptions = new BrokerClientOptions(), - callback: (msg: RpcRequestMessage< - InferOutput, - InferOutput - >) => Promise>, - ): RpcClient { - return new RpcClient( - this, - topic, - key, - options, - requestSchema, - responseSchema, - callback, - ); - } - - private registerSubclient(subclient: BaseSubclient): void { - const topic = subclient.topic; - if (!this.topics.has(topic)) { - this.topics.set(topic, new TopicMetadata(this.connection, topic)); - } - const metadata = this.topics.get(topic)!; - metadata.registerSubclient(subclient); - } - - // Private API - public deregisterSubclient(subclient: BaseSubclient): void { - const topic = subclient.topic; - if (this.topics.has(topic)) { - const metadata = this.topics.get(topic)!; - metadata.deregisterSubclient(subclient); - if (metadata.isEmpty) { - this.topics.delete(topic); - } - } - } - - public destroy(): void { - Logger.debug(BrokerClient.TAG, `Destroying BrokerClient with active topics: ${[...this.topics.keys()]}`); - while (this.topics.size > 0) { - const [topic, metadata] = this.topics.entries().next().value as [string, TopicMetadata]; - metadata.destroy(); - this.topics.delete(topic); - } - } - + private readonly topics: Map = new Map(); + + public constructor(public readonly connection: BrokerConnection) {} + + public consumer>>( + topic: string, + key: string, + schema: TSchema, + callback: (msg: BrokerMessage>) => Promise, + options: BrokerClientOptions = new BrokerClientOptions(), + ): ConsumerSubclient { + Logger.debug(TAG, `Creating consumer for key '${key}' in topic '${topic}'`); + const client = new ConsumerSubclient(this.connection, this, topic, key, options, schema, callback); + this.registerSubclient(client); + return client; + } + + public producer( + topic: string, + key: string, + options: BrokerClientOptions = new BrokerClientOptions(), + ): ProducerSubclient { + Logger.debug(TAG, `Creating producer for key '${key}' in topic '${topic}'`); + const client = new ProducerSubclient(this.connection, this, topic, key, options); + this.registerSubclient(client); + return client; + } + + public rpc< + RequestTSchema extends BaseSchema>, + ResponseTSchema extends BaseSchema>, + >( + topic: string, + key: string, + requestSchema: RequestTSchema, + responseSchema: ResponseTSchema, + callback: ( + msg: RpcRequestMessage, InferOutput>, + ) => Promise>>, + options: BrokerClientOptions = new BrokerClientOptions(), + ): RpcClient { + return new RpcClient(this, topic, key, options, requestSchema, responseSchema, callback); + } + + private registerSubclient(subclient: BaseSubclient): void { + const topic = subclient.topic; + if (!this.topics.has(topic)) { + this.topics.set(topic, new TopicMetadata(this.connection, topic)); + } + const metadata = this.topics.get(topic)!; + metadata.registerSubclient(subclient); + } + + // Private API + public deregisterSubclient(subclient: BaseSubclient): void { + const topic = subclient.topic; + if (this.topics.has(topic)) { + const metadata = this.topics.get(topic)!; + metadata.deregisterSubclient(subclient); + if (metadata.isEmpty) { + this.topics.delete(topic); + } + } + } + + public destroy(): void { + Logger.debug(TAG, `Destroying BrokerClient with active topics: ${[...this.topics.keys()]}`); + while (this.topics.size > 0) { + const [topic, metadata] = this.topics.entries().next().value as [string, TopicMetadata]; + metadata.destroy(); + this.topics.delete(topic); + } + } } class TopicMetadata { - - private static readonly TAG = "TopicMetadata"; - - private readonly keys: Map = new Map(); - private isDestroyed: boolean = false; - private connectionListener: TopicListener | null = null; - - public constructor( - private readonly connection: BrokerConnection, - private readonly topic: string, - ) { } - - public get isEmpty(): boolean { - return this.keys.size === 0; - } - - public registerSubclient(subclient: BaseSubclient): void { - if (subclient.topic !== this.topic) { - throw new Error(`Attempting to register subclient with topic '${subclient.topic}' in TopicMetadata of '${this.topic}'`); - } - Logger.debug(TopicMetadata.TAG, `Adding ${this.constructor.name} for key '${subclient.key}' in topic '${this.topic}'`); - const metadata = this.getOrCreateKeyMetadata(subclient.key); - if (subclient instanceof ConsumerSubclient) { - if (metadata.consumers.size === 0 && this.connectionListener === null) { - Logger.debug(TopicMetadata.TAG, `Creating new connection listener for topic '${this.topic}'`); - this.connectionListener = (topic, key, value, headers) => this.onTopicMessage(topic, key, value, headers); - this.connection.on(this.topic, this.connectionListener); - } - metadata.consumers.add(subclient); - } else if (subclient instanceof ProducerSubclient) { - metadata.producers.add(subclient); - } - } - - public deregisterSubclient(subclient: BaseSubclient): void { - Logger.debug(TopicMetadata.TAG, `Removing ${this.constructor.name} for key '${subclient.key}' in topic '${this.topic}'`); - const metadata = this.getExistingKeyMetadata(subclient.key); - if (metadata) { - metadata.producers.delete(subclient as ProducerSubclient); - metadata.consumers.delete(subclient as ConsumerSubclient); - this.maybeCleanupKeyMetadata(metadata); - } - } - - private maybeCleanupKeyMetadata(metadata: KeyMetadata): void { - if (metadata.isEmpty) { - this.keys.delete(metadata.key); - } - if (this.isEmpty && this.connectionListener !== null) { - Logger.debug(TopicMetadata.TAG, `Removing connection listener for topic '${this.topic}' after key cleanup`); - this.connection.off(this.topic, this.connectionListener); - this.connectionListener = null; - } - } - - private getOrCreateKeyMetadata(key: string): KeyMetadata { - if (!this.keys.has(key)) { - this.keys.set(key, new KeyMetadata(key)); - } - return this.keys.get(key)!; - } - - private getExistingKeyMetadata(key: string): KeyMetadata | null { - return this.keys.get(key) ?? null; - } - - private onTopicMessage( - topic: string, - key: string, - value: string, - headers: BrokerMessageHeaders, - ): void { - const metadata = this.getExistingKeyMetadata(key); - if (!metadata) { - return; - } - for (const consumer of metadata.consumers) { - consumer - .onIncomingMessage(value, headers) - .catch(e => Logger.error(TopicMetadata.TAG, - `Uncaught error in BrokerClient listener for key '${key}' in topic '${topic}'`, e)); - } - } - - public destroy(): void { - if (this.isDestroyed) { - return; - } - while (this.keys.size > 0) { - const [key, metadata] = this.keys.entries().next().value as [string, KeyMetadata]; - metadata.destroy(); - this.keys.delete(key); - } - if (this.connectionListener !== null) { - Logger.debug(TopicMetadata.TAG, `Removing connection listener for topic '${this.topic}' during destroy`); - this.connection.off(this.topic, this.connectionListener); - this.connectionListener = null; - } - } - + private static readonly TAG = "TopicMetadata"; + + private readonly keys: Map = new Map(); + private isDestroyed: boolean = false; + private connectionListener: TopicListener | null = null; + + public constructor( + private readonly connection: BrokerConnection, + private readonly topic: string, + ) {} + + public get isEmpty(): boolean { + return this.keys.size === 0; + } + + public registerSubclient(subclient: BaseSubclient): void { + if (subclient.topic !== this.topic) { + throw new Error( + `Attempting to register subclient with topic '${subclient.topic}' in TopicMetadata of '${this.topic}'`, + ); + } + Logger.debug( + TopicMetadata.TAG, + `Adding ${this.constructor.name} for key '${subclient.key}' in topic '${this.topic}'`, + ); + const metadata = this.getOrCreateKeyMetadata(subclient.key); + if (subclient instanceof ConsumerSubclient) { + if (metadata.consumers.size === 0 && this.connectionListener === null) { + Logger.debug(TopicMetadata.TAG, `Creating new connection listener for topic '${this.topic}'`); + this.connectionListener = (topic, key, value, headers) => this.onTopicMessage(topic, key, value, headers); + this.connection.on(this.topic, this.connectionListener); + } + metadata.consumers.add(subclient); + } else if (subclient instanceof ProducerSubclient) { + metadata.producers.add(subclient); + } + } + + public deregisterSubclient(subclient: BaseSubclient): void { + Logger.debug( + TopicMetadata.TAG, + `Removing ${this.constructor.name} for key '${subclient.key}' in topic '${this.topic}'`, + ); + const metadata = this.getExistingKeyMetadata(subclient.key); + if (metadata) { + metadata.producers.delete(subclient as ProducerSubclient); + metadata.consumers.delete(subclient as ConsumerSubclient); + this.maybeCleanupKeyMetadata(metadata); + } + } + + private maybeCleanupKeyMetadata(metadata: KeyMetadata): void { + if (metadata.isEmpty) { + this.keys.delete(metadata.key); + } + if (this.isEmpty && this.connectionListener !== null) { + Logger.debug(TopicMetadata.TAG, `Removing connection listener for topic '${this.topic}' after key cleanup`); + this.connection.off(this.topic, this.connectionListener); + this.connectionListener = null; + } + } + + private getOrCreateKeyMetadata(key: string): KeyMetadata { + if (!this.keys.has(key)) { + this.keys.set(key, new KeyMetadata(key)); + } + return this.keys.get(key)!; + } + + private getExistingKeyMetadata(key: string): KeyMetadata | null { + return this.keys.get(key) ?? null; + } + + private onTopicMessage(topic: string, key: string, value: string, headers: BrokerMessageHeaders): void { + const metadata = this.getExistingKeyMetadata(key); + if (!metadata) { + return; + } + for (const consumer of metadata.consumers) { + consumer + .onIncomingMessage(value, headers) + .catch(e => + Logger.error( + TopicMetadata.TAG, + `Uncaught error in BrokerClient listener for key '${key}' in topic '${topic}'`, + e, + ), + ); + } + } + + public destroy(): void { + if (this.isDestroyed) { + return; + } + while (this.keys.size > 0) { + const [key, metadata] = this.keys.entries().next().value as [string, KeyMetadata]; + metadata.destroy(); + this.keys.delete(key); + } + if (this.connectionListener !== null) { + Logger.debug(TopicMetadata.TAG, `Removing connection listener for topic '${this.topic}' during destroy`); + this.connection.off(this.topic, this.connectionListener); + this.connectionListener = null; + } + } } class KeyMetadata { + public readonly producers: Set> = new Set(); + public readonly consumers: Set> = new Set(); - public readonly producers: Set> = new Set(); - public readonly consumers: Set> = new Set(); - - public constructor( - public readonly key: string, - ) { } + public constructor(public readonly key: string) {} - public get isEmpty(): boolean { - return this.producers.size === 0 && this.consumers.size === 0; - } + public get isEmpty(): boolean { + return this.producers.size === 0 && this.consumers.size === 0; + } - public destroy(): void { - this.producers.forEach(producer => producer.destroy()); - this.consumers.forEach(consumer => consumer.destroy()); - this.producers.clear(); - this.consumers.clear(); - } + public destroy(): void { + this.producers.forEach(producer => producer.destroy()); + this.consumers.forEach(consumer => consumer.destroy()); + this.producers.clear(); + this.consumers.clear(); + } } diff --git a/water/src/broker/BrokerConnection.ts b/water/src/broker/BrokerConnection.ts index 0d97e21..8d0da68 100644 --- a/water/src/broker/BrokerConnection.ts +++ b/water/src/broker/BrokerConnection.ts @@ -1,6 +1,5 @@ -import { Logger } from "../index.js"; -import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; - +import type { BrokerMessageHeaders } from "@"; +import { Logger } from "@"; export type TopicListener = (topic: string, key: string, value: string, headers: BrokerMessageHeaders) => void; @@ -9,124 +8,135 @@ export type MessageId = string; const TAG = "BrokerConnection"; export abstract class BrokerConnection { - - public abstract readonly serviceName: string; - public abstract readonly instanceId: string; - public abstract readonly supportsTopicHotSwap: boolean; - public abstract readonly deferInitialTopicCreation: boolean; - - protected readonly topicListeners: Map> = new Map(); - private readonly deferredTopicsToCreate: Set = new Set(); - private hasStarted: boolean = false; - - public async start(): Promise { - await this.abstractStart(); - this.hasStarted = true; - for (const topic of this.deferredTopicsToCreate) { - this.createTopic(topic); - } - this.deferredTopicsToCreate.clear(); - } - - public async destroy(): Promise { - Logger.debug(TAG, "Destroying BrokerConnection"); - this.topicListeners.clear(); - } - - // Internal API - public abstract abstractStart(): Promise; - // Internal API - public abstract abstractSend(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise; - - public send(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { - Logger.verbose(TAG, `Sending message ${headers.messageId} with key '${key}' in topic '${topic}'`); - return this.abstractSend(topic, key, value, headers); - } - - protected abstract createTopic(topic: string): void; - protected abstract removeTopic(topic: string): void; - - public on(topic: string, cb: TopicListener): void { - let listeners = this.topicListeners.get(topic); - if (!listeners) { - listeners = new Set(); - if (!this.hasStarted && this.deferInitialTopicCreation) { - Logger.debug(TAG, `Deferring creation of topic '${topic}' to after connected`); - this.deferredTopicsToCreate.add(topic); - } else { - Logger.debug(TAG, `Creating new topic '${topic}'`); - this.createTopic(topic); - } - this.topicListeners.set(topic, listeners); - } - listeners.add(cb); - } - - public off(topic: string, cb: TopicListener): void { - const listeners = this.topicListeners.get(topic); - if (listeners) { - listeners.delete(cb); - if (listeners.size === 0) { - Logger.debug(TAG, `Removing topic '${topic}'`); - this.deferredTopicsToCreate.delete(topic); - this.removeTopic(topic); - this.topicListeners.delete(topic); - } - } - } - - // To be called by implementers - protected dispatchIncomingMessage(topic: string, key: string, value: string, headers: BrokerMessageHeaders): void { - if ( - (headers.targetServices.size > 0 && !headers.targetServices.has(this.serviceName)) || - (headers.targetInstances.size > 0 && !headers.targetInstances.has(this.instanceId)) - ) { - // If there is a target cluster restriction and this message wasn't meant for us, - // discard it immediately without notifying any listeners. - return; - } - if (headers.sourceInstance === this.instanceId && headers.sourceService === this.serviceName) { - // If this message was sent by ourselves, discard it too, as we already dispatch events - // to our listeners in `send()` to avoid the round trip through an external service. - return; - } - this.invokeLocalCallbacks(topic, key, value, headers); - } - - protected shouldDispatchExternallyAfterShortCircuit(topic: string, key: string, value: string, headers: BrokerMessageHeaders): boolean { - const targetServices = headers.targetServices; - const targetInstances = headers.targetInstances; - const isThisConnectionTargeted = (targetServices.size === 0 || targetServices.has(this.serviceName)) && - (targetInstances.size === 0 || targetInstances.has(this.instanceId)); - - // If the message is meant for ourselves (amongst other clusters), - // immediately dispatch it to the listeners. - if (isThisConnectionTargeted) { - this.invokeLocalCallbacks(topic, key, value, headers); - } - - // Return whether implementers should dispatch this message to external services - return ( - // For all services/instances - targetServices.size === 0 || targetInstances.size === 0 || - // Not for us, so it must be for somebody else - !isThisConnectionTargeted || - // For us, so check if it is also for someone else - targetServices.size > 1 || targetInstances.size > 1 - ); - } - - private invokeLocalCallbacks(topic: string, key: string, value: string, headers: BrokerMessageHeaders): void { - const listeners = this.topicListeners.get(topic); - if (listeners) { - for (const listener of listeners) { - try { - listener(topic, key, value, headers); - } catch (e) { - Logger.error(TAG, `Uncaught error in BrokerConnection listener for key '${key}' in topic '${topic}'`, e); - } - } - } - } - + public abstract readonly serviceName: string; + public abstract readonly instanceId: string; + public abstract readonly supportsTopicHotSwap: boolean; + public abstract readonly deferInitialTopicCreation: boolean; + + protected readonly topicListeners: Map> = new Map(); + private readonly deferredTopicsToCreate: Set = new Set(); + private hasStarted: boolean = false; + + public async start(): Promise { + await this.abstractStart(); + this.hasStarted = true; + for (const topic of this.deferredTopicsToCreate) { + this.createTopic(topic); + } + this.deferredTopicsToCreate.clear(); + } + + public async destroy(): Promise { + Logger.debug(TAG, "Destroying BrokerConnection"); + this.topicListeners.clear(); + } + + // Internal API + public abstract abstractStart(): Promise; + // Internal API + public abstract abstractSend( + topic: string, + key: string, + value: string, + headers: BrokerMessageHeaders, + ): Promise; + + public send(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { + Logger.verbose(TAG, `Sending message ${headers.messageId} with key '${key}' in topic '${topic}'`); + return this.abstractSend(topic, key, value, headers); + } + + protected abstract createTopic(topic: string): void; + protected abstract removeTopic(topic: string): void; + + public on(topic: string, cb: TopicListener): void { + let listeners = this.topicListeners.get(topic); + if (!listeners) { + listeners = new Set(); + if (!this.hasStarted && this.deferInitialTopicCreation) { + Logger.debug(TAG, `Deferring creation of topic '${topic}' to after connected`); + this.deferredTopicsToCreate.add(topic); + } else { + Logger.debug(TAG, `Creating new topic '${topic}'`); + this.createTopic(topic); + } + this.topicListeners.set(topic, listeners); + } + listeners.add(cb); + } + + public off(topic: string, cb: TopicListener): void { + const listeners = this.topicListeners.get(topic); + if (listeners) { + listeners.delete(cb); + if (listeners.size === 0) { + Logger.debug(TAG, `Removing topic '${topic}'`); + this.deferredTopicsToCreate.delete(topic); + this.removeTopic(topic); + this.topicListeners.delete(topic); + } + } + } + + // To be called by implementers + protected dispatchIncomingMessage(topic: string, key: string, value: string, headers: BrokerMessageHeaders): void { + if ( + (headers.targetServices.size > 0 && !headers.targetServices.has(this.serviceName)) || + (headers.targetInstances.size > 0 && !headers.targetInstances.has(this.instanceId)) + ) { + // If there is a target cluster restriction and this message wasn't meant for us, + // discard it immediately without notifying any listeners. + return; + } + if (headers.sourceInstance === this.instanceId && headers.sourceService === this.serviceName) { + // If this message was sent by ourselves, discard it too, as we already dispatch events + // to our listeners in `send()` to avoid the round trip through an external service. + return; + } + this.invokeLocalCallbacks(topic, key, value, headers); + } + + protected shouldDispatchExternallyAfterShortCircuit( + topic: string, + key: string, + value: string, + headers: BrokerMessageHeaders, + ): boolean { + const targetServices = headers.targetServices; + const targetInstances = headers.targetInstances; + const isThisConnectionTargeted = + (targetServices.size === 0 || targetServices.has(this.serviceName)) && + (targetInstances.size === 0 || targetInstances.has(this.instanceId)); + + // If the message is meant for ourselves (amongst other clusters), + // immediately dispatch it to the listeners. + if (isThisConnectionTargeted) { + this.invokeLocalCallbacks(topic, key, value, headers); + } + + // Return whether implementers should dispatch this message to external services + return ( + // For all services/instances + targetServices.size === 0 || + targetInstances.size === 0 || + // Not for us, so it must be for somebody else + !isThisConnectionTargeted || + // For us, so check if it is also for someone else + targetServices.size > 1 || + targetInstances.size > 1 + ); + } + + private invokeLocalCallbacks(topic: string, key: string, value: string, headers: BrokerMessageHeaders): void { + const listeners = this.topicListeners.get(topic); + if (listeners) { + for (const listener of listeners) { + try { + listener(topic, key, value, headers); + } catch (e) { + Logger.error(TAG, `Uncaught error in BrokerConnection listener for key '${key}' in topic '${topic}'`, e); + } + } + } + } } diff --git a/water/src/broker/BrokerMessage.ts b/water/src/broker/BrokerMessage.ts index 0b4bdb9..fce9e1f 100644 --- a/water/src/broker/BrokerMessage.ts +++ b/water/src/broker/BrokerMessage.ts @@ -1,29 +1,27 @@ -import { RpcMessageHeaders, RpcRequestMessage, RpcResponseMessage } from "../index.js"; -import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; -import type { RpcStatus } from "./rpc/RpcStatus.js"; +import type { BrokerMessageHeaders, RpcStatus } from "@"; +import { RpcMessageHeaders, RpcRequestMessage, RpcResponseMessage } from "@"; export class BrokerMessage { - public constructor( - public readonly topic: string, - public readonly key: string, - public readonly value: T, - public readonly headers: H - ) { } + public constructor( + public readonly topic: string, + public readonly key: string, + public readonly value: T, + public readonly headers: H, + ) {} - public get messageId(): string { - return this.headers.messageId; - } + public get messageId(): string { + return this.headers.messageId; + } - // Private API - public toRpcRequestMessage( - updateSender: (status: RpcStatus, data: ResponseT) => Promise, - ): RpcRequestMessage { - return new RpcRequestMessage(this.topic, this.key, this.value, this.headers, updateSender); - } - - // Private API - public toRpcResponseMessage(): RpcResponseMessage { - return new RpcResponseMessage(this.topic, this.key, this.value, new RpcMessageHeaders(this.headers)); - } + // Private API + public toRpcRequestMessage( + updateSender: (status: RpcStatus, data: ResponseT) => Promise, + ): RpcRequestMessage { + return new RpcRequestMessage(this.topic, this.key, this.value, this.headers, updateSender); + } + // Private API + public toRpcResponseMessage(): RpcResponseMessage { + return new RpcResponseMessage(this.topic, this.key, this.value, new RpcMessageHeaders(this.headers)); + } } diff --git a/water/src/broker/BrokerMessageHeaders.ts b/water/src/broker/BrokerMessageHeaders.ts index 9fa6d22..7b9f1bf 100644 --- a/water/src/broker/BrokerMessageHeaders.ts +++ b/water/src/broker/BrokerMessageHeaders.ts @@ -1,86 +1,73 @@ import { randomUUID } from "node:crypto"; -import { BrokerConnection } from "../index.js"; -import type { MessageId } from "./BrokerConnection.js"; +import { getOrThrow } from "../util/internal.js"; +import type { MessageId } from "@"; +import { BrokerConnection } from "@"; const BROKER_HEADER_NAMES = { - SOURCE_SERVICE: "source-service", - SOURCE_INSTANCE: "source-instance", - TARGET_SERVICES: "target-services", - TARGET_INSTANCES: "target-instances", - MESSAGE_ID: "message-id", + SOURCE_SERVICE: "source-service", + SOURCE_INSTANCE: "source-instance", + TARGET_SERVICES: "target-services", + TARGET_INSTANCES: "target-instances", + MESSAGE_ID: "message-id", }; export class BrokerMessageHeaders { + public readonly headers: Record; - public readonly headers: Record; + public get sourceService(): string { + return getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_SERVICE); + } + public get sourceInstance(): string { + return getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_INSTANCE); + } + public get targetServices(): Set { + return new Set((this.headers[BROKER_HEADER_NAMES.TARGET_SERVICES] ?? "").split(",").filter(s => s)); + } + public get targetInstances(): Set { + return new Set((this.headers[BROKER_HEADER_NAMES.TARGET_INSTANCES] ?? "").split(",").filter(s => s)); + } + public get messageId(): MessageId { + return getOrThrow(this.headers, BROKER_HEADER_NAMES.MESSAGE_ID); + } - public get sourceService(): string { - return getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_SERVICE); - } - public get sourceInstance(): string { - return getOrThrow(this.headers, BROKER_HEADER_NAMES.SOURCE_INSTANCE); - } - public get targetServices(): Set { - return new Set((this.headers[BROKER_HEADER_NAMES.TARGET_SERVICES] ?? "").split(",").filter(s => s)); - } - public get targetInstances(): Set { - return new Set((this.headers[BROKER_HEADER_NAMES.TARGET_INSTANCES] ?? "").split(",").filter(s => s)); - } - public get messageId(): MessageId { - return getOrThrow(this.headers, BROKER_HEADER_NAMES.MESSAGE_ID); - } + public constructor(headers: Record); + public constructor(connection: BrokerConnection, targetServices: Set, targetInstances: Set); + public constructor( + headersOrConnection: BrokerConnection | Record, + targetServices?: Set, + targetInstances?: Set, + ) { + // I hate that JS doesn't have native method overloading + if (headersOrConnection instanceof BrokerConnection) { + const connection = headersOrConnection; + this.headers = BrokerMessageHeaders.createHeadersMap( + connection.serviceName, + connection.instanceId, + targetServices!, + targetInstances!, + null, + ); + } else { + this.headers = headersOrConnection; + } + } - public constructor(headers: Record); - public constructor( - connection: BrokerConnection, - targetServices: Set, - targetInstances: Set, - ); - public constructor( - headersOrConnection: BrokerConnection | Record, - targetServices?: Set, - targetInstances?: Set, - ) { - // I hate that JS doesn't have native method overloading - if (headersOrConnection instanceof BrokerConnection) { - const connection = headersOrConnection; - this.headers = BrokerMessageHeaders.createHeadersMap( - connection.serviceName, - connection.instanceId, - targetServices!, - targetInstances!, - null, - ); - } else { - this.headers = headersOrConnection; - } - } - - protected static createHeadersMap( - sourceService: string, - sourceInstance: string, - targetServices: Set, - targetInstances: Set, - messageId: MessageId | null, - extra: Record = {}, - ): Record { - const headers: Record = { - [BROKER_HEADER_NAMES.SOURCE_SERVICE]: sourceService, - [BROKER_HEADER_NAMES.SOURCE_INSTANCE]: sourceInstance, - [BROKER_HEADER_NAMES.TARGET_SERVICES]: [...targetServices].join(","), - [BROKER_HEADER_NAMES.TARGET_INSTANCES]: [...targetInstances].join(","), - [BROKER_HEADER_NAMES.MESSAGE_ID]: messageId ?? randomUUID(), - ...extra, - }; - return headers; - } - -} - -export function getOrThrow(headers: Record, key: string): string { - const value = headers[key]; - if (value === undefined) { - throw new Error(`Missing broker message header '${key}'`); - } - return value; + protected static createHeadersMap( + sourceService: string, + sourceInstance: string, + targetServices: Set, + targetInstances: Set, + messageId: MessageId | null, + extra: Record = {}, + ): Record { + const headers: Record = { + [BROKER_HEADER_NAMES.SOURCE_SERVICE]: sourceService, + [BROKER_HEADER_NAMES.SOURCE_INSTANCE]: sourceInstance, + [BROKER_HEADER_NAMES.TARGET_SERVICES]: [...targetServices].join(","), + [BROKER_HEADER_NAMES.TARGET_INSTANCES]: [...targetInstances].join(","), + [BROKER_HEADER_NAMES.MESSAGE_ID]: messageId ?? randomUUID(), + ...extra, + }; + return headers; + } } diff --git a/water/src/broker/Exceptions.ts b/water/src/broker/Exceptions.ts index 8a2568e..b9581b7 100644 --- a/water/src/broker/Exceptions.ts +++ b/water/src/broker/Exceptions.ts @@ -1,25 +1,25 @@ -import type { RpcStatus } from "../index.js"; +import type { RpcStatus } from "@"; export class BrokerException extends Error { - public constructor(message?: string) { - super(message); - } + public constructor(message?: string) { + super(message); + } } export class RpcRequestTimeout extends BrokerException { - public constructor(message: string) { - super(message); - } + public constructor(message: string) { + super(message); + } } export class IgnoreRpcRequest extends BrokerException { - public constructor() { - super("Ignoring RPC request"); - } + public constructor() { + super("Ignoring RPC request"); + } } export class RpcException extends BrokerException { - public constructor(public readonly status: RpcStatus) { - super(`RPC failed with status ${status}`); - } + public constructor(public readonly status: RpcStatus) { + super(`RPC failed with status ${status}`); + } } diff --git a/water/src/broker/LocalConnection.ts b/water/src/broker/LocalConnection.ts index 2b369e8..3fe1112 100644 --- a/water/src/broker/LocalConnection.ts +++ b/water/src/broker/LocalConnection.ts @@ -1,38 +1,47 @@ -import { BrokerConnection } from "../index.js"; -import type { MessageId } from "./BrokerConnection.js"; -import type { BrokerMessageHeaders } from "./BrokerMessageHeaders.js"; +import type { BrokerMessageHeaders, MessageId } from "@"; +import { BrokerConnection } from "@"; export class LocalConnection extends BrokerConnection { - - public override readonly supportsTopicHotSwap: boolean = true; - public override readonly deferInitialTopicCreation: boolean = false; - - public constructor(public override readonly serviceName: string, public override readonly instanceId: string) { - super(); - } - - public override async abstractSend(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { - if (this.shouldDispatchExternallyAfterShortCircuit(topic, key, value, headers)) { - const isForExternalService = headers.targetServices.size > 0 && [...headers.targetServices][0] !== this.serviceName; - const isForExternalInstance = headers.targetInstances.size > 0 && [...headers.targetInstances][0] !== this.instanceId; - if (isForExternalService || isForExternalInstance) { - throw new Error("Attempting to send message to other services/instances in a LocalConnection " + - `(services=${[...headers.targetServices]}; instances=${[...headers.targetInstances]})`); - } - } - return headers.messageId; - } - - public override async abstractStart(): Promise { - // Nothing to start :) - } - - public override createTopic(topic: string): void { - // noop - } - - public override removeTopic(topic: string): void { - // noop - } - + public override readonly supportsTopicHotSwap: boolean = true; + public override readonly deferInitialTopicCreation: boolean = false; + + public constructor( + public override readonly serviceName: string, + public override readonly instanceId: string, + ) { + super(); + } + + public override async abstractSend( + topic: string, + key: string, + value: string, + headers: BrokerMessageHeaders, + ): Promise { + if (this.shouldDispatchExternallyAfterShortCircuit(topic, key, value, headers)) { + const isForExternalService = + headers.targetServices.size > 0 && [...headers.targetServices][0] !== this.serviceName; + const isForExternalInstance = + headers.targetInstances.size > 0 && [...headers.targetInstances][0] !== this.instanceId; + if (isForExternalService || isForExternalInstance) { + throw new Error( + "Attempting to send message to other services/instances in a LocalConnection " + + `(services=${[...headers.targetServices]}; instances=${[...headers.targetInstances]})`, + ); + } + } + return headers.messageId; + } + + public override async abstractStart(): Promise { + // Nothing to start :) + } + + public override createTopic(topic: string): void { + // noop + } + + public override removeTopic(topic: string): void { + // noop + } } diff --git a/water/src/broker/Subclients.ts b/water/src/broker/Subclients.ts index 40d538e..3f68632 100644 --- a/water/src/broker/Subclients.ts +++ b/water/src/broker/Subclients.ts @@ -1,127 +1,119 @@ import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; import { parse } from "valibot"; -import { BrokerMessage, BrokerMessageHeaders, Logger } from "../index.js"; -import type { BrokerClient } from "./BrokerClient.js"; -import type { BrokerConnection, MessageId } from "./BrokerConnection.js"; +import { BrokerMessage, BrokerMessageHeaders, Logger } from "@"; +import type { BrokerClient, BrokerConnection, MessageId } from "@"; export class BrokerClientOptions { - // No options exist in the JS universe at the moment + // No options exist in the JS universe at the moment } export abstract class BaseSubclient { - - private isDestroyed: boolean = false; - - public constructor( - protected readonly connection: BrokerConnection, - protected readonly client: BrokerClient, - public readonly topic: string, - public readonly key: string, - protected readonly options: BrokerClientOptions, - ) { } - - protected abstract doDestroy(): void; - - public destroy(): void { - if (!this.isDestroyed) { - this.isDestroyed = true; - this.doDestroy(); - } - } + private isDestroyed: boolean = false; + + public constructor( + protected readonly connection: BrokerConnection, + protected readonly client: BrokerClient, + public readonly topic: string, + public readonly key: string, + protected readonly options: BrokerClientOptions, + ) {} + + protected abstract doDestroy(): void; + + public destroy(): void { + if (!this.isDestroyed) { + this.isDestroyed = true; + this.doDestroy(); + } + } } export class ProducerSubclient extends BaseSubclient { - - private static readonly TAG = "ProducerSubclient"; - - public constructor( - connection: BrokerConnection, - client: BrokerClient, - topic: string, - key: string, - options: BrokerClientOptions, - ) { - super(connection, client, topic, key, options); - } - - protected doDestroy(): void { - this.client.deregisterSubclient(this); - } - - public async send( - data: T, - services: Set = new Set(), - instances: Set = new Set(), - ): Promise { - const msg = new BrokerMessage( - this.topic, - this.key, - data, - new BrokerMessageHeaders( - this.connection, - services, - instances, - ), - ); - return this.internalSend(msg); - } - - // Private API - public async internalSend(msg: BrokerMessage): Promise { - const stringifiedData = this.stringifyOutgoing(msg.value); - Logger.verbose(ProducerSubclient.TAG, `Sending message ${msg.messageId} with key '${msg.key}' in topic '${msg.topic}'`); - return this.connection.send(msg.topic, msg.key, stringifiedData, msg.headers); - } - - private stringifyOutgoing(data: T | null): string { - return JSON.stringify(data); - } - + private static readonly TAG = "ProducerSubclient"; + + public constructor( + connection: BrokerConnection, + client: BrokerClient, + topic: string, + key: string, + options: BrokerClientOptions, + ) { + super(connection, client, topic, key, options); + } + + protected doDestroy(): void { + this.client.deregisterSubclient(this); + } + + public async send( + data: T, + services: Set = new Set(), + instances: Set = new Set(), + ): Promise { + const msg = new BrokerMessage( + this.topic, + this.key, + data, + new BrokerMessageHeaders(this.connection, services, instances), + ); + return this.internalSend(msg); + } + + // Private API + public async internalSend(msg: BrokerMessage): Promise { + const stringifiedData = this.stringifyOutgoing(msg.value); + Logger.verbose( + ProducerSubclient.TAG, + `Sending message ${msg.messageId} with key '${msg.key}' in topic '${msg.topic}'`, + ); + return this.connection.send(msg.topic, msg.key, stringifiedData, msg.headers); + } + + private stringifyOutgoing(data: T | null): string { + return JSON.stringify(data); + } } export class ConsumerSubclient>> extends BaseSubclient { - - private static readonly TAG = "ConsumerSubclient"; - - public constructor( - connection: BrokerConnection, - client: BrokerClient, - topic: string, - key: string, - options: BrokerClientOptions, - private readonly schema: TSchema, - private readonly callback: (msg: BrokerMessage>) => Promise, - ) { - super(connection, client, topic, key, options); - } - - protected doDestroy(): void { - this.client.deregisterSubclient(this); - } - - // Private API - public async onIncomingMessage( - value: string, - headers: BrokerMessageHeaders, - ): Promise { - const data = this.parseIncoming(value); - const msg = new BrokerMessage( - this.topic, - this.key, - data, - headers, - ); - Logger.verbose(ConsumerSubclient.TAG, `Received message ${msg.messageId} with key '${msg.key}' in topic '${msg.topic}'`); - try { - await this.callback(msg); - } catch (e) { - Logger.error(ConsumerSubclient.TAG, "Uncaught consumer callback error while processing message " + - `${headers.messageId} with key '${this.key}' in topic '${this.topic}'`, e); - } - } - - private parseIncoming(data: string): InferOutput { - return parse(this.schema, JSON.parse(data)); - } - + private static readonly TAG = "ConsumerSubclient"; + + public constructor( + connection: BrokerConnection, + client: BrokerClient, + topic: string, + key: string, + options: BrokerClientOptions, + private readonly schema: TSchema, + private readonly callback: (msg: BrokerMessage>) => Promise, + ) { + super(connection, client, topic, key, options); + } + + protected doDestroy(): void { + this.client.deregisterSubclient(this); + } + + // Private API + public async onIncomingMessage(value: string, headers: BrokerMessageHeaders): Promise { + const data = this.parseIncoming(value); + const msg = new BrokerMessage(this.topic, this.key, data, headers); + Logger.verbose( + ConsumerSubclient.TAG, + `Received message ${msg.messageId} with key '${msg.key}' in topic '${msg.topic}'`, + ); + try { + await this.callback(msg); + } catch (e) { + Logger.error( + ConsumerSubclient.TAG, + "Uncaught consumer callback error while processing message " + + `${headers.messageId} with key '${this.key}' in topic '${this.topic}'`, + e, + ); + } + } + + private parseIncoming(data: string): InferOutput { + return parse(this.schema, JSON.parse(data)); + } } diff --git a/water/src/broker/rabbitmq/RabbitConnection.ts b/water/src/broker/rabbitmq/RabbitConnection.ts index 8541899..9d4e569 100644 --- a/water/src/broker/rabbitmq/RabbitConnection.ts +++ b/water/src/broker/rabbitmq/RabbitConnection.ts @@ -1,149 +1,165 @@ import type { Consumer, Publisher } from "rabbitmq-client"; import { Connection } from "rabbitmq-client"; -import type { MessageId } from "../BrokerConnection.js"; -import { BrokerConnection, BrokerMessageHeaders, Logger } from "../../index.js"; +import type { MessageId } from "@"; +import { BrokerConnection, BrokerMessageHeaders, Logger } from "@"; export class RabbitConnection extends BrokerConnection { + private static readonly TAG = "RabbitConnection"; - private static readonly TAG = "RabbitConnection"; + public override supportsTopicHotSwap: boolean = true; + public override deferInitialTopicCreation: boolean = true; - public override supportsTopicHotSwap: boolean = true; - public override deferInitialTopicCreation: boolean = true; + private connection: Connection | null = null; + private readonly consumers: Map = new Map(); + private readonly publishers: Map = new Map(); - private connection: Connection | null = null; - private readonly consumers: Map = new Map(); - private readonly publishers: Map = new Map(); + public constructor( + private readonly rabbitHosts: string[], + public override readonly serviceName: string, + public override readonly instanceId: string, + private readonly useTls: boolean = false, + private readonly username: string = "guest", + private readonly password: string = "guest", + ) { + super(); + } - public constructor( - private readonly rabbitHosts: string[], - public override readonly serviceName: string, - public override readonly instanceId: string, - private readonly useTls: boolean = false, - private readonly username: string = "guest", - private readonly password: string = "guest", - ) { - super(); - } + public override async abstractStart(): Promise { + this.connection = new Connection({ + hosts: this.rabbitHosts, + username: this.username, + password: this.password, + tls: this.useTls, + connectionName: `${this.serviceName}: ${this.instanceId}`, + }); + this.connection.on("connection.blocked", reason => { + Logger.warn(RabbitConnection.TAG, `RabbitMQ server blocked connection for reason: ${reason}`); + }); + this.connection.on("connection.unblocked", () => { + Logger.info(RabbitConnection.TAG, "RabbitMQ server unblocked connection"); + }); + await new Promise((resolve, reject) => { + // Use `on` instead of `once` to re-use the same logging code for future connection updates + this.connection!.on("connection", () => { + Logger.info(RabbitConnection.TAG, "RabbitMQ connection (re)established"); + resolve(); + }); + this.connection!.on("error", error => { + Logger.error(RabbitConnection.TAG, "Error in RabbitMQ connection", error); + reject(error); + }); + }); + } - public override async abstractStart(): Promise { - this.connection = new Connection({ - hosts: this.rabbitHosts, - username: this.username, - password: this.password, - tls: this.useTls, - connectionName: `${this.serviceName}: ${this.instanceId}`, - }); - this.connection.on("connection.blocked", reason => { - Logger.warn(RabbitConnection.TAG, `RabbitMQ server blocked connection for reason: ${reason}`); - }); - this.connection.on("connection.unblocked", () => { - Logger.info(RabbitConnection.TAG, "RabbitMQ server unblocked connection"); - }); - await new Promise((resolve, reject) => { - // Use `on` instead of `once` to re-use the same logging code for future connection updates - this.connection!.on("connection", () => { - Logger.info(RabbitConnection.TAG, "RabbitMQ connection (re)established"); - resolve(); - }); - this.connection!.on("error", error => { - Logger.error(RabbitConnection.TAG, "Error in RabbitMQ connection", error); - reject(error); - }); - }); - } + public override async destroy(): Promise { + await super.destroy(); + await this.connection?.close(); + } - public override async destroy(): Promise { - await super.destroy(); - await this.connection?.close(); - } + public override async abstractSend( + topic: string, + key: string, + value: string, + headers: BrokerMessageHeaders, + ): Promise { + if (this.shouldDispatchExternallyAfterShortCircuit(topic, key, value, headers)) { + let publisher = this.publishers.get(topic); + if (!publisher) { + publisher = this.ensureConnection().createPublisher(this.commonPublisherConsumerArguments(topic)); + this.publishers.set(topic, publisher); + } + await publisher.send( + { + exchange: topic, + routingKey: key, + headers: headers.headers, + messageId: headers.messageId, + durable: true, + }, + value, + ); + } + return headers.messageId; + } - public override async abstractSend(topic: string, key: string, value: string, headers: BrokerMessageHeaders): Promise { - if (this.shouldDispatchExternallyAfterShortCircuit(topic, key, value, headers)) { - let publisher = this.publishers.get(topic); - if (!publisher) { - publisher = this - .ensureConnection() - .createPublisher(this.commonPublisherConsumerArguments(topic)); - this.publishers.set(topic, publisher); - } - await publisher.send({ - exchange: topic, - routingKey: key, - headers: headers.headers, - messageId: headers.messageId, - durable: true, - }, value); - } - return headers.messageId; - } + protected override createTopic(topic: string): void { + if (this.consumers.has(topic)) { + return; + } + const queueName = this.createQueueName(topic); + this.consumers.set( + topic, + this.ensureConnection().createConsumer( + { + queue: queueName, + ...this.commonPublisherConsumerArguments(topic), + }, + async (msg, reply) => { + const key = msg.routingKey; + const value = msg.body instanceof Buffer ? msg.body.toString("utf8") : String(msg.body); + const headers = new BrokerMessageHeaders(msg.headers ?? {}); + this.dispatchIncomingMessage(topic, key, value, headers); + }, + ), + ); + } - protected override createTopic(topic: string): void { - if (this.consumers.has(topic)) { - return; - } - const queueName = this.createQueueName(topic); - this.consumers.set(topic, this.ensureConnection().createConsumer({ - queue: queueName, - ...this.commonPublisherConsumerArguments(topic), - }, async (msg, reply) => { - const key = msg.routingKey; - const value = msg.body instanceof Buffer ? msg.body.toString("utf8") : String(msg.body); - const headers = new BrokerMessageHeaders(msg.headers ?? {}); - this.dispatchIncomingMessage(topic, key, value, headers); - })); - } + protected override removeTopic(topic: string): void { + const consumer = this.consumers.get(topic); + if (consumer) { + consumer.close().catch(e => { + Logger.error(RabbitConnection.TAG, `Error closing consumer for topic ${topic}`, e); + }); + this.consumers.delete(topic); + } + const publisher = this.publishers.get(topic); + if (publisher) { + publisher.close().catch(e => { + Logger.error(RabbitConnection.TAG, `Error closing publisher for topic ${topic}`, e); + }); + this.publishers.delete(topic); + } + } - protected override removeTopic(topic: string): void { - const consumer = this.consumers.get(topic); - if (consumer) { - consumer.close().catch(e => { - Logger.error(RabbitConnection.TAG, `Error closing consumer for topic ${topic}`, e); - }); - this.consumers.delete(topic); - } - const publisher = this.publishers.get(topic); - if (publisher) { - publisher.close().catch(e => { - Logger.error(RabbitConnection.TAG, `Error closing publisher for topic ${topic}`, e); - }); - this.publishers.delete(topic); - } - } + private ensureConnection(): Connection { + const connection = this.connection; + if (!connection) { + throw new Error("Connection not open"); + } + return connection; + } - private ensureConnection(): Connection { - const connection = this.connection; - if (!connection) { - throw new Error("Connection not open"); - } - return connection; - } - - private createQueueName(topic: string): string { - return `${this.serviceName}.${this.instanceId}.${topic}`; - } - - private commonPublisherConsumerArguments(topic: string) { - const exchangeName = topic; - const queueName = this.createQueueName(topic); - const routingKey = "#"; - return { - queues: [{ - queue: queueName, - durable: true, - exclusive: false, - autoDelete: false, - }], - exchanges: [{ - exchange: exchangeName, - type: "topic", - durable: true, - }], - queueBindings: [{ - queue: queueName, - exchange: exchangeName, - routingKey, - }], - }; - } + private createQueueName(topic: string): string { + return `${this.serviceName}.${this.instanceId}.${topic}`; + } + private commonPublisherConsumerArguments(topic: string) { + const exchangeName = topic; + const queueName = this.createQueueName(topic); + const routingKey = "#"; + return { + queues: [ + { + queue: queueName, + durable: true, + exclusive: false, + autoDelete: false, + }, + ], + exchanges: [ + { + exchange: exchangeName, + type: "topic", + durable: true, + }, + ], + queueBindings: [ + { + queue: queueName, + exchange: exchangeName, + routingKey, + }, + ], + }; + } } diff --git a/water/src/broker/rpc/RpcClient.ts b/water/src/broker/rpc/RpcClient.ts index 5c27cfc..927b4f7 100644 --- a/water/src/broker/rpc/RpcClient.ts +++ b/water/src/broker/rpc/RpcClient.ts @@ -1,186 +1,185 @@ import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; -import { Logger } from "../../logging/Logger.js"; -import type { BrokerClient } from "../BrokerClient.js"; -import { IgnoreRpcRequest, RpcException, RpcRequestTimeout } from "../Exceptions.js"; -import type { BrokerClientOptions, ConsumerSubclient, ProducerSubclient } from "../Subclients.js"; -import { BaseSubclient } from "../Subclients.js"; import { OnlineEmitter } from "../../util/OnlineEmitter.js"; -import type { BrokerMessage } from "../BrokerMessage.js"; -import { CountDownLatch } from "../../util/CountDownLatch.js"; -import type { RpcRequestMessage, RpcResponse } from "./RpcMessage.js"; -import { RpcResponseMessage } from "./RpcMessage.js"; -import { RpcMessageHeaders } from "./RpcMessageHeaders.js"; -import type { RpcStatus } from "./RpcStatus.js"; +import type { + BrokerClient, + BrokerClientOptions, + BrokerMessage, + ConsumerSubclient, + ProducerSubclient, + RpcRequestMessage, + RpcResponse, + RpcStatus, +} from "@"; +import { + BaseSubclient, + CountDownLatch, + IgnoreRpcRequest, + Logger, + RpcException, + RpcMessageHeaders, + RpcRequestTimeout, + RpcResponseMessage, +} from "@"; export class RpcClient< - RequestTSchema extends BaseSchema>, - ResponseTSchema extends BaseSchema>, + RequestTSchema extends BaseSchema>, + ResponseTSchema extends BaseSchema>, > extends BaseSubclient { + private static readonly TAG = "RpcClient"; - private static readonly TAG = "RpcClient"; + private readonly requestProducer: ProducerSubclient>; + private readonly requestConsumer: ConsumerSubclient; + private readonly responseProducer: ProducerSubclient>; + private readonly responseConsumer: ConsumerSubclient; + private readonly responses: OnlineEmitter>> = new OnlineEmitter(); - private readonly requestProducer: ProducerSubclient>; - private readonly requestConsumer: ConsumerSubclient; - private readonly responseProducer: ProducerSubclient>; - private readonly responseConsumer: ConsumerSubclient; - private readonly responses: OnlineEmitter>> = new OnlineEmitter(); + public constructor( + client: BrokerClient, + topic: string, + key: string, + options: BrokerClientOptions, + requestSchema: RequestTSchema, + responseSchema: ResponseTSchema, + private readonly callback: ( + msg: RpcRequestMessage, InferOutput>, + ) => Promise>>, + ) { + super(client.connection, client, topic, key, options); - public constructor( - client: BrokerClient, - topic: string, - key: string, - options: BrokerClientOptions, - requestSchema: RequestTSchema, - responseSchema: ResponseTSchema, - private readonly callback: (msg: RpcRequestMessage< - InferOutput, - InferOutput - >) => Promise>, - ) { - super(client.connection, client, topic, key, options); + this.requestProducer = this.client.producer(topic, key, options); + this.requestConsumer = this.client.consumer( + topic, + key, + requestSchema, + async msg => { + const sendResponse = async ( + response: InferOutput | null, + status: RpcStatus, + isException: boolean, + isUpdate: boolean, + ) => { + const responseMsg = new RpcResponseMessage( + this.toResponseTopic(topic), + this.toResponseKey(key), + response, + new RpcMessageHeaders( + this.connection, + new Set([msg.headers.sourceService]), + new Set([msg.headers.sourceInstance]), + msg.headers.messageId, + status, + isException, + isUpdate, + ), + ); + await this.responseProducer.internalSend(responseMsg); + }; - this.requestProducer = this.client.producer( - topic, - key, - options, - ); - this.requestConsumer = this.client.consumer( - topic, - key, - requestSchema, - options, - async msg => { - const sendResponse = async ( - response: InferOutput | null, - status: RpcStatus, - isException: boolean, - isUpdate: boolean, - ) => { - const responseMsg = new RpcResponseMessage( - this.toResponseTopic(topic), - this.toResponseKey(key), - response, - new RpcMessageHeaders( - this.connection, - new Set([msg.headers.sourceService]), - new Set([msg.headers.sourceInstance]), - msg.headers.messageId, - status, - isException, - isUpdate, - ), - ); - await this.responseProducer.internalSend(responseMsg); - }; + const rpcMessage = msg.toRpcRequestMessage>(async (status, data) => { + await sendResponse(data, status, false, true); + }); + try { + const [status, response] = await this.callback(rpcMessage); + await sendResponse(response, status, false, false); + } catch (e) { + if (e instanceof IgnoreRpcRequest) { + return; + } else if (e instanceof RpcException) { + await sendResponse(null, e.status, true, false); + return; + } + Logger.error( + RpcClient.TAG, + "Uncaught RPC callback error while processing message " + + `${msg.headers.messageId} with key '$key' in topic '$topic'`, + e, + ); + } + }, + options, + ); + this.responseProducer = this.client.producer(this.toResponseTopic(topic), this.toResponseKey(key), options); + this.responseConsumer = this.client.consumer( + this.toResponseTopic(topic), + this.toResponseKey(key), + responseSchema, + async msg => { + this.responses.emit(msg); + }, + options, + ); + } - const rpcMessage = msg.toRpcRequestMessage>(async (status, data) => { - await sendResponse(data, status, false, true); - }); - try { - const [status, response] = await this.callback(rpcMessage); - await sendResponse(response, status, false, false); - } catch (e) { - if (e instanceof IgnoreRpcRequest) { - return; - } else if (e instanceof RpcException) { - await sendResponse(null, e.status, true, false); - return; - } - Logger.error(RpcClient.TAG, "Uncaught RPC callback error while processing message " - + `${msg.headers.messageId} with key '$key' in topic '$topic'`, e); - } - }, - ); - this.responseProducer = this.client.producer( - this.toResponseTopic(topic), - this.toResponseKey(key), - options, - ); - this.responseConsumer = this.client.consumer( - this.toResponseTopic(topic), - this.toResponseKey(key), - responseSchema, - options, - async msg => { - this.responses.emit(msg); - }, - ); - } + public async call( + request: InferOutput, + services: Set = new Set(), + instances: Set = new Set(), + timeout: number = 10 * 1000, + ): Promise>> { + const generator = this.stream(request, services, instances, timeout, 1); + const msg = (await generator.next()).value; + if (!msg) { + throw new Error("Unexpected end of single-response stream"); + } + return msg; + } - public async call( - request: InferOutput, - services: Set = new Set(), - instances: Set = new Set(), - timeout: number = 10 * 1000, - ): Promise>> { - const generator = this.stream(request, services, instances, timeout, 1); - const msg = (await generator.next()).value; - if (!msg) { - throw new Error("Unexpected end of single-response stream"); - } - return msg; - } + public async *stream( + request: InferOutput, + services: Set = new Set(), + instances: Set = new Set(), + timeout: number = 10 * 1000, + maxResponses: number = Infinity, + ): AsyncGenerator>, void> { + if (timeout === Infinity && maxResponses === Infinity) { + throw new Error("Must specify either a timeout or a max number of responses"); + } + if (maxResponses !== Infinity && maxResponses <= 0) { + throw new Error("maxResponses must be at least 1"); + } + let responseCounter = 0; + const timeoutLatch = maxResponses ? new CountDownLatch(maxResponses) : null; + const timeoutPromise = timeoutLatch?.await(timeout) ?? new Promise(() => {}); - public async *stream( - request: InferOutput, - services: Set = new Set(), - instances: Set = new Set(), - timeout: number = 10 * 1000, - maxResponses: number = Infinity, - ): AsyncGenerator>, void> { - if (timeout === Infinity && maxResponses === Infinity) { - throw new Error("Must specify either a timeout or a max number of responses"); - } - if (maxResponses !== Infinity && maxResponses <= 0) { - throw new Error("maxResponses must be at least 1"); - } - let responseCounter = 0; - const timeoutLatch = maxResponses ? new CountDownLatch(maxResponses) : null; - const timeoutPromise = timeoutLatch?.await(timeout) ?? new Promise(() => { }); + const messageId = await this.requestProducer.send(request, services, instances); - const messageId = await this.requestProducer.send(request, services, instances); + while (true) { + const result = await Promise.race([this.responses.awaitValue(), timeoutPromise]); + if (typeof result === "boolean") { + if (result) { + return; + } else { + throw new RpcRequestTimeout(`RPC request timed out after ${timeout} ms`); + } + } + const msg = result.toRpcResponseMessage(); + if (msg.headers.inReplyTo !== messageId) { + return; + } + if (msg.headers.isException) { + throw new RpcException(msg.headers.status); + } + yield msg; + timeoutLatch?.countDown(); + responseCounter++; + if (responseCounter >= maxResponses) { + return; + } + } + } - while (true) { - const result = await Promise.race([ - this.responses.awaitValue(), - timeoutPromise, - ]); - if (typeof result === "boolean") { - if (result) { - return; - } else { - throw new RpcRequestTimeout(`RPC request timed out after ${timeout} ms`); - } - } - const msg = result.toRpcResponseMessage(); - if (msg.headers.inReplyTo !== messageId) { - return; - } - if (msg.headers.isException) { - throw new RpcException(msg.headers.status); - } - yield msg; - timeoutLatch?.countDown(); - responseCounter++; - if (responseCounter >= maxResponses) { - return; - } - } - } + protected override doDestroy(): void { + this.requestProducer.destroy(); + this.requestConsumer.destroy(); + this.responseProducer.destroy(); + this.responseConsumer.destroy(); + } - protected override doDestroy(): void { - this.requestProducer.destroy(); - this.requestConsumer.destroy(); - this.responseProducer.destroy(); - this.responseConsumer.destroy(); - } - - private toResponseTopic(topic: string): string { - return this.connection.supportsTopicHotSwap ? `${topic}.responses` : topic; - } - - private toResponseKey(key: string): string { - return `${key}.response`; - } + private toResponseTopic(topic: string): string { + return this.connection.supportsTopicHotSwap ? `${topic}.responses` : topic; + } + private toResponseKey(key: string): string { + return `${key}.response`; + } } diff --git a/water/src/broker/rpc/RpcMessage.ts b/water/src/broker/rpc/RpcMessage.ts index bf9b300..69247a5 100644 --- a/water/src/broker/rpc/RpcMessage.ts +++ b/water/src/broker/rpc/RpcMessage.ts @@ -1,42 +1,34 @@ -import { BrokerMessage } from "../BrokerMessage.js"; -import type { BrokerMessageHeaders } from "../BrokerMessageHeaders.js"; -import type { RpcMessageHeaders } from "./RpcMessageHeaders.js"; -import type { RpcStatus } from "./RpcStatus.js"; +import type { BrokerMessageHeaders, RpcMessageHeaders, RpcStatus } from "@"; +import { BrokerMessage } from "@"; export type RpcResponse = [RpcStatus, ResponseT]; -export class RpcRequestMessage - extends BrokerMessage { - - public constructor( - topic: string, - key: string, - value: RequestT, - headers: H, - private readonly updateSender: (stats: RpcStatus, data: ResponseT) => Promise, - ) { - super(topic, key, value, headers); - } - - public async sendUpdate(status: RpcStatus, data: ResponseT): Promise { - await this.updateSender(status, data); - } - +export class RpcRequestMessage< + RequestT, + ResponseT, + H extends BrokerMessageHeaders = BrokerMessageHeaders, +> extends BrokerMessage { + public constructor( + topic: string, + key: string, + value: RequestT, + headers: H, + private readonly updateSender: (stats: RpcStatus, data: ResponseT) => Promise, + ) { + super(topic, key, value, headers); + } + + public async sendUpdate(status: RpcStatus, data: ResponseT): Promise { + await this.updateSender(status, data); + } } export class RpcResponseMessage extends BrokerMessage { + public constructor(topic: string, key: string, value: ResponseT, headers: RpcMessageHeaders) { + super(topic, key, value, headers); + } - public constructor( - topic: string, - key: string, - value: ResponseT, - headers: RpcMessageHeaders, - ) { - super(topic, key, value, headers); - } - - public get status(): RpcStatus { - return this.headers.status; - } - + public get status(): RpcStatus { + return this.headers.status; + } } diff --git a/water/src/broker/rpc/RpcMessageHeaders.ts b/water/src/broker/rpc/RpcMessageHeaders.ts index ba4fafd..28e394e 100644 --- a/water/src/broker/rpc/RpcMessageHeaders.ts +++ b/water/src/broker/rpc/RpcMessageHeaders.ts @@ -1,81 +1,76 @@ -import type { MessageId } from "../BrokerConnection.js"; -import { BrokerConnection } from "../BrokerConnection.js"; -import { BrokerMessageHeaders, getOrThrow } from "../BrokerMessageHeaders.js"; -import { RpcStatus } from "./RpcStatus.js"; +import { getOrThrow } from "../../util/internal.js"; +import type { MessageId } from "@"; +import { BrokerConnection, BrokerMessageHeaders, RpcStatus } from "@"; const RPC_HEADER_NAMES = { - IN_REPLY_TO: "rpc-in-reply-to", - STATUS: "rpc-response-status", - IS_EXCEPTION: "rpc-is-exception", - IS_UPDATE: "rpc-is-update", + IN_REPLY_TO: "rpc-in-reply-to", + STATUS: "rpc-response-status", + IS_EXCEPTION: "rpc-is-exception", + IS_UPDATE: "rpc-is-update", }; export class RpcMessageHeaders extends BrokerMessageHeaders { + public get inReplyTo(): MessageId { + return getOrThrow(this.headers, RPC_HEADER_NAMES.IN_REPLY_TO); + } + public get status(): RpcStatus { + return new RpcStatus(safeToInt(getOrThrow(this.headers, RPC_HEADER_NAMES.STATUS))); + } + public get isException(): boolean { + return this.headers[RPC_HEADER_NAMES.IS_EXCEPTION]?.toLowerCase() === "true"; + } + public get isUpdate(): boolean { + return this.headers[RPC_HEADER_NAMES.IS_UPDATE]?.toLowerCase() === "true"; + } - public get inReplyTo(): MessageId { - return getOrThrow(this.headers, RPC_HEADER_NAMES.IN_REPLY_TO); - } - public get status(): RpcStatus { - return new RpcStatus(safeToInt(getOrThrow(this.headers, RPC_HEADER_NAMES.STATUS))); - } - public get isException(): boolean { - return this.headers[RPC_HEADER_NAMES.IS_EXCEPTION]?.toLowerCase() === "true"; - } - public get isUpdate(): boolean { - return this.headers[RPC_HEADER_NAMES.IS_UPDATE]?.toLowerCase() === "true"; - } - - public constructor( - base: Record | BrokerMessageHeaders, - ); - public constructor( - connection: BrokerConnection, - targetServices: Set, - targetInstances: Set, - inReplyTo: MessageId, - status: RpcStatus, - isException: boolean, - isUpdate: boolean, - ); - public constructor( - baseOrConnection: Record | BrokerMessageHeaders | BrokerConnection, - targetServices?: Set, - targetInstances?: Set, - inReplyTo?: MessageId, - status?: RpcStatus, - isException?: boolean, - isUpdate?: boolean, - ) { - if (baseOrConnection instanceof BrokerMessageHeaders) { - super(baseOrConnection.headers); - } else if (baseOrConnection instanceof BrokerConnection) { - const connection = baseOrConnection; - super( - BrokerMessageHeaders.createHeadersMap( - connection.serviceName, - connection.instanceId, - targetServices!, - targetInstances!, - null, - { - [RPC_HEADER_NAMES.IN_REPLY_TO]: inReplyTo!, - [RPC_HEADER_NAMES.STATUS]: status!.code.toString(), - [RPC_HEADER_NAMES.IS_EXCEPTION]: isException!.toString(), - [RPC_HEADER_NAMES.IS_UPDATE]: isUpdate!.toString(), - }, - ), - ); - } else { - super(baseOrConnection); - } - } - + public constructor(base: Record | BrokerMessageHeaders); + public constructor( + connection: BrokerConnection, + targetServices: Set, + targetInstances: Set, + inReplyTo: MessageId, + status: RpcStatus, + isException: boolean, + isUpdate: boolean, + ); + public constructor( + baseOrConnection: Record | BrokerMessageHeaders | BrokerConnection, + targetServices?: Set, + targetInstances?: Set, + inReplyTo?: MessageId, + status?: RpcStatus, + isException?: boolean, + isUpdate?: boolean, + ) { + if (baseOrConnection instanceof BrokerMessageHeaders) { + super(baseOrConnection.headers); + } else if (baseOrConnection instanceof BrokerConnection) { + const connection = baseOrConnection; + super( + BrokerMessageHeaders.createHeadersMap( + connection.serviceName, + connection.instanceId, + targetServices!, + targetInstances!, + null, + { + [RPC_HEADER_NAMES.IN_REPLY_TO]: inReplyTo!, + [RPC_HEADER_NAMES.STATUS]: status!.code.toString(), + [RPC_HEADER_NAMES.IS_EXCEPTION]: isException!.toString(), + [RPC_HEADER_NAMES.IS_UPDATE]: isUpdate!.toString(), + }, + ), + ); + } else { + super(baseOrConnection); + } + } } function safeToInt(value: string): number { - const parsed = parseInt(value, 10); - if (isNaN(parsed)) { - throw new Error(`Failed to parse integer from value: ${value}`); - } - return parsed; + const parsed = parseInt(value, 10); + if (isNaN(parsed)) { + throw new Error(`Failed to parse integer from value: ${value}`); + } + return parsed; } diff --git a/water/src/broker/rpc/RpcStatus.ts b/water/src/broker/rpc/RpcStatus.ts index 19803a5..4be6163 100644 --- a/water/src/broker/rpc/RpcStatus.ts +++ b/water/src/broker/rpc/RpcStatus.ts @@ -1,16 +1,14 @@ export class RpcStatus { + public static readonly OK = new RpcStatus(0); + public static readonly UNKNOWN = new RpcStatus(999_999); - public static readonly OK = new RpcStatus(0); - public static readonly UNKNOWN = new RpcStatus(999_999); + public constructor(public readonly code: number) {} - public constructor(public readonly code: number) { } - - public equals(other: RpcStatus): boolean { - return this.code === other.code; - } - - public toString(): string { - return `RpcStatus(${this.code})`; - } + public equals(other: RpcStatus): boolean { + return this.code === other.code; + } + public toString(): string { + return `RpcStatus(${this.code})`; + } } diff --git a/water/src/util/CountDownLatch.ts b/water/src/util/CountDownLatch.ts index b3e9f9c..16bd1c8 100644 --- a/water/src/util/CountDownLatch.ts +++ b/water/src/util/CountDownLatch.ts @@ -1,57 +1,59 @@ type PromiseResolver = (timeout: boolean) => void; export class CountDownLatch { - private counter: number; - private resolvers = new Map(); - - public constructor(initialCount: number) { - if (!Number.isInteger(initialCount)) { - throw new Error("Initial count must be an integer"); - } - if (initialCount <= 0) { - throw new Error(`Initial count has to be greater than zero (is ${initialCount})`); - } - this.counter = initialCount; - } - - public countDown(): void { - if (this.counter <= 0) { - return; - } - this.counter--; - - if (this.counter === 0) { - for (const [resolver, timeout] of this.resolvers.entries()) { - if (timeout) { - clearTimeout(timeout); - } - resolver(true); - } - this.resolvers.clear(); - } - } - - public getCount(): number { - return this.counter; - } - - public await(waitTime: number | null = null): Promise { - if (this.counter === 0) { - return Promise.resolve(true); - } - - let resolver: PromiseResolver; - - const timeout = waitTime ? setTimeout(() => { - this.resolvers.delete(resolver); - resolver(false); - }, waitTime) : null; - timeout?.unref(); - - const promise = new Promise(r => { - resolver = r; - this.resolvers.set(r, timeout); - }); - return promise; - } + private counter: number; + private resolvers = new Map(); + + public constructor(initialCount: number) { + if (!Number.isInteger(initialCount)) { + throw new Error("Initial count must be an integer"); + } + if (initialCount <= 0) { + throw new Error(`Initial count has to be greater than zero (is ${initialCount})`); + } + this.counter = initialCount; + } + + public countDown(): void { + if (this.counter <= 0) { + return; + } + this.counter--; + + if (this.counter === 0) { + for (const [resolver, timeout] of this.resolvers.entries()) { + if (timeout) { + clearTimeout(timeout); + } + resolver(true); + } + this.resolvers.clear(); + } + } + + public getCount(): number { + return this.counter; + } + + public await(waitTime: number | null = null): Promise { + if (this.counter === 0) { + return Promise.resolve(true); + } + + let resolver: PromiseResolver; + + const timeout = waitTime + ? setTimeout(() => { + this.resolvers.delete(resolver); + resolver(false); + }, waitTime) + : null; + timeout?.unref(); + + const promise = new Promise(r => { + resolver = r; + this.resolvers.set(r, timeout); + }); + return promise; + } } diff --git a/water/src/util/OnlineEmitter.ts b/water/src/util/OnlineEmitter.ts index 6b24709..e85be89 100644 --- a/water/src/util/OnlineEmitter.ts +++ b/water/src/util/OnlineEmitter.ts @@ -1,43 +1,40 @@ export class OnlineEmitter { - - private readonly listeners: Set<(data: T) => void> = new Set(); - - public emit(data: T): void { - for (const listener of this.listeners) { - listener(data); - } - } - - public addListener(listener: (data: T) => void): void { - this.listeners.add(listener); - } - - - public removeListener(listener: (data: T) => void): void { - this.listeners.delete(listener); - } - - public async awaitValue(): Promise { - return new Promise(resolve => { - const listener = (data: T) => { - this.removeListener(listener); - resolve(data); - }; - this.addListener(listener); - }); - } - - public [Symbol.asyncIterator](): AsyncIterableIterator { - const awaitValue = this.awaitValue.bind(this); - return { - async next(): Promise> { - const value = await awaitValue(); - return { value, done: false }; - }, - [Symbol.asyncIterator](): AsyncIterableIterator { - return this; - }, - }; - } - + private readonly listeners: Set<(data: T) => void> = new Set(); + + public emit(data: T): void { + for (const listener of this.listeners) { + listener(data); + } + } + + public addListener(listener: (data: T) => void): void { + this.listeners.add(listener); + } + + public removeListener(listener: (data: T) => void): void { + this.listeners.delete(listener); + } + + public async awaitValue(): Promise { + return new Promise(resolve => { + const listener = (data: T) => { + this.removeListener(listener); + resolve(data); + }; + this.addListener(listener); + }); + } + + public [Symbol.asyncIterator](): AsyncIterableIterator { + const awaitValue = this.awaitValue.bind(this); + return { + async next(): Promise> { + const value = await awaitValue(); + return { value, done: false }; + }, + [Symbol.asyncIterator](): AsyncIterableIterator { + return this; + }, + }; + } } diff --git a/water/src/util/internal.ts b/water/src/util/internal.ts new file mode 100644 index 0000000..de0a26e --- /dev/null +++ b/water/src/util/internal.ts @@ -0,0 +1,7 @@ +export function getOrThrow(headers: Record, key: string): string { + const value = headers[key]; + if (value === undefined) { + throw new Error(`Missing broker message header '${key}'`); + } + return value; +} diff --git a/water/tsconfig.json b/water/tsconfig.json index 0e873c2..aeaea05 100644 --- a/water/tsconfig.json +++ b/water/tsconfig.json @@ -20,7 +20,11 @@ "sourceMap": true, "strict": true, "useDefineForClassFields": true, - "outDir": "dist" + "outDir": "dist", + "paths": { + "@": ["./src/index.js"], + "@/*": ["./src/*"] + } }, "include": ["src/**/*"] } From 00844ad8e37bc7feb88d40c5d9ee21b7f01f2ef8 Mon Sep 17 00:00:00 2001 From: Adrian Paschkowski Date: Wed, 17 Jul 2024 21:11:48 +0200 Subject: [PATCH 12/12] Add BrokerClient tests --- water/.eslintignore | 1 + water/package.json | 7 +- water/src/broker/BrokerClient.test.ts | 52 ++ water/src/broker/BrokerMessageHeaders.ts | 2 +- water/src/broker/LocalConnection.test.ts | 8 + water/src/broker/rpc/RpcClient.ts | 55 +- water/src/broker/rpc/RpcMessageHeaders.ts | 2 +- water/src/index.ts | 1 - water/src/util/internal/BufferedEmitter.ts | 43 ++ .../src/util/{ => internal}/OnlineEmitter.ts | 12 +- .../util/{internal.ts => internal/util.ts} | 0 water/vite.config.ts | 6 + water/yarn.lock | 628 +++++++++++++++++- 13 files changed, 786 insertions(+), 31 deletions(-) create mode 100644 water/src/broker/BrokerClient.test.ts create mode 100644 water/src/broker/LocalConnection.test.ts create mode 100644 water/src/util/internal/BufferedEmitter.ts rename water/src/util/{ => internal}/OnlineEmitter.ts (70%) rename water/src/util/{internal.ts => internal/util.ts} (100%) create mode 100644 water/vite.config.ts diff --git a/water/.eslintignore b/water/.eslintignore index 2c0af11..e7ff46e 100644 --- a/water/.eslintignore +++ b/water/.eslintignore @@ -1,3 +1,4 @@ node_modules dist .eslintrc.cjs +vite.config.ts diff --git a/water/package.json b/water/package.json index 68a9134..10c1c77 100644 --- a/water/package.json +++ b/water/package.json @@ -9,7 +9,8 @@ "scripts": { "build": "rm -rf dist/ && tsc", "format": "prettier --write .", - "lint": "eslint ." + "lint": "eslint .", + "test": "vitest" }, "files": [ "dist" @@ -25,6 +26,8 @@ "eslint": "^8.57.0", "eslint-plugin-import": "^2.29.1", "prettier": "^3.3.3", - "typescript": "^5.5.3" + "typescript": "^5.5.3", + "vite-tsconfig-paths": "^4.3.2", + "vitest": "^2.0.3" } } diff --git a/water/src/broker/BrokerClient.test.ts b/water/src/broker/BrokerClient.test.ts new file mode 100644 index 0000000..35fb7e6 --- /dev/null +++ b/water/src/broker/BrokerClient.test.ts @@ -0,0 +1,52 @@ +import { describe, expect, test } from "vitest"; +import { null_, object, string } from "valibot"; +import { LocalConnection } from "./LocalConnection.js"; +import { BrokerClient } from "./BrokerClient.js"; +import { Logger, RpcException, RpcStatus } from "@"; + +describe("BrokerClient", () => { + const client = new TestBrokerClient(new LocalConnection("service", "instance")); + test("greeting RPC", async () => { + const response = await client.greetingRpc.call({ name: "Beemo" }); + expect(response.value.greeting).toBe("Hello, Beemo"); + expect(response.status.code).toBe(RpcStatus.OK.code); + }); + test("null RPC", async () => { + const response = await client.nullRpc.call(null); + expect(response.value).toBe(null); + expect(response.status.code).toBe(1337); + }); + test("exception RPC", async () => { + // Ideally the exception status should be checked here, but vitest + // doesn't yet seem to have an API to inspect an error's properties. + await expect(client.exceptionRpc.call(null)).rejects.toThrowError(RpcException); + }); +}); + +const greetingRequest = object({ + name: string(), +}); + +const greetingResponse = object({ + greeting: string(), +}); + +class TestBrokerClient extends BrokerClient { + private static readonly TAG = "TestBrokerClient"; + + public readonly greetingRpc = this.rpc("rpc.greetings", "greet", greetingRequest, greetingResponse, async msg => { + Logger.info(TestBrokerClient.TAG, `greetingRpc received request: ${msg.value.name}`); + return [RpcStatus.OK, { greeting: `Hello, ${msg.value.name}` }]; + }); + + public readonly nullRpc = this.rpc("null", "null", null_(), null_(), async msg => { + Logger.info(TestBrokerClient.TAG, `nullRpc received request: ${msg.value}`); + expect(msg.value).toBe(null); + return [new RpcStatus(1337), null]; + }); + + public readonly exceptionRpc = this.rpc("exception", "exception", null_(), null_(), async msg => { + Logger.info(TestBrokerClient.TAG, `exceptionRpc received request: ${msg.value}`); + throw new RpcException(new RpcStatus(1337)); + }); +} diff --git a/water/src/broker/BrokerMessageHeaders.ts b/water/src/broker/BrokerMessageHeaders.ts index 7b9f1bf..669a4d1 100644 --- a/water/src/broker/BrokerMessageHeaders.ts +++ b/water/src/broker/BrokerMessageHeaders.ts @@ -1,5 +1,5 @@ import { randomUUID } from "node:crypto"; -import { getOrThrow } from "../util/internal.js"; +import { getOrThrow } from "../util/internal/util.js"; import type { MessageId } from "@"; import { BrokerConnection } from "@"; diff --git a/water/src/broker/LocalConnection.test.ts b/water/src/broker/LocalConnection.test.ts new file mode 100644 index 0000000..65029a4 --- /dev/null +++ b/water/src/broker/LocalConnection.test.ts @@ -0,0 +1,8 @@ +import { expect, test } from "vitest"; +import { LocalConnection } from "./LocalConnection.js"; + +test("LocalConnection", () => { + const connection = new LocalConnection("service-name", "instance-id"); + expect(connection.serviceName).toBe("service-name"); + expect(connection.instanceId).toBe("instance-id"); +}); diff --git a/water/src/broker/rpc/RpcClient.ts b/water/src/broker/rpc/RpcClient.ts index 927b4f7..2c4d309 100644 --- a/water/src/broker/rpc/RpcClient.ts +++ b/water/src/broker/rpc/RpcClient.ts @@ -1,5 +1,6 @@ import type { BaseIssue, BaseSchema, InferOutput } from "valibot"; -import { OnlineEmitter } from "../../util/OnlineEmitter.js"; +import { OnlineEmitter } from "../../util/internal/OnlineEmitter.js"; +import { BufferedEmitter } from "../../util/internal/BufferedEmitter.js"; import type { BrokerClient, BrokerClientOptions, @@ -141,30 +142,42 @@ export class RpcClient< const timeoutLatch = maxResponses ? new CountDownLatch(maxResponses) : null; const timeoutPromise = timeoutLatch?.await(timeout) ?? new Promise(() => {}); - const messageId = await this.requestProducer.send(request, services, instances); + // Start to listen for responses before sending the request to avoid missing any, + // especially with local short-circuit which will run synchronoously to the request. + const responses = new BufferedEmitter>>(); + const responseListener = (msg: BrokerMessage>) => { + responses.emit(msg); + }; - while (true) { - const result = await Promise.race([this.responses.awaitValue(), timeoutPromise]); - if (typeof result === "boolean") { - if (result) { + try { + this.responses.addListener(responseListener); + const messageId = await this.requestProducer.send(request, services, instances); + + while (true) { + const result = await Promise.race([responses.next(), timeoutPromise]); + if (typeof result === "boolean") { + if (result) { + return; + } else { + throw new RpcRequestTimeout(`RPC request timed out after ${timeout} ms`); + } + } + const msg = result.toRpcResponseMessage(); + if (msg.headers.inReplyTo !== messageId) { + return; + } + if (msg.headers.isException) { + throw new RpcException(msg.headers.status); + } + yield msg; + timeoutLatch?.countDown(); + responseCounter++; + if (responseCounter >= maxResponses) { return; - } else { - throw new RpcRequestTimeout(`RPC request timed out after ${timeout} ms`); } } - const msg = result.toRpcResponseMessage(); - if (msg.headers.inReplyTo !== messageId) { - return; - } - if (msg.headers.isException) { - throw new RpcException(msg.headers.status); - } - yield msg; - timeoutLatch?.countDown(); - responseCounter++; - if (responseCounter >= maxResponses) { - return; - } + } finally { + this.responses.removeListener(responseListener); } } diff --git a/water/src/broker/rpc/RpcMessageHeaders.ts b/water/src/broker/rpc/RpcMessageHeaders.ts index 28e394e..91495ab 100644 --- a/water/src/broker/rpc/RpcMessageHeaders.ts +++ b/water/src/broker/rpc/RpcMessageHeaders.ts @@ -1,4 +1,4 @@ -import { getOrThrow } from "../../util/internal.js"; +import { getOrThrow } from "../../util/internal/util.js"; import type { MessageId } from "@"; import { BrokerConnection, BrokerMessageHeaders, RpcStatus } from "@"; diff --git a/water/src/index.ts b/water/src/index.ts index f2b410f..5684a60 100644 --- a/water/src/index.ts +++ b/water/src/index.ts @@ -16,4 +16,3 @@ export * from "./broker/rabbitmq/RabbitConnection.js"; export * from "./logging/Logger.js"; export * from "./util/CountDownLatch.js"; -export * from "./util/OnlineEmitter.js"; diff --git a/water/src/util/internal/BufferedEmitter.ts b/water/src/util/internal/BufferedEmitter.ts new file mode 100644 index 0000000..eb1ce93 --- /dev/null +++ b/water/src/util/internal/BufferedEmitter.ts @@ -0,0 +1,43 @@ +import { OnlineEmitter } from "./OnlineEmitter.js"; + +/** + * A class that allows for emitting events to multiple listeners. + * Events emitted when no listeners are registered are buffered until a listener is registered, + * at which point the buffered events are emitted in order. + */ +export class BufferedEmitter extends OnlineEmitter { + private readonly buffer: T[] = []; + + /** + * Emit data to listeners if any are registered, otherwise buffer the data. + * + * @param data The data to emit. + */ + public override emit(data: T): void { + if (this.listeners.size > 0) { + super.emit(data); + } else { + this.buffer.push(data); + } + } + + public override addListener(listener: (data: T) => void): void { + super.addListener(listener); + for (const data of this.buffer) { + listener(data); + } + this.buffer.length = 0; + } + + /** + * Returns the next value from the buffer, if any, or waits for the next emitted value. + * + * @returns A promise that resolves to the next emitted value. + */ + public override async next(): Promise { + if (this.buffer.length > 0) { + return this.buffer.shift()!; + } + return super.next(); + } +} diff --git a/water/src/util/OnlineEmitter.ts b/water/src/util/internal/OnlineEmitter.ts similarity index 70% rename from water/src/util/OnlineEmitter.ts rename to water/src/util/internal/OnlineEmitter.ts index e85be89..c4ecc1a 100644 --- a/water/src/util/OnlineEmitter.ts +++ b/water/src/util/internal/OnlineEmitter.ts @@ -1,5 +1,9 @@ +/** + * A class that allows for emitting events to multiple listeners. + * Events emitted when no listeners are registered are lost. + */ export class OnlineEmitter { - private readonly listeners: Set<(data: T) => void> = new Set(); + protected readonly listeners: Set<(data: T) => void> = new Set(); public emit(data: T): void { for (const listener of this.listeners) { @@ -15,7 +19,7 @@ export class OnlineEmitter { this.listeners.delete(listener); } - public async awaitValue(): Promise { + public async next(): Promise { return new Promise(resolve => { const listener = (data: T) => { this.removeListener(listener); @@ -26,10 +30,10 @@ export class OnlineEmitter { } public [Symbol.asyncIterator](): AsyncIterableIterator { - const awaitValue = this.awaitValue.bind(this); + const next = this.next.bind(this); return { async next(): Promise> { - const value = await awaitValue(); + const value = await next(); return { value, done: false }; }, [Symbol.asyncIterator](): AsyncIterableIterator { diff --git a/water/src/util/internal.ts b/water/src/util/internal/util.ts similarity index 100% rename from water/src/util/internal.ts rename to water/src/util/internal/util.ts diff --git a/water/vite.config.ts b/water/vite.config.ts new file mode 100644 index 0000000..d9bd11f --- /dev/null +++ b/water/vite.config.ts @@ -0,0 +1,6 @@ +import { defineConfig } from "vite"; +import tsconfigPaths from "vite-tsconfig-paths"; + +export default defineConfig({ + plugins: [tsconfigPaths()], +}); diff --git a/water/yarn.lock b/water/yarn.lock index 7c23e06..5d3c9db 100644 --- a/water/yarn.lock +++ b/water/yarn.lock @@ -7,6 +7,129 @@ resolved "https://registry.yarnpkg.com/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz#bd9154aec9983f77b3a034ecaa015c2e4201f6cf" integrity sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA== +"@ampproject/remapping@^2.3.0": + version "2.3.0" + resolved "https://registry.yarnpkg.com/@ampproject/remapping/-/remapping-2.3.0.tgz#ed441b6fa600072520ce18b43d2c8cc8caecc7f4" + integrity sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw== + dependencies: + "@jridgewell/gen-mapping" "^0.3.5" + "@jridgewell/trace-mapping" "^0.3.24" + +"@esbuild/aix-ppc64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz#c7184a326533fcdf1b8ee0733e21c713b975575f" + integrity sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ== + +"@esbuild/android-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz#09d9b4357780da9ea3a7dfb833a1f1ff439b4052" + integrity sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A== + +"@esbuild/android-arm@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.21.5.tgz#9b04384fb771926dfa6d7ad04324ecb2ab9b2e28" + integrity sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg== + +"@esbuild/android-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.21.5.tgz#29918ec2db754cedcb6c1b04de8cd6547af6461e" + integrity sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA== + +"@esbuild/darwin-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz#e495b539660e51690f3928af50a76fb0a6ccff2a" + integrity sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ== + +"@esbuild/darwin-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz#c13838fa57372839abdddc91d71542ceea2e1e22" + integrity sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw== + +"@esbuild/freebsd-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz#646b989aa20bf89fd071dd5dbfad69a3542e550e" + integrity sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g== + +"@esbuild/freebsd-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz#aa615cfc80af954d3458906e38ca22c18cf5c261" + integrity sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ== + +"@esbuild/linux-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz#70ac6fa14f5cb7e1f7f887bcffb680ad09922b5b" + integrity sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q== + +"@esbuild/linux-arm@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz#fc6fd11a8aca56c1f6f3894f2bea0479f8f626b9" + integrity sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA== + +"@esbuild/linux-ia32@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz#3271f53b3f93e3d093d518d1649d6d68d346ede2" + integrity sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg== + +"@esbuild/linux-loong64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz#ed62e04238c57026aea831c5a130b73c0f9f26df" + integrity sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg== + +"@esbuild/linux-mips64el@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz#e79b8eb48bf3b106fadec1ac8240fb97b4e64cbe" + integrity sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg== + +"@esbuild/linux-ppc64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz#5f2203860a143b9919d383ef7573521fb154c3e4" + integrity sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w== + +"@esbuild/linux-riscv64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz#07bcafd99322d5af62f618cb9e6a9b7f4bb825dc" + integrity sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA== + +"@esbuild/linux-s390x@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz#b7ccf686751d6a3e44b8627ababc8be3ef62d8de" + integrity sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A== + +"@esbuild/linux-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz#6d8f0c768e070e64309af8004bb94e68ab2bb3b0" + integrity sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ== + +"@esbuild/netbsd-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz#bbe430f60d378ecb88decb219c602667387a6047" + integrity sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg== + +"@esbuild/openbsd-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz#99d1cf2937279560d2104821f5ccce220cb2af70" + integrity sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow== + +"@esbuild/sunos-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz#08741512c10d529566baba837b4fe052c8f3487b" + integrity sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg== + +"@esbuild/win32-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz#675b7385398411240735016144ab2e99a60fc75d" + integrity sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A== + +"@esbuild/win32-ia32@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz#1bfc3ce98aa6ca9a0969e4d2af72144c59c1193b" + integrity sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA== + +"@esbuild/win32-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz#acad351d582d157bb145535db2a6ff53dd514b5c" + integrity sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw== + "@eslint-community/eslint-utils@^4.2.0", "@eslint-community/eslint-utils@^4.4.0": version "4.4.0" resolved "https://registry.yarnpkg.com/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz#a23514e8fb9af1269d5f7788aa556798d61c6b59" @@ -63,6 +186,38 @@ resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-2.0.2.tgz#d9fae00a2d5cb40f92cfe64b47ad749fbc38f917" integrity sha512-6EwiSjwWYP7pTckG6I5eyFANjPhmPjUX9JRLUSfNPC7FX7zK9gyZAfUEaECL6ALTpGX5AjnBq3C9XmVWPitNpw== +"@jridgewell/gen-mapping@^0.3.5": + version "0.3.5" + resolved "https://registry.yarnpkg.com/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz#dcce6aff74bdf6dad1a95802b69b04a2fcb1fb36" + integrity sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg== + dependencies: + "@jridgewell/set-array" "^1.2.1" + "@jridgewell/sourcemap-codec" "^1.4.10" + "@jridgewell/trace-mapping" "^0.3.24" + +"@jridgewell/resolve-uri@^3.1.0": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz#7a0ee601f60f99a20c7c7c5ff0c80388c1189bd6" + integrity sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw== + +"@jridgewell/set-array@^1.2.1": + version "1.2.1" + resolved "https://registry.yarnpkg.com/@jridgewell/set-array/-/set-array-1.2.1.tgz#558fb6472ed16a4c850b889530e6b36438c49280" + integrity sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A== + +"@jridgewell/sourcemap-codec@^1.4.10", "@jridgewell/sourcemap-codec@^1.4.14", "@jridgewell/sourcemap-codec@^1.4.15": + version "1.5.0" + resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz#3188bcb273a414b0d215fd22a58540b989b9409a" + integrity sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ== + +"@jridgewell/trace-mapping@^0.3.24": + version "0.3.25" + resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz#15f190e98895f3fc23276ee14bc76b675c2e50f0" + integrity sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ== + dependencies: + "@jridgewell/resolve-uri" "^3.1.0" + "@jridgewell/sourcemap-codec" "^1.4.14" + "@nodelib/fs.scandir@2.1.5": version "2.1.5" resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" @@ -84,6 +239,91 @@ "@nodelib/fs.scandir" "2.1.5" fastq "^1.6.0" +"@rollup/rollup-android-arm-eabi@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.18.1.tgz#f0da481244b7d9ea15296b35f7fe39cd81157396" + integrity sha512-lncuC4aHicncmbORnx+dUaAgzee9cm/PbIqgWz1PpXuwc+sa1Ct83tnqUDy/GFKleLiN7ZIeytM6KJ4cAn1SxA== + +"@rollup/rollup-android-arm64@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.18.1.tgz#82ab3c575f4235fb647abea5e08eec6cf325964e" + integrity sha512-F/tkdw0WSs4ojqz5Ovrw5r9odqzFjb5LIgHdHZG65dFI1lWTWRVy32KDJLKRISHgJvqUeUhdIvy43fX41znyDg== + +"@rollup/rollup-darwin-arm64@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.18.1.tgz#6a530452e68a9152809ce58de1f89597632a085b" + integrity sha512-vk+ma8iC1ebje/ahpxpnrfVQJibTMyHdWpOGZ3JpQ7Mgn/3QNHmPq7YwjZbIE7km73dH5M1e6MRRsnEBW7v5CQ== + +"@rollup/rollup-darwin-x64@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.18.1.tgz#47727479f5ca292cf434d7e75af2725b724ecbc7" + integrity sha512-IgpzXKauRe1Tafcej9STjSSuG0Ghu/xGYH+qG6JwsAUxXrnkvNHcq/NL6nz1+jzvWAnQkuAJ4uIwGB48K9OCGA== + +"@rollup/rollup-linux-arm-gnueabihf@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.18.1.tgz#46193c498aa7902a8db89ac00128060320e84fef" + integrity sha512-P9bSiAUnSSM7EmyRK+e5wgpqai86QOSv8BwvkGjLwYuOpaeomiZWifEos517CwbG+aZl1T4clSE1YqqH2JRs+g== + +"@rollup/rollup-linux-arm-musleabihf@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.18.1.tgz#22d831fe239643c1d05c98906420325cee439d85" + integrity sha512-5RnjpACoxtS+aWOI1dURKno11d7krfpGDEn19jI8BuWmSBbUC4ytIADfROM1FZrFhQPSoP+KEa3NlEScznBTyQ== + +"@rollup/rollup-linux-arm64-gnu@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.18.1.tgz#19abd33695ec9d588b4a858d122631433084e4a3" + integrity sha512-8mwmGD668m8WaGbthrEYZ9CBmPug2QPGWxhJxh/vCgBjro5o96gL04WLlg5BA233OCWLqERy4YUzX3bJGXaJgQ== + +"@rollup/rollup-linux-arm64-musl@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.18.1.tgz#d60af8c0b9be424424ff96a0ba19fce65d26f6ab" + integrity sha512-dJX9u4r4bqInMGOAQoGYdwDP8lQiisWb9et+T84l2WXk41yEej8v2iGKodmdKimT8cTAYt0jFb+UEBxnPkbXEQ== + +"@rollup/rollup-linux-powerpc64le-gnu@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.18.1.tgz#b1194e5ed6d138fdde0842d126fccde74a90f457" + integrity sha512-V72cXdTl4EI0x6FNmho4D502sy7ed+LuVW6Ym8aI6DRQ9hQZdp5sj0a2usYOlqvFBNKQnLQGwmYnujo2HvjCxQ== + +"@rollup/rollup-linux-riscv64-gnu@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.18.1.tgz#f5a635c017b9bff8b856b0221fbd5c0e3373b7ec" + integrity sha512-f+pJih7sxoKmbjghrM2RkWo2WHUW8UbfxIQiWo5yeCaCM0TveMEuAzKJte4QskBp1TIinpnRcxkquY+4WuY/tg== + +"@rollup/rollup-linux-s390x-gnu@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.18.1.tgz#f1043d9f4026bf6995863cb3f8dd4732606e4baa" + integrity sha512-qb1hMMT3Fr/Qz1OKovCuUM11MUNLUuHeBC2DPPAWUYYUAOFWaxInaTwTQmc7Fl5La7DShTEpmYwgdt2hG+4TEg== + +"@rollup/rollup-linux-x64-gnu@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.18.1.tgz#1e781730be445119f06c9df5f185e193bc82c610" + integrity sha512-7O5u/p6oKUFYjRbZkL2FLbwsyoJAjyeXHCU3O4ndvzg2OFO2GinFPSJFGbiwFDaCFc+k7gs9CF243PwdPQFh5g== + +"@rollup/rollup-linux-x64-musl@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.18.1.tgz#08f12e1965d6f27d6898ff932592121cca6abc4b" + integrity sha512-pDLkYITdYrH/9Cv/Vlj8HppDuLMDUBmgsM0+N+xLtFd18aXgM9Nyqupb/Uw+HeidhfYg2lD6CXvz6CjoVOaKjQ== + +"@rollup/rollup-win32-arm64-msvc@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.18.1.tgz#4a5dcbbe7af7d41cac92b09798e7c1831da1f599" + integrity sha512-W2ZNI323O/8pJdBGil1oCauuCzmVd9lDmWBBqxYZcOqWD6aWqJtVBQ1dFrF4dYpZPks6F+xCZHfzG5hYlSHZ6g== + +"@rollup/rollup-win32-ia32-msvc@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.18.1.tgz#075b0713de627843a73b4cf0e087c56b53e9d780" + integrity sha512-ELfEX1/+eGZYMaCIbK4jqLxO1gyTSOIlZr6pbC4SRYFaSIDVKOnZNMdoZ+ON0mrFDp4+H5MhwNC1H/AhE3zQLg== + +"@rollup/rollup-win32-x64-msvc@4.18.1": + version "4.18.1" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.18.1.tgz#0cb240c147c0dfd0e3eaff4cc060a772d39e155c" + integrity sha512-yjk2MAkQmoaPYCSu35RLJ62+dz358nE83VfTePJRp8CG7aMg25mEJYpXFiD+NcevhX8LxD5OP5tktPXnXN7GDw== + +"@types/estree@1.0.5", "@types/estree@^1.0.0": + version "1.0.5" + resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.5.tgz#a6ce3e556e00fd9895dd872dd172ad0d4bd687f4" + integrity sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw== + "@types/json5@^0.0.29": version "0.0.29" resolved "https://registry.yarnpkg.com/@types/json5/-/json5-0.0.29.tgz#ee28707ae94e11d2b827bcbe5270bcea7f3e71ee" @@ -182,6 +422,57 @@ resolved "https://registry.yarnpkg.com/@ungap/structured-clone/-/structured-clone-1.2.0.tgz#756641adb587851b5ccb3e095daf27ae581c8406" integrity sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ== +"@vitest/expect@2.0.3": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@vitest/expect/-/expect-2.0.3.tgz#367727256f2a253e21a3e69cd996af51fc7899b1" + integrity sha512-X6AepoOYePM0lDNUPsGXTxgXZAl3EXd0GYe/MZyVE4HzkUqyUVC6S3PrY5mClDJ6/7/7vALLMV3+xD/Ko60Hqg== + dependencies: + "@vitest/spy" "2.0.3" + "@vitest/utils" "2.0.3" + chai "^5.1.1" + tinyrainbow "^1.2.0" + +"@vitest/pretty-format@2.0.3", "@vitest/pretty-format@^2.0.3": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@vitest/pretty-format/-/pretty-format-2.0.3.tgz#30af705250cd055890091999e467968e41872c82" + integrity sha512-URM4GLsB2xD37nnTyvf6kfObFafxmycCL8un3OC9gaCs5cti2u+5rJdIflZ2fUJUen4NbvF6jCufwViAFLvz1g== + dependencies: + tinyrainbow "^1.2.0" + +"@vitest/runner@2.0.3": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@vitest/runner/-/runner-2.0.3.tgz#4310ff4583d7874f57b5a8a194062bb85f07b0df" + integrity sha512-EmSP4mcjYhAcuBWwqgpjR3FYVeiA4ROzRunqKltWjBfLNs1tnMLtF+qtgd5ClTwkDP6/DGlKJTNa6WxNK0bNYQ== + dependencies: + "@vitest/utils" "2.0.3" + pathe "^1.1.2" + +"@vitest/snapshot@2.0.3": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@vitest/snapshot/-/snapshot-2.0.3.tgz#31acf5906f8c12f9c7fde21b84cc28f043e983b1" + integrity sha512-6OyA6v65Oe3tTzoSuRPcU6kh9m+mPL1vQ2jDlPdn9IQoUxl8rXhBnfICNOC+vwxWY684Vt5UPgtcA2aPFBb6wg== + dependencies: + "@vitest/pretty-format" "2.0.3" + magic-string "^0.30.10" + pathe "^1.1.2" + +"@vitest/spy@2.0.3": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@vitest/spy/-/spy-2.0.3.tgz#62a14f6d7ec4f13caeeecac42d37f903f68c83c1" + integrity sha512-sfqyAw/ypOXlaj4S+w8689qKM1OyPOqnonqOc9T91DsoHbfN5mU7FdifWWv3MtQFf0lEUstEwR9L/q/M390C+A== + dependencies: + tinyspy "^3.0.0" + +"@vitest/utils@2.0.3": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@vitest/utils/-/utils-2.0.3.tgz#3c57f5338e49c91e3c4ac5be8c74ae22a3c2d5b4" + integrity sha512-c/UdELMuHitQbbc/EVctlBaxoYAwQPQdSNwv7z/vHyBKy2edYZaFgptE27BRueZB7eW8po+cllotMNTDpL3HWg== + dependencies: + "@vitest/pretty-format" "2.0.3" + estree-walker "^3.0.3" + loupe "^3.1.1" + tinyrainbow "^1.2.0" + acorn-jsx@^5.3.2: version "5.3.2" resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937" @@ -299,6 +590,11 @@ arraybuffer.prototype.slice@^1.0.3: is-array-buffer "^3.0.4" is-shared-array-buffer "^1.0.2" +assertion-error@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/assertion-error/-/assertion-error-2.0.1.tgz#f641a196b335690b1070bf00b6e7593fec190bf7" + integrity sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA== + available-typed-arrays@^1.0.6: version "1.0.6" resolved "https://registry.yarnpkg.com/available-typed-arrays/-/available-typed-arrays-1.0.6.tgz#ac812d8ce5a6b976d738e1c45f08d0b00bc7d725" @@ -331,6 +627,11 @@ braces@^3.0.2: dependencies: fill-range "^7.0.1" +cac@^6.7.14: + version "6.7.14" + resolved "https://registry.yarnpkg.com/cac/-/cac-6.7.14.tgz#804e1e6f506ee363cb0e3ccbb09cad5dd9870959" + integrity sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ== + call-bind@^1.0.2, call-bind@^1.0.5, call-bind@^1.0.6, call-bind@^1.0.7: version "1.0.7" resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.7.tgz#06016599c40c56498c18769d2730be242b6fa3b9" @@ -347,6 +648,17 @@ callsites@^3.0.0: resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73" integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ== +chai@^5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/chai/-/chai-5.1.1.tgz#f035d9792a22b481ead1c65908d14bb62ec1c82c" + integrity sha512-pT1ZgP8rPNqUgieVaEY+ryQr6Q4HXNg8Ei9UnLUrjN4IA7dvQC5JB+/kxVcPNDHyBcc/26CXPkbNzq3qwrOEKA== + dependencies: + assertion-error "^2.0.1" + check-error "^2.1.1" + deep-eql "^5.0.1" + loupe "^3.1.0" + pathval "^2.0.0" + chalk@^4.0.0: version "4.1.2" resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" @@ -355,6 +667,11 @@ chalk@^4.0.0: ansi-styles "^4.1.0" supports-color "^7.1.0" +check-error@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/check-error/-/check-error-2.1.1.tgz#87eb876ae71ee388fa0471fe423f494be1d96ccc" + integrity sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw== + color-convert@^2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" @@ -372,7 +689,7 @@ concat-map@0.0.1: resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg== -cross-spawn@^7.0.2: +cross-spawn@^7.0.2, cross-spawn@^7.0.3: version "7.0.3" resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w== @@ -388,6 +705,13 @@ debug@^3.2.7: dependencies: ms "^2.1.1" +debug@^4.1.1, debug@^4.3.5: + version "4.3.5" + resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.5.tgz#e83444eceb9fedd4a1da56d671ae2446a01a6e1e" + integrity sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg== + dependencies: + ms "2.1.2" + debug@^4.3.1, debug@^4.3.2, debug@^4.3.4: version "4.3.4" resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" @@ -395,6 +719,11 @@ debug@^4.3.1, debug@^4.3.2, debug@^4.3.4: dependencies: ms "2.1.2" +deep-eql@^5.0.1: + version "5.0.2" + resolved "https://registry.yarnpkg.com/deep-eql/-/deep-eql-5.0.2.tgz#4b756d8d770a9257300825d52a2c2cff99c3a341" + integrity sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q== + deep-is@^0.1.3: version "0.1.4" resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831" @@ -528,6 +857,35 @@ es-to-primitive@^1.2.1: is-date-object "^1.0.1" is-symbol "^1.0.2" +esbuild@^0.21.3: + version "0.21.5" + resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.21.5.tgz#9ca301b120922959b766360d8ac830da0d02997d" + integrity sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw== + optionalDependencies: + "@esbuild/aix-ppc64" "0.21.5" + "@esbuild/android-arm" "0.21.5" + "@esbuild/android-arm64" "0.21.5" + "@esbuild/android-x64" "0.21.5" + "@esbuild/darwin-arm64" "0.21.5" + "@esbuild/darwin-x64" "0.21.5" + "@esbuild/freebsd-arm64" "0.21.5" + "@esbuild/freebsd-x64" "0.21.5" + "@esbuild/linux-arm" "0.21.5" + "@esbuild/linux-arm64" "0.21.5" + "@esbuild/linux-ia32" "0.21.5" + "@esbuild/linux-loong64" "0.21.5" + "@esbuild/linux-mips64el" "0.21.5" + "@esbuild/linux-ppc64" "0.21.5" + "@esbuild/linux-riscv64" "0.21.5" + "@esbuild/linux-s390x" "0.21.5" + "@esbuild/linux-x64" "0.21.5" + "@esbuild/netbsd-x64" "0.21.5" + "@esbuild/openbsd-x64" "0.21.5" + "@esbuild/sunos-x64" "0.21.5" + "@esbuild/win32-arm64" "0.21.5" + "@esbuild/win32-ia32" "0.21.5" + "@esbuild/win32-x64" "0.21.5" + escape-string-regexp@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" @@ -657,11 +1015,33 @@ estraverse@^5.1.0, estraverse@^5.2.0: resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-5.3.0.tgz#2eea5290702f26ab8fe5370370ff86c965d21123" integrity sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA== +estree-walker@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/estree-walker/-/estree-walker-3.0.3.tgz#67c3e549ec402a487b4fc193d1953a524752340d" + integrity sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g== + dependencies: + "@types/estree" "^1.0.0" + esutils@^2.0.2: version "2.0.3" resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64" integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== +execa@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/execa/-/execa-8.0.1.tgz#51f6a5943b580f963c3ca9c6321796db8cc39b8c" + integrity sha512-VyhnebXciFV2DESc+p6B+y0LjSm0krU4OgJN44qFAhBY0TJ+1V61tYD2+wHusZ6F9n5K+vl8k0sTy7PEfV4qpg== + dependencies: + cross-spawn "^7.0.3" + get-stream "^8.0.1" + human-signals "^5.0.0" + is-stream "^3.0.0" + merge-stream "^2.0.0" + npm-run-path "^5.1.0" + onetime "^6.0.0" + signal-exit "^4.1.0" + strip-final-newline "^3.0.0" + fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3: version "3.1.3" resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" @@ -743,6 +1123,11 @@ fs.realpath@^1.0.0: resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw== +fsevents@~2.3.2, fsevents@~2.3.3: + version "2.3.3" + resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" + integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw== + function-bind@^1.1.2: version "1.1.2" resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.2.tgz#2c02d864d97f3ea6c8830c464cbd11ab6eab7a1c" @@ -763,6 +1148,11 @@ functions-have-names@^1.2.3: resolved "https://registry.yarnpkg.com/functions-have-names/-/functions-have-names-1.2.3.tgz#0404fe4ee2ba2f607f0e0ec3c80bae994133b834" integrity sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ== +get-func-name@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/get-func-name/-/get-func-name-2.0.2.tgz#0d7cf20cd13fda808669ffa88f4ffc7a3943fc41" + integrity sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ== + get-intrinsic@^1.1.3, get-intrinsic@^1.2.1, get-intrinsic@^1.2.2, get-intrinsic@^1.2.3, get-intrinsic@^1.2.4: version "1.2.4" resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.2.4.tgz#e385f5a4b5227d449c3eabbad05494ef0abbeadd" @@ -774,6 +1164,11 @@ get-intrinsic@^1.1.3, get-intrinsic@^1.2.1, get-intrinsic@^1.2.2, get-intrinsic@ has-symbols "^1.0.3" hasown "^2.0.0" +get-stream@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-8.0.1.tgz#def9dfd71742cd7754a7761ed43749a27d02eca2" + integrity sha512-VaUJspBffn/LMCJVoMvSAdmscJyS1auj5Zulnn5UoYcY531UWmdwhRWkcGKnGU93m5HSXP9LP2usOryrBtQowA== + get-symbol-description@^1.0.2: version "1.0.2" resolved "https://registry.yarnpkg.com/get-symbol-description/-/get-symbol-description-1.0.2.tgz#533744d5aa20aca4e079c8e5daf7fd44202821f5" @@ -835,6 +1230,11 @@ globby@^11.1.0: merge2 "^1.4.1" slash "^3.0.0" +globrex@^0.1.2: + version "0.1.2" + resolved "https://registry.yarnpkg.com/globrex/-/globrex-0.1.2.tgz#dd5d9ec826232730cd6793a5e33a9302985e6098" + integrity sha512-uHJgbwAMwNFf5mLst7IWLNg14x1CkeqglJb/K3doi4dw6q2IvAAmM/Y81kevy83wP+Sst+nutFTYOGg3d1lsxg== + gopd@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.0.1.tgz#29ff76de69dac7489b7c0918a5788e56477c332c" @@ -888,6 +1288,11 @@ hasown@^2.0.0, hasown@^2.0.1: dependencies: function-bind "^1.1.2" +human-signals@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-5.0.0.tgz#42665a284f9ae0dade3ba41ebc37eb4b852f3a28" + integrity sha512-AXcZb6vzzrFAUE61HnN4mpLqd/cSIwNQjtNWR0euPm6y0iqx3G4gOXaIDdtdDwZmhwe82LA6+zinmW4UBWVePQ== + ignore@^5.2.0, ignore@^5.3.1: version "5.3.1" resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.3.1.tgz#5073e554cd42c5b33b394375f538b8593e34d4ef" @@ -1019,6 +1424,11 @@ is-shared-array-buffer@^1.0.2: dependencies: call-bind "^1.0.2" +is-stream@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-3.0.0.tgz#e6bfd7aa6bef69f4f472ce9bb681e3e57b4319ac" + integrity sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA== + is-string@^1.0.5, is-string@^1.0.7: version "1.0.7" resolved "https://registry.yarnpkg.com/is-string/-/is-string-1.0.7.tgz#0dd12bf2006f255bb58f695110eff7491eebc0fd" @@ -1113,6 +1523,25 @@ lodash.merge@^4.6.2: resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a" integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ== +loupe@^3.1.0, loupe@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/loupe/-/loupe-3.1.1.tgz#71d038d59007d890e3247c5db97c1ec5a92edc54" + integrity sha512-edNu/8D5MKVfGVFRhFf8aAxiTM6Wumfz5XsaatSxlD3w4R1d/WEKUTydCdPGbl9K7QG/Ca3GnDV2sIKIpXRQcw== + dependencies: + get-func-name "^2.0.1" + +magic-string@^0.30.10: + version "0.30.10" + resolved "https://registry.yarnpkg.com/magic-string/-/magic-string-0.30.10.tgz#123d9c41a0cb5640c892b041d4cfb3bd0aa4b39e" + integrity sha512-iIRwTIf0QKV3UAnYK4PU8uiEc4SRh5jX0mwpIwETPpHdhVM4f53RSwS/vXvN1JhGX+Cs7B8qIq3d6AH49O5fAQ== + dependencies: + "@jridgewell/sourcemap-codec" "^1.4.15" + +merge-stream@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" + integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== + merge2@^1.3.0, merge2@^1.4.1: version "1.4.1" resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.4.1.tgz#4368892f885e907455a6fd7dc55c0c9d404990ae" @@ -1126,6 +1555,11 @@ micromatch@^4.0.4: braces "^3.0.2" picomatch "^2.3.1" +mimic-fn@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-4.0.0.tgz#60a90550d5cb0b239cca65d893b1a53b29871ecc" + integrity sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw== + minimatch@^3.0.5, minimatch@^3.1.1, minimatch@^3.1.2: version "3.1.2" resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" @@ -1155,11 +1589,23 @@ ms@^2.1.1: resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== +nanoid@^3.3.7: + version "3.3.7" + resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.7.tgz#d0c301a691bc8d54efa0a2226ccf3fe2fd656bd8" + integrity sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g== + natural-compare@^1.4.0: version "1.4.0" resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== +npm-run-path@^5.1.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-5.3.0.tgz#e23353d0ebb9317f174e93417e4a4d82d0249e9f" + integrity sha512-ppwTtiJZq0O/ai0z7yfudtBpWIoxM8yE6nHi1X47eFR2EWORqfbu6CnPlNsjeN683eT0qG6H/Pyf9fCcvjnnnQ== + dependencies: + path-key "^4.0.0" + object-inspect@^1.13.1: version "1.13.1" resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.13.1.tgz#b96c6109324ccfef6b12216a956ca4dc2ff94bc2" @@ -1216,6 +1662,13 @@ once@^1.3.0: dependencies: wrappy "1" +onetime@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-6.0.0.tgz#7c24c18ed1fd2e9bca4bd26806a33613c77d34b4" + integrity sha512-1FlR+gjXK7X+AsAHso35MnyN5KqGwJRi/31ft6x0M194ht7S+rWAvd7PHss9xSKMzE0asv1pyIHaJYq+BbacAQ== + dependencies: + mimic-fn "^4.0.0" + optionator@^0.9.3: version "0.9.3" resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.9.3.tgz#007397d44ed1872fdc6ed31360190f81814e2c64" @@ -1264,6 +1717,11 @@ path-key@^3.1.0: resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375" integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q== +path-key@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-4.0.0.tgz#295588dc3aee64154f877adb9d780b81c554bf18" + integrity sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ== + path-parse@^1.0.7: version "1.0.7" resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.7.tgz#fbc114b60ca42b30d9daf5858e4bd68bbedb6735" @@ -1274,11 +1732,35 @@ path-type@^4.0.0: resolved "https://registry.yarnpkg.com/path-type/-/path-type-4.0.0.tgz#84ed01c0a7ba380afe09d90a8c180dcd9d03043b" integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw== +pathe@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/pathe/-/pathe-1.1.2.tgz#6c4cb47a945692e48a1ddd6e4094d170516437ec" + integrity sha512-whLdWMYL2TwI08hn8/ZqAbrVemu0LNaNNJZX73O6qaIdCTfXutsLhMkjdENX0qhsQ9uIimo4/aQOmXkoon2nDQ== + +pathval@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/pathval/-/pathval-2.0.0.tgz#7e2550b422601d4f6b8e26f1301bc8f15a741a25" + integrity sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA== + +picocolors@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.1.tgz#a8ad579b571952f0e5d25892de5445bcfe25aaa1" + integrity sha512-anP1Z8qwhkbmu7MFP5iTt+wQKXgwzf7zTyGlcdzabySa9vd0Xt392U0rVmz9poOaBj0uHJKyyo9/upk0HrEQew== + picomatch@^2.3.1: version "2.3.1" resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== +postcss@^8.4.39: + version "8.4.39" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.39.tgz#aa3c94998b61d3a9c259efa51db4b392e1bde0e3" + integrity sha512-0vzE+lAiG7hZl1/9I8yzKLx3aR9Xbof3fBHKunvMfOCYAtMhrsnccJY2iTURb9EZd5+pLuiNV9/c/GZJOHsgIw== + dependencies: + nanoid "^3.3.7" + picocolors "^1.0.1" + source-map-js "^1.2.0" + prelude-ls@^1.2.1: version "1.2.1" resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396" @@ -1340,6 +1822,31 @@ rimraf@^3.0.2: dependencies: glob "^7.1.3" +rollup@^4.13.0: + version "4.18.1" + resolved "https://registry.yarnpkg.com/rollup/-/rollup-4.18.1.tgz#18a606df5e76ca53b8a69f2d8eab256d69dda851" + integrity sha512-Elx2UT8lzxxOXMpy5HWQGZqkrQOtrVDDa/bm9l10+U4rQnVzbL/LgZ4NOM1MPIDyHk69W4InuYDF5dzRh4Kw1A== + dependencies: + "@types/estree" "1.0.5" + optionalDependencies: + "@rollup/rollup-android-arm-eabi" "4.18.1" + "@rollup/rollup-android-arm64" "4.18.1" + "@rollup/rollup-darwin-arm64" "4.18.1" + "@rollup/rollup-darwin-x64" "4.18.1" + "@rollup/rollup-linux-arm-gnueabihf" "4.18.1" + "@rollup/rollup-linux-arm-musleabihf" "4.18.1" + "@rollup/rollup-linux-arm64-gnu" "4.18.1" + "@rollup/rollup-linux-arm64-musl" "4.18.1" + "@rollup/rollup-linux-powerpc64le-gnu" "4.18.1" + "@rollup/rollup-linux-riscv64-gnu" "4.18.1" + "@rollup/rollup-linux-s390x-gnu" "4.18.1" + "@rollup/rollup-linux-x64-gnu" "4.18.1" + "@rollup/rollup-linux-x64-musl" "4.18.1" + "@rollup/rollup-win32-arm64-msvc" "4.18.1" + "@rollup/rollup-win32-ia32-msvc" "4.18.1" + "@rollup/rollup-win32-x64-msvc" "4.18.1" + fsevents "~2.3.2" + run-parallel@^1.1.9: version "1.2.0" resolved "https://registry.yarnpkg.com/run-parallel/-/run-parallel-1.2.0.tgz#66d1368da7bdf921eb9d95bd1a9229e7f21a43ee" @@ -1419,11 +1926,36 @@ side-channel@^1.0.4: get-intrinsic "^1.2.4" object-inspect "^1.13.1" +siginfo@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/siginfo/-/siginfo-2.0.0.tgz#32e76c70b79724e3bb567cb9d543eb858ccfaf30" + integrity sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g== + +signal-exit@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-4.1.0.tgz#952188c1cbd546070e2dd20d0f41c0ae0530cb04" + integrity sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw== + slash@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634" integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== +source-map-js@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.2.0.tgz#16b809c162517b5b8c3e7dcd315a2a5c2612b2af" + integrity sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg== + +stackback@0.0.2: + version "0.0.2" + resolved "https://registry.yarnpkg.com/stackback/-/stackback-0.0.2.tgz#1ac8a0d9483848d1695e418b6d031a3c3ce68e3b" + integrity sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw== + +std-env@^3.7.0: + version "3.7.0" + resolved "https://registry.yarnpkg.com/std-env/-/std-env-3.7.0.tgz#c9f7386ced6ecf13360b6c6c55b8aaa4ef7481d2" + integrity sha512-JPbdCEQLj1w5GilpiHAx3qJvFndqybBysA3qUOnznweH4QbNYUsW/ea8QzSrnh0vNsezMMw5bcVool8lM0gwzg== + string.prototype.trim@^1.2.8: version "1.2.8" resolved "https://registry.yarnpkg.com/string.prototype.trim/-/string.prototype.trim-1.2.8.tgz#f9ac6f8af4bd55ddfa8895e6aea92a96395393bd" @@ -1463,6 +1995,11 @@ strip-bom@^3.0.0: resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3" integrity sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA== +strip-final-newline@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-final-newline/-/strip-final-newline-3.0.0.tgz#52894c313fbff318835280aed60ff71ebf12b8fd" + integrity sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw== + strip-json-comments@^3.1.1: version "3.1.1" resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006" @@ -1485,6 +2022,26 @@ text-table@^0.2.0: resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw== +tinybench@^2.8.0: + version "2.8.0" + resolved "https://registry.yarnpkg.com/tinybench/-/tinybench-2.8.0.tgz#30e19ae3a27508ee18273ffed9ac7018949acd7b" + integrity sha512-1/eK7zUnIklz4JUUlL+658n58XO2hHLQfSk1Zf2LKieUjxidN16eKFEoDEfjHc3ohofSSqK3X5yO6VGb6iW8Lw== + +tinypool@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/tinypool/-/tinypool-1.0.0.tgz#a68965218e04f4ad9de037d2a1cd63cda9afb238" + integrity sha512-KIKExllK7jp3uvrNtvRBYBWBOAXSX8ZvoaD8T+7KB/QHIuoJW3Pmr60zucywjAlMb5TeXUkcs/MWeWLu0qvuAQ== + +tinyrainbow@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/tinyrainbow/-/tinyrainbow-1.2.0.tgz#5c57d2fc0fb3d1afd78465c33ca885d04f02abb5" + integrity sha512-weEDEq7Z5eTHPDh4xjX789+fHfF+P8boiFB+0vbWzpbnbsEr/GRaohi/uMKxg8RZMXnl1ItAi/IUHWMsjDV7kQ== + +tinyspy@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/tinyspy/-/tinyspy-3.0.0.tgz#cb61644f2713cd84dee184863f4642e06ddf0585" + integrity sha512-q5nmENpTHgiPVd1cJDDc9cVoYN5x4vCvwT3FMilvKPKneCBZAxn2YWQjDF0UMcE9k0Cay1gBiDfTMU0g+mPMQA== + to-regex-range@^5.0.1: version "5.0.1" resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" @@ -1497,6 +2054,11 @@ ts-api-utils@^1.3.0: resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-1.3.0.tgz#4b490e27129f1e8e686b45cc4ab63714dc60eea1" integrity sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ== +tsconfck@^3.0.3: + version "3.1.1" + resolved "https://registry.yarnpkg.com/tsconfck/-/tsconfck-3.1.1.tgz#c7284913262c293b43b905b8b034f524de4a3162" + integrity sha512-00eoI6WY57SvZEVjm13stEVE90VkEdJAFGgpFLTsZbJyW/LwFQ7uQxJHWpZ2hzSWgCPKc9AnBnNP+0X7o3hAmQ== + tsconfig-paths@^3.15.0: version "3.15.0" resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz#5299ec605e55b1abb23ec939ef15edaf483070d4" @@ -1591,6 +2153,62 @@ valibot@^0.36.0: resolved "https://registry.yarnpkg.com/valibot/-/valibot-0.36.0.tgz#74e746694b1abcc1879e4393db551d4ce1e10578" integrity sha512-CjF1XN4sUce8sBK9TixrDqFM7RwNkuXdJu174/AwmQUB62QbCQADg5lLe8ldBalFgtj1uKj+pKwDJiNo4Mn+eQ== +vite-node@2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/vite-node/-/vite-node-2.0.3.tgz#449b1524178304ba764bd33062bd31a09c5e673f" + integrity sha512-14jzwMx7XTcMB+9BhGQyoEAmSl0eOr3nrnn+Z12WNERtOvLN+d2scbRUvyni05rT3997Bg+rZb47NyP4IQPKXg== + dependencies: + cac "^6.7.14" + debug "^4.3.5" + pathe "^1.1.2" + tinyrainbow "^1.2.0" + vite "^5.0.0" + +vite-tsconfig-paths@^4.3.2: + version "4.3.2" + resolved "https://registry.yarnpkg.com/vite-tsconfig-paths/-/vite-tsconfig-paths-4.3.2.tgz#321f02e4b736a90ff62f9086467faf4e2da857a9" + integrity sha512-0Vd/a6po6Q+86rPlntHye7F31zA2URZMbH8M3saAZ/xR9QoGN/L21bxEGfXdWmFdNkqPpRdxFT7nmNe12e9/uA== + dependencies: + debug "^4.1.1" + globrex "^0.1.2" + tsconfck "^3.0.3" + +vite@^5.0.0: + version "5.3.4" + resolved "https://registry.yarnpkg.com/vite/-/vite-5.3.4.tgz#b36ebd47c8a5e3a8727046375d5f10bf9fdf8715" + integrity sha512-Cw+7zL3ZG9/NZBB8C+8QbQZmR54GwqIz+WMI4b3JgdYJvX+ny9AjJXqkGQlDXSXRP9rP0B4tbciRMOVEKulVOA== + dependencies: + esbuild "^0.21.3" + postcss "^8.4.39" + rollup "^4.13.0" + optionalDependencies: + fsevents "~2.3.3" + +vitest@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/vitest/-/vitest-2.0.3.tgz#daf7e43c9415c6825922ae3a63cac452d1ac705f" + integrity sha512-o3HRvU93q6qZK4rI2JrhKyZMMuxg/JRt30E6qeQs6ueaiz5hr1cPj+Sk2kATgQzMMqsa2DiNI0TIK++1ULx8Jw== + dependencies: + "@ampproject/remapping" "^2.3.0" + "@vitest/expect" "2.0.3" + "@vitest/pretty-format" "^2.0.3" + "@vitest/runner" "2.0.3" + "@vitest/snapshot" "2.0.3" + "@vitest/spy" "2.0.3" + "@vitest/utils" "2.0.3" + chai "^5.1.1" + debug "^4.3.5" + execa "^8.0.1" + magic-string "^0.30.10" + pathe "^1.1.2" + std-env "^3.7.0" + tinybench "^2.8.0" + tinypool "^1.0.0" + tinyrainbow "^1.2.0" + vite "^5.0.0" + vite-node "2.0.3" + why-is-node-running "^2.2.2" + which-boxed-primitive@^1.0.2: version "1.0.2" resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz#13757bc89b209b049fe5d86430e21cf40a89a8e6" @@ -1620,6 +2238,14 @@ which@^2.0.1: dependencies: isexe "^2.0.0" +why-is-node-running@^2.2.2: + version "2.3.0" + resolved "https://registry.yarnpkg.com/why-is-node-running/-/why-is-node-running-2.3.0.tgz#a3f69a97107f494b3cdc3bdddd883a7d65cebf04" + integrity sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w== + dependencies: + siginfo "^2.0.0" + stackback "0.0.2" + wrappy@1: version "1.0.2" resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f"