diff --git a/.eslintrc.json b/.eslintrc.json index 532fe2a..92b12d9 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -25,7 +25,8 @@ "error", { "argsIgnorePattern": "^_", - "varsIgnorePattern": "^_" + "varsIgnorePattern": "^_", + "caughtErrorsIgnorePattern": "^_" } ], "@typescript-eslint/no-non-null-assertion": "warn", @@ -64,13 +65,15 @@ "error", { "argsIgnorePattern": "^_", - "varsIgnorePattern": "^_" + "varsIgnorePattern": "^_", + "caughtErrorsIgnorePattern": "^_" } ], "@typescript-eslint/no-unsafe-assignment": "off", "@typescript-eslint/no-unsafe-call": "off", "@typescript-eslint/no-unsafe-member-access": "off", - "@typescript-eslint/no-redundant-type-constituents": "off" + "@typescript-eslint/no-redundant-type-constituents": "off", + "@typescript-eslint/no-require-imports": "off" } }, { @@ -90,14 +93,16 @@ "error", { "argsIgnorePattern": "^_", - "varsIgnorePattern": "^_" + "varsIgnorePattern": "^_", + "caughtErrorsIgnorePattern": "^_" } ], "@typescript-eslint/no-unsafe-assignment": "off", "@typescript-eslint/no-unsafe-call": "off", "@typescript-eslint/no-unsafe-member-access": "off", "@typescript-eslint/no-unsafe-return": "off", - "@typescript-eslint/no-unsafe-argument": "off" + "@typescript-eslint/no-unsafe-argument": "off", + "@typescript-eslint/no-require-imports": "off" } }, { diff --git a/package-lock.json b/package-lock.json index d1d6ff1..e814953 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "@willwade/aac-processors", - "version": "0.0.0", + "version": "0.0.0-development", "lockfileVersion": 3, "requires": true, "packages": { @@ -13,7 +13,7 @@ "adm-zip": "^0.5.16", "axios": "^1.11.0", "better-sqlite3": "^12.5.0", - "commander": "^13.1.0", + "commander": "^14.0.3", "exceljs": "^4.4.0", "fast-xml-parser": "^5.2.0", "jszip": "^3.10.1", @@ -29,20 +29,21 @@ "@types/adm-zip": "^0.5.7", "@types/better-sqlite3": "^7.6.13", "@types/exceljs": "^0.5.3", - "@types/jest": "^29.5.12", - "@types/node": "^20.11.24", + "@types/jest": "^30.0.0", + "@types/node": "^22.19.17", + "@types/pako": "^2.0.4", "@types/plist": "^3.0.5", "@types/sql.js": "^1.4.9", - "@typescript-eslint/eslint-plugin": "^7.1.0", - "@typescript-eslint/parser": "^7.1.0", - "eslint": "^8.56.0", - "eslint-config-prettier": "^9.1.0", + "@typescript-eslint/eslint-plugin": "^8.58.2", + "@typescript-eslint/parser": "^8.58.2", + "eslint": "^8.57.1", + "eslint-config-prettier": "^10.1.8", "eslint-plugin-prettier": "^5.1.3", "fast-check": "^4.2.0", - "jest": "^29.7.0", + "jest": "^30.3.0", "prettier": "^3.5.3", - "rimraf": "^5.0.5", - "ts-jest": "^29.1.2", + "rimraf": "^6.1.3", + "ts-jest": "^29.4.9", "typedoc": "^0.28.14", "typescript": "~5.5.0" }, @@ -51,28 +52,14 @@ "npm": ">=9.0.0" } }, - "node_modules/@ampproject/remapping": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", - "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@jridgewell/gen-mapping": "^0.3.5", - "@jridgewell/trace-mapping": "^0.3.24" - }, - "engines": { - "node": ">=6.0.0" - } - }, "node_modules/@babel/code-frame": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.27.1.tgz", - "integrity": "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg==", + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz", + "integrity": "sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==", "dev": true, "license": "MIT", "dependencies": { - "@babel/helper-validator-identifier": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5", "js-tokens": "^4.0.0", "picocolors": "^1.1.1" }, @@ -81,9 +68,9 @@ } }, "node_modules/@babel/compat-data": { - "version": "7.28.0", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.0.tgz", - "integrity": "sha512-60X7qkglvrap8mn1lh2ebxXdZYtUcpd7gsmy9kLaBJ4i/WdY8PqTSdxyA8qraikqKQK5C1KRBKXqznrVapyNaw==", + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.0.tgz", + "integrity": "sha512-T1NCJqT/j9+cn8fvkt7jtwbLBfLC/1y1c7NtCeXFRgzGTsafi68MRv8yzkYSapBnFA6L3U2VSc02ciDzoAJhJg==", "dev": true, "license": "MIT", "engines": { @@ -91,22 +78,22 @@ } }, "node_modules/@babel/core": { - "version": "7.28.0", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.0.tgz", - "integrity": "sha512-UlLAnTPrFdNGoFtbSXwcGFQBtQZJCNjaN6hQNP3UPvuNXT1i82N26KL3dZeIpNalWywr9IuQuncaAfUaS1g6sQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@ampproject/remapping": "^2.2.0", - "@babel/code-frame": "^7.27.1", - "@babel/generator": "^7.28.0", - "@babel/helper-compilation-targets": "^7.27.2", - "@babel/helper-module-transforms": "^7.27.3", - "@babel/helpers": "^7.27.6", - "@babel/parser": "^7.28.0", - "@babel/template": "^7.27.2", - "@babel/traverse": "^7.28.0", - "@babel/types": "^7.28.0", + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.0.tgz", + "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-module-transforms": "^7.28.6", + "@babel/helpers": "^7.28.6", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/traverse": "^7.29.0", + "@babel/types": "^7.29.0", + "@jridgewell/remapping": "^2.3.5", "convert-source-map": "^2.0.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", @@ -132,14 +119,14 @@ } }, "node_modules/@babel/generator": { - "version": "7.28.0", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.0.tgz", - "integrity": "sha512-lJjzvrbEeWrhB4P3QBsH7tey117PjLZnDbLiQEKjQ/fNJTjuq4HSqgFA+UNSwZT8D7dxxbnuSBMsa1lrWzKlQg==", + "version": "7.29.1", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz", + "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==", "dev": true, "license": "MIT", "dependencies": { - "@babel/parser": "^7.28.0", - "@babel/types": "^7.28.0", + "@babel/parser": "^7.29.0", + "@babel/types": "^7.29.0", "@jridgewell/gen-mapping": "^0.3.12", "@jridgewell/trace-mapping": "^0.3.28", "jsesc": "^3.0.2" @@ -149,13 +136,13 @@ } }, "node_modules/@babel/helper-compilation-targets": { - "version": "7.27.2", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.2.tgz", - "integrity": "sha512-2+1thGUUWWjLTYTHZWK1n8Yga0ijBz1XAhUXcKy81rd5g6yh7hGqMp45v7cadSbEHc9G3OTv45SyneRN3ps4DQ==", + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz", + "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==", "dev": true, "license": "MIT", "dependencies": { - "@babel/compat-data": "^7.27.2", + "@babel/compat-data": "^7.28.6", "@babel/helper-validator-option": "^7.27.1", "browserslist": "^4.24.0", "lru-cache": "^5.1.1", @@ -186,29 +173,29 @@ } }, "node_modules/@babel/helper-module-imports": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.27.1.tgz", - "integrity": "sha512-0gSFWUPNXNopqtIPQvlD5WgXYI5GY2kP2cCvoT8kczjbfcfuIljTbcWrulD1CIPIX2gt1wghbDy08yE1p+/r3w==", + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz", + "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==", "dev": true, "license": "MIT", "dependencies": { - "@babel/traverse": "^7.27.1", - "@babel/types": "^7.27.1" + "@babel/traverse": "^7.28.6", + "@babel/types": "^7.28.6" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/helper-module-transforms": { - "version": "7.27.3", - "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.27.3.tgz", - "integrity": "sha512-dSOvYwvyLsWBeIRyOeHXp5vPj5l1I011r52FM1+r1jCERv+aFXYk4whgQccYEGYxK2H3ZAIA8nuPkQ0HaUo3qg==", + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz", + "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==", "dev": true, "license": "MIT", "dependencies": { - "@babel/helper-module-imports": "^7.27.1", - "@babel/helper-validator-identifier": "^7.27.1", - "@babel/traverse": "^7.27.3" + "@babel/helper-module-imports": "^7.28.6", + "@babel/helper-validator-identifier": "^7.28.5", + "@babel/traverse": "^7.28.6" }, "engines": { "node": ">=6.9.0" @@ -218,9 +205,9 @@ } }, "node_modules/@babel/helper-plugin-utils": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.27.1.tgz", - "integrity": "sha512-1gn1Up5YXka3YYAHGKpbideQ5Yjf1tDa9qYcgysz+cNCXukyLl6DjPXhD3VRwSb8c0J9tA4b2+rHEZtc6R0tlw==", + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.28.6.tgz", + "integrity": "sha512-S9gzZ/bz83GRysI7gAD4wPT/AI3uCnY+9xn+Mx/KPs2JwHJIz1W8PZkg2cqyt3RNOBM8ejcXhV6y8Og7ly/Dug==", "dev": true, "license": "MIT", "engines": { @@ -238,9 +225,9 @@ } }, "node_modules/@babel/helper-validator-identifier": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.27.1.tgz", - "integrity": "sha512-D2hP9eA+Sqx1kBZgzxZh0y1trbuU+JoDkiEwqhQ36nodYqJwyEIhPSdMNd7lOm/4io72luTPWH20Yda0xOuUow==", + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", "dev": true, "license": "MIT", "engines": { @@ -258,27 +245,27 @@ } }, "node_modules/@babel/helpers": { - "version": "7.28.2", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.2.tgz", - "integrity": "sha512-/V9771t+EgXz62aCcyofnQhGM8DQACbRhvzKFsXKC9QM+5MadF8ZmIm0crDMaz3+o0h0zXfJnd4EhbYbxsrcFw==", + "version": "7.29.2", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.29.2.tgz", + "integrity": "sha512-HoGuUs4sCZNezVEKdVcwqmZN8GoHirLUcLaYVNBK2J0DadGtdcqgr3BCbvH8+XUo4NGjNl3VOtSjEKNzqfFgKw==", "dev": true, "license": "MIT", "dependencies": { - "@babel/template": "^7.27.2", - "@babel/types": "^7.28.2" + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/parser": { - "version": "7.28.0", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.0.tgz", - "integrity": "sha512-jVZGvOxOuNSsuQuLRTh13nU0AogFlw32w/MT+LV6D3sP5WdbW61E77RnkbaO2dUvmPAYrBDJXGn5gGS6tH4j8g==", + "version": "7.29.2", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.2.tgz", + "integrity": "sha512-4GgRzy/+fsBa72/RZVJmGKPmZu9Byn8o4MoLpmNe1m8ZfYnz5emHLQz3U4gLud6Zwl0RZIcgiLD7Uq7ySFuDLA==", "dev": true, "license": "MIT", "dependencies": { - "@babel/types": "^7.28.0" + "@babel/types": "^7.29.0" }, "bin": { "parser": "bin/babel-parser.js" @@ -343,13 +330,13 @@ } }, "node_modules/@babel/plugin-syntax-import-attributes": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.27.1.tgz", - "integrity": "sha512-oFT0FrKHgF53f4vOsZGi2Hh3I35PfSmVs4IBFLFj4dnafP+hIWDLg3VyKmUHfLoLHlyxY4C7DGtmHuJgn+IGww==", + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.28.6.tgz", + "integrity": "sha512-jiLC0ma9XkQT3TKJ9uYvlakm66Pamywo+qwL+oL8HJOvc6TWdZXVfhqJr8CCzbSGUAbDOzlGHJC1U+vRfLQDvw==", "dev": true, "license": "MIT", "dependencies": { - "@babel/helper-plugin-utils": "^7.27.1" + "@babel/helper-plugin-utils": "^7.28.6" }, "engines": { "node": ">=6.9.0" @@ -385,13 +372,13 @@ } }, "node_modules/@babel/plugin-syntax-jsx": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.27.1.tgz", - "integrity": "sha512-y8YTNIeKoyhGd9O0Jiyzyyqk8gdjnumGTQPsz0xOZOQ2RmkVJeZ1vmmfIvFEKqucBG6axJGBZDE/7iI5suUI/w==", + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.28.6.tgz", + "integrity": "sha512-wgEmr06G6sIpqr8YDwA2dSRTE3bJ+V0IfpzfSY3Lfgd7YWOaAdlykvJi13ZKBt8cZHfgH1IXN+CL656W3uUa4w==", "dev": true, "license": "MIT", "dependencies": { - "@babel/helper-plugin-utils": "^7.27.1" + "@babel/helper-plugin-utils": "^7.28.6" }, "engines": { "node": ">=6.9.0" @@ -511,13 +498,13 @@ } }, "node_modules/@babel/plugin-syntax-typescript": { - "version": "7.27.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.27.1.tgz", - "integrity": "sha512-xfYCBMxveHrRMnAWl1ZlPXOZjzkN82THFvLhQhFXFt81Z5HnN+EtUkZhv/zcKpmT3fzmWZB0ywiBrbC3vogbwQ==", + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.28.6.tgz", + "integrity": "sha512-+nDNmQye7nlnuuHDboPbGm00Vqg3oO8niRRL27/4LYHUsHYh0zJ1xWOz0uRwNFmM1Avzk8wZbc6rdiYhomzv/A==", "dev": true, "license": "MIT", "dependencies": { - "@babel/helper-plugin-utils": "^7.27.1" + "@babel/helper-plugin-utils": "^7.28.6" }, "engines": { "node": ">=6.9.0" @@ -527,33 +514,33 @@ } }, "node_modules/@babel/template": { - "version": "7.27.2", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.2.tgz", - "integrity": "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==", + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz", + "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==", "dev": true, "license": "MIT", "dependencies": { - "@babel/code-frame": "^7.27.1", - "@babel/parser": "^7.27.2", - "@babel/types": "^7.27.1" + "@babel/code-frame": "^7.28.6", + "@babel/parser": "^7.28.6", + "@babel/types": "^7.28.6" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/traverse": { - "version": "7.28.0", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.0.tgz", - "integrity": "sha512-mGe7UK5wWyh0bKRfupsUchrQGqvDbZDbKJw+kcRGSmdHVYrv+ltd0pnpDTVpiTqnaBru9iEvA8pz8W46v0Amwg==", + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz", + "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==", "dev": true, "license": "MIT", "dependencies": { - "@babel/code-frame": "^7.27.1", - "@babel/generator": "^7.28.0", + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", "@babel/helper-globals": "^7.28.0", - "@babel/parser": "^7.28.0", - "@babel/template": "^7.27.2", - "@babel/types": "^7.28.0", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0", "debug": "^4.3.1" }, "engines": { @@ -561,14 +548,14 @@ } }, "node_modules/@babel/types": { - "version": "7.28.2", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.2.tgz", - "integrity": "sha512-ruv7Ae4J5dUYULmeXw1gmb7rYRz57OWCPM57pHojnLq/3Z1CK2lNSLTCVjxVk1F/TZHwOZZrOWi0ur95BbLxNQ==", + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz", + "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==", "dev": true, "license": "MIT", "dependencies": { "@babel/helper-string-parser": "^7.27.1", - "@babel/helper-validator-identifier": "^7.27.1" + "@babel/helper-validator-identifier": "^7.28.5" }, "engines": { "node": ">=6.9.0" @@ -581,10 +568,44 @@ "dev": true, "license": "MIT" }, + "node_modules/@emnapi/core": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/core/-/core-1.10.0.tgz", + "integrity": "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/wasi-threads": "1.2.1", + "tslib": "^2.4.0" + } + }, + "node_modules/@emnapi/runtime": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.10.0.tgz", + "integrity": "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@emnapi/wasi-threads": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz", + "integrity": "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" + } + }, "node_modules/@eslint-community/eslint-utils": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.7.0.tgz", - "integrity": "sha512-dyybb3AcajC7uha6CvhdVRJqaKyn7w2YKqKyAN37NKYgZT36w+iRb0Dymmc5qEJ549c/S31cMMSFd75bteCpCw==", + "version": "4.9.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz", + "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==", "dev": true, "license": "MIT", "dependencies": { @@ -601,9 +622,9 @@ } }, "node_modules/@eslint-community/regexpp": { - "version": "4.12.1", - "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz", - "integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==", + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", "dev": true, "license": "MIT", "engines": { @@ -634,10 +655,17 @@ "url": "https://opencollective.com/eslint" } }, + "node_modules/@eslint/eslintrc/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, "node_modules/@eslint/eslintrc/node_modules/brace-expansion": { - "version": "1.1.12", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", - "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.14.tgz", + "integrity": "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g==", "dev": true, "license": "MIT", "dependencies": { @@ -645,10 +673,20 @@ "concat-map": "0.0.1" } }, + "node_modules/@eslint/eslintrc/node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, "node_modules/@eslint/eslintrc/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz", + "integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==", "dev": true, "license": "ISC", "dependencies": { @@ -710,16 +748,16 @@ "license": "MIT" }, "node_modules/@gerrit0/mini-shiki": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/@gerrit0/mini-shiki/-/mini-shiki-3.15.0.tgz", - "integrity": "sha512-L5IHdZIDa4bG4yJaOzfasOH/o22MCesY0mx+n6VATbaiCtMeR59pdRqYk4bEiQkIHfxsHPNgdi7VJlZb2FhdMQ==", + "version": "3.23.0", + "resolved": "https://registry.npmjs.org/@gerrit0/mini-shiki/-/mini-shiki-3.23.0.tgz", + "integrity": "sha512-bEMORlG0cqdjVyCEuU0cDQbORWX+kYCeo0kV1lbxF5bt4r7SID2l9bqsxJEM0zndaxpOUT7riCyIVEuqq/Ynxg==", "dev": true, "license": "MIT", "dependencies": { - "@shikijs/engine-oniguruma": "^3.15.0", - "@shikijs/langs": "^3.15.0", - "@shikijs/themes": "^3.15.0", - "@shikijs/types": "^3.15.0", + "@shikijs/engine-oniguruma": "^3.23.0", + "@shikijs/langs": "^3.23.0", + "@shikijs/themes": "^3.23.0", + "@shikijs/types": "^3.23.0", "@shikijs/vscode-textmate": "^10.0.2" } }, @@ -739,10 +777,17 @@ "node": ">=10.10.0" } }, + "node_modules/@humanwhocodes/config-array/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, "node_modules/@humanwhocodes/config-array/node_modules/brace-expansion": { - "version": "1.1.12", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", - "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.14.tgz", + "integrity": "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g==", "dev": true, "license": "MIT", "dependencies": { @@ -751,9 +796,9 @@ } }, "node_modules/@humanwhocodes/config-array/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz", + "integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==", "dev": true, "license": "ISC", "dependencies": { @@ -804,9 +849,9 @@ } }, "node_modules/@isaacs/cliui/node_modules/ansi-regex": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz", - "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==", + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.2.2.tgz", + "integrity": "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg==", "dev": true, "license": "MIT", "engines": { @@ -817,9 +862,9 @@ } }, "node_modules/@isaacs/cliui/node_modules/ansi-styles": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz", - "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==", + "version": "6.2.3", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.3.tgz", + "integrity": "sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg==", "dev": true, "license": "MIT", "engines": { @@ -855,13 +900,13 @@ } }, "node_modules/@isaacs/cliui/node_modules/strip-ansi": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", - "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.2.0.tgz", + "integrity": "sha512-yDPMNjp4WyfYBkHnjIRLfca1i6KMyGCtsVgoKe/z1+6vukgaENdgGBZt+ZmKPc4gavvEZ5OgHfHdrazhgNyG7w==", "dev": true, "license": "MIT", "dependencies": { - "ansi-regex": "^6.0.1" + "ansi-regex": "^6.2.2" }, "engines": { "node": ">=12" @@ -930,9 +975,9 @@ } }, "node_modules/@istanbuljs/load-nyc-config/node_modules/js-yaml": { - "version": "3.14.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", - "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "version": "3.14.2", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.2.tgz", + "integrity": "sha512-PMSmkqxr106Xa156c2M265Z+FTrPl+oxd/rgOQy2tijQeK5TxQ43psO1ZCwhVOSdnn+RzkzlRz/eY4BgJBYVpg==", "dev": true, "license": "MIT", "dependencies": { @@ -996,9 +1041,9 @@ } }, "node_modules/@istanbuljs/schema": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", - "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.6.tgz", + "integrity": "sha512-+Sg6GCR/wy1oSmQDFq4LQDAhm3ETKnorxN+y5nbLULOR3P0c14f2Wurzj3/xqPXtasLFfHd5iRFQ7AJt4KH2cw==", "dev": true, "license": "MIT", "engines": { @@ -1006,61 +1051,60 @@ } }, "node_modules/@jest/console": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/console/-/console-29.7.0.tgz", - "integrity": "sha512-5Ni4CU7XHQi32IJ398EEP4RrB8eV09sXP2ROqD4bksHrnTree52PsxvX8tpL8LvTZ3pFzXyPbNQReSN41CAhOg==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-30.3.0.tgz", + "integrity": "sha512-PAwCvFJ4696XP2qZj+LAn1BWjZaJ6RjG6c7/lkMaUJnkyMS34ucuIsfqYvfskVNvUI27R/u4P1HMYFnlVXG/Ww==", "dev": true, "license": "MIT", "dependencies": { - "@jest/types": "^29.6.3", + "@jest/types": "30.3.0", "@types/node": "*", - "chalk": "^4.0.0", - "jest-message-util": "^29.7.0", - "jest-util": "^29.7.0", + "chalk": "^4.1.2", + "jest-message-util": "30.3.0", + "jest-util": "30.3.0", "slash": "^3.0.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/core": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/core/-/core-29.7.0.tgz", - "integrity": "sha512-n7aeXWKMnGtDA48y8TLWJPJmLmmZ642Ceo78cYWEpiD7FzDgmNDV/GCVRorPABdXLJZ/9wzzgZAlHjXjxDHGsg==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/core/-/core-30.3.0.tgz", + "integrity": "sha512-U5mVPsBxLSO6xYbf+tgkymLx+iAhvZX43/xI1+ej2ZOPnPdkdO1CzDmFKh2mZBn2s4XZixszHeQnzp1gm/DIxw==", "dev": true, "license": "MIT", "dependencies": { - "@jest/console": "^29.7.0", - "@jest/reporters": "^29.7.0", - "@jest/test-result": "^29.7.0", - "@jest/transform": "^29.7.0", - "@jest/types": "^29.6.3", + "@jest/console": "30.3.0", + "@jest/pattern": "30.0.1", + "@jest/reporters": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", "@types/node": "*", - "ansi-escapes": "^4.2.1", - "chalk": "^4.0.0", - "ci-info": "^3.2.0", - "exit": "^0.1.2", - "graceful-fs": "^4.2.9", - "jest-changed-files": "^29.7.0", - "jest-config": "^29.7.0", - "jest-haste-map": "^29.7.0", - "jest-message-util": "^29.7.0", - "jest-regex-util": "^29.6.3", - "jest-resolve": "^29.7.0", - "jest-resolve-dependencies": "^29.7.0", - "jest-runner": "^29.7.0", - "jest-runtime": "^29.7.0", - "jest-snapshot": "^29.7.0", - "jest-util": "^29.7.0", - "jest-validate": "^29.7.0", - "jest-watcher": "^29.7.0", - "micromatch": "^4.0.4", - "pretty-format": "^29.7.0", - "slash": "^3.0.0", - "strip-ansi": "^6.0.0" + "ansi-escapes": "^4.3.2", + "chalk": "^4.1.2", + "ci-info": "^4.2.0", + "exit-x": "^0.2.2", + "graceful-fs": "^4.2.11", + "jest-changed-files": "30.3.0", + "jest-config": "30.3.0", + "jest-haste-map": "30.3.0", + "jest-message-util": "30.3.0", + "jest-regex-util": "30.0.1", + "jest-resolve": "30.3.0", + "jest-resolve-dependencies": "30.3.0", + "jest-runner": "30.3.0", + "jest-runtime": "30.3.0", + "jest-snapshot": "30.3.0", + "jest-util": "30.3.0", + "jest-validate": "30.3.0", + "jest-watcher": "30.3.0", + "pretty-format": "30.3.0", + "slash": "^3.0.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" }, "peerDependencies": { "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" @@ -1071,117 +1115,150 @@ } } }, + "node_modules/@jest/diff-sequences": { + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/diff-sequences/-/diff-sequences-30.3.0.tgz", + "integrity": "sha512-cG51MVnLq1ecVUaQ3fr6YuuAOitHK1S4WUJHnsPFE/quQr33ADUx1FfrTCpMCRxvy0Yr9BThKpDjSlcTi91tMA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, "node_modules/@jest/environment": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-29.7.0.tgz", - "integrity": "sha512-aQIfHDq33ExsN4jP1NWGXhxgQ/wixs60gDiKO+XVMd8Mn0NWPWgc34ZQDTb2jKaUWQ7MuwoitXAsN2XVXNMpAw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-30.3.0.tgz", + "integrity": "sha512-SlLSF4Be735yQXyh2+mctBOzNDx5s5uLv88/j8Qn1wH679PDcwy67+YdADn8NJnGjzlXtN62asGH/T4vWOkfaw==", "dev": true, "license": "MIT", "dependencies": { - "@jest/fake-timers": "^29.7.0", - "@jest/types": "^29.6.3", + "@jest/fake-timers": "30.3.0", + "@jest/types": "30.3.0", "@types/node": "*", - "jest-mock": "^29.7.0" + "jest-mock": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/expect": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/expect/-/expect-29.7.0.tgz", - "integrity": "sha512-8uMeAMycttpva3P1lBHB8VciS9V0XAr3GymPpipdyQXbBcuhkLQOSe8E/p92RyAdToS6ZD1tFkX+CkhoECE0dQ==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/expect/-/expect-30.3.0.tgz", + "integrity": "sha512-76Nlh4xJxk2D/9URCn3wFi98d2hb19uWE1idLsTt2ywhvdOldbw3S570hBgn25P4ICUZ/cBjybrBex2g17IDbg==", "dev": true, "license": "MIT", "dependencies": { - "expect": "^29.7.0", - "jest-snapshot": "^29.7.0" + "expect": "30.3.0", + "jest-snapshot": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/expect-utils": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/expect-utils/-/expect-utils-29.7.0.tgz", - "integrity": "sha512-GlsNBWiFQFCVi9QVSx7f5AgMeLxe9YCCs5PuP2O2LdjDAA8Jh9eX7lA1Jq/xdXw3Wb3hyvlFNfZIfcRetSzYcA==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/expect-utils/-/expect-utils-30.3.0.tgz", + "integrity": "sha512-j0+W5iQQ8hBh7tHZkTQv3q2Fh/M7Je72cIsYqC4OaktgtO7v1So9UTjp6uPBHIaB6beoF/RRsCgMJKvti0wADA==", "dev": true, "license": "MIT", "dependencies": { - "jest-get-type": "^29.6.3" + "@jest/get-type": "30.1.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/fake-timers": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-29.7.0.tgz", - "integrity": "sha512-q4DH1Ha4TTFPdxLsqDXK1d3+ioSL7yL5oCMJZgDYm6i+6CygW5E5xVr/D1HdsGxjt1ZWSfUAs9OxSB/BNelWrQ==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-30.3.0.tgz", + "integrity": "sha512-WUQDs8SOP9URStX1DzhD425CqbN/HxUYCTwVrT8sTVBfMvFqYt/s61EK5T05qnHu0po6RitXIvP9otZxYDzTGQ==", "dev": true, "license": "MIT", "dependencies": { - "@jest/types": "^29.6.3", - "@sinonjs/fake-timers": "^10.0.2", + "@jest/types": "30.3.0", + "@sinonjs/fake-timers": "^15.0.0", "@types/node": "*", - "jest-message-util": "^29.7.0", - "jest-mock": "^29.7.0", - "jest-util": "^29.7.0" + "jest-message-util": "30.3.0", + "jest-mock": "30.3.0", + "jest-util": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/get-type": { + "version": "30.1.0", + "resolved": "https://registry.npmjs.org/@jest/get-type/-/get-type-30.1.0.tgz", + "integrity": "sha512-eMbZE2hUnx1WV0pmURZY9XoXPkUYjpc55mb0CrhtdWLtzMQPFvu/rZkTLZFTsdaVQa+Tr4eWAteqcUzoawq/uA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/globals": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-29.7.0.tgz", - "integrity": "sha512-mpiz3dutLbkW2MNFubUGUEVLkTGiqW6yLVTA+JbP6fI6J5iL9Y0Nlg8k95pcF8ctKwCS7WVxteBs29hhfAotzQ==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-30.3.0.tgz", + "integrity": "sha512-+owLCBBdfpgL3HU+BD5etr1SvbXpSitJK0is1kiYjJxAAJggYMRQz5hSdd5pq1sSggfxPbw2ld71pt4x5wwViA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "30.3.0", + "@jest/expect": "30.3.0", + "@jest/types": "30.3.0", + "jest-mock": "30.3.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/pattern": { + "version": "30.0.1", + "resolved": "https://registry.npmjs.org/@jest/pattern/-/pattern-30.0.1.tgz", + "integrity": "sha512-gWp7NfQW27LaBQz3TITS8L7ZCQ0TLvtmI//4OwlQRx4rnWxcPNIYjxZpDcN4+UlGxgm3jS5QPz8IPTCkb59wZA==", "dev": true, "license": "MIT", "dependencies": { - "@jest/environment": "^29.7.0", - "@jest/expect": "^29.7.0", - "@jest/types": "^29.6.3", - "jest-mock": "^29.7.0" + "@types/node": "*", + "jest-regex-util": "30.0.1" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/reporters": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-29.7.0.tgz", - "integrity": "sha512-DApq0KJbJOEzAFYjHADNNxAE3KbhxQB1y5Kplb5Waqw6zVbuWatSnMjE5gs8FUgEPmNsnZA3NCWl9NG0ia04Pg==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-30.3.0.tgz", + "integrity": "sha512-a09z89S+PkQnL055bVj8+pe2Caed2PBOaczHcXCykW5ngxX9EWx/1uAwncxc/HiU0oZqfwseMjyhxgRjS49qPw==", "dev": true, "license": "MIT", "dependencies": { "@bcoe/v8-coverage": "^0.2.3", - "@jest/console": "^29.7.0", - "@jest/test-result": "^29.7.0", - "@jest/transform": "^29.7.0", - "@jest/types": "^29.6.3", - "@jridgewell/trace-mapping": "^0.3.18", + "@jest/console": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", + "@jridgewell/trace-mapping": "^0.3.25", "@types/node": "*", - "chalk": "^4.0.0", - "collect-v8-coverage": "^1.0.0", - "exit": "^0.1.2", - "glob": "^7.1.3", - "graceful-fs": "^4.2.9", + "chalk": "^4.1.2", + "collect-v8-coverage": "^1.0.2", + "exit-x": "^0.2.2", + "glob": "^10.5.0", + "graceful-fs": "^4.2.11", "istanbul-lib-coverage": "^3.0.0", "istanbul-lib-instrument": "^6.0.0", "istanbul-lib-report": "^3.0.0", - "istanbul-lib-source-maps": "^4.0.0", + "istanbul-lib-source-maps": "^5.0.0", "istanbul-reports": "^3.1.3", - "jest-message-util": "^29.7.0", - "jest-util": "^29.7.0", - "jest-worker": "^29.7.0", + "jest-message-util": "30.3.0", + "jest-util": "30.3.0", + "jest-worker": "30.3.0", "slash": "^3.0.0", - "string-length": "^4.0.1", - "strip-ansi": "^6.0.0", + "string-length": "^4.0.2", "v8-to-istanbul": "^9.0.1" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" }, "peerDependencies": { "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" @@ -1192,115 +1269,186 @@ } } }, + "node_modules/@jest/reporters/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@jest/reporters/node_modules/brace-expansion": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.1.0.tgz", + "integrity": "sha512-TN1kCZAgdgweJhWWpgKYrQaMNHcDULHkWwQIspdtjV4Y5aurRdZpjAqn6yX3FPqTA9ngHCc4hJxMAMgGfve85w==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/@jest/reporters/node_modules/glob": { + "version": "10.5.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.5.0.tgz", + "integrity": "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg==", + "deprecated": "Old versions of glob are not supported, and contain widely publicized security vulnerabilities, which have been fixed in the current version. Please update. Support for old versions may be purchased (at exorbitant rates) by contacting i@izs.me", + "dev": true, + "license": "ISC", + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^3.1.2", + "minimatch": "^9.0.4", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^1.11.1" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/@jest/reporters/node_modules/minimatch": { + "version": "9.0.9", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.9.tgz", + "integrity": "sha512-OBwBN9AL4dqmETlpS2zasx+vTeWclWzkblfZk7KTA5j3jeOONz/tRCnZomUyvNg83wL5Zv9Ss6HMJXAgL8R2Yg==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.2" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, "node_modules/@jest/schemas": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.6.3.tgz", - "integrity": "sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA==", + "version": "30.0.5", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-30.0.5.tgz", + "integrity": "sha512-DmdYgtezMkh3cpU8/1uyXakv3tJRcmcXxBOcO0tbaozPwpmh4YMsnWrQm9ZmZMfa5ocbxzbFk6O4bDPEc/iAnA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@sinclair/typebox": "^0.34.0" + }, + "engines": { + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/@jest/snapshot-utils": { + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/snapshot-utils/-/snapshot-utils-30.3.0.tgz", + "integrity": "sha512-ORbRN9sf5PP82v3FXNSwmO1OTDR2vzR2YTaR+E3VkSBZ8zadQE6IqYdYEeFH1NIkeB2HIGdF02dapb6K0Mj05g==", "dev": true, "license": "MIT", "dependencies": { - "@sinclair/typebox": "^0.27.8" + "@jest/types": "30.3.0", + "chalk": "^4.1.2", + "graceful-fs": "^4.2.11", + "natural-compare": "^1.4.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/source-map": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-29.6.3.tgz", - "integrity": "sha512-MHjT95QuipcPrpLM+8JMSzFx6eHp5Bm+4XeFDJlwsvVBjmKNiIAvasGK2fxz2WbGRlnvqehFbh07MMa7n3YJnw==", + "version": "30.0.1", + "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-30.0.1.tgz", + "integrity": "sha512-MIRWMUUR3sdbP36oyNyhbThLHyJ2eEDClPCiHVbrYAe5g3CHRArIVpBw7cdSB5fr+ofSfIb2Tnsw8iEHL0PYQg==", "dev": true, "license": "MIT", "dependencies": { - "@jridgewell/trace-mapping": "^0.3.18", - "callsites": "^3.0.0", - "graceful-fs": "^4.2.9" + "@jridgewell/trace-mapping": "^0.3.25", + "callsites": "^3.1.0", + "graceful-fs": "^4.2.11" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/test-result": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-29.7.0.tgz", - "integrity": "sha512-Fdx+tv6x1zlkJPcWXmMDAG2HBnaR9XPSd5aDWQVsfrZmLVT3lU1cwyxLgRmXR9yrq4NBoEm9BMsfgFzTQAbJYA==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-30.3.0.tgz", + "integrity": "sha512-e/52nJGuD74AKTSe0P4y5wFRlaXP0qmrS17rqOMHeSwm278VyNyXE3gFO/4DTGF9w+65ra3lo3VKj0LBrzmgdQ==", "dev": true, "license": "MIT", "dependencies": { - "@jest/console": "^29.7.0", - "@jest/types": "^29.6.3", - "@types/istanbul-lib-coverage": "^2.0.0", - "collect-v8-coverage": "^1.0.0" + "@jest/console": "30.3.0", + "@jest/types": "30.3.0", + "@types/istanbul-lib-coverage": "^2.0.6", + "collect-v8-coverage": "^1.0.2" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/test-sequencer": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-29.7.0.tgz", - "integrity": "sha512-GQwJ5WZVrKnOJuiYiAF52UNUJXgTZx1NHjFSEB0qEMmSZKAkdMoIzw/Cj6x6NF4AvV23AUqDpFzQkN/eYCYTxw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-30.3.0.tgz", + "integrity": "sha512-dgbWy9b8QDlQeRZcv7LNF+/jFiiYHTKho1xirauZ7kVwY7avjFF6uTT0RqlgudB5OuIPagFdVtfFMosjVbk1eA==", "dev": true, "license": "MIT", "dependencies": { - "@jest/test-result": "^29.7.0", - "graceful-fs": "^4.2.9", - "jest-haste-map": "^29.7.0", + "@jest/test-result": "30.3.0", + "graceful-fs": "^4.2.11", + "jest-haste-map": "30.3.0", "slash": "^3.0.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/transform": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-29.7.0.tgz", - "integrity": "sha512-ok/BTPFzFKVMwO5eOHRrvnBVHdRy9IrsrW1GpMaQ9MCnilNLXQKmAX8s1YXDFaai9xJpac2ySzV0YeRRECr2Vw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-30.3.0.tgz", + "integrity": "sha512-TLKY33fSLVd/lKB2YI1pH69ijyUblO/BQvCj566YvnwuzoTNr648iE0j22vRvVNk2HsPwByPxATg3MleS3gf5A==", "dev": true, "license": "MIT", "dependencies": { - "@babel/core": "^7.11.6", - "@jest/types": "^29.6.3", - "@jridgewell/trace-mapping": "^0.3.18", - "babel-plugin-istanbul": "^6.1.1", - "chalk": "^4.0.0", + "@babel/core": "^7.27.4", + "@jest/types": "30.3.0", + "@jridgewell/trace-mapping": "^0.3.25", + "babel-plugin-istanbul": "^7.0.1", + "chalk": "^4.1.2", "convert-source-map": "^2.0.0", "fast-json-stable-stringify": "^2.1.0", - "graceful-fs": "^4.2.9", - "jest-haste-map": "^29.7.0", - "jest-regex-util": "^29.6.3", - "jest-util": "^29.7.0", - "micromatch": "^4.0.4", - "pirates": "^4.0.4", + "graceful-fs": "^4.2.11", + "jest-haste-map": "30.3.0", + "jest-regex-util": "30.0.1", + "jest-util": "30.3.0", + "pirates": "^4.0.7", "slash": "^3.0.0", - "write-file-atomic": "^4.0.2" + "write-file-atomic": "^5.0.1" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jest/types": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/@jest/types/-/types-29.6.3.tgz", - "integrity": "sha512-u3UPsIilWKOM3F9CXtrG8LEJmNxwoCQC/XVj4IKYXvvpx7QIi/Kg1LI5uDmDpKlac62NUtX7eLjRh+jVZcLOzw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-30.3.0.tgz", + "integrity": "sha512-JHm87k7bA33hpBngtU8h6UBub/fqqA9uXfw+21j5Hmk7ooPHlboRNxHq0JcMtC+n8VJGP1mcfnD3Mk+XKe1oSw==", "dev": true, "license": "MIT", "dependencies": { - "@jest/schemas": "^29.6.3", - "@types/istanbul-lib-coverage": "^2.0.0", - "@types/istanbul-reports": "^3.0.0", + "@jest/pattern": "30.0.1", + "@jest/schemas": "30.0.5", + "@types/istanbul-lib-coverage": "^2.0.6", + "@types/istanbul-reports": "^3.0.4", "@types/node": "*", - "@types/yargs": "^17.0.8", - "chalk": "^4.0.0" + "@types/yargs": "^17.0.33", + "chalk": "^4.1.2" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/@jridgewell/gen-mapping": { - "version": "0.3.12", - "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.12.tgz", - "integrity": "sha512-OuLGC46TjB5BbN1dH8JULVVZY4WTdkF7tV9Ys6wLL1rubZnCMstOhNHueU5bLCrnRuDhKPDM4g6sw4Bel5Gzqg==", + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", + "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", "dev": true, "license": "MIT", "dependencies": { @@ -1308,6 +1456,17 @@ "@jridgewell/trace-mapping": "^0.3.24" } }, + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", + "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, "node_modules/@jridgewell/resolve-uri": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", @@ -1319,16 +1478,16 @@ } }, "node_modules/@jridgewell/sourcemap-codec": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.4.tgz", - "integrity": "sha512-VT2+G1VQs/9oz078bLrYbecdZKs912zQlkelYpuf+SXF+QvZDYJlbx/LSx+meSAwdDFnF8FVXW92AVjjkVmgFw==", + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", "dev": true, "license": "MIT" }, "node_modules/@jridgewell/trace-mapping": { - "version": "0.3.29", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.29.tgz", - "integrity": "sha512-uw6guiW/gcAGPDhLmd77/6lW8QLeiV5RUTsAX46Db6oLhGaVj4lhnPwb184s1bkc8kdVg/+h988dro8GRDpmYQ==", + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", "dev": true, "license": "MIT", "dependencies": { @@ -1336,6 +1495,31 @@ "@jridgewell/sourcemap-codec": "^1.4.14" } }, + "node_modules/@napi-rs/wasm-runtime": { + "version": "0.2.12", + "resolved": "https://registry.npmjs.org/@napi-rs/wasm-runtime/-/wasm-runtime-0.2.12.tgz", + "integrity": "sha512-ZVWUcfwY4E/yPitQJl481FjFo3K22D6qF0DuFH6Y/nbnE11GY5uguDxZMGXPQ8WQ0128MXQD7TnfHyK4oWoIJQ==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/core": "^1.4.3", + "@emnapi/runtime": "^1.4.3", + "@tybys/wasm-util": "^0.10.0" + } + }, + "node_modules/@nodable/entities": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@nodable/entities/-/entities-2.1.0.tgz", + "integrity": "sha512-nyT7T3nbMyBI/lvr6L5TyWbFJAI9FTgVRakNoBqCD+PmID8DzFrrNdLLtHMwMszOtqZa8PAOV24ZqDnQrhQINA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/nodable" + } + ], + "license": "MIT" + }, "node_modules/@nodelib/fs.scandir": { "version": "2.1.5", "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", @@ -1399,40 +1583,40 @@ } }, "node_modules/@shikijs/engine-oniguruma": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/@shikijs/engine-oniguruma/-/engine-oniguruma-3.15.0.tgz", - "integrity": "sha512-HnqFsV11skAHvOArMZdLBZZApRSYS4LSztk2K3016Y9VCyZISnlYUYsL2hzlS7tPqKHvNqmI5JSUJZprXloMvA==", + "version": "3.23.0", + "resolved": "https://registry.npmjs.org/@shikijs/engine-oniguruma/-/engine-oniguruma-3.23.0.tgz", + "integrity": "sha512-1nWINwKXxKKLqPibT5f4pAFLej9oZzQTsby8942OTlsJzOBZ0MWKiwzMsd+jhzu8YPCHAswGnnN1YtQfirL35g==", "dev": true, "license": "MIT", "dependencies": { - "@shikijs/types": "3.15.0", + "@shikijs/types": "3.23.0", "@shikijs/vscode-textmate": "^10.0.2" } }, "node_modules/@shikijs/langs": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/@shikijs/langs/-/langs-3.15.0.tgz", - "integrity": "sha512-WpRvEFvkVvO65uKYW4Rzxs+IG0gToyM8SARQMtGGsH4GDMNZrr60qdggXrFOsdfOVssG/QQGEl3FnJ3EZ+8w8A==", + "version": "3.23.0", + "resolved": "https://registry.npmjs.org/@shikijs/langs/-/langs-3.23.0.tgz", + "integrity": "sha512-2Ep4W3Re5aB1/62RSYQInK9mM3HsLeB91cHqznAJMuylqjzNVAVCMnNWRHFtcNHXsoNRayP9z1qj4Sq3nMqYXg==", "dev": true, "license": "MIT", "dependencies": { - "@shikijs/types": "3.15.0" + "@shikijs/types": "3.23.0" } }, "node_modules/@shikijs/themes": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/@shikijs/themes/-/themes-3.15.0.tgz", - "integrity": "sha512-8ow2zWb1IDvCKjYb0KiLNrK4offFdkfNVPXb1OZykpLCzRU6j+efkY+Y7VQjNlNFXonSw+4AOdGYtmqykDbRiQ==", + "version": "3.23.0", + "resolved": "https://registry.npmjs.org/@shikijs/themes/-/themes-3.23.0.tgz", + "integrity": "sha512-5qySYa1ZgAT18HR/ypENL9cUSGOeI2x+4IvYJu4JgVJdizn6kG4ia5Q1jDEOi7gTbN4RbuYtmHh0W3eccOrjMA==", "dev": true, "license": "MIT", "dependencies": { - "@shikijs/types": "3.15.0" + "@shikijs/types": "3.23.0" } }, "node_modules/@shikijs/types": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/@shikijs/types/-/types-3.15.0.tgz", - "integrity": "sha512-BnP+y/EQnhihgHy4oIAN+6FFtmfTekwOLsQbRw9hOKwqgNy8Bdsjq8B05oAt/ZgvIWWFrshV71ytOrlPfYjIJw==", + "version": "3.23.0", + "resolved": "https://registry.npmjs.org/@shikijs/types/-/types-3.23.0.tgz", + "integrity": "sha512-3JZ5HXOZfYjsYSk0yPwBrkupyYSLpAE26Qc0HLghhZNGTZg/SKxXIIgoxOpmmeQP0RRSDJTk1/vPfw9tbw+jSQ==", "dev": true, "license": "MIT", "dependencies": { @@ -1448,9 +1632,9 @@ "license": "MIT" }, "node_modules/@sinclair/typebox": { - "version": "0.27.8", - "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.27.8.tgz", - "integrity": "sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==", + "version": "0.34.49", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.34.49.tgz", + "integrity": "sha512-brySQQs7Jtn0joV8Xh9ZV/hZb9Ozb0pmazDIASBkYKCjXrXU3mpcFahmK/z4YDhGkQvP9mWJbVyahdtU5wQA+A==", "dev": true, "license": "MIT" }, @@ -1465,19 +1649,30 @@ } }, "node_modules/@sinonjs/fake-timers": { - "version": "10.3.0", - "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-10.3.0.tgz", - "integrity": "sha512-V4BG07kuYSUkTCSBHG8G8TNhM+F19jXFWnQtzj+we8DrkpSBCee9Z3Ms8yiGer/dlmhe35/Xdgyo3/0rQKg7YA==", + "version": "15.3.2", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-15.3.2.tgz", + "integrity": "sha512-mrn35Jl2pCpns+mE3HaZa1yPN5EYCRgiMI+135COjr2hr8Cls9DXqIZ57vZe2cz7y2XVSq92tcs6kGQcT1J8Rw==", "dev": true, "license": "BSD-3-Clause", "dependencies": { - "@sinonjs/commons": "^3.0.0" + "@sinonjs/commons": "^3.0.1" + } + }, + "node_modules/@tybys/wasm-util": { + "version": "0.10.1", + "resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.1.tgz", + "integrity": "sha512-9tTaPJLSiejZKx+Bmog4uSubteqTvFrVrURwkmHixBo0G4seD0zUxp98E1DzUBJxLQ3NPwXrGKDiVjwx/DpPsg==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "tslib": "^2.4.0" } }, "node_modules/@types/adm-zip": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/@types/adm-zip/-/adm-zip-0.5.7.tgz", - "integrity": "sha512-DNEs/QvmyRLurdQPChqq0Md4zGvPwHerAJYWk9l2jCbD1VPpnzRJorOdiq4zsw09NFbYnhfsoEhWtxIzXpn2yw==", + "version": "0.5.8", + "resolved": "https://registry.npmjs.org/@types/adm-zip/-/adm-zip-0.5.8.tgz", + "integrity": "sha512-RVVH7QvZYbN+ihqZ4kX/dMiowf6o+Jk1fNwiSdx0NahBJLU787zkULhGhJM8mf/obmLGmgdMM0bXsQTmyfbR7Q==", "dev": true, "license": "MIT", "dependencies": { @@ -1520,13 +1715,13 @@ } }, "node_modules/@types/babel__traverse": { - "version": "7.20.7", - "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.20.7.tgz", - "integrity": "sha512-dkO5fhS7+/oos4ciWxyEyjWe48zmG6wbCheo/G2ZnHx4fs3EU6YC6UM8rk56gAjNJ9P3MTH2jo5jb92/K6wbng==", + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz", + "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==", "dev": true, "license": "MIT", "dependencies": { - "@babel/types": "^7.20.7" + "@babel/types": "^7.28.2" } }, "node_modules/@types/better-sqlite3": { @@ -1556,16 +1751,6 @@ "@types/node": "*" } }, - "node_modules/@types/graceful-fs": { - "version": "4.1.9", - "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.9.tgz", - "integrity": "sha512-olP3sd1qOEe5dXTSaFvQG+02VdRXcdytWLAZsAq1PecU8uqQAhkrnbli7DagjtXKW/Bl7YJbUsa8MPcuc8LHEQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*" - } - }, "node_modules/@types/hast": { "version": "3.0.4", "resolved": "https://registry.npmjs.org/@types/hast/-/hast-3.0.4.tgz", @@ -1604,25 +1789,32 @@ } }, "node_modules/@types/jest": { - "version": "29.5.14", - "resolved": "https://registry.npmjs.org/@types/jest/-/jest-29.5.14.tgz", - "integrity": "sha512-ZN+4sdnLUbo8EVvVc2ao0GFW6oVrQRPn4K2lglySj7APvSrgzxHiNNK99us4WDMi57xxA2yggblIAMNhXOotLQ==", + "version": "30.0.0", + "resolved": "https://registry.npmjs.org/@types/jest/-/jest-30.0.0.tgz", + "integrity": "sha512-XTYugzhuwqWjws0CVz8QpM36+T+Dz5mTEBKhNs/esGLnCIlGdRy+Dq78NRjd7ls7r8BC8ZRMOrKlkO1hU0JOwA==", "dev": true, "license": "MIT", "dependencies": { - "expect": "^29.0.0", - "pretty-format": "^29.0.0" + "expect": "^30.0.0", + "pretty-format": "^30.0.0" } }, "node_modules/@types/node": { - "version": "20.19.9", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.9.tgz", - "integrity": "sha512-cuVNgarYWZqxRJDQHEB58GEONhOK79QVR/qYx4S7kcUObQvUwvFnYxJuuHUKm2aieN9X3yZB4LZsuYNU1Qphsw==", + "version": "22.19.17", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.19.17.tgz", + "integrity": "sha512-wGdMcf+vPYM6jikpS/qhg6WiqSV/OhG+jeeHT/KlVqxYfD40iYJf9/AE1uQxVWFvU7MipKRkRv8NSHiCGgPr8Q==", "license": "MIT", "dependencies": { "undici-types": "~6.21.0" } }, + "node_modules/@types/pako": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/@types/pako/-/pako-2.0.4.tgz", + "integrity": "sha512-VWDCbrLeVXJM9fihYodcLiIv0ku+AlOa/TQ1SvYOaBuyrSKgEcro95LJyIsJ4vSo6BXIxOKxiJAat04CmST9Fw==", + "dev": true, + "license": "MIT" + }, "node_modules/@types/plist": { "version": "3.0.5", "resolved": "https://registry.npmjs.org/@types/plist/-/plist-3.0.5.tgz", @@ -1635,9 +1827,9 @@ } }, "node_modules/@types/sql.js": { - "version": "1.4.9", - "resolved": "https://registry.npmjs.org/@types/sql.js/-/sql.js-1.4.9.tgz", - "integrity": "sha512-ep8b36RKHlgWPqjNG9ToUrPiwkhwh0AEzy883mO5Xnd+cL6VBH1EvSjBAAuxLUFF2Vn/moE3Me6v9E1Lo+48GQ==", + "version": "1.4.11", + "resolved": "https://registry.npmjs.org/@types/sql.js/-/sql.js-1.4.11.tgz", + "integrity": "sha512-QXIx38p2ZThJaK9vP5ZdqdlRe1FG9I8SmCZOS7FHfB/2qPAjZwkL7/vlfPg6N/oWHuuOaGg/P/IRwfP2W0kWVQ==", "dev": true, "license": "MIT", "dependencies": { @@ -1669,9 +1861,9 @@ } }, "node_modules/@types/yargs": { - "version": "17.0.33", - "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.33.tgz", - "integrity": "sha512-WpxBCKWPLr4xSsHgz511rFJAM+wS28w2zEO1QDNY5zM/S8ok70NNfztH0xwhqKyaK0OHCbN98LDAZuy1ctxDkA==", + "version": "17.0.35", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.35.tgz", + "integrity": "sha512-qUHkeCyQFxMXg79wQfTtfndEC+N9ZZg76HJftDJp+qH2tV7Gj4OJi7l+PiWwJ+pWtW8GwSmqsDj/oymhrTWXjg==", "dev": true, "license": "MIT", "dependencies": { @@ -1686,122 +1878,149 @@ "license": "MIT" }, "node_modules/@typescript-eslint/eslint-plugin": { - "version": "7.18.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.18.0.tgz", - "integrity": "sha512-94EQTWZ40mzBc42ATNIBimBEDltSJ9RQHCC8vc/PDbxi4k8dVwUAv4o98dk50M1zB+JGFxp43FP7f8+FP8R6Sw==", + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.58.2.tgz", + "integrity": "sha512-aC2qc5thQahutKjP+cl8cgN9DWe3ZUqVko30CMSZHnFEHyhOYoZSzkGtAI2mcwZ38xeImDucI4dnqsHiOYuuCw==", "dev": true, "license": "MIT", "dependencies": { - "@eslint-community/regexpp": "^4.10.0", - "@typescript-eslint/scope-manager": "7.18.0", - "@typescript-eslint/type-utils": "7.18.0", - "@typescript-eslint/utils": "7.18.0", - "@typescript-eslint/visitor-keys": "7.18.0", - "graphemer": "^1.4.0", - "ignore": "^5.3.1", + "@eslint-community/regexpp": "^4.12.2", + "@typescript-eslint/scope-manager": "8.58.2", + "@typescript-eslint/type-utils": "8.58.2", + "@typescript-eslint/utils": "8.58.2", + "@typescript-eslint/visitor-keys": "8.58.2", + "ignore": "^7.0.5", "natural-compare": "^1.4.0", - "ts-api-utils": "^1.3.0" + "ts-api-utils": "^2.5.0" }, "engines": { - "node": "^18.18.0 || >=20.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" }, "peerDependencies": { - "@typescript-eslint/parser": "^7.0.0", - "eslint": "^8.56.0" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "@typescript-eslint/parser": "^8.58.2", + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" } }, "node_modules/@typescript-eslint/parser": { - "version": "7.18.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-7.18.0.tgz", - "integrity": "sha512-4Z+L8I2OqhZV8qA132M4wNL30ypZGYOQVBfMgxDH/K5UX0PNqTu1c6za9ST5r9+tavvHiTWmBnKzpCJ/GlVFtg==", + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.58.2.tgz", + "integrity": "sha512-/Zb/xaIDfxeJnvishjGdcR4jmr7S+bda8PKNhRGdljDM+elXhlvN0FyPSsMnLmJUrVG9aPO6dof80wjMawsASg==", "dev": true, - "license": "BSD-2-Clause", + "license": "MIT", "dependencies": { - "@typescript-eslint/scope-manager": "7.18.0", - "@typescript-eslint/types": "7.18.0", - "@typescript-eslint/typescript-estree": "7.18.0", - "@typescript-eslint/visitor-keys": "7.18.0", - "debug": "^4.3.4" + "@typescript-eslint/scope-manager": "8.58.2", + "@typescript-eslint/types": "8.58.2", + "@typescript-eslint/typescript-estree": "8.58.2", + "@typescript-eslint/visitor-keys": "8.58.2", + "debug": "^4.4.3" }, "engines": { - "node": "^18.18.0 || >=20.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" }, "peerDependencies": { - "eslint": "^8.56.0" + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/project-service": { + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.58.2.tgz", + "integrity": "sha512-Cq6UfpZZk15+r87BkIh5rDpi38W4b+Sjnb8wQCPPDDweS/LRCFjCyViEbzHk5Ck3f2QDfgmlxqSa7S7clDtlfg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/tsconfig-utils": "^8.58.2", + "@typescript-eslint/types": "^8.58.2", + "debug": "^4.4.3" }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" } }, "node_modules/@typescript-eslint/scope-manager": { - "version": "7.18.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-7.18.0.tgz", - "integrity": "sha512-jjhdIE/FPF2B7Z1uzc6i3oWKbGcHb87Qw7AWj6jmEqNOfDFbJWtjt/XfwCpvNkpGWlcJaog5vTR+VV8+w9JflA==", + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.58.2.tgz", + "integrity": "sha512-SgmyvDPexWETQek+qzZnrG6844IaO02UVyOLhI4wpo82dpZJY9+6YZCKAMFzXb7qhx37mFK1QcPQ18tud+vo6Q==", "dev": true, "license": "MIT", "dependencies": { - "@typescript-eslint/types": "7.18.0", - "@typescript-eslint/visitor-keys": "7.18.0" + "@typescript-eslint/types": "8.58.2", + "@typescript-eslint/visitor-keys": "8.58.2" }, "engines": { - "node": "^18.18.0 || >=20.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" } }, + "node_modules/@typescript-eslint/tsconfig-utils": { + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.58.2.tgz", + "integrity": "sha512-3SR+RukipDvkkKp/d0jP0dyzuls3DbGmwDpVEc5wqk5f38KFThakqAAO0XMirWAE+kT00oTauTbzMFGPoAzB0A==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" + } + }, "node_modules/@typescript-eslint/type-utils": { - "version": "7.18.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-7.18.0.tgz", - "integrity": "sha512-XL0FJXuCLaDuX2sYqZUUSOJ2sG5/i1AAze+axqmLnSkNEVMVYLF+cbwlB2w8D1tinFuSikHmFta+P+HOofrLeA==", + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.58.2.tgz", + "integrity": "sha512-Z7EloNR/B389FvabdGeTo2XMs4W9TjtPiO9DAsmT0yom0bwlPyRjkJ1uCdW1DvrrrYP50AJZ9Xc3sByZA9+dcg==", "dev": true, "license": "MIT", "dependencies": { - "@typescript-eslint/typescript-estree": "7.18.0", - "@typescript-eslint/utils": "7.18.0", - "debug": "^4.3.4", - "ts-api-utils": "^1.3.0" + "@typescript-eslint/types": "8.58.2", + "@typescript-eslint/typescript-estree": "8.58.2", + "@typescript-eslint/utils": "8.58.2", + "debug": "^4.4.3", + "ts-api-utils": "^2.5.0" }, "engines": { - "node": "^18.18.0 || >=20.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" }, "peerDependencies": { - "eslint": "^8.56.0" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" } }, "node_modules/@typescript-eslint/types": { - "version": "7.18.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-7.18.0.tgz", - "integrity": "sha512-iZqi+Ds1y4EDYUtlOOC+aUmxnE9xS/yCigkjA7XpTKV6nCBd3Hp/PRGGmdwnfkV2ThMyYldP1wRpm/id99spTQ==", + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.58.2.tgz", + "integrity": "sha512-9TukXyATBQf/Jq9AMQXfvurk+G5R2MwfqQGDR2GzGz28HvY/lXNKGhkY+6IOubwcquikWk5cjlgPvD2uAA7htQ==", "dev": true, "license": "MIT", "engines": { - "node": "^18.18.0 || >=20.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", @@ -1809,75 +2028,88 @@ } }, "node_modules/@typescript-eslint/typescript-estree": { - "version": "7.18.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-7.18.0.tgz", - "integrity": "sha512-aP1v/BSPnnyhMHts8cf1qQ6Q1IFwwRvAQGRvBFkWlo3/lH29OXA3Pts+c10nxRxIBrDnoMqzhgdwVe5f2D6OzA==", + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.58.2.tgz", + "integrity": "sha512-ELGuoofuhhoCvNbQjFFiobFcGgcDCEm0ThWdmO4Z0UzLqPXS3KFvnEZ+SHewwOYHjM09tkzOWXNTv9u6Gqtyuw==", "dev": true, - "license": "BSD-2-Clause", + "license": "MIT", "dependencies": { - "@typescript-eslint/types": "7.18.0", - "@typescript-eslint/visitor-keys": "7.18.0", - "debug": "^4.3.4", - "globby": "^11.1.0", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^1.3.0" + "@typescript-eslint/project-service": "8.58.2", + "@typescript-eslint/tsconfig-utils": "8.58.2", + "@typescript-eslint/types": "8.58.2", + "@typescript-eslint/visitor-keys": "8.58.2", + "debug": "^4.4.3", + "minimatch": "^10.2.2", + "semver": "^7.7.3", + "tinyglobby": "^0.2.15", + "ts-api-utils": "^2.5.0" }, "engines": { - "node": "^18.18.0 || >=20.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" } }, "node_modules/@typescript-eslint/utils": { - "version": "7.18.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-7.18.0.tgz", - "integrity": "sha512-kK0/rNa2j74XuHVcoCZxdFBMF+aq/vH83CXAOHieC+2Gis4mF8jJXT5eAfyD3K0sAxtPuwxaIOIOvhwzVDt/kw==", + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.58.2.tgz", + "integrity": "sha512-QZfjHNEzPY8+l0+fIXMvuQ2sJlplB4zgDZvA+NmvZsZv3EQwOcc1DuIU1VJUTWZ/RKouBMhDyNaBMx4sWvrzRA==", "dev": true, "license": "MIT", "dependencies": { - "@eslint-community/eslint-utils": "^4.4.0", - "@typescript-eslint/scope-manager": "7.18.0", - "@typescript-eslint/types": "7.18.0", - "@typescript-eslint/typescript-estree": "7.18.0" + "@eslint-community/eslint-utils": "^4.9.1", + "@typescript-eslint/scope-manager": "8.58.2", + "@typescript-eslint/types": "8.58.2", + "@typescript-eslint/typescript-estree": "8.58.2" }, "engines": { - "node": "^18.18.0 || >=20.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" }, "peerDependencies": { - "eslint": "^8.56.0" + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" } }, "node_modules/@typescript-eslint/visitor-keys": { - "version": "7.18.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-7.18.0.tgz", - "integrity": "sha512-cDF0/Gf81QpY3xYyJKDV14Zwdmid5+uuENhjH2EqFaF0ni+yAyq/LzMaIJdhNJXZI7uLzwIlA+V7oWoyn6Curg==", + "version": "8.58.2", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.58.2.tgz", + "integrity": "sha512-f1WO2Lx8a9t8DARmcWAUPJbu0G20bJlj8L4z72K00TMeJAoyLr/tHhI/pzYBLrR4dXWkcxO1cWYZEOX8DKHTqA==", "dev": true, "license": "MIT", "dependencies": { - "@typescript-eslint/types": "7.18.0", - "eslint-visitor-keys": "^3.4.3" + "@typescript-eslint/types": "8.58.2", + "eslint-visitor-keys": "^5.0.0" }, "engines": { - "node": "^18.18.0 || >=20.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" } }, + "node_modules/@typescript-eslint/visitor-keys/node_modules/eslint-visitor-keys": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-5.0.1.tgz", + "integrity": "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, "node_modules/@ungap/structured-clone": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.3.0.tgz", @@ -1885,25 +2117,294 @@ "dev": true, "license": "ISC" }, - "node_modules/@xmldom/xmldom": { - "version": "0.8.10", - "resolved": "https://registry.npmjs.org/@xmldom/xmldom/-/xmldom-0.8.10.tgz", - "integrity": "sha512-2WALfTl4xo2SkGCYRt6rDTFfk9R1czmBvUQy12gK2KuRKIpWEhcbbzy8EZXtz/jkRqHX8bFEc6FC1HjX4TUWYw==", + "node_modules/@unrs/resolver-binding-android-arm-eabi": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-android-arm-eabi/-/resolver-binding-android-arm-eabi-1.11.1.tgz", + "integrity": "sha512-ppLRUgHVaGRWUx0R0Ut06Mjo9gBaBkg3v/8AxusGLhsIotbBLuRk51rAzqLC8gq6NyyAojEXglNjzf6R948DNw==", + "cpu": [ + "arm" + ], + "dev": true, "license": "MIT", - "engines": { - "node": ">=10.0.0" - } + "optional": true, + "os": [ + "android" + ] }, - "node_modules/acorn": { - "version": "8.15.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", - "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", + "node_modules/@unrs/resolver-binding-android-arm64": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-android-arm64/-/resolver-binding-android-arm64-1.11.1.tgz", + "integrity": "sha512-lCxkVtb4wp1v+EoN+HjIG9cIIzPkX5OtM03pQYkG+U5O/wL53LC4QbIeazgiKqluGeVEeBlZahHalCaBvU1a2g==", + "cpu": [ + "arm64" + ], "dev": true, "license": "MIT", - "bin": { - "acorn": "bin/acorn" - }, - "engines": { + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@unrs/resolver-binding-darwin-arm64": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-darwin-arm64/-/resolver-binding-darwin-arm64-1.11.1.tgz", + "integrity": "sha512-gPVA1UjRu1Y/IsB/dQEsp2V1pm44Of6+LWvbLc9SDk1c2KhhDRDBUkQCYVWe6f26uJb3fOK8saWMgtX8IrMk3g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@unrs/resolver-binding-darwin-x64": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-darwin-x64/-/resolver-binding-darwin-x64-1.11.1.tgz", + "integrity": "sha512-cFzP7rWKd3lZaCsDze07QX1SC24lO8mPty9vdP+YVa3MGdVgPmFc59317b2ioXtgCMKGiCLxJ4HQs62oz6GfRQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@unrs/resolver-binding-freebsd-x64": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-freebsd-x64/-/resolver-binding-freebsd-x64-1.11.1.tgz", + "integrity": "sha512-fqtGgak3zX4DCB6PFpsH5+Kmt/8CIi4Bry4rb1ho6Av2QHTREM+47y282Uqiu3ZRF5IQioJQ5qWRV6jduA+iGw==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ] + }, + "node_modules/@unrs/resolver-binding-linux-arm-gnueabihf": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm-gnueabihf/-/resolver-binding-linux-arm-gnueabihf-1.11.1.tgz", + "integrity": "sha512-u92mvlcYtp9MRKmP+ZvMmtPN34+/3lMHlyMj7wXJDeXxuM0Vgzz0+PPJNsro1m3IZPYChIkn944wW8TYgGKFHw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-arm-musleabihf": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm-musleabihf/-/resolver-binding-linux-arm-musleabihf-1.11.1.tgz", + "integrity": "sha512-cINaoY2z7LVCrfHkIcmvj7osTOtm6VVT16b5oQdS4beibX2SYBwgYLmqhBjA1t51CarSaBuX5YNsWLjsqfW5Cw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-arm64-gnu": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm64-gnu/-/resolver-binding-linux-arm64-gnu-1.11.1.tgz", + "integrity": "sha512-34gw7PjDGB9JgePJEmhEqBhWvCiiWCuXsL9hYphDF7crW7UgI05gyBAi6MF58uGcMOiOqSJ2ybEeCvHcq0BCmQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-arm64-musl": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-arm64-musl/-/resolver-binding-linux-arm64-musl-1.11.1.tgz", + "integrity": "sha512-RyMIx6Uf53hhOtJDIamSbTskA99sPHS96wxVE/bJtePJJtpdKGXO1wY90oRdXuYOGOTuqjT8ACccMc4K6QmT3w==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-ppc64-gnu": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-ppc64-gnu/-/resolver-binding-linux-ppc64-gnu-1.11.1.tgz", + "integrity": "sha512-D8Vae74A4/a+mZH0FbOkFJL9DSK2R6TFPC9M+jCWYia/q2einCubX10pecpDiTmkJVUH+y8K3BZClycD8nCShA==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-riscv64-gnu": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-riscv64-gnu/-/resolver-binding-linux-riscv64-gnu-1.11.1.tgz", + "integrity": "sha512-frxL4OrzOWVVsOc96+V3aqTIQl1O2TjgExV4EKgRY09AJ9leZpEg8Ak9phadbuX0BA4k8U5qtvMSQQGGmaJqcQ==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-riscv64-musl": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-riscv64-musl/-/resolver-binding-linux-riscv64-musl-1.11.1.tgz", + "integrity": "sha512-mJ5vuDaIZ+l/acv01sHoXfpnyrNKOk/3aDoEdLO/Xtn9HuZlDD6jKxHlkN8ZhWyLJsRBxfv9GYM2utQ1SChKew==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-s390x-gnu": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-s390x-gnu/-/resolver-binding-linux-s390x-gnu-1.11.1.tgz", + "integrity": "sha512-kELo8ebBVtb9sA7rMe1Cph4QHreByhaZ2QEADd9NzIQsYNQpt9UkM9iqr2lhGr5afh885d/cB5QeTXSbZHTYPg==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-x64-gnu": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-x64-gnu/-/resolver-binding-linux-x64-gnu-1.11.1.tgz", + "integrity": "sha512-C3ZAHugKgovV5YvAMsxhq0gtXuwESUKc5MhEtjBpLoHPLYM+iuwSj3lflFwK3DPm68660rZ7G8BMcwSro7hD5w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-linux-x64-musl": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-linux-x64-musl/-/resolver-binding-linux-x64-musl-1.11.1.tgz", + "integrity": "sha512-rV0YSoyhK2nZ4vEswT/QwqzqQXw5I6CjoaYMOX0TqBlWhojUf8P94mvI7nuJTeaCkkds3QE4+zS8Ko+GdXuZtA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@unrs/resolver-binding-wasm32-wasi": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-wasm32-wasi/-/resolver-binding-wasm32-wasi-1.11.1.tgz", + "integrity": "sha512-5u4RkfxJm+Ng7IWgkzi3qrFOvLvQYnPBmjmZQ8+szTK/b31fQCnleNl1GgEt7nIsZRIf5PLhPwT0WM+q45x/UQ==", + "cpu": [ + "wasm32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@napi-rs/wasm-runtime": "^0.2.11" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@unrs/resolver-binding-win32-arm64-msvc": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-arm64-msvc/-/resolver-binding-win32-arm64-msvc-1.11.1.tgz", + "integrity": "sha512-nRcz5Il4ln0kMhfL8S3hLkxI85BXs3o8EYoattsJNdsX4YUU89iOkVn7g0VHSRxFuVMdM4Q1jEpIId1Ihim/Uw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@unrs/resolver-binding-win32-ia32-msvc": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-ia32-msvc/-/resolver-binding-win32-ia32-msvc-1.11.1.tgz", + "integrity": "sha512-DCEI6t5i1NmAZp6pFonpD5m7i6aFrpofcp4LA2i8IIq60Jyo28hamKBxNrZcyOwVOZkgsRp9O2sXWBWP8MnvIQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@unrs/resolver-binding-win32-x64-msvc": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/@unrs/resolver-binding-win32-x64-msvc/-/resolver-binding-win32-x64-msvc-1.11.1.tgz", + "integrity": "sha512-lrW200hZdbfRtztbygyaq/6jP6AKE8qQN2KvPcJ+x7wiD038YtnYtZ82IMNJ69GJibV7bwL3y9FgK+5w/pYt6g==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@xmldom/xmldom": { + "version": "0.8.13", + "resolved": "https://registry.npmjs.org/@xmldom/xmldom/-/xmldom-0.8.13.tgz", + "integrity": "sha512-KRYzxepc14G/CEpEGc3Yn+JKaAeT63smlDr+vjB8jRfgTBBI9wRj/nkQEO+ucV8p8I9bfKLWp37uHgFrbntPvw==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/acorn": { + "version": "8.16.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", + "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==", + "dev": true, + "license": "MIT", + "bin": { + "acorn": "bin/acorn" + }, + "engines": { "node": ">=0.4.0" } }, @@ -1918,18 +2419,18 @@ } }, "node_modules/adm-zip": { - "version": "0.5.16", - "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.5.16.tgz", - "integrity": "sha512-TGw5yVi4saajsSEgz25grObGHEUaDrniwvA2qwSC060KfqGPdglhvPMA2lPIoxs3PQIItj2iag35fONcQqgUaQ==", + "version": "0.5.17", + "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.5.17.tgz", + "integrity": "sha512-+Ut8d9LLqwEvHHJl1+PIHqoyDxFgVN847JTVM3Izi3xHDWPE4UtzzXysMZQs64DMcrJfBeS/uoEP4AD3HQHnQQ==", "license": "MIT", "engines": { "node": ">=12.0" } }, "node_modules/ajv": { - "version": "6.12.6", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", - "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.14.0.tgz", + "integrity": "sha512-IWrosm/yrn43eiKqkfkHis7QioDleaXQHdDVPKg0FSwwd/DuvyX79TZnFOnYpB7dcsFAMmtFztZuXPDvSePkFw==", "dev": true, "license": "MIT", "dependencies": { @@ -2012,6 +2513,19 @@ "node": ">= 8" } }, + "node_modules/anymatch/node_modules/picomatch": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.2.tgz", + "integrity": "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, "node_modules/archiver": { "version": "5.3.2", "resolved": "https://registry.npmjs.org/archiver/-/archiver-5.3.2.tgz", @@ -2088,16 +2602,6 @@ "dev": true, "license": "Python-2.0" }, - "node_modules/array-union": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", - "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, "node_modules/async": { "version": "3.2.6", "resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz", @@ -2111,102 +2615,75 @@ "license": "MIT" }, "node_modules/axios": { - "version": "1.12.2", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.12.2.tgz", - "integrity": "sha512-vMJzPewAlRyOgxV2dU0Cuz2O8zzzx9VYtbJOaBgXFeLc4IV/Eg50n4LowmehOOR61S8ZMpc2K5Sa7g6A4jfkUw==", + "version": "1.15.1", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.15.1.tgz", + "integrity": "sha512-WOG+Jj8ZOvR0a3rAn+Tuf1UQJRxw5venr6DgdbJzngJE3qG7X0kL83CZGpdHMxEm+ZK3seAbvFsw4FfOfP9vxg==", "license": "MIT", "dependencies": { - "follow-redirects": "^1.15.6", - "form-data": "^4.0.4", - "proxy-from-env": "^1.1.0" + "follow-redirects": "^1.15.11", + "form-data": "^4.0.5", + "proxy-from-env": "^2.1.0" } }, "node_modules/babel-jest": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-29.7.0.tgz", - "integrity": "sha512-BrvGY3xZSwEcCzKvKsCi2GgHqDqsYkOP4/by5xCgIwGXQxIEh+8ew3gmrE1y7XRR6LHZIj6yLYnUi/mm2KXKBg==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-30.3.0.tgz", + "integrity": "sha512-gRpauEU2KRrCox5Z296aeVHR4jQ98BCnu0IO332D/xpHNOsIH/bgSRk9k6GbKIbBw8vFeN6ctuu6tV8WOyVfYQ==", "dev": true, "license": "MIT", "dependencies": { - "@jest/transform": "^29.7.0", - "@types/babel__core": "^7.1.14", - "babel-plugin-istanbul": "^6.1.1", - "babel-preset-jest": "^29.6.3", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.9", + "@jest/transform": "30.3.0", + "@types/babel__core": "^7.20.5", + "babel-plugin-istanbul": "^7.0.1", + "babel-preset-jest": "30.3.0", + "chalk": "^4.1.2", + "graceful-fs": "^4.2.11", "slash": "^3.0.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" }, "peerDependencies": { - "@babel/core": "^7.8.0" + "@babel/core": "^7.11.0 || ^8.0.0-0" } }, "node_modules/babel-plugin-istanbul": { - "version": "6.1.1", - "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", - "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-7.0.1.tgz", + "integrity": "sha512-D8Z6Qm8jCvVXtIRkBnqNHX0zJ37rQcFJ9u8WOS6tkYOsRdHBzypCstaxWiu5ZIlqQtviRYbgnRLSoCEvjqcqbA==", "dev": true, "license": "BSD-3-Clause", + "workspaces": [ + "test/babel-8" + ], "dependencies": { "@babel/helper-plugin-utils": "^7.0.0", "@istanbuljs/load-nyc-config": "^1.0.0", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-instrument": "^5.0.4", + "@istanbuljs/schema": "^0.1.3", + "istanbul-lib-instrument": "^6.0.2", "test-exclude": "^6.0.0" }, "engines": { - "node": ">=8" - } - }, - "node_modules/babel-plugin-istanbul/node_modules/istanbul-lib-instrument": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.2.1.tgz", - "integrity": "sha512-pzqtp31nLv/XFOzXGuvhCb8qhjmTVo5vjVk19XE4CRlSWz0KoeJ3bw9XsA7nOp9YBf4qHjwBxkDzKcME/J29Yg==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "@babel/core": "^7.12.3", - "@babel/parser": "^7.14.7", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-coverage": "^3.2.0", - "semver": "^6.3.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/babel-plugin-istanbul/node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "dev": true, - "license": "ISC", - "bin": { - "semver": "bin/semver.js" + "node": ">=12" } }, "node_modules/babel-plugin-jest-hoist": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-29.6.3.tgz", - "integrity": "sha512-ESAc/RJvGTFEzRwOTT4+lNDk/GNHMkKbNzsvT0qKRfDyyYTskxB5rnU2njIDYVxXCBHHEI1c0YwHob3WaYujOg==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-30.3.0.tgz", + "integrity": "sha512-+TRkByhsws6sfPjVaitzadk1I0F5sPvOVUH5tyTSzhePpsGIVrdeunHSw/C36QeocS95OOk8lunc4rlu5Anwsg==", "dev": true, "license": "MIT", "dependencies": { - "@babel/template": "^7.3.3", - "@babel/types": "^7.3.3", - "@types/babel__core": "^7.1.14", - "@types/babel__traverse": "^7.0.6" + "@types/babel__core": "^7.20.5" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/babel-preset-current-node-syntax": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.1.0.tgz", - "integrity": "sha512-ldYss8SbBlWva1bs28q78Ju5Zq1F+8BrqBZZ0VFhLBvhh6lCpC2o3gDJi/5DRLs9FgYZCnmPYIVFU4lRXCkyUw==", + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.2.0.tgz", + "integrity": "sha512-E/VlAEzRrsLEb2+dv8yp3bo4scof3l9nR4lrld+Iy5NyVqgVYUJnDAmunkhPMisRI32Qc4iRiz425d8vM++2fg==", "dev": true, "license": "MIT", "dependencies": { @@ -2227,31 +2704,35 @@ "@babel/plugin-syntax-top-level-await": "^7.14.5" }, "peerDependencies": { - "@babel/core": "^7.0.0" + "@babel/core": "^7.0.0 || ^8.0.0-0" } }, "node_modules/babel-preset-jest": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-29.6.3.tgz", - "integrity": "sha512-0B3bhxR6snWXJZtR/RliHTDPRgn1sNHOR0yVtq/IiQFyuOVjFS+wuio/R4gSNkyYmKmJB4wGZv2NZanmKmTnNA==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-30.3.0.tgz", + "integrity": "sha512-6ZcUbWHC+dMz2vfzdNwi87Z1gQsLNK2uLuK1Q89R11xdvejcivlYYwDlEv0FHX3VwEXpbBQ9uufB/MUNpZGfhQ==", "dev": true, "license": "MIT", "dependencies": { - "babel-plugin-jest-hoist": "^29.6.3", - "babel-preset-current-node-syntax": "^1.0.0" + "babel-plugin-jest-hoist": "30.3.0", + "babel-preset-current-node-syntax": "^1.2.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" }, "peerDependencies": { - "@babel/core": "^7.0.0" + "@babel/core": "^7.11.0 || ^8.0.0-beta.1" } }, "node_modules/balanced-match": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", - "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", - "license": "MIT" + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", + "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "18 || 20 || >=22" + } }, "node_modules/base64-js": { "version": "1.5.1", @@ -2273,10 +2754,23 @@ ], "license": "MIT" }, + "node_modules/baseline-browser-mapping": { + "version": "2.10.20", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.20.tgz", + "integrity": "sha512-1AaXxEPfXT+GvTBJFuy4yXVHWJBXa4OdbIebGN/wX5DlsIkU0+wzGnd2lOzokSk51d5LUmqjgBLRLlypLUqInQ==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.cjs" + }, + "engines": { + "node": ">=6.0.0" + } + }, "node_modules/better-sqlite3": { - "version": "12.5.0", - "resolved": "https://registry.npmjs.org/better-sqlite3/-/better-sqlite3-12.5.0.tgz", - "integrity": "sha512-WwCZ/5Diz7rsF29o27o0Gcc1Du+l7Zsv7SYtVPG0X3G/uUI1LqdxrQI7c9Hs2FWpqXXERjW9hp6g3/tH7DlVKg==", + "version": "12.9.0", + "resolved": "https://registry.npmjs.org/better-sqlite3/-/better-sqlite3-12.9.0.tgz", + "integrity": "sha512-wqUv4Gm3toFpHDQmaKD4QhZm3g1DjUBI0yzS4UBl6lElUmXFYdTQmmEDpAFa5o8FiFiymURypEnfVHzILKaxqQ==", "hasInstallScript": true, "license": "MIT", "dependencies": { @@ -2336,31 +2830,22 @@ "license": "MIT" }, "node_modules/brace-expansion": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", - "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", - "license": "MIT", - "dependencies": { - "balanced-match": "^1.0.0" - } - }, - "node_modules/braces": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", - "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "version": "5.0.5", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.5.tgz", + "integrity": "sha512-VZznLgtwhn+Mact9tfiwx64fA9erHH/MCXEUfB/0bX/6Fz6ny5EGTXYltMocqg4xFAQZtnO3DHWWXi8RiuN7cQ==", "dev": true, "license": "MIT", "dependencies": { - "fill-range": "^7.1.1" + "balanced-match": "^4.0.2" }, "engines": { - "node": ">=8" + "node": "18 || 20 || >=22" } }, "node_modules/browserslist": { - "version": "4.25.1", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.25.1.tgz", - "integrity": "sha512-KGj0KoOMXLpSNkkEI6Z6mShmQy0bc1I+T7K9N81k4WWMrfz+6fQ6es80B/YLAeRoKvjYE1YSHHOW1qe9xIVzHw==", + "version": "4.28.2", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.2.tgz", + "integrity": "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg==", "dev": true, "funding": [ { @@ -2378,10 +2863,11 @@ ], "license": "MIT", "dependencies": { - "caniuse-lite": "^1.0.30001726", - "electron-to-chromium": "^1.5.173", - "node-releases": "^2.0.19", - "update-browserslist-db": "^1.1.3" + "baseline-browser-mapping": "^2.10.12", + "caniuse-lite": "^1.0.30001782", + "electron-to-chromium": "^1.5.328", + "node-releases": "^2.0.36", + "update-browserslist-db": "^1.2.3" }, "bin": { "browserslist": "cli.js" @@ -2504,9 +2990,9 @@ } }, "node_modules/caniuse-lite": { - "version": "1.0.30001727", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001727.tgz", - "integrity": "sha512-pB68nIHmbN6L/4C6MH1DokyR3bYqFwjaSs/sWDHGj4CTcFtQUQMuJftVwWkXq7mNWOybD3KhUv3oWHoGxgP14Q==", + "version": "1.0.30001788", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001788.tgz", + "integrity": "sha512-6q8HFp+lOQtcf7wBK+uEenxymVWkGKkjFpCvw5W25cmMwEDU45p1xQFBQv8JDlMMry7eNxyBaR+qxgmTUZkIRQ==", "dev": true, "funding": [ { @@ -2570,9 +3056,9 @@ "license": "ISC" }, "node_modules/ci-info": { - "version": "3.9.0", - "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", - "integrity": "sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==", + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-4.4.0.tgz", + "integrity": "sha512-77PSwercCZU2Fc4sX94eF8k8Pxte6JAwL4/ICZLFjJLqegs7kCuAsqqj/70NQF6TvDpgFjkubQB2FW2ZZddvQg==", "dev": true, "funding": [ { @@ -2586,9 +3072,9 @@ } }, "node_modules/cjs-module-lexer": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.4.3.tgz", - "integrity": "sha512-9z8TZaGM1pfswYeXrUpzPrkx8UnWYdhJclsiYMm6x/w5+nN+8Tf/LnAgfLGQCm59qAOxU8WwHEq2vNwF6i4j+Q==", + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-2.2.0.tgz", + "integrity": "sha512-4bHTS2YuzUvtoLjdy+98ykbNB5jS0+07EvFNXerqZQJ89F7DI6ET7OQo/HJuW6K0aVsKA9hj9/RVb2kQVOrPDQ==", "dev": true, "license": "MIT" }, @@ -2619,9 +3105,9 @@ } }, "node_modules/collect-v8-coverage": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.2.tgz", - "integrity": "sha512-lHl4d5/ONEbLlJvaJNtsF/Lz+WvB07u2ycqTYbdrq7UypDXailES4valYb2eWiJFxZlVmpGekfqoxQhzyFdT4Q==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.3.tgz", + "integrity": "sha512-1L5aqIkwPfiodaMgQunkF1zRhNqifHBmtbbbxcr6yVxxBnliw4TDOW6NxpO8DJLgJ16OT+Y4ztZqP6p/FtXnAw==", "dev": true, "license": "MIT" }, @@ -2658,12 +3144,12 @@ } }, "node_modules/commander": { - "version": "13.1.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-13.1.0.tgz", - "integrity": "sha512-/rFeCpNJQbhSZjGVwO9RFV3xPqbnERS8MmIQzCtD/zl6gpJuV/bMLuN92oG3F7d8oDEHHRrujSXNUr8fpjntKw==", + "version": "14.0.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-14.0.3.tgz", + "integrity": "sha512-H+y0Jo/T1RZ9qPP4Eh1pkcQcLRglraJaSLoyOtHxu6AapkjWVCy2Sit1QQ4x3Dng8qDlSsZEet7g5Pq06MvTgw==", "license": "MIT", "engines": { - "node": ">=18" + "node": ">=20" } }, "node_modules/compress-commons": { @@ -2725,28 +3211,6 @@ "node": ">= 10" } }, - "node_modules/create-jest": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", - "integrity": "sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jest/types": "^29.6.3", - "chalk": "^4.0.0", - "exit": "^0.1.2", - "graceful-fs": "^4.2.9", - "jest-config": "^29.7.0", - "jest-util": "^29.7.0", - "prompts": "^2.0.1" - }, - "bin": { - "create-jest": "bin/create-jest.js" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, "node_modules/cross-spawn": { "version": "7.0.6", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", @@ -2763,15 +3227,15 @@ } }, "node_modules/dayjs": { - "version": "1.11.13", - "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.13.tgz", - "integrity": "sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==", + "version": "1.11.20", + "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.20.tgz", + "integrity": "sha512-YbwwqR/uYpeoP4pu043q+LTDLFBLApUP6VxRihdfNTqu4ubqMlGDLd6ErXhEgsyvY0K6nCs7nggYumAN+9uEuQ==", "license": "MIT" }, "node_modules/debug": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.1.tgz", - "integrity": "sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ==", + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", "dev": true, "license": "MIT", "dependencies": { @@ -2802,9 +3266,9 @@ } }, "node_modules/dedent": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.6.0.tgz", - "integrity": "sha512-F1Z+5UCFpmQUzJa11agbyPVMbpgT/qA3/SKyJ1jyBgm7dUcUEa8v9JwDkerSQXfakBwFljIxhOJqGkjUwZ9FSA==", + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.7.2.tgz", + "integrity": "sha512-WzMx3mW98SN+zn3hgemf4OzdmyNhhhKz5Ay0pUfQiMQ3e1g+xmTJWp/pKdwKVXhdSkAEGIIzqeuWrL3mV/AXbA==", "dev": true, "license": "MIT", "peerDependencies": { @@ -2870,29 +3334,6 @@ "node": ">=8" } }, - "node_modules/diff-sequences": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", - "integrity": "sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/dir-glob": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", - "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", - "dev": true, - "license": "MIT", - "dependencies": { - "path-type": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/doctrine": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", @@ -2966,26 +3407,10 @@ "dev": true, "license": "MIT" }, - "node_modules/ejs": { - "version": "3.1.10", - "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.10.tgz", - "integrity": "sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "jake": "^10.8.5" - }, - "bin": { - "ejs": "bin/cli.js" - }, - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/electron-to-chromium": { - "version": "1.5.191", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.191.tgz", - "integrity": "sha512-xcwe9ELcuxYLUFqZZxL19Z6HVKcvNkIwhbHUz7L3us6u12yR+7uY89dSl570f/IqNthx8dAw3tojG7i4Ni4tDA==", + "version": "1.5.340", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.340.tgz", + "integrity": "sha512-908qahOGocRMinT2nM3ajCEM99H4iPdv84eagPP3FfZy/1ZGeOy2CZYzjhms81ckOPCXPlW7LkY4XpxD8r1DrA==", "dev": true, "license": "ISC" }, @@ -3032,9 +3457,9 @@ } }, "node_modules/error-ex": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", - "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.4.tgz", + "integrity": "sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==", "dev": true, "license": "MIT", "dependencies": { @@ -3167,27 +3592,30 @@ } }, "node_modules/eslint-config-prettier": { - "version": "9.1.2", - "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-9.1.2.tgz", - "integrity": "sha512-iI1f+D2ViGn+uvv5HuHVUamg8ll4tN+JRHGc6IJi4TP9Kl976C57fzPXgseXNs8v0iA8aSJpHsTWjDb9QJamGQ==", + "version": "10.1.8", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-10.1.8.tgz", + "integrity": "sha512-82GZUjRS0p/jganf6q1rEO25VSoHH0hKPCTrgillPjdI/3bgBhAE1QzHrHTizjpRvy6pGAvKjDJtk2pF9NDq8w==", "dev": true, "license": "MIT", "bin": { "eslint-config-prettier": "bin/cli.js" }, + "funding": { + "url": "https://opencollective.com/eslint-config-prettier" + }, "peerDependencies": { "eslint": ">=7.0.0" } }, "node_modules/eslint-plugin-prettier": { - "version": "5.5.3", - "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.5.3.tgz", - "integrity": "sha512-NAdMYww51ehKfDyDhv59/eIItUVzU0Io9H2E8nHNGKEeeqlnci+1gCvrHib6EmZdf6GxF+LCV5K7UC65Ezvw7w==", + "version": "5.5.5", + "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.5.5.tgz", + "integrity": "sha512-hscXkbqUZ2sPithAuLm5MXL+Wph+U7wHngPBv9OMWwlP8iaflyxpjTYZkmdgB4/vPIhemRlBEoLrH7UC1n7aUw==", "dev": true, "license": "MIT", "dependencies": { - "prettier-linter-helpers": "^1.0.0", - "synckit": "^0.11.7" + "prettier-linter-helpers": "^1.0.1", + "synckit": "^0.11.12" }, "engines": { "node": "^14.18.0 || >=16.0.0" @@ -3240,10 +3668,17 @@ "url": "https://opencollective.com/eslint" } }, + "node_modules/eslint/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, "node_modules/eslint/node_modules/brace-expansion": { - "version": "1.1.12", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", - "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.14.tgz", + "integrity": "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g==", "dev": true, "license": "MIT", "dependencies": { @@ -3251,10 +3686,20 @@ "concat-map": "0.0.1" } }, + "node_modules/eslint/node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, "node_modules/eslint/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz", + "integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==", "dev": true, "license": "ISC", "dependencies": { @@ -3297,9 +3742,9 @@ } }, "node_modules/esquery": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", - "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", + "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==", "dev": true, "license": "BSD-3-Clause", "dependencies": { @@ -3386,11 +3831,12 @@ "url": "https://github.com/sindresorhus/execa?sponsor=1" } }, - "node_modules/exit": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", - "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "node_modules/exit-x": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/exit-x/-/exit-x-0.2.2.tgz", + "integrity": "sha512-+I6B/IkJc1o/2tiURyz/ivu/O0nKNEArIUB5O7zBrlDVJr22SCLH3xTeEry428LvFhRzIA1g8izguxJ/gbNcVQ==", "dev": true, + "license": "MIT", "engines": { "node": ">= 0.8.0" } @@ -3405,26 +3851,27 @@ } }, "node_modules/expect": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz", - "integrity": "sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/expect/-/expect-30.3.0.tgz", + "integrity": "sha512-1zQrciTiQfRdo7qJM1uG4navm8DayFa2TgCSRlzUyNkhcJ6XUZF3hjnpkyr3VhAqPH7i/9GkG7Tv5abz6fqz0Q==", "dev": true, "license": "MIT", "dependencies": { - "@jest/expect-utils": "^29.7.0", - "jest-get-type": "^29.6.3", - "jest-matcher-utils": "^29.7.0", - "jest-message-util": "^29.7.0", - "jest-util": "^29.7.0" + "@jest/expect-utils": "30.3.0", + "@jest/get-type": "30.1.0", + "jest-matcher-utils": "30.3.0", + "jest-message-util": "30.3.0", + "jest-mock": "30.3.0", + "jest-util": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/fast-check": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/fast-check/-/fast-check-4.2.0.tgz", - "integrity": "sha512-buxrKEaSseOwFjt6K1REcGMeFOrb0wk3cXifeMAG8yahcE9kV20PjQn1OdzPGL6OBFTbYXfjleNBARf/aCfV1A==", + "version": "4.7.0", + "resolved": "https://registry.npmjs.org/fast-check/-/fast-check-4.7.0.tgz", + "integrity": "sha512-NsZRtqvSSoCP0HbNjUD+r1JH8zqZalyp6gLY9e7OYs7NK9b6AHOs2baBFeBG7bVNsuoukh89x2Yg3rPsul8ziQ==", "dev": true, "funding": [ { @@ -3438,7 +3885,7 @@ ], "license": "MIT", "dependencies": { - "pure-rand": "^7.0.0" + "pure-rand": "^8.0.0" }, "engines": { "node": ">=12.17.0" @@ -3471,36 +3918,6 @@ "dev": true, "license": "Apache-2.0" }, - "node_modules/fast-glob": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", - "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", - "dev": true, - "license": "MIT", - "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.8" - }, - "engines": { - "node": ">=8.6.0" - } - }, - "node_modules/fast-glob/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "license": "ISC", - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, "node_modules/fast-json-stable-stringify": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", @@ -3515,10 +3932,25 @@ "dev": true, "license": "MIT" }, + "node_modules/fast-xml-builder": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/fast-xml-builder/-/fast-xml-builder-1.1.5.tgz", + "integrity": "sha512-4TJn/8FKLeslLAH3dnohXqE3QSoxkhvaMzepOIZytwJXZO69Bfz0HBdDHzOTOon6G59Zrk6VQ2bEiv1t61rfkA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/NaturalIntelligence" + } + ], + "license": "MIT", + "dependencies": { + "path-expression-matcher": "^1.1.3" + } + }, "node_modules/fast-xml-parser": { - "version": "5.2.5", - "resolved": "https://registry.npmjs.org/fast-xml-parser/-/fast-xml-parser-5.2.5.tgz", - "integrity": "sha512-pfX9uG9Ki0yekDHx2SiuRIyFdyAr1kMIMitPvb0YBo8SUfKvia7w7FIyd/l6av85pFYRhZscS75MwMnbvY+hcQ==", + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/fast-xml-parser/-/fast-xml-parser-5.7.1.tgz", + "integrity": "sha512-8Cc3f8GUGUULg34pBch/KGyPLglS+OFs05deyOlY7fL2MTagYPKrVQNmR1fLF/yJ9PH5ZSTd3YDF6pnmeZU+zA==", "funding": [ { "type": "github", @@ -3527,16 +3959,19 @@ ], "license": "MIT", "dependencies": { - "strnum": "^2.1.0" + "@nodable/entities": "^2.1.0", + "fast-xml-builder": "^1.1.5", + "path-expression-matcher": "^1.5.0", + "strnum": "^2.2.3" }, "bin": { "fxparser": "src/cli/cli.js" } }, "node_modules/fastq": { - "version": "1.19.1", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", - "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", + "version": "1.20.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.20.1.tgz", + "integrity": "sha512-GGToxJ/w1x32s/D2EKND7kTil4n8OVk/9mycTc4VDza13lOvpUZTGX3mFSCtV9ksdGBVzvsyAVLM6mHFThxXxw==", "dev": true, "license": "ISC", "dependencies": { @@ -3553,6 +3988,24 @@ "bser": "2.1.1" } }, + "node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, "node_modules/file-entry-cache": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", @@ -3572,42 +4025,6 @@ "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", "license": "MIT" }, - "node_modules/filelist": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.4.tgz", - "integrity": "sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "minimatch": "^5.0.1" - } - }, - "node_modules/filelist/node_modules/minimatch": { - "version": "5.1.6", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", - "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==", - "dev": true, - "license": "ISC", - "dependencies": { - "brace-expansion": "^2.0.1" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/fill-range": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", - "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", - "dev": true, - "license": "MIT", - "dependencies": { - "to-regex-range": "^5.0.1" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/find-up": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", @@ -3658,16 +4075,16 @@ } }, "node_modules/flatted": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", - "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.4.2.tgz", + "integrity": "sha512-PjDse7RzhcPkIJwy5t7KPWQSZ9cAbzQXcafsetQoD7sOJRQlGikNbx7yZp2OotDnJyrDcbyRq3Ttb18iYOqkxA==", "dev": true, "license": "ISC" }, "node_modules/follow-redirects": { - "version": "1.15.9", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", - "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", + "version": "1.16.0", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.16.0.tgz", + "integrity": "sha512-y5rN/uOsadFT/JfYwhxRS5R7Qce+g3zG97+JrtFZlC9klX/W5hD7iiLzScI4nZqUS7DNUdhPgw4xI8W2LuXlUw==", "funding": [ { "type": "individual", @@ -3715,9 +4132,9 @@ } }, "node_modules/form-data": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.4.tgz", - "integrity": "sha512-KrGhL9Q4zjj0kiUt5OO4Mr/A/jlI2jDYs5eHBpYHPcBEVSiipAvn2Ko2HnPe20rmcuuvMHNdZFp+4IlGTMF0Ow==", + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.5.tgz", + "integrity": "sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w==", "license": "MIT", "dependencies": { "asynckit": "^0.4.0", @@ -3885,7 +4302,7 @@ "version": "7.2.3", "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", - "deprecated": "Glob versions prior to v9 are no longer supported", + "deprecated": "Old versions of glob are not supported, and contain widely publicized security vulnerabilities, which have been fixed in the current version. Please update. Support for old versions may be purchased (at exorbitant rates) by contacting i@izs.me", "license": "ISC", "dependencies": { "fs.realpath": "^1.0.0", @@ -3915,10 +4332,16 @@ "node": ">=10.13.0" } }, + "node_modules/glob/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "license": "MIT" + }, "node_modules/glob/node_modules/brace-expansion": { - "version": "1.1.12", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", - "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.14.tgz", + "integrity": "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g==", "license": "MIT", "dependencies": { "balanced-match": "^1.0.0", @@ -3926,9 +4349,9 @@ } }, "node_modules/glob/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz", + "integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==", "license": "ISC", "dependencies": { "brace-expansion": "^1.1.7" @@ -3953,27 +4376,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/globby": { - "version": "11.1.0", - "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", - "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", - "dev": true, - "license": "MIT", - "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" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, "node_modules/gopd": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", @@ -3999,6 +4401,28 @@ "dev": true, "license": "MIT" }, + "node_modules/handlebars": { + "version": "4.7.9", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.9.tgz", + "integrity": "sha512-4E71E0rpOaQuJR2A3xDZ+GM1HyWYv1clR58tC8emQNeQe3RH7MAzSbat+V0wG78LQBo6m6bzSG/L4pBuCsgnUQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "minimist": "^1.2.5", + "neo-async": "^2.6.2", + "source-map": "^0.6.1", + "wordwrap": "^1.0.0" + }, + "bin": { + "handlebars": "bin/handlebars" + }, + "engines": { + "node": ">=0.4.7" + }, + "optionalDependencies": { + "uglify-js": "^3.1.4" + } + }, "node_modules/has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", @@ -4037,9 +4461,9 @@ } }, "node_modules/hasown": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", - "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.3.tgz", + "integrity": "sha512-ej4AhfhfL2Q2zpMmLo7U1Uv9+PyhIZpgQLGT1F9miIGmiCJIoCgSmczFdrc97mWT4kVY72KA+WnnhJ5pghSvSg==", "license": "MIT", "dependencies": { "function-bind": "^1.1.2" @@ -4086,9 +4510,9 @@ "license": "BSD-3-Clause" }, "node_modules/ignore": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", - "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "version": "7.0.5", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz", + "integrity": "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==", "dev": true, "license": "MIT", "engines": { @@ -4178,22 +4602,6 @@ "dev": true, "license": "MIT" }, - "node_modules/is-core-module": { - "version": "2.16.1", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", - "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", - "dev": true, - "license": "MIT", - "dependencies": { - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", @@ -4237,16 +4645,6 @@ "node": ">=0.10.0" } }, - "node_modules/is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.12.0" - } - }, "node_modules/is-path-inside": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", @@ -4326,24 +4724,24 @@ } }, "node_modules/istanbul-lib-source-maps": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", - "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "version": "5.0.6", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-5.0.6.tgz", + "integrity": "sha512-yg2d+Em4KizZC5niWhQaIomgf5WlL4vOOjZ5xGCmF8SnPE/mDWWXgvRExdcpCgh9lLRRa1/fSYp2ymmbJ1pI+A==", "dev": true, "license": "BSD-3-Clause", "dependencies": { + "@jridgewell/trace-mapping": "^0.3.23", "debug": "^4.1.1", - "istanbul-lib-coverage": "^3.0.0", - "source-map": "^0.6.1" + "istanbul-lib-coverage": "^3.0.0" }, "engines": { "node": ">=10" } }, "node_modules/istanbul-reports": { - "version": "3.1.7", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.7.tgz", - "integrity": "sha512-BewmUXImeuRk2YY0PVbxgKAysvhRPUQE0h5QRM++nVWyubKGV0l8qQ5op8+B2DOmwSe63Jivj0BjkPQVf8fP5g==", + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.2.0.tgz", + "integrity": "sha512-HGYWWS/ehqTV3xN10i23tkPkpH46MLCIMFNCaaKNavAXTF1RkqxawEPtnjnGZ6XKSInBKkiOA5BKS+aZiY3AvA==", "dev": true, "license": "BSD-3-Clause", "dependencies": { @@ -4370,66 +4768,23 @@ "@pkgjs/parseargs": "^0.11.0" } }, - "node_modules/jake": { - "version": "10.9.2", - "resolved": "https://registry.npmjs.org/jake/-/jake-10.9.2.tgz", - "integrity": "sha512-2P4SQ0HrLQ+fw6llpLnOaGAvN2Zu6778SJMrCUwns4fOoG9ayrTiZk3VV8sCPkVZF8ab0zksVpS8FDY5pRCNBA==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "async": "^3.2.3", - "chalk": "^4.0.2", - "filelist": "^1.0.4", - "minimatch": "^3.1.2" - }, - "bin": { - "jake": "bin/cli.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/jake/node_modules/brace-expansion": { - "version": "1.1.12", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", - "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", - "dev": true, - "license": "MIT", - "dependencies": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "node_modules/jake/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", - "dev": true, - "license": "ISC", - "dependencies": { - "brace-expansion": "^1.1.7" - }, - "engines": { - "node": "*" - } - }, "node_modules/jest": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest/-/jest-29.7.0.tgz", - "integrity": "sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest/-/jest-30.3.0.tgz", + "integrity": "sha512-AkXIIFcaazymvey2i/+F94XRnM6TsVLZDhBMLsd1Sf/W0wzsvvpjeyUrCZD6HGG4SDYPgDJDBKeiJTBb10WzMg==", "dev": true, "license": "MIT", "dependencies": { - "@jest/core": "^29.7.0", - "@jest/types": "^29.6.3", - "import-local": "^3.0.2", - "jest-cli": "^29.7.0" + "@jest/core": "30.3.0", + "@jest/types": "30.3.0", + "import-local": "^3.2.0", + "jest-cli": "30.3.0" }, "bin": { "jest": "bin/jest.js" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" }, "peerDependencies": { "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" @@ -4441,56 +4796,56 @@ } }, "node_modules/jest-changed-files": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-29.7.0.tgz", - "integrity": "sha512-fEArFiwf1BpQ+4bXSprcDc3/x4HSzL4al2tozwVpDFpsxALjLYdyiIK4e5Vz66GQJIbXJ82+35PtysofptNX2w==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-30.3.0.tgz", + "integrity": "sha512-B/7Cny6cV5At6M25EWDgf9S617lHivamL8vl6KEpJqkStauzcG4e+WPfDgMMF+H4FVH4A2PLRyvgDJan4441QA==", "dev": true, "license": "MIT", "dependencies": { - "execa": "^5.0.0", - "jest-util": "^29.7.0", + "execa": "^5.1.1", + "jest-util": "30.3.0", "p-limit": "^3.1.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-circus": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-29.7.0.tgz", - "integrity": "sha512-3E1nCMgipcTkCocFwM90XXQab9bS+GMsjdpmPrlelaxwD93Ad8iVEjX/vvHPdLPnFf+L40u+5+iutRdA1N9myw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-30.3.0.tgz", + "integrity": "sha512-PyXq5szeSfR/4f1lYqCmmQjh0vqDkURUYi9N6whnHjlRz4IUQfMcXkGLeEoiJtxtyPqgUaUUfyQlApXWBSN1RA==", "dev": true, "license": "MIT", "dependencies": { - "@jest/environment": "^29.7.0", - "@jest/expect": "^29.7.0", - "@jest/test-result": "^29.7.0", - "@jest/types": "^29.6.3", + "@jest/environment": "30.3.0", + "@jest/expect": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/types": "30.3.0", "@types/node": "*", - "chalk": "^4.0.0", + "chalk": "^4.1.2", "co": "^4.6.0", - "dedent": "^1.0.0", - "is-generator-fn": "^2.0.0", - "jest-each": "^29.7.0", - "jest-matcher-utils": "^29.7.0", - "jest-message-util": "^29.7.0", - "jest-runtime": "^29.7.0", - "jest-snapshot": "^29.7.0", - "jest-util": "^29.7.0", + "dedent": "^1.6.0", + "is-generator-fn": "^2.1.0", + "jest-each": "30.3.0", + "jest-matcher-utils": "30.3.0", + "jest-message-util": "30.3.0", + "jest-runtime": "30.3.0", + "jest-snapshot": "30.3.0", + "jest-util": "30.3.0", "p-limit": "^3.1.0", - "pretty-format": "^29.7.0", - "pure-rand": "^6.0.0", + "pretty-format": "30.3.0", + "pure-rand": "^7.0.0", "slash": "^3.0.0", - "stack-utils": "^2.0.3" + "stack-utils": "^2.0.6" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-circus/node_modules/pure-rand": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-6.1.0.tgz", - "integrity": "sha512-bVWawvoZoBYpp6yIoQtQXHZjmz35RSVHnUOTefl8Vcjr8snTPY1wnpSPMWekcFwbxI6gtmT7rSYPFvz71ldiOA==", + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-7.0.1.tgz", + "integrity": "sha512-oTUZM/NAZS8p7ANR3SHh30kXB+zK2r2BPcEn/awJIbOvq82WoMN4p62AWWp3Hhw50G0xMsw1mhIBLqHw64EcNQ==", "dev": true, "funding": [ { @@ -4505,29 +4860,28 @@ "license": "MIT" }, "node_modules/jest-cli": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-29.7.0.tgz", - "integrity": "sha512-OVVobw2IubN/GSYsxETi+gOe7Ka59EFMR/twOU3Jb2GnKKeMGJB5SGUUrEz3SFVmJASUdZUzy83sLNNQ2gZslg==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-30.3.0.tgz", + "integrity": "sha512-l6Tqx+j1fDXJEW5bqYykDQQ7mQg+9mhWXtnj+tQZrTWYHyHoi6Be8HPumDSA+UiX2/2buEgjA58iJzdj146uCw==", "dev": true, "license": "MIT", "dependencies": { - "@jest/core": "^29.7.0", - "@jest/test-result": "^29.7.0", - "@jest/types": "^29.6.3", - "chalk": "^4.0.0", - "create-jest": "^29.7.0", - "exit": "^0.1.2", - "import-local": "^3.0.2", - "jest-config": "^29.7.0", - "jest-util": "^29.7.0", - "jest-validate": "^29.7.0", - "yargs": "^17.3.1" + "@jest/core": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/types": "30.3.0", + "chalk": "^4.1.2", + "exit-x": "^0.2.2", + "import-local": "^3.2.0", + "jest-config": "30.3.0", + "jest-util": "30.3.0", + "jest-validate": "30.3.0", + "yargs": "^17.7.2" }, "bin": { "jest": "bin/jest.js" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" }, "peerDependencies": { "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" @@ -4539,215 +4893,265 @@ } }, "node_modules/jest-config": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-29.7.0.tgz", - "integrity": "sha512-uXbpfeQ7R6TZBqI3/TxCU4q4ttk3u0PJeC+E0zbfSoSjq6bJ7buBPxzQPL0ifrkY4DNu4JUdk0ImlBUYi840eQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/core": "^7.11.6", - "@jest/test-sequencer": "^29.7.0", - "@jest/types": "^29.6.3", - "babel-jest": "^29.7.0", - "chalk": "^4.0.0", - "ci-info": "^3.2.0", - "deepmerge": "^4.2.2", - "glob": "^7.1.3", - "graceful-fs": "^4.2.9", - "jest-circus": "^29.7.0", - "jest-environment-node": "^29.7.0", - "jest-get-type": "^29.6.3", - "jest-regex-util": "^29.6.3", - "jest-resolve": "^29.7.0", - "jest-runner": "^29.7.0", - "jest-util": "^29.7.0", - "jest-validate": "^29.7.0", - "micromatch": "^4.0.4", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-30.3.0.tgz", + "integrity": "sha512-WPMAkMAtNDY9P/oKObtsRG/6KTrhtgPJoBTmk20uDn4Uy6/3EJnnaZJre/FMT1KVRx8cve1r7/FlMIOfRVWL4w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.27.4", + "@jest/get-type": "30.1.0", + "@jest/pattern": "30.0.1", + "@jest/test-sequencer": "30.3.0", + "@jest/types": "30.3.0", + "babel-jest": "30.3.0", + "chalk": "^4.1.2", + "ci-info": "^4.2.0", + "deepmerge": "^4.3.1", + "glob": "^10.5.0", + "graceful-fs": "^4.2.11", + "jest-circus": "30.3.0", + "jest-docblock": "30.2.0", + "jest-environment-node": "30.3.0", + "jest-regex-util": "30.0.1", + "jest-resolve": "30.3.0", + "jest-runner": "30.3.0", + "jest-util": "30.3.0", + "jest-validate": "30.3.0", "parse-json": "^5.2.0", - "pretty-format": "^29.7.0", + "pretty-format": "30.3.0", "slash": "^3.0.0", "strip-json-comments": "^3.1.1" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" }, "peerDependencies": { "@types/node": "*", + "esbuild-register": ">=3.4.0", "ts-node": ">=9.0.0" }, "peerDependenciesMeta": { "@types/node": { "optional": true }, + "esbuild-register": { + "optional": true + }, "ts-node": { "optional": true } } }, - "node_modules/jest-diff": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.7.0.tgz", - "integrity": "sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw==", + "node_modules/jest-config/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-config/node_modules/brace-expansion": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.1.0.tgz", + "integrity": "sha512-TN1kCZAgdgweJhWWpgKYrQaMNHcDULHkWwQIspdtjV4Y5aurRdZpjAqn6yX3FPqTA9ngHCc4hJxMAMgGfve85w==", "dev": true, "license": "MIT", "dependencies": { - "chalk": "^4.0.0", - "diff-sequences": "^29.6.3", - "jest-get-type": "^29.6.3", - "pretty-format": "^29.7.0" + "balanced-match": "^1.0.0" + } + }, + "node_modules/jest-config/node_modules/glob": { + "version": "10.5.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.5.0.tgz", + "integrity": "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg==", + "deprecated": "Old versions of glob are not supported, and contain widely publicized security vulnerabilities, which have been fixed in the current version. Please update. Support for old versions may be purchased (at exorbitant rates) by contacting i@izs.me", + "dev": true, + "license": "ISC", + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^3.1.2", + "minimatch": "^9.0.4", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^1.11.1" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/jest-config/node_modules/minimatch": { + "version": "9.0.9", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.9.tgz", + "integrity": "sha512-OBwBN9AL4dqmETlpS2zasx+vTeWclWzkblfZk7KTA5j3jeOONz/tRCnZomUyvNg83wL5Zv9Ss6HMJXAgL8R2Yg==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.2" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/jest-docblock": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-29.7.0.tgz", - "integrity": "sha512-q617Auw3A612guyaFgsbFeYpNP5t2aoUNLwBUbc/0kD1R4t9ixDbyFTHd1nok4epoVFpr7PmeWHrhvuV3XaJ4g==", + "node_modules/jest-diff": { + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-30.3.0.tgz", + "integrity": "sha512-n3q4PDQjS4LrKxfWB3Z5KNk1XjXtZTBwQp71OP0Jo03Z6V60x++K5L8k6ZrW8MY8pOFylZvHM0zsjS1RqlHJZQ==", "dev": true, "license": "MIT", "dependencies": { - "detect-newline": "^3.0.0" + "@jest/diff-sequences": "30.3.0", + "@jest/get-type": "30.1.0", + "chalk": "^4.1.2", + "pretty-format": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, - "node_modules/jest-each": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-29.7.0.tgz", - "integrity": "sha512-gns+Er14+ZrEoC5fhOfYCY1LOHHr0TI+rQUHZS8Ttw2l7gl+80eHc/gFf2Ktkw0+SIACDTeWvpFcv3B04VembQ==", + "node_modules/jest-docblock": { + "version": "30.2.0", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-30.2.0.tgz", + "integrity": "sha512-tR/FFgZKS1CXluOQzZvNH3+0z9jXr3ldGSD8bhyuxvlVUwbeLOGynkunvlTMxchC5urrKndYiwCFC0DLVjpOCA==", "dev": true, "license": "MIT", "dependencies": { - "@jest/types": "^29.6.3", - "chalk": "^4.0.0", - "jest-get-type": "^29.6.3", - "jest-util": "^29.7.0", - "pretty-format": "^29.7.0" + "detect-newline": "^3.1.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, - "node_modules/jest-environment-node": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-29.7.0.tgz", - "integrity": "sha512-DOSwCRqXirTOyheM+4d5YZOrWcdu0LNZ87ewUoywbcb2XR4wKgqiG8vNeYwhjFMbEkfju7wx2GYH0P2gevGvFw==", + "node_modules/jest-each": { + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-30.3.0.tgz", + "integrity": "sha512-V8eMndg/aZ+3LnCJgSm13IxS5XSBM22QSZc9BtPK8Dek6pm+hfUNfwBdvsB3d342bo1q7wnSkC38zjX259qZNA==", "dev": true, "license": "MIT", "dependencies": { - "@jest/environment": "^29.7.0", - "@jest/fake-timers": "^29.7.0", - "@jest/types": "^29.6.3", - "@types/node": "*", - "jest-mock": "^29.7.0", - "jest-util": "^29.7.0" + "@jest/get-type": "30.1.0", + "@jest/types": "30.3.0", + "chalk": "^4.1.2", + "jest-util": "30.3.0", + "pretty-format": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, - "node_modules/jest-get-type": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", - "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "node_modules/jest-environment-node": { + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-30.3.0.tgz", + "integrity": "sha512-4i6HItw/JSiJVsC5q0hnKIe/hbYfZLVG9YJ/0pU9Hz2n/9qZe3Rhn5s5CUZA5ORZlcdT/vmAXRMyONXJwPrmYQ==", "dev": true, "license": "MIT", + "dependencies": { + "@jest/environment": "30.3.0", + "@jest/fake-timers": "30.3.0", + "@jest/types": "30.3.0", + "@types/node": "*", + "jest-mock": "30.3.0", + "jest-util": "30.3.0", + "jest-validate": "30.3.0" + }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-haste-map": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-29.7.0.tgz", - "integrity": "sha512-fP8u2pyfqx0K1rGn1R9pyE0/KTn+G7PxktWidOBTqFPLYX0b9ksaMFkhK5vrS3DVun09pckLdlx90QthlW7AmA==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-30.3.0.tgz", + "integrity": "sha512-mMi2oqG4KRU0R9QEtscl87JzMXfUhbKaFqOxmjb2CKcbHcUGFrJCBWHmnTiUqi6JcnzoBlO4rWfpdl2k/RfLCA==", "dev": true, "license": "MIT", "dependencies": { - "@jest/types": "^29.6.3", - "@types/graceful-fs": "^4.1.3", + "@jest/types": "30.3.0", "@types/node": "*", - "anymatch": "^3.0.3", - "fb-watchman": "^2.0.0", - "graceful-fs": "^4.2.9", - "jest-regex-util": "^29.6.3", - "jest-util": "^29.7.0", - "jest-worker": "^29.7.0", - "micromatch": "^4.0.4", + "anymatch": "^3.1.3", + "fb-watchman": "^2.0.2", + "graceful-fs": "^4.2.11", + "jest-regex-util": "30.0.1", + "jest-util": "30.3.0", + "jest-worker": "30.3.0", + "picomatch": "^4.0.3", "walker": "^1.0.8" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" }, "optionalDependencies": { - "fsevents": "^2.3.2" + "fsevents": "^2.3.3" } }, "node_modules/jest-leak-detector": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-29.7.0.tgz", - "integrity": "sha512-kYA8IJcSYtST2BY9I+SMC32nDpBT3J2NvWJx8+JCuCdl/CR1I4EKUJROiP8XtCcxqgTTBGJNdbB1A8XRKbTetw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-30.3.0.tgz", + "integrity": "sha512-cuKmUUGIjfXZAiGJ7TbEMx0bcqNdPPI6P1V+7aF+m/FUJqFDxkFR4JqkTu8ZOiU5AaX/x0hZ20KaaIPXQzbMGQ==", "dev": true, "license": "MIT", "dependencies": { - "jest-get-type": "^29.6.3", - "pretty-format": "^29.7.0" + "@jest/get-type": "30.1.0", + "pretty-format": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-matcher-utils": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz", - "integrity": "sha512-sBkD+Xi9DtcChsI3L3u0+N0opgPYnCRPtGcQYrgXmR+hmt/fYfWAL0xRXYU8eWOdfuLgBe0YCW3AFtnRLagq/g==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-30.3.0.tgz", + "integrity": "sha512-HEtc9uFQgaUHkC7nLSlQL3Tph4Pjxt/yiPvkIrrDCt9jhoLIgxaubo1G+CFOnmHYMxHwwdaSN7mkIFs6ZK8OhA==", "dev": true, "license": "MIT", "dependencies": { - "chalk": "^4.0.0", - "jest-diff": "^29.7.0", - "jest-get-type": "^29.6.3", - "pretty-format": "^29.7.0" + "@jest/get-type": "30.1.0", + "chalk": "^4.1.2", + "jest-diff": "30.3.0", + "pretty-format": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-message-util": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.7.0.tgz", - "integrity": "sha512-GBEV4GRADeP+qtB2+6u61stea8mGcOT4mCtrYISZwfu9/ISHFJ/5zOMXYbpBE9RsS5+Gb63DW4FgmnKJ79Kf6w==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-30.3.0.tgz", + "integrity": "sha512-Z/j4Bo+4ySJ+JPJN3b2Qbl9hDq3VrXmnjjGEWD/x0BCXeOXPTV1iZYYzl2X8c1MaCOL+ewMyNBcm88sboE6YWw==", "dev": true, "license": "MIT", "dependencies": { - "@babel/code-frame": "^7.12.13", - "@jest/types": "^29.6.3", - "@types/stack-utils": "^2.0.0", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.9", - "micromatch": "^4.0.4", - "pretty-format": "^29.7.0", + "@babel/code-frame": "^7.27.1", + "@jest/types": "30.3.0", + "@types/stack-utils": "^2.0.3", + "chalk": "^4.1.2", + "graceful-fs": "^4.2.11", + "picomatch": "^4.0.3", + "pretty-format": "30.3.0", "slash": "^3.0.0", - "stack-utils": "^2.0.3" + "stack-utils": "^2.0.6" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-mock": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-29.7.0.tgz", - "integrity": "sha512-ITOMZn+UkYS4ZFh83xYAOzWStloNzJFO2s8DWrE4lhtGD+AorgnbkiKERe4wQVBydIGPx059g6riW5Btp6Llnw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-30.3.0.tgz", + "integrity": "sha512-OTzICK8CpE+t4ndhKrwlIdbM6Pn8j00lvmSmq5ejiO+KxukbLjgOflKWMn3KE34EZdQm5RqTuKj+5RIEniYhog==", "dev": true, "license": "MIT", "dependencies": { - "@jest/types": "^29.6.3", + "@jest/types": "30.3.0", "@types/node": "*", - "jest-util": "^29.7.0" + "jest-util": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-pnp-resolver": { @@ -4769,183 +5173,239 @@ } }, "node_modules/jest-regex-util": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-29.6.3.tgz", - "integrity": "sha512-KJJBsRCyyLNWCNBOvZyRDnAIfUiRJ8v+hOBQYGn8gDyF3UegwiP4gwRR3/SDa42g1YbVycTidUF3rKjyLFDWbg==", + "version": "30.0.1", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-30.0.1.tgz", + "integrity": "sha512-jHEQgBXAgc+Gh4g0p3bCevgRCVRkB4VB70zhoAE48gxeSr1hfUOsM/C2WoJgVL7Eyg//hudYENbm3Ne+/dRVVA==", "dev": true, "license": "MIT", "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-resolve": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-29.7.0.tgz", - "integrity": "sha512-IOVhZSrg+UvVAshDSDtHyFCCBUl/Q3AAJv8iZ6ZjnZ74xzvwuzLXid9IIIPgTnY62SJjfuupMKZsZQRsCvxEgA==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-30.3.0.tgz", + "integrity": "sha512-NRtTAHQlpd15F9rUR36jqwelbrDV/dY4vzNte3S2kxCKUJRYNd5/6nTSbYiak1VX5g8IoFF23Uj5TURkUW8O5g==", "dev": true, "license": "MIT", "dependencies": { - "chalk": "^4.0.0", - "graceful-fs": "^4.2.9", - "jest-haste-map": "^29.7.0", - "jest-pnp-resolver": "^1.2.2", - "jest-util": "^29.7.0", - "jest-validate": "^29.7.0", - "resolve": "^1.20.0", - "resolve.exports": "^2.0.0", - "slash": "^3.0.0" + "chalk": "^4.1.2", + "graceful-fs": "^4.2.11", + "jest-haste-map": "30.3.0", + "jest-pnp-resolver": "^1.2.3", + "jest-util": "30.3.0", + "jest-validate": "30.3.0", + "slash": "^3.0.0", + "unrs-resolver": "^1.7.11" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-resolve-dependencies": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-29.7.0.tgz", - "integrity": "sha512-un0zD/6qxJ+S0et7WxeI3H5XSe9lTBBR7bOHCHXkKR6luG5mwDDlIzVQ0V5cZCuoTgEdcdwzTghYkTWfubi+nA==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-30.3.0.tgz", + "integrity": "sha512-9ev8s3YN6Hsyz9LV75XUwkCVFlwPbaFn6Wp75qnI0wzAINYWY8Fb3+6y59Rwd3QaS3kKXffHXsZMziMavfz/nw==", "dev": true, "license": "MIT", "dependencies": { - "jest-regex-util": "^29.6.3", - "jest-snapshot": "^29.7.0" + "jest-regex-util": "30.0.1", + "jest-snapshot": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-runner": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-29.7.0.tgz", - "integrity": "sha512-fsc4N6cPCAahybGBfTRcq5wFR6fpLznMg47sY5aDpsoejOcVYFb07AHuSnR0liMcPTgBsA3ZJL6kFOjPdoNipQ==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-30.3.0.tgz", + "integrity": "sha512-gDv6C9LGKWDPLia9TSzZwf4h3kMQCqyTpq+95PODnTRDO0g9os48XIYYkS6D236vjpBir2fF63YmJFtqkS5Duw==", "dev": true, "license": "MIT", "dependencies": { - "@jest/console": "^29.7.0", - "@jest/environment": "^29.7.0", - "@jest/test-result": "^29.7.0", - "@jest/transform": "^29.7.0", - "@jest/types": "^29.6.3", + "@jest/console": "30.3.0", + "@jest/environment": "30.3.0", + "@jest/test-result": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", "@types/node": "*", - "chalk": "^4.0.0", + "chalk": "^4.1.2", "emittery": "^0.13.1", - "graceful-fs": "^4.2.9", - "jest-docblock": "^29.7.0", - "jest-environment-node": "^29.7.0", - "jest-haste-map": "^29.7.0", - "jest-leak-detector": "^29.7.0", - "jest-message-util": "^29.7.0", - "jest-resolve": "^29.7.0", - "jest-runtime": "^29.7.0", - "jest-util": "^29.7.0", - "jest-watcher": "^29.7.0", - "jest-worker": "^29.7.0", + "exit-x": "^0.2.2", + "graceful-fs": "^4.2.11", + "jest-docblock": "30.2.0", + "jest-environment-node": "30.3.0", + "jest-haste-map": "30.3.0", + "jest-leak-detector": "30.3.0", + "jest-message-util": "30.3.0", + "jest-resolve": "30.3.0", + "jest-runtime": "30.3.0", + "jest-util": "30.3.0", + "jest-watcher": "30.3.0", + "jest-worker": "30.3.0", "p-limit": "^3.1.0", "source-map-support": "0.5.13" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-runtime": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-29.7.0.tgz", - "integrity": "sha512-gUnLjgwdGqW7B4LvOIkbKs9WGbn+QLqRQQ9juC6HndeDiezIwhDP+mhMwHWCEcfQ5RUXa6OPnFF8BJh5xegwwQ==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-30.3.0.tgz", + "integrity": "sha512-CgC+hIBJbuh78HEffkhNKcbXAytQViplcl8xupqeIWyKQF50kCQA8J7GeJCkjisC6hpnC9Muf8jV5RdtdFbGng==", "dev": true, "license": "MIT", "dependencies": { - "@jest/environment": "^29.7.0", - "@jest/fake-timers": "^29.7.0", - "@jest/globals": "^29.7.0", - "@jest/source-map": "^29.6.3", - "@jest/test-result": "^29.7.0", - "@jest/transform": "^29.7.0", - "@jest/types": "^29.6.3", + "@jest/environment": "30.3.0", + "@jest/fake-timers": "30.3.0", + "@jest/globals": "30.3.0", + "@jest/source-map": "30.0.1", + "@jest/test-result": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", "@types/node": "*", - "chalk": "^4.0.0", - "cjs-module-lexer": "^1.0.0", - "collect-v8-coverage": "^1.0.0", - "glob": "^7.1.3", - "graceful-fs": "^4.2.9", - "jest-haste-map": "^29.7.0", - "jest-message-util": "^29.7.0", - "jest-mock": "^29.7.0", - "jest-regex-util": "^29.6.3", - "jest-resolve": "^29.7.0", - "jest-snapshot": "^29.7.0", - "jest-util": "^29.7.0", + "chalk": "^4.1.2", + "cjs-module-lexer": "^2.1.0", + "collect-v8-coverage": "^1.0.2", + "glob": "^10.5.0", + "graceful-fs": "^4.2.11", + "jest-haste-map": "30.3.0", + "jest-message-util": "30.3.0", + "jest-mock": "30.3.0", + "jest-regex-util": "30.0.1", + "jest-resolve": "30.3.0", + "jest-snapshot": "30.3.0", + "jest-util": "30.3.0", "slash": "^3.0.0", "strip-bom": "^4.0.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" + } + }, + "node_modules/jest-runtime/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/jest-runtime/node_modules/brace-expansion": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.1.0.tgz", + "integrity": "sha512-TN1kCZAgdgweJhWWpgKYrQaMNHcDULHkWwQIspdtjV4Y5aurRdZpjAqn6yX3FPqTA9ngHCc4hJxMAMgGfve85w==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/jest-runtime/node_modules/glob": { + "version": "10.5.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.5.0.tgz", + "integrity": "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg==", + "deprecated": "Old versions of glob are not supported, and contain widely publicized security vulnerabilities, which have been fixed in the current version. Please update. Support for old versions may be purchased (at exorbitant rates) by contacting i@izs.me", + "dev": true, + "license": "ISC", + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^3.1.2", + "minimatch": "^9.0.4", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^1.11.1" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/jest-runtime/node_modules/minimatch": { + "version": "9.0.9", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.9.tgz", + "integrity": "sha512-OBwBN9AL4dqmETlpS2zasx+vTeWclWzkblfZk7KTA5j3jeOONz/tRCnZomUyvNg83wL5Zv9Ss6HMJXAgL8R2Yg==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.2" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, "node_modules/jest-snapshot": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-29.7.0.tgz", - "integrity": "sha512-Rm0BMWtxBcioHr1/OX5YCP8Uov4riHvKPknOGs804Zg9JGZgmIBkbtlxJC/7Z4msKYVbIJtfU+tKb8xlYNfdkw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/core": "^7.11.6", - "@babel/generator": "^7.7.2", - "@babel/plugin-syntax-jsx": "^7.7.2", - "@babel/plugin-syntax-typescript": "^7.7.2", - "@babel/types": "^7.3.3", - "@jest/expect-utils": "^29.7.0", - "@jest/transform": "^29.7.0", - "@jest/types": "^29.6.3", - "babel-preset-current-node-syntax": "^1.0.0", - "chalk": "^4.0.0", - "expect": "^29.7.0", - "graceful-fs": "^4.2.9", - "jest-diff": "^29.7.0", - "jest-get-type": "^29.6.3", - "jest-matcher-utils": "^29.7.0", - "jest-message-util": "^29.7.0", - "jest-util": "^29.7.0", - "natural-compare": "^1.4.0", - "pretty-format": "^29.7.0", - "semver": "^7.5.3" + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-30.3.0.tgz", + "integrity": "sha512-f14c7atpb4O2DeNhwcvS810Y63wEn8O1HqK/luJ4F6M4NjvxmAKQwBUWjbExUtMxWJQ0wVgmCKymeJK6NZMnfQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.27.4", + "@babel/generator": "^7.27.5", + "@babel/plugin-syntax-jsx": "^7.27.1", + "@babel/plugin-syntax-typescript": "^7.27.1", + "@babel/types": "^7.27.3", + "@jest/expect-utils": "30.3.0", + "@jest/get-type": "30.1.0", + "@jest/snapshot-utils": "30.3.0", + "@jest/transform": "30.3.0", + "@jest/types": "30.3.0", + "babel-preset-current-node-syntax": "^1.2.0", + "chalk": "^4.1.2", + "expect": "30.3.0", + "graceful-fs": "^4.2.11", + "jest-diff": "30.3.0", + "jest-matcher-utils": "30.3.0", + "jest-message-util": "30.3.0", + "jest-util": "30.3.0", + "pretty-format": "30.3.0", + "semver": "^7.7.2", + "synckit": "^0.11.8" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-util": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.7.0.tgz", - "integrity": "sha512-z6EbKajIpqGKU56y5KBUgy1dt1ihhQJgWzUlZHArA/+X2ad7Cb5iF+AK1EWVL/Bo7Rz9uurpqw6SiBCefUbCGA==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-30.3.0.tgz", + "integrity": "sha512-/jZDa00a3Sz7rdyu55NLrQCIrbyIkbBxareejQI315f/i8HjYN+ZWsDLLpoQSiUIEIyZF/R8fDg3BmB8AtHttg==", "dev": true, "license": "MIT", "dependencies": { - "@jest/types": "^29.6.3", + "@jest/types": "30.3.0", "@types/node": "*", - "chalk": "^4.0.0", - "ci-info": "^3.2.0", - "graceful-fs": "^4.2.9", - "picomatch": "^2.2.3" + "chalk": "^4.1.2", + "ci-info": "^4.2.0", + "graceful-fs": "^4.2.11", + "picomatch": "^4.0.3" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-validate": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-29.7.0.tgz", - "integrity": "sha512-ZB7wHqaRGVw/9hST/OuFUReG7M8vKeq0/J2egIGLdvjHCmYqGARhzXmtgi+gVeZ5uXFF219aOc3Ls2yLg27tkw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-30.3.0.tgz", + "integrity": "sha512-I/xzC8h5G+SHCb2P2gWkJYrNiTbeL47KvKeW5EzplkyxzBRBw1ssSHlI/jXec0ukH2q7x2zAWQm7015iusg62Q==", "dev": true, "license": "MIT", "dependencies": { - "@jest/types": "^29.6.3", - "camelcase": "^6.2.0", - "chalk": "^4.0.0", - "jest-get-type": "^29.6.3", + "@jest/get-type": "30.1.0", + "@jest/types": "30.3.0", + "camelcase": "^6.3.0", + "chalk": "^4.1.2", "leven": "^3.1.0", - "pretty-format": "^29.7.0" + "pretty-format": "30.3.0" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-validate/node_modules/camelcase": { @@ -4962,39 +5422,40 @@ } }, "node_modules/jest-watcher": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-29.7.0.tgz", - "integrity": "sha512-49Fg7WXkU3Vl2h6LbLtMQ/HyB6rXSIX7SqvBLQmssRBGN9I0PNvPmAmCWSOY6SOvrjhI/F7/bGAv9RtnsPA03g==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-30.3.0.tgz", + "integrity": "sha512-PJ1d9ThtTR8aMiBWUdcownq9mDdLXsQzJayTk4kmaBRHKvwNQn+ANveuhEBUyNI2hR1TVhvQ8D5kHubbzBHR/w==", "dev": true, "license": "MIT", "dependencies": { - "@jest/test-result": "^29.7.0", - "@jest/types": "^29.6.3", + "@jest/test-result": "30.3.0", + "@jest/types": "30.3.0", "@types/node": "*", - "ansi-escapes": "^4.2.1", - "chalk": "^4.0.0", + "ansi-escapes": "^4.3.2", + "chalk": "^4.1.2", "emittery": "^0.13.1", - "jest-util": "^29.7.0", - "string-length": "^4.0.1" + "jest-util": "30.3.0", + "string-length": "^4.0.2" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-worker": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-29.7.0.tgz", - "integrity": "sha512-eIz2msL/EzL9UFTFFx7jBTkeZfku0yUAyZZZmJ93H2TYEiroIx2PQjEXcwYtYl8zXCxb+PAmA2hLIt/6ZEkPHw==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-30.3.0.tgz", + "integrity": "sha512-DrCKkaQwHexjRUFTmPzs7sHQe0TSj9nvDALKGdwmK5mW9v7j90BudWirKAJHt3QQ9Dhrg1F7DogPzhChppkJpQ==", "dev": true, "license": "MIT", "dependencies": { "@types/node": "*", - "jest-util": "^29.7.0", + "@ungap/structured-clone": "^1.3.0", + "jest-util": "30.3.0", "merge-stream": "^2.0.0", - "supports-color": "^8.0.0" + "supports-color": "^8.1.1" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/jest-worker/node_modules/supports-color": { @@ -5021,9 +5482,9 @@ "license": "MIT" }, "node_modules/js-yaml": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", - "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz", + "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==", "dev": true, "license": "MIT", "dependencies": { @@ -5139,16 +5600,6 @@ "json-buffer": "3.0.1" } }, - "node_modules/kleur": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", - "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, "node_modules/lazystream": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/lazystream/-/lazystream-1.0.1.tgz", @@ -5407,9 +5858,9 @@ } }, "node_modules/markdown-it": { - "version": "14.1.0", - "resolved": "https://registry.npmjs.org/markdown-it/-/markdown-it-14.1.0.tgz", - "integrity": "sha512-a54IwgWPaeBCAAsv13YgmALOF1elABB08FxO9i+r4VFk5Vl4pKokRPeX8u5TCgSsPi6ec1otfLjdOpVcgbpshg==", + "version": "14.1.1", + "resolved": "https://registry.npmjs.org/markdown-it/-/markdown-it-14.1.1.tgz", + "integrity": "sha512-BuU2qnTti9YKgK5N+IeMubp14ZUKUUw7yeJbkjtosvHiP0AZ5c8IAgEMk79D0eC8F23r4Ac/q8cAIFdm2FtyoA==", "dev": true, "license": "MIT", "dependencies": { @@ -5447,30 +5898,6 @@ "dev": true, "license": "MIT" }, - "node_modules/merge2": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", - "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 8" - } - }, - "node_modules/micromatch": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", - "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", - "dev": true, - "license": "MIT", - "dependencies": { - "braces": "^3.0.3", - "picomatch": "^2.3.1" - }, - "engines": { - "node": ">=8.6" - } - }, "node_modules/mime-db": { "version": "1.52.0", "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", @@ -5515,16 +5942,16 @@ } }, "node_modules/minimatch": { - "version": "9.0.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", - "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "version": "10.2.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz", + "integrity": "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg==", "dev": true, - "license": "ISC", + "license": "BlueOak-1.0.0", "dependencies": { - "brace-expansion": "^2.0.1" + "brace-expansion": "^5.0.5" }, "engines": { - "node": ">=16 || 14 >=14.17" + "node": "18 || 20 || >=22" }, "funding": { "url": "https://github.com/sponsors/isaacs" @@ -5540,11 +5967,11 @@ } }, "node_modules/minipass": { - "version": "7.1.2", - "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", - "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.3.tgz", + "integrity": "sha512-tEBHqDnIoM/1rXME1zgka9g6Q2lcoCkxHLuc7ODJ5BxbP5d4c2Z5cGgtXAku59200Cx7diuHTOYfSBD8n6mm8A==", "dev": true, - "license": "ISC", + "license": "BlueOak-1.0.0", "engines": { "node": ">=16 || 14 >=14.17" } @@ -5580,6 +6007,22 @@ "integrity": "sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA==", "license": "MIT" }, + "node_modules/napi-postinstall": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/napi-postinstall/-/napi-postinstall-0.3.4.tgz", + "integrity": "sha512-PHI5f1O0EP5xJ9gQmFGMS6IZcrVvTjpXjz7Na41gTE7eE2hK11lg04CECCYEEjdc17EV4DO+fkGEtt7TpTaTiQ==", + "dev": true, + "license": "MIT", + "bin": { + "napi-postinstall": "lib/cli.js" + }, + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/napi-postinstall" + } + }, "node_modules/natural-compare": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", @@ -5587,10 +6030,17 @@ "dev": true, "license": "MIT" }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true, + "license": "MIT" + }, "node_modules/node-abi": { - "version": "3.85.0", - "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.85.0.tgz", - "integrity": "sha512-zsFhmbkAzwhTft6nd3VxcG0cvJsT70rL+BIGHWVq5fi6MwGrHwzqKaxXE+Hl2GmnGItnDKPPkO5/LQqjVkIdFg==", + "version": "3.89.0", + "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.89.0.tgz", + "integrity": "sha512-6u9UwL0HlAl21+agMN3YAMXcKByMqwGx+pq+P76vii5f7hTPtKDp08/H9py6DY+cfDw7kQNTGEj/rly3IgbNQA==", "license": "MIT", "dependencies": { "semver": "^7.3.5" @@ -5607,9 +6057,9 @@ "license": "MIT" }, "node_modules/node-releases": { - "version": "2.0.19", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.19.tgz", - "integrity": "sha512-xxOWJsBKtzAq7DY0J+DTzuz58K8e7sJbdgwkbMWQe8UYB6ekmsQ45q0M/tJDsGaZmbC+l7n57UV8Hl5tHxO9uw==", + "version": "2.0.37", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.37.tgz", + "integrity": "sha512-1h5gKZCF+pO/o3Iqt5Jp7wc9rH3eJJ0+nh/CIoiRwjRxde/hAHyLPXYN4V3CqKAbiZPSeJFSWHmJsbkicta0Eg==", "dev": true, "license": "MIT" }, @@ -5775,6 +6225,21 @@ "node": ">=8" } }, + "node_modules/path-expression-matcher": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/path-expression-matcher/-/path-expression-matcher-1.5.0.tgz", + "integrity": "sha512-cbrerZV+6rvdQrrD+iGMcZFEiiSrbv9Tfdkvnusy6y0x0GKBXREFg/Y65GhIfm0tnLntThhzCnfKwp1WRjeCyQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/NaturalIntelligence" + } + ], + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, "node_modules/path-is-absolute": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", @@ -5794,13 +6259,6 @@ "node": ">=8" } }, - "node_modules/path-parse": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", - "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", - "dev": true, - "license": "MIT" - }, "node_modules/path-scurry": { "version": "1.11.1", "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz", @@ -5825,16 +6283,6 @@ "dev": true, "license": "ISC" }, - "node_modules/path-type": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", - "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, "node_modules/pend": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz", @@ -5849,13 +6297,13 @@ "license": "ISC" }, "node_modules/picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", + "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", "dev": true, "license": "MIT", "engines": { - "node": ">=8.6" + "node": ">=12" }, "funding": { "url": "https://github.com/sponsors/jonschlinkert" @@ -5958,6 +6406,7 @@ "version": "7.1.3", "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.3.tgz", "integrity": "sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug==", + "deprecated": "No longer maintained. Please contact the author of the relevant native addon; alternatives are available.", "license": "MIT", "dependencies": { "detect-libc": "^2.0.0", @@ -5991,9 +6440,9 @@ } }, "node_modules/prettier": { - "version": "3.6.2", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.6.2.tgz", - "integrity": "sha512-I7AIg5boAr5R0FFtJ6rCfD+LFsWHp81dolrFD8S79U9tb8Az2nGrJncnMSnys+bpQJfRUzqs9hnA81OAA3hCuQ==", + "version": "3.8.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.8.3.tgz", + "integrity": "sha512-7igPTM53cGHMW8xWuVTydi2KO233VFiTNyF5hLJqpilHfmn8C8gPf+PS7dUT64YcXFbiMGZxS9pCSxL/Dxm/Jw==", "dev": true, "license": "MIT", "bin": { @@ -6007,9 +6456,9 @@ } }, "node_modules/prettier-linter-helpers": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz", - "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==", + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.1.tgz", + "integrity": "sha512-SxToR7P8Y2lWmv/kTzVLC1t/GDI2WGjMwNhLLE9qtH8Q13C+aEmuRlzDst4Up4s0Wc8sF2M+J57iB3cMLqftfg==", "dev": true, "license": "MIT", "dependencies": { @@ -6020,18 +6469,18 @@ } }, "node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "version": "30.3.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-30.3.0.tgz", + "integrity": "sha512-oG4T3wCbfeuvljnyAzhBvpN45E8iOTXCU/TD3zXW80HA3dQ4ahdqMkWGiPWZvjpQwlbyHrPTWUAqUzGzv4l1JQ==", "dev": true, "license": "MIT", "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" + "@jest/schemas": "30.0.5", + "ansi-styles": "^5.2.0", + "react-is": "^18.3.1" }, "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + "node": "^18.14.0 || ^20.0.0 || ^22.0.0 || >=24.0.0" } }, "node_modules/pretty-format/node_modules/ansi-styles": { @@ -6053,30 +6502,19 @@ "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", "license": "MIT" }, - "node_modules/prompts": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", - "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", - "dev": true, + "node_modules/proxy-from-env": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-2.1.0.tgz", + "integrity": "sha512-cJ+oHTW1VAEa8cJslgmUZrc+sjRKgAKl3Zyse6+PV38hZe/V6Z14TbCuXcan9F9ghlz4QrFr2c92TNF82UkYHA==", "license": "MIT", - "dependencies": { - "kleur": "^3.0.3", - "sisteransi": "^1.0.5" - }, "engines": { - "node": ">= 6" + "node": ">=10" } }, - "node_modules/proxy-from-env": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", - "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", - "license": "MIT" - }, "node_modules/pump": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.3.tgz", - "integrity": "sha512-todwxLMY7/heScKmntwQG8CXVkWUOdYxIvY2s0VWAAMh/nd8SoYiRaKjlr7+iCs984f2P8zvrfWcDDYVb73NfA==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.4.tgz", + "integrity": "sha512-VS7sjc6KR7e1ukRFhQSY5LM2uBWAUPiOPa/A3mkKmiMwSmRFUITt0xuj+/lesgnCv+dPIEYlkzrcyXgquIHMcA==", "license": "MIT", "dependencies": { "end-of-stream": "^1.1.0", @@ -6104,9 +6542,9 @@ } }, "node_modules/pure-rand": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-7.0.1.tgz", - "integrity": "sha512-oTUZM/NAZS8p7ANR3SHh30kXB+zK2r2BPcEn/awJIbOvq82WoMN4p62AWWp3Hhw50G0xMsw1mhIBLqHw64EcNQ==", + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-8.4.0.tgz", + "integrity": "sha512-IoM8YF/jY0hiugFo/wOWqfmarlE6J0wc6fDK1PhftMk7MGhVZl88sZimmqBBFomLOCSmcCCpsfj7wXASCpvK9A==", "dev": true, "funding": [ { @@ -6195,10 +6633,25 @@ "minimatch": "^5.1.0" } }, + "node_modules/readdir-glob/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "license": "MIT" + }, + "node_modules/readdir-glob/node_modules/brace-expansion": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.1.0.tgz", + "integrity": "sha512-TN1kCZAgdgweJhWWpgKYrQaMNHcDULHkWwQIspdtjV4Y5aurRdZpjAqn6yX3FPqTA9ngHCc4hJxMAMgGfve85w==", + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, "node_modules/readdir-glob/node_modules/minimatch": { - "version": "5.1.6", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", - "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==", + "version": "5.1.9", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.9.tgz", + "integrity": "sha512-7o1wEA2RyMP7Iu7GNba9vc0RWWGACJOCZBJX2GJWip0ikV+wcOsgVuY9uE8CPiyQhkGFSlhuSkZPavN7u1c2Fw==", "license": "ISC", "dependencies": { "brace-expansion": "^2.0.1" @@ -6217,27 +6670,6 @@ "node": ">=0.10.0" } }, - "node_modules/resolve": { - "version": "1.22.10", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", - "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-core-module": "^2.16.0", - "path-parse": "^1.0.7", - "supports-preserve-symlinks-flag": "^1.0.0" - }, - "bin": { - "resolve": "bin/resolve" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/resolve-cwd": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", @@ -6271,16 +6703,6 @@ "node": ">=4" } }, - "node_modules/resolve.exports": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-2.0.3.tgz", - "integrity": "sha512-OcXjMsGdhL4XnbShKpAcSqPMzQoYkYyhbEaeSko47MjRP9NfEQMhZkXL1DoFlt9LWQn4YttrdnV6X2OiyzBi+A==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - } - }, "node_modules/reusify": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", @@ -6293,37 +6715,65 @@ } }, "node_modules/rimraf": { - "version": "5.0.10", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-5.0.10.tgz", - "integrity": "sha512-l0OE8wL34P4nJH/H2ffoaniAokM2qSmrtXHmlpvYr5AVVX8msAyW0l8NVJFDxlSK4u3Uh/f41cQheDVdnYijwQ==", + "version": "6.1.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-6.1.3.tgz", + "integrity": "sha512-LKg+Cr2ZF61fkcaK1UdkH2yEBBKnYjTyWzTJT6KNPcSPaiT7HSdhtMXQuN5wkTX0Xu72KQ1l8S42rlmexS2hSA==", "dev": true, - "license": "ISC", + "license": "BlueOak-1.0.0", "dependencies": { - "glob": "^10.3.7" + "glob": "^13.0.3", + "package-json-from-dist": "^1.0.1" }, "bin": { "rimraf": "dist/esm/bin.mjs" }, + "engines": { + "node": "20 || >=22" + }, "funding": { "url": "https://github.com/sponsors/isaacs" } }, "node_modules/rimraf/node_modules/glob": { - "version": "10.4.5", - "resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz", - "integrity": "sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==", + "version": "13.0.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-13.0.6.tgz", + "integrity": "sha512-Wjlyrolmm8uDpm/ogGyXZXb1Z+Ca2B8NbJwqBVg0axK9GbBeoS7yGV6vjXnYdGm6X53iehEuxxbyiKp8QmN4Vw==", "dev": true, - "license": "ISC", + "license": "BlueOak-1.0.0", "dependencies": { - "foreground-child": "^3.1.0", - "jackspeak": "^3.1.2", - "minimatch": "^9.0.4", - "minipass": "^7.1.2", - "package-json-from-dist": "^1.0.0", - "path-scurry": "^1.11.1" + "minimatch": "^10.2.2", + "minipass": "^7.1.3", + "path-scurry": "^2.0.2" }, - "bin": { - "glob": "dist/esm/bin.mjs" + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rimraf/node_modules/lru-cache": { + "version": "11.3.5", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-11.3.5.tgz", + "integrity": "sha512-NxVFwLAnrd9i7KUBxC4DrUhmgjzOs+1Qm50D3oF1/oL+r1NpZ4gA7xvG0/zJ8evR7zIKn4vLf7qTNduWFtCrRw==", + "dev": true, + "license": "BlueOak-1.0.0", + "engines": { + "node": "20 || >=22" + } + }, + "node_modules/rimraf/node_modules/path-scurry": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-2.0.2.tgz", + "integrity": "sha512-3O/iVVsJAPsOnpwWIeD+d6z/7PmqApyQePUtCndjatj/9I5LylHvt5qluFaBT3I5h3r1ejfR056c+FCv+NnNXg==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "lru-cache": "^11.0.0", + "minipass": "^7.1.2" + }, + "engines": { + "node": "18 || 20 || >=22" }, "funding": { "url": "https://github.com/sponsors/isaacs" @@ -6374,10 +6824,13 @@ "license": "MIT" }, "node_modules/sax": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/sax/-/sax-1.4.3.tgz", - "integrity": "sha512-yqYn1JhPczigF94DMS+shiDMjDowYO6y9+wB/4WgO0Y19jWYk0lQ4tuG5KI7kj4FTp1wxPj5IFfcrz/s1c3jjQ==", - "license": "BlueOak-1.0.0" + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.6.0.tgz", + "integrity": "sha512-6R3J5M4AcbtLUdZmRv2SygeVaM7IhrLXu9BmnOGmmACak8fiUtOsYNWUS4uK7upbmHIBbLBeFeI//477BKLBzA==", + "license": "BlueOak-1.0.0", + "engines": { + "node": ">=11.0.0" + } }, "node_modules/saxes": { "version": "5.0.1", @@ -6392,9 +6845,9 @@ } }, "node_modules/semver": { - "version": "7.7.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.2.tgz", - "integrity": "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==", + "version": "7.7.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.4.tgz", + "integrity": "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==", "license": "ISC", "bin": { "semver": "bin/semver.js" @@ -6484,13 +6937,6 @@ "simple-concat": "^1.0.0" } }, - "node_modules/sisteransi": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", - "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", - "dev": true, - "license": "MIT" - }, "node_modules/slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -6673,9 +7119,9 @@ } }, "node_modules/strnum": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/strnum/-/strnum-2.1.1.tgz", - "integrity": "sha512-7ZvoFTiCnGxBtDqJ//Cu6fWtZtc7Y3x+QOirG15wztbdngGSkht27o2pyGWrVy0b4WAy3jbKmnoK6g5VlVNUUw==", + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/strnum/-/strnum-2.2.3.tgz", + "integrity": "sha512-oKx6RUCuHfT3oyVjtnrmn19H1SiCqgJSg+54XqURKp5aCMbrXrhLjRN9TjuwMjiYstZ0MzDrHqkGZ5dFTKd+zg==", "funding": [ { "type": "github", @@ -6697,23 +7143,10 @@ "node": ">=8" } }, - "node_modules/supports-preserve-symlinks-flag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", - "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/synckit": { - "version": "0.11.11", - "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.11.11.tgz", - "integrity": "sha512-MeQTA1r0litLUf0Rp/iisCaL8761lKAZHaimlbGK4j0HysC4PLfqygQj9srcs0m2RdtDYnF8UuYyKpbjHYp7Jw==", + "version": "0.11.12", + "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.11.12.tgz", + "integrity": "sha512-Bh7QjT8/SuKUIfObSXNHNSK6WHo6J1tHCqJsuaFDP7gP0fkzSfTxI8y85JrppZ0h8l0maIgc2tfuZQ6/t3GtnQ==", "dev": true, "license": "MIT", "dependencies": { @@ -6769,10 +7202,17 @@ "node": ">=8" } }, + "node_modules/test-exclude/node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, "node_modules/test-exclude/node_modules/brace-expansion": { - "version": "1.1.12", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", - "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.14.tgz", + "integrity": "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g==", "dev": true, "license": "MIT", "dependencies": { @@ -6781,9 +7221,9 @@ } }, "node_modules/test-exclude/node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz", + "integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==", "dev": true, "license": "ISC", "dependencies": { @@ -6800,6 +7240,23 @@ "dev": true, "license": "MIT" }, + "node_modules/tinyglobby": { + "version": "0.2.16", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz", + "integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==", + "dev": true, + "license": "MIT", + "dependencies": { + "fdir": "^6.5.0", + "picomatch": "^4.0.4" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" + } + }, "node_modules/tmp": { "version": "0.2.5", "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.5.tgz", @@ -6816,19 +7273,6 @@ "dev": true, "license": "BSD-3-Clause" }, - "node_modules/to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-number": "^7.0.0" - }, - "engines": { - "node": ">=8.0" - } - }, "node_modules/traverse": { "version": "0.3.9", "resolved": "https://registry.npmjs.org/traverse/-/traverse-0.3.9.tgz", @@ -6839,32 +7283,32 @@ } }, "node_modules/ts-api-utils": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.4.3.tgz", - "integrity": "sha512-i3eMG77UTMD0hZhgRS562pv83RC6ukSAC2GMNWc+9dieh/+jDM5u5YG+NHX6VNDRHQcHwmsTHctP9LhbC3WxVw==", + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.5.0.tgz", + "integrity": "sha512-OJ/ibxhPlqrMM0UiNHJ/0CKQkoKF243/AEmplt3qpRgkW8VG7IfOS41h7V8TjITqdByHzrjcS/2si+y4lIh8NA==", "dev": true, "license": "MIT", "engines": { - "node": ">=16" + "node": ">=18.12" }, "peerDependencies": { - "typescript": ">=4.2.0" + "typescript": ">=4.8.4" } }, "node_modules/ts-jest": { - "version": "29.4.0", - "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-29.4.0.tgz", - "integrity": "sha512-d423TJMnJGu80/eSgfQ5w/R+0zFJvdtTxwtF9KzFFunOpSeD+79lHJQIiAhluJoyGRbvj9NZJsl9WjCUo0ND7Q==", + "version": "29.4.9", + "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-29.4.9.tgz", + "integrity": "sha512-LTb9496gYPMCqjeDLdPrKuXtncudeV1yRZnF4Wo5l3SFi0RYEnYRNgMrFIdg+FHvfzjCyQk1cLncWVqiSX+EvQ==", "dev": true, "license": "MIT", "dependencies": { "bs-logger": "^0.2.6", - "ejs": "^3.1.10", "fast-json-stable-stringify": "^2.1.0", + "handlebars": "^4.7.9", "json5": "^2.2.3", "lodash.memoize": "^4.1.2", "make-error": "^1.3.6", - "semver": "^7.7.2", + "semver": "^7.7.4", "type-fest": "^4.41.0", "yargs-parser": "^21.1.1" }, @@ -6881,7 +7325,7 @@ "babel-jest": "^29.0.0 || ^30.0.0", "jest": "^29.0.0 || ^30.0.0", "jest-util": "^29.0.0 || ^30.0.0", - "typescript": ">=4.3 <6" + "typescript": ">=4.3 <7" }, "peerDependenciesMeta": { "@babel/core": { @@ -6917,6 +7361,14 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD", + "optional": true + }, "node_modules/tunnel-agent": { "version": "0.6.0", "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", @@ -6966,17 +7418,17 @@ } }, "node_modules/typedoc": { - "version": "0.28.14", - "resolved": "https://registry.npmjs.org/typedoc/-/typedoc-0.28.14.tgz", - "integrity": "sha512-ftJYPvpVfQvFzpkoSfHLkJybdA/geDJ8BGQt/ZnkkhnBYoYW6lBgPQXu6vqLxO4X75dA55hX8Af847H5KXlEFA==", + "version": "0.28.19", + "resolved": "https://registry.npmjs.org/typedoc/-/typedoc-0.28.19.tgz", + "integrity": "sha512-wKh+lhdmMFivMlc6vRRcMGXeGEHGU2g8a2CkPTJjJlwRf1iXbimWIPcFolCqe4E0d/FRtGszpIrsp3WLpDB8Pw==", "dev": true, "license": "Apache-2.0", "dependencies": { - "@gerrit0/mini-shiki": "^3.12.0", + "@gerrit0/mini-shiki": "^3.23.0", "lunr": "^2.3.9", - "markdown-it": "^14.1.0", - "minimatch": "^9.0.5", - "yaml": "^2.8.1" + "markdown-it": "^14.1.1", + "minimatch": "^10.2.5", + "yaml": "^2.8.3" }, "bin": { "typedoc": "bin/typedoc" @@ -6986,7 +7438,7 @@ "pnpm": ">= 10" }, "peerDependencies": { - "typescript": "5.0.x || 5.1.x || 5.2.x || 5.3.x || 5.4.x || 5.5.x || 5.6.x || 5.7.x || 5.8.x || 5.9.x" + "typescript": "5.0.x || 5.1.x || 5.2.x || 5.3.x || 5.4.x || 5.5.x || 5.6.x || 5.7.x || 5.8.x || 5.9.x || 6.0.x" } }, "node_modules/typescript": { @@ -7010,12 +7462,61 @@ "dev": true, "license": "MIT" }, + "node_modules/uglify-js": { + "version": "3.19.3", + "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.19.3.tgz", + "integrity": "sha512-v3Xu+yuwBXisp6QYTcH4UbH+xYJXqnq2m/LtQVWKWzYc1iehYnLixoQDN9FH6/j9/oybfd6W9Ghwkl8+UMKTKQ==", + "dev": true, + "license": "BSD-2-Clause", + "optional": true, + "bin": { + "uglifyjs": "bin/uglifyjs" + }, + "engines": { + "node": ">=0.8.0" + } + }, "node_modules/undici-types": { "version": "6.21.0", "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz", "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==", "license": "MIT" }, + "node_modules/unrs-resolver": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/unrs-resolver/-/unrs-resolver-1.11.1.tgz", + "integrity": "sha512-bSjt9pjaEBnNiGgc9rUiHGKv5l4/TGzDmYw3RhnkJGtLhbnnA/5qJj7x3dNDCRx/PJxu774LlH8lCOlB4hEfKg==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "dependencies": { + "napi-postinstall": "^0.3.0" + }, + "funding": { + "url": "https://opencollective.com/unrs-resolver" + }, + "optionalDependencies": { + "@unrs/resolver-binding-android-arm-eabi": "1.11.1", + "@unrs/resolver-binding-android-arm64": "1.11.1", + "@unrs/resolver-binding-darwin-arm64": "1.11.1", + "@unrs/resolver-binding-darwin-x64": "1.11.1", + "@unrs/resolver-binding-freebsd-x64": "1.11.1", + "@unrs/resolver-binding-linux-arm-gnueabihf": "1.11.1", + "@unrs/resolver-binding-linux-arm-musleabihf": "1.11.1", + "@unrs/resolver-binding-linux-arm64-gnu": "1.11.1", + "@unrs/resolver-binding-linux-arm64-musl": "1.11.1", + "@unrs/resolver-binding-linux-ppc64-gnu": "1.11.1", + "@unrs/resolver-binding-linux-riscv64-gnu": "1.11.1", + "@unrs/resolver-binding-linux-riscv64-musl": "1.11.1", + "@unrs/resolver-binding-linux-s390x-gnu": "1.11.1", + "@unrs/resolver-binding-linux-x64-gnu": "1.11.1", + "@unrs/resolver-binding-linux-x64-musl": "1.11.1", + "@unrs/resolver-binding-wasm32-wasi": "1.11.1", + "@unrs/resolver-binding-win32-arm64-msvc": "1.11.1", + "@unrs/resolver-binding-win32-ia32-msvc": "1.11.1", + "@unrs/resolver-binding-win32-x64-msvc": "1.11.1" + } + }, "node_modules/unzipper": { "version": "0.10.14", "resolved": "https://registry.npmjs.org/unzipper/-/unzipper-0.10.14.tgz", @@ -7065,9 +7566,9 @@ } }, "node_modules/update-browserslist-db": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.3.tgz", - "integrity": "sha512-UxhIZQ+QInVdunkDAaiazvvT/+fXL5Osr0JZlJulepYu6Jd7qJtDZjlur0emRlT71EN3ScPoE7gvsuIKKNavKw==", + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz", + "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==", "dev": true, "funding": [ { @@ -7171,6 +7672,13 @@ "node": ">=0.10.0" } }, + "node_modules/wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==", + "dev": true, + "license": "MIT" + }, "node_modules/wrap-ansi": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", @@ -7215,17 +7723,30 @@ "license": "ISC" }, "node_modules/write-file-atomic": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-4.0.2.tgz", - "integrity": "sha512-7KxauUdBmSdWnmpaGFg+ppNjKF8uNLry8LyzjauQDOVONfFLNKrKvQOxZ/VuTIcS/gge/YNahf5RIIQWTSarlg==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-5.0.1.tgz", + "integrity": "sha512-+QU2zd6OTD8XWIJCbffaiQeH9U73qIqafo1x6V1snCWYGJf6cVE0cDR4D8xRzcEnfI21IFrUPzPGtcPf8AC+Rw==", "dev": true, "license": "ISC", "dependencies": { "imurmurhash": "^0.1.4", - "signal-exit": "^3.0.7" + "signal-exit": "^4.0.1" }, "engines": { - "node": "^12.13.0 || ^14.15.0 || >=16.0.0" + "node": "^14.17.0 || ^16.13.0 || >=18.0.0" + } + }, + "node_modules/write-file-atomic/node_modules/signal-exit": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", + "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, "node_modules/xml2js": { @@ -7283,9 +7804,9 @@ "license": "ISC" }, "node_modules/yaml": { - "version": "2.8.1", - "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.8.1.tgz", - "integrity": "sha512-lcYcMxX2PO9XMGvAJkJ3OsNMw+/7FKes7/hgerGUYWIoWu5j/+YQqcZr5JnPZWzOsEBgMbSbiSTn/dv/69Mkpw==", + "version": "2.8.3", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.8.3.tgz", + "integrity": "sha512-AvbaCLOO2Otw/lW5bmh9d/WEdcDFdQp2Z2ZUH3pX9U2ihyUY0nvLv7J6TrWowklRGPYbB/IuIMfYgxaCPg5Bpg==", "dev": true, "license": "ISC", "bin": { @@ -7293,6 +7814,9 @@ }, "engines": { "node": ">= 14.6" + }, + "funding": { + "url": "https://github.com/sponsors/eemeli" } }, "node_modules/yargs": { @@ -7325,9 +7849,9 @@ } }, "node_modules/yauzl": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-3.2.0.tgz", - "integrity": "sha512-Ow9nuGZE+qp1u4JIPvg+uCiUr7xGQWdff7JQSk5VGYTAZMDe2q8lxJ10ygv10qmSj031Ty/6FNJpLO4o1Sgc+w==", + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-3.3.0.tgz", + "integrity": "sha512-PtGEvEP30p7sbIBJKUBjUnqgTVOyMURc4dLo9iNyAJnNIEz9pm88cCXF21w94Kg3k6RXkeZh5DHOGS0qEONvNQ==", "license": "MIT", "dependencies": { "buffer-crc32": "~0.2.3", diff --git a/package.json b/package.json index 2c4bf43..08a92a3 100644 --- a/package.json +++ b/package.json @@ -154,20 +154,21 @@ "@types/adm-zip": "^0.5.7", "@types/better-sqlite3": "^7.6.13", "@types/exceljs": "^0.5.3", - "@types/jest": "^29.5.12", - "@types/node": "^20.11.24", + "@types/jest": "^30.0.0", + "@types/node": "^22.19.17", + "@types/pako": "^2.0.4", "@types/plist": "^3.0.5", "@types/sql.js": "^1.4.9", - "@typescript-eslint/eslint-plugin": "^7.1.0", - "@typescript-eslint/parser": "^7.1.0", - "eslint": "^8.56.0", - "eslint-config-prettier": "^9.1.0", + "@typescript-eslint/eslint-plugin": "^8.58.2", + "@typescript-eslint/parser": "^8.58.2", + "eslint": "^8.57.1", + "eslint-config-prettier": "^10.1.8", "eslint-plugin-prettier": "^5.1.3", "fast-check": "^4.2.0", - "jest": "^29.7.0", + "jest": "^30.3.0", "prettier": "^3.5.3", - "rimraf": "^5.0.5", - "ts-jest": "^29.1.2", + "rimraf": "^6.1.3", + "ts-jest": "^29.4.9", "typedoc": "^0.28.14", "typescript": "~5.5.0" }, @@ -176,7 +177,7 @@ "adm-zip": "^0.5.16", "axios": "^1.11.0", "better-sqlite3": "^12.5.0", - "commander": "^13.1.0", + "commander": "^14.0.3", "exceljs": "^4.4.0", "fast-xml-parser": "^5.2.0", "jszip": "^3.10.1", diff --git a/src/core/treeStructure.ts b/src/core/treeStructure.ts index e556525..a346ce5 100644 --- a/src/core/treeStructure.ts +++ b/src/core/treeStructure.ts @@ -193,7 +193,15 @@ export class AACButton { parameters?: { [key: string]: any }; // Predictions: Array of predicted word forms for smart grammar predictions?: string[]; - // Metrics support: Motor planning identifiers + // Part-of-speech tag from gridset (e.g., 'Verb', 'Noun', 'Pronoun', 'Unknown') + pos?: string; + wordForms?: Array<{ + lang?: string; + tags: string[]; + value: string; + pronunciation?: string; + base?: string; + }>; semantic_id?: string; // Unique ID for buttons with same semantic meaning across boards clone_id?: string; // Unique ID for buttons with same label+location across boards @@ -221,6 +229,8 @@ export class AACButton { directActivate, parameters, predictions, + pos, + wordForms, semantic_id, clone_id, // Legacy input support @@ -255,6 +265,14 @@ export class AACButton { directActivate?: boolean; parameters?: { [key: string]: any }; predictions?: string[]; + pos?: string; + wordForms?: Array<{ + lang?: string; + tags: string[]; + value: string; + pronunciation?: string; + base?: string; + }>; semantic_id?: string; clone_id?: string; // Legacy constructor properties for backward compatibility @@ -288,6 +306,8 @@ export class AACButton { this.directActivate = directActivate; this.parameters = parameters; this.predictions = predictions; + this.pos = pos; + this.wordForms = wordForms; this.semantic_id = semantic_id; this.clone_id = clone_id; diff --git a/src/index.node.ts b/src/index.node.ts index e3477a8..1c32b8f 100644 --- a/src/index.node.ts +++ b/src/index.node.ts @@ -31,6 +31,10 @@ export * as Validation from './validation'; // Metrics namespace (pageset analytics) export * as Metrics from './metrics'; +// Node-only morphology utilities (Grid 3 verbs parser) +export { Grid3VerbsParser } from './utilities/analytics/morphology/grid3VerbsParser'; +export { WordFormGenerator } from './utilities/analytics/morphology/wordFormGenerator'; + // Processor namespaces (platform-specific utilities) export * as Gridset from './gridset'; export * as Snap from './snap'; diff --git a/src/metrics.ts b/src/metrics.ts index 819d81b..5ae2bc5 100644 --- a/src/metrics.ts +++ b/src/metrics.ts @@ -13,6 +13,17 @@ export { MetricsCalculator } from './utilities/analytics/metrics/core'; export { VocabularyAnalyzer } from './utilities/analytics/metrics/vocabulary'; export { SentenceAnalyzer } from './utilities/analytics/metrics/sentence'; export { ComparisonAnalyzer } from './utilities/analytics/metrics/comparison'; +export { MorphologyEngine } from './utilities/analytics/morphology'; +export { WordFormGenerator } from './utilities/analytics/morphology'; +export type { + MorphRuleSet, + MorphRule, + MorphWordForms, + Grid3VerbForms, + AstericsWordForm, + VerbFormWithConditions, + Grid3VerbFormsDetailed, +} from './utilities/analytics/morphology'; export { ReferenceLoader } from './utilities/analytics/reference'; export { InMemoryReferenceLoader, diff --git a/src/processors/astericsGridProcessor.ts b/src/processors/astericsGridProcessor.ts index 44c7994..e925210 100644 --- a/src/processors/astericsGridProcessor.ts +++ b/src/processors/astericsGridProcessor.ts @@ -790,7 +790,7 @@ class AstericsGridProcessor extends BaseProcessor { }); }); }); - } catch (error) { + } catch (_error) { // If JSON parsing fails, return empty array } @@ -1269,7 +1269,7 @@ class AstericsGridProcessor extends BaseProcessor { // Use detected format for filename imageName = element.image.id || `image.${imageFormat}`; - } catch (e) { + } catch (_e) { // Invalid base64 data, skip image } } @@ -1285,19 +1285,25 @@ class AstericsGridProcessor extends BaseProcessor { audioRecording: audioRecording, visibility: mapAstericsVisibility(element.hidden), image: imageName, // Store image filename/reference - parameters: imageData - ? { - ...{ imageData: imageData }, // Store actual image data in parameters for conversion - } - : undefined, style: { backgroundColor: finalBackgroundColor, borderColor: colorStyles.borderColor || colorConfig?.elementBorderColor || '#CCCCCC', borderWidth: colorConfig?.borderWidth || 1, fontFamily: colorConfig?.fontFamily || 'Arial', - fontSize: colorConfig?.fontSizePct ? colorConfig.fontSizePct * 16 : 16, // Default to 16px + fontSize: colorConfig?.fontSizePct ? colorConfig.fontSizePct * 16 : 16, fontColor: fontColor, }, + wordForms: element.wordForms && element.wordForms.length > 0 ? element.wordForms : undefined, + parameters: { + ...(imageData ? { imageData: imageData } : {}), + ...(element.actions?.some((a: GridAction) => a.modelName === 'GridActionWordForm') + ? { + wordFormActions: element.actions.filter( + (a: GridAction) => a.modelName === 'GridActionWordForm' + ), + } + : {}), + }, }); } @@ -1584,6 +1590,14 @@ class AstericsGridProcessor extends BaseProcessor { const locale = tree.metadata?.locale || 'en'; + if ( + button.parameters?.wordFormActions && + Array.isArray(button.parameters.wordFormActions) + ) { + // eslint-disable-next-line @typescript-eslint/no-unsafe-argument + actions.push(...button.parameters.wordFormActions); + } + return { id: button.id, modelName: 'GridElement', @@ -1593,7 +1607,7 @@ class AstericsGridProcessor extends BaseProcessor { x: calculatedX, y: calculatedY, label: { [locale]: button.label }, - wordForms: [], + wordForms: button.wordForms || [], image: { data: null, author: undefined, @@ -1702,7 +1716,7 @@ class AstericsGridProcessor extends BaseProcessor { audioAction.mimeType = parsedMetadata.mimeType || audioAction.mimeType; audioAction.durationMs = parsedMetadata.durationMs || audioAction.durationMs; audioAction.filename = parsedMetadata.filename || audioAction.filename; - } catch (e) { + } catch (_e) { // Use defaults if metadata parsing fails } } @@ -1773,7 +1787,7 @@ class AstericsGridProcessor extends BaseProcessor { elementIds.push(element.id); }); }); - } catch (error) { + } catch (_error) { // If JSON parsing fails, return empty array } @@ -1802,7 +1816,7 @@ class AstericsGridProcessor extends BaseProcessor { } } } - } catch (error) { + } catch (_error) { // If JSON parsing fails, return false } diff --git a/src/processors/excelProcessor.ts b/src/processors/excelProcessor.ts index 91431c8..923ee74 100644 --- a/src/processors/excelProcessor.ts +++ b/src/processors/excelProcessor.ts @@ -356,7 +356,12 @@ export class ExcelProcessor extends BaseProcessor { // Create internal link to another worksheet const sanitizedTargetName = this.sanitizeWorksheetName(targetPageId); cell.value = { - text: cell.value?.toString() || '', + text: + typeof cell.value === 'string' + ? cell.value + : typeof cell.value === 'number' || typeof cell.value === 'boolean' + ? String(cell.value) + : '', hyperlink: `#'${sanitizedTargetName}'!A1`, }; } diff --git a/src/processors/gridset/helpers.ts b/src/processors/gridset/helpers.ts index 3c44ea0..9f00bda 100644 --- a/src/processors/gridset/helpers.ts +++ b/src/processors/gridset/helpers.ts @@ -84,7 +84,7 @@ export async function openImage( return Buffer.from(data); } return data; - } catch (error) { + } catch (_error) { return null; } } @@ -226,7 +226,7 @@ export function getCommonDocumentsPath(): string { if (match && match[1]) { return match[1].trim(); } - } catch (error) { + } catch (_error) { // Registry access failed, fall back to default } @@ -293,7 +293,7 @@ export async function findGrid3UserPaths( } } } - } catch (error) { + } catch (_error) { // Silently fail if directory access fails } diff --git a/src/processors/gridset/imageDebug.ts b/src/processors/gridset/imageDebug.ts index ab1fd69..1781859 100644 --- a/src/processors/gridset/imageDebug.ts +++ b/src/processors/gridset/imageDebug.ts @@ -117,7 +117,7 @@ export async function auditGridsetImages( dynamicFilesMap.set(staticFile, list); } } - } catch (e) { + } catch (_e) { // FileMap parsing failed, continue without it } } @@ -206,7 +206,7 @@ export async function auditGridsetImages( }); } } - } catch (e) { + } catch (_e) { // Skip grids that can't be processed continue; } diff --git a/src/processors/gridsetProcessor.ts b/src/processors/gridsetProcessor.ts index 50fcf83..1d337ca 100644 --- a/src/processors/gridsetProcessor.ts +++ b/src/processors/gridsetProcessor.ts @@ -508,7 +508,7 @@ class GridsetProcessor extends BaseProcessor { } } } - } catch (e) { + } catch (_e) { /* ignore: optional FileMap.xml may be missing or malformed */ } @@ -588,7 +588,7 @@ class GridsetProcessor extends BaseProcessor { : Buffer.from(raw); const normalizedEntry = imageEntry.entryName.replace(/\\/g, '/'); imageDataCache.set(normalizedEntry, data); - } catch (err) { + } catch (_err) { // Silently fail - individual image loading failures shouldn't break the entire load } } @@ -625,7 +625,7 @@ class GridsetProcessor extends BaseProcessor { } } } - } catch (e) { + } catch (_e) { // Skip errors in first pass } } @@ -643,7 +643,7 @@ class GridsetProcessor extends BaseProcessor { `[Gridset] Raw XML content (first 200 chars) for ${entry.entryName}:`, xmlContent.substring(0, 200) ); - } catch (e) { + } catch (_e) { // Skip unreadable files continue; } @@ -1076,6 +1076,7 @@ class GridsetProcessor extends BaseProcessor { // infer action type implicitly from commands; no explicit enum needed let navigationTarget: string | undefined; let detectedCommands: any[] = []; // Store detected command metadata + let buttonPos: string | undefined; // Part-of-speech from Action.InsertText const commands = content.Commands?.Command || content.commands?.command; let predictionWords: string[] | undefined; @@ -1386,8 +1387,11 @@ class GridsetProcessor extends BaseProcessor { } case 'Action.InsertText': { - // speak const insertText = getParam('text'); + const posParam = getParam('pos'); + if (posParam) { + buttonPos = posParam; + } semanticAction = { category: AACSemanticCategory.COMMUNICATION, intent: AACSemanticIntent.INSERT_TEXT, @@ -1395,7 +1399,7 @@ class GridsetProcessor extends BaseProcessor { platformData: { grid3: { commandId, - parameters: { text: insertText }, + parameters: { text: insertText, pos: posParam }, }, }, fallback: { @@ -1626,13 +1630,15 @@ class GridsetProcessor extends BaseProcessor { // Extract grammar tags from commands (Smart Grammar) const grammar: Record = {}; + if (buttonPos) grammar.pos = buttonPos; detectedCommands.forEach((cmd) => { - if (cmd.parameters.pos) grammar.pos = cmd.parameters.pos; + if (!grammar.pos && cmd.parameters.pos) grammar.pos = cmd.parameters.pos; if (cmd.parameters.person) grammar.person = cmd.parameters.person; if (cmd.parameters.number) grammar.number = cmd.parameters.number; if (cmd.parameters.feature) grammar.feature = cmd.parameters.feature; }); const isSmartGrammarCell = Object.keys(grammar).length > 0; + const effectivePos = buttonPos || grammar.pos || undefined; const button = new AACButton({ id: `${gridId}_btn_${idx}`, @@ -1673,6 +1679,7 @@ class GridsetProcessor extends BaseProcessor { : gridPredictionWords.length > 0 ? [...gridPredictionWords] : undefined, + pos: effectivePos, parameters: { pluginMetadata: pluginMetadata, // Store full plugin metadata for future use grid3Commands: detectedCommands, // Store detected command metadata @@ -1874,7 +1881,7 @@ class GridsetProcessor extends BaseProcessor { metadata.defaultKeyboardPageId = gridNameToIdMap.get(keyboardGridName); } } - } catch (e) { + } catch (_e) { // If settings.xml parsing fails, tree.rootId will default to first page } diff --git a/src/processors/obfProcessor.ts b/src/processors/obfProcessor.ts index 96f8441..e38bd5e 100644 --- a/src/processors/obfProcessor.ts +++ b/src/processors/obfProcessor.ts @@ -121,7 +121,7 @@ class ObfProcessor extends BaseProcessor { } return null; } - } catch (err) { + } catch (_err) { continue; } } @@ -173,7 +173,7 @@ class ObfProcessor extends BaseProcessor { this.imageCache.set(imageId, dataUrl); return dataUrl; } - } catch (err) { + } catch (_err) { // Continue to next path continue; } @@ -431,7 +431,7 @@ class ObfProcessor extends BaseProcessor { } return obj as ObfBoard; } - } catch (error: any) { + } catch (_error: any) { // Log parsing errors for debugging but don't throw } return null; @@ -899,7 +899,7 @@ class ObfProcessor extends BaseProcessor { try { // eslint-disable-next-line @typescript-eslint/no-var-requires, @typescript-eslint/no-unsafe-return return require('../validation/obfValidator').ObfValidator; - } catch (error) { + } catch (_error) { throw new Error('Validation utilities are not available in this environment.'); } } diff --git a/src/processors/snap/helpers.ts b/src/processors/snap/helpers.ts index 9995ba7..af3957e 100644 --- a/src/processors/snap/helpers.ts +++ b/src/processors/snap/helpers.ts @@ -41,7 +41,7 @@ async function collectFiles( let entries: string[]; try { entries = await listDir(current.dir); - } catch (error) { + } catch (_error) { continue; } @@ -153,7 +153,7 @@ export async function openImage( await removePath(dbPath); const dir = dirname(dbPath); await removePath(dir); - } catch (e) { + } catch (_e) { // Ignore cleanup errors } } @@ -240,7 +240,7 @@ export async function findSnapPackages( }); } } - } catch (error) { + } catch (_error) { // Silently fail if directory access fails } diff --git a/src/processors/snapProcessor.ts b/src/processors/snapProcessor.ts index 3d380bb..15e021e 100644 --- a/src/processors/snapProcessor.ts +++ b/src/processors/snapProcessor.ts @@ -302,7 +302,7 @@ class SnapProcessor extends BaseProcessor { let positions: Array<{ Column: number; Row: number }> = []; try { positions = JSON.parse(sg.SerializedGridPositions as string); - } catch (e) { + } catch (_e) { // Invalid JSON, skip this group return; } @@ -539,7 +539,7 @@ class SnapProcessor extends BaseProcessor { targetPageUniqueId = dashboardPageId; } } - } catch (e) { + } catch (_e) { // Ignore JSON parse errors in commands } } diff --git a/src/processors/touchchatProcessor.ts b/src/processors/touchchatProcessor.ts index 9d0dca5..3cc772a 100644 --- a/src/processors/touchchatProcessor.ts +++ b/src/processors/touchchatProcessor.ts @@ -190,7 +190,7 @@ class TouchChatProcessor extends BaseProcessor { mappings.forEach((mapping) => { idMappings.set(mapping.numeric_id, mapping.string_id); }); - } catch (e) { + } catch (_e) { // No mapping table, use numeric IDs as strings } @@ -208,7 +208,7 @@ class TouchChatProcessor extends BaseProcessor { pageStyleRows.forEach((style) => { pageStyles.set(style.id, style); }); - } catch (e) { + } catch (_e) { // console.log('No styles found:', e); } @@ -451,7 +451,7 @@ class TouchChatProcessor extends BaseProcessor { } } }); - } catch (e) { + } catch (_e) { // console.log('No button box cells found:', e); } @@ -514,7 +514,7 @@ class TouchChatProcessor extends BaseProcessor { ); if (page) page.addButton(button); }); - } catch (e) { + } catch (_e) { // console.log('No direct page buttons found:', e); } @@ -568,7 +568,7 @@ class TouchChatProcessor extends BaseProcessor { } } }); - } catch (e) { + } catch (_e) { // console.log('No navigation actions found:', e); } @@ -618,7 +618,7 @@ class TouchChatProcessor extends BaseProcessor { tree.rootId = rootPageId; tree.metadata.defaultHomePageId = rootPageId; } - } catch (e) { + } catch (_e) { // No metadata table or other error, use first page as root if (rootPageId) { tree.rootId = rootPageId; diff --git a/src/utilities/analytics/docs/AAC_METRICS_GUIDE.md b/src/utilities/analytics/docs/AAC_METRICS_GUIDE.md index d5f8c60..740b52b 100644 --- a/src/utilities/analytics/docs/AAC_METRICS_GUIDE.md +++ b/src/utilities/analytics/docs/AAC_METRICS_GUIDE.md @@ -25,6 +25,7 @@ The algorithm rewards systems that support motor planning. If a button appears i ### Switch Scanning Support For users who use scanning access methods, the implementation supports evaluating scanning effort. Instead of movement distance, the algorithm calculates: + 1. **Scan Steps**: The number of highlight movements (items or groups) to reach the target. 2. **Selections**: The number of switch activations required. @@ -41,38 +42,42 @@ Supported scanning types include `linear`, `row-column`, `column-row`, and `bloc The `MetricsCalculator` is format-agnostic. You can use any processor to load a pageset into an `AACTree`, then pass that tree to the calculator. #### Loading an OBFSET + ```typescript -import { ObfsetProcessor, Analytics } from '@willwade/aac-processors'; +import { ObfsetProcessor, Analytics } from "@willwade/aac-processors"; const processor = new ObfsetProcessor(); -const tree = await processor.loadIntoTree('set.obfset'); +const tree = await processor.loadIntoTree("set.obfset"); const result = new Analytics.MetricsCalculator().analyze(tree); ``` #### Loading Grid 3 (.gridset) + ```typescript -import { GridsetProcessor, Analytics } from '@willwade/aac-processors'; +import { GridsetProcessor, Analytics } from "@willwade/aac-processors"; const processor = new GridsetProcessor(); -const tree = await processor.loadIntoTree('my_file.gridset'); +const tree = await processor.loadIntoTree("my_file.gridset"); const result = new Analytics.MetricsCalculator().analyze(tree); ``` #### Loading Snap (.pageSet / .spb) + ```typescript -import { SnapProcessor, Analytics } from '@willwade/aac-processors'; +import { SnapProcessor, Analytics } from "@willwade/aac-processors"; const processor = new SnapProcessor(); -const tree = await processor.loadIntoTree('SnapBackup.pageSet'); +const tree = await processor.loadIntoTree("SnapBackup.pageSet"); const result = new Analytics.MetricsCalculator().analyze(tree); ``` #### Loading TouchChat (.zip) + ```typescript -import { TouchChatProcessor, Analytics } from '@willwade/aac-processors'; +import { TouchChatProcessor, Analytics } from "@willwade/aac-processors"; const processor = new TouchChatProcessor(); -const tree = await processor.loadIntoTree('TouchChatBackup.zip'); +const tree = await processor.loadIntoTree("TouchChatBackup.zip"); const result = new Analytics.MetricsCalculator().analyze(tree); ``` @@ -95,25 +100,26 @@ The `analyze` function returns: You can evaluate scanning efficiency by setting the `scanType` on `AACPage` objects: ```typescript -import { AACScanType } from '@willwade/aac-processors'; +import { AACScanType } from "@willwade/aac-processors"; // Set scanning behavior for a page -page.scanType = AACScanType.ROW_COLUMN; +page.scanType = AACScanType.ROW_COLUMN; // Or using blocks page.scanType = AACScanType.BLOCK_ROW_COLUMN; page.scanBlocksConfig = [ - { id: 1, name: 'Main', order: 1 }, - { id: 2, name: 'Numbers', order: 2 } + { id: 1, name: "Main", order: 1 }, + { id: 2, name: "Numbers", order: 2 }, ]; // Assign buttons to blocks -button.scanBlocks = [2]; +button.scanBlocks = [2]; ``` ### 5. Unified History & OBL Support This implementation supports gathering and analyzing historical usage data from multiple sources: + - **Local History**: Automatically discover and read logs from Grid 3 and TD Snap on the local machine. - **OBL Format**: Support for the [Open Board Logging (OBL)](./OBL_SUPPORT_GUIDE.md) standard for cross-platform log sharing. @@ -151,3 +157,69 @@ This version is optimized for Node.js and adheres strictly to the **v0.2 Algorit - **Improved Logic**: Resolves "double-discounting" issues present in legacy implementations, favoring a more realistic motor-planning model. For detailed technical notes on implementation decisions, see [ALGORITHM_IMPLEMENTATION_NOTES.md](./ALGORITHM_IMPLEMENTATION_NOTES.md). + +--- + +## 🔤 Smart Grammar / Morphological Expansion + +Grid 3's Super Core vocabulary sets tag word buttons with a `pos` (part-of-speech) parameter (e.g., `Verb`, `Pronoun`). At runtime, Grid 3 uses this to offer morphological inflections — pressing "go" (tagged `Verb`) offers "going", "goes", "gone", "went" as one-tap predictions. + +### How it works in `aac-processors` + +1. **POS extraction** — The GridsetProcessor extracts the `pos` parameter from `Action.InsertText` commands and stores it as `button.pos`. + +2. **Auto-detection** — `MetricsCalculator.analyze()` automatically detects whether any button has POS tags. For Grid 3 trees, smart grammar is enabled automatically. For TD Snap, TouchChat, and OBF trees, it's automatically disabled with zero overhead. + +3. **Morphological expansion** — The `MorphologyEngine` generates inflected word forms from POS tags using: + - Built-in English rules (~120 irregular verbs, 90+ irregular nouns, pronouns, adjectives, plus regex-based regular rules) + - Grid 3's native verb data via `Grid3VerbsParser` (reads `verbs.zip` from Grid 3's locale directories for 24+ languages) + +4. **Word form metrics** — Generated forms are assigned effort scores and included in vocabulary coverage analysis. + +### API + +```typescript +import { + Metrics, + Grid3VerbsParser, + WordFormGenerator, +} from "@willwade/aac-processors"; + +// MorphologyEngine with built-in English rules +const engine = new Metrics.MorphologyEngine("en-gb"); +engine.inflect("go", "Verb"); // → ['goes', 'going', 'gone', 'went'] + +// Grid 3 verbs parser (reads from Grid 3 installation) +const parser = new Grid3VerbsParser(); +const enVerbs = parser.parseZip("C:\\...\\Locale\\en-GB\\verbs\\verbs.zip"); + +// Word form generator (bridges POS → AsTeRICS Grid tag format) +const generator = new Metrics.WordFormGenerator(); +generator.generateFromEngineSlots("go", "Verb", engine, "en"); +// → [{tags:['BASE'], value:'go'}, {tags:['3.PERS'], value:'goes'}, ...] +``` + +### Multi-language support + +The `Grid3VerbsParser` reads Grid 3's native morphology data for 24+ locales including English, Norwegian, Swedish, German, French, Spanish, and more. Each language has its own conjugation rule sets stored in `verbs.zip`. + +```typescript +// Parse any installed locale +const nbVerbs = await parser.parseLocale("nb-NO"); +const svVerbs = await parser.parseLocale("sv-SE"); + +// Or scan all installed locales +const all = await parser.parseInstalledLocales(); +``` + +### AsTeRICS Grid word forms + +AsTeRICS Grid (`.grd` files) stores word forms per-element with tagged tuples: + +```json +{ "value": "were", "tags": ["2.PERS", "PAST"], "lang": "en" } +``` + +The `WordFormGenerator` converts Grid 3 POS data to this format for cross-platform conversion. The Asterics Grid processor now preserves word forms on import and writes them on export — enabling `.gridset` → `.grd` conversion with morphology intact. + +**Note:** Word forms on individual buttons are generated automatically, but grammar control buttons (e.g., "You" → `2.PERS`, "Past" → `PAST`) are vocabulary layout decisions that are not auto-generated. diff --git a/src/utilities/analytics/docs/VOCABULARY_ANALYSIS_GUIDE.md b/src/utilities/analytics/docs/VOCABULARY_ANALYSIS_GUIDE.md index 27ddd1e..40cc71e 100644 --- a/src/utilities/analytics/docs/VOCABULARY_ANALYSIS_GUIDE.md +++ b/src/utilities/analytics/docs/VOCABULARY_ANALYSIS_GUIDE.md @@ -20,11 +20,11 @@ Analyzes how well a board set covers core vocabulary lists and identifies gaps. **Usage:** ```typescript -import { ObfsetProcessor, Analytics } from '@willwade/aac-processors'; +import { ObfsetProcessor, Analytics } from "@willwade/aac-processors"; // Load board set const processor = new ObfsetProcessor(); -const tree = await processor.loadIntoTree('path/to/boardset.obfset'); +const tree = await processor.loadIntoTree("path/to/boardset.obfset"); // Calculate metrics const metrics = new Analytics.MetricsCalculator().analyze(tree); @@ -75,7 +75,7 @@ Calculates the effort required to construct test sentences from the board set. **Usage:** ```typescript -import { Analytics } from '@willwade/aac-processors'; +import { Analytics } from "@willwade/aac-processors"; const sentenceAnalyzer = new Analytics.SentenceAnalyzer(); const refLoader = new Analytics.ReferenceLoader(); @@ -124,7 +124,7 @@ Compares two board sets to identify differences and generate CARE component scor **Usage:** ```typescript -import { Analytics } from '@willwade/aac-processors'; +import { Analytics } from "@willwade/aac-processors"; // Load two board sets const targetResult = calculator.analyze(targetTree); @@ -132,9 +132,13 @@ const compareResult = calculator.analyze(compareTree); // Compare const comparisonAnalyzer = new Analytics.ComparisonAnalyzer(); -const comparison = await comparisonAnalyzer.compare(targetResult, compareResult, { - includeSentences: true, -}); +const comparison = await comparisonAnalyzer.compare( + targetResult, + compareResult, + { + includeSentences: true, + }, +); console.log("Missing words:", comparison.missing_words); console.log("CARE components:", comparison.care_components); @@ -227,10 +231,10 @@ npx ts-node test-comparison-analysis.ts All processors can now use these analysis features: ```typescript -import { ObfProcessor, Analytics } from '@willwade/aac-processors'; +import { ObfProcessor, Analytics } from "@willwade/aac-processors"; const processor = new ObfProcessor(); -const tree = await processor.loadIntoTree('my-board.obf'); +const tree = await processor.loadIntoTree("my-board.obf"); const metrics = new Analytics.MetricsCalculator().analyze(tree); @@ -285,3 +289,41 @@ aac-processors coverage my-boardset.obf --core-lists default,unc --format markdo - `src/optional/analytics/reference/index.ts` - Reference data loader - `test-vocabulary-analysis.ts` - Vocabulary analysis demo - `test-comparison-analysis.ts` - Comparative analysis demo + +## Morphological Vocabulary Coverage + +### Problem + +Standard vocabulary coverage analysis only checks exact button label matches. For Grid 3 users, morphological inflections like "going", "went", "things", "books" are one tap away from their base form but reported as missing. + +### Solution + +`hasWord()` now checks both direct button labels AND smart grammar word forms: + +```typescript +const analyzer = new VocabularyAnalyzer(); +analyzer.hasWord("went", metrics); // true — found as inflection of "go" +analyzer.hasWord("books", metrics); // true — found as plural of "book" +``` + +This is auto-detected: only Grid 3 gridsets have POS tags, so for other formats the check degrades gracefully to exact-match only. + +### Impact + +In children's shared-reading transcript analysis against Super Core 50: + +- **Regular inflections** move from "missing" to "smart_grammar": going/go, things/thing, books/book, flying/fly, watching/watch +- **Irregular forms** need the irregular table: went/go, got/get, came/come +- Words where the base itself is missing remain genuinely missing + +### Cross-platform morphology + +Different AAC platforms handle morphology differently: + +| Platform | Approach | POS data | +| ----------------------- | --------------------------------------------- | ---------------------------------------------- | +| **Grid 3** | Runtime POS-based inflection from `verbs.zip` | `pos` parameter on buttons | +| **AsTeRICS Grid** | Per-element tagged word forms | `{value, tags[], lang}` stored on each element | +| **TD Snap / TouchChat** | Static buttons (no morphology) | None | + +The `WordFormGenerator` bridges these approaches, enabling cross-format conversion with morphology preserved. For future work, AsTeRICS Grid's tag-based approach could enable direct word-form lookups in vocabulary coverage analysis for `.grd` files. diff --git a/src/utilities/analytics/metrics/core.ts b/src/utilities/analytics/metrics/core.ts index 48807d5..e410733 100644 --- a/src/utilities/analytics/metrics/core.ts +++ b/src/utilities/analytics/metrics/core.ts @@ -24,6 +24,7 @@ import { localScanEffort, scanningEffort, } from './effort'; +import { MorphologyEngine } from '../morphology'; interface ToVisitItem { board: AACPage; @@ -118,9 +119,12 @@ export class MetricsCalculator { // Update buttons using dynamic spelling effort if applicable const buttons = Array.from(knownButtons.values()).sort((a, b) => a.effort - b.effort); - // Calculate metrics for word forms (smart grammar predictions) if enabled - // Default to true if not specified - const useSmartGrammar = options.useSmartGrammar !== false; + // Expand morphological predictions from POS tags if enabled or auto-detected + const useSmartGrammar = options.useSmartGrammar === true || this.treeHasPosTags(tree); + if (useSmartGrammar) { + this.expandMorphologicalPredictions(tree, options); + } + if (useSmartGrammar) { const { wordFormMetrics, replacedLabels } = this.calculateWordFormMetrics( tree, @@ -215,14 +219,23 @@ export class MetricsCalculator { } if (!spellingPage) - return { spellingPage: null, spellingBaseEffort: 10, spellingAvgLetterEffort: 2.5 }; + return { + spellingPage: null, + spellingBaseEffort: 10, + spellingAvgLetterEffort: 2.5, + }; // Calculate effort to reach this page from root const rootBoard = tree.rootId ? tree.pages[tree.rootId] : Object.values(tree.pages).find((p) => !p.parentId); - if (!rootBoard) return { spellingPage, spellingBaseEffort: 10, spellingAvgLetterEffort: 2.5 }; + if (!rootBoard) + return { + spellingPage, + spellingBaseEffort: 10, + spellingAvgLetterEffort: 2.5, + }; // Analyze specifically to find the lowest effort path to the spelling page const result = this.analyzeFrom(tree, rootBoard, setPcts, true, options); @@ -741,6 +754,163 @@ export class MetricsCalculator { return boardPcts; } + /** + * Quick check whether any button in the tree has a POS tag. + * Used to auto-enable smart grammar without requiring explicit opt-in. + */ + private treeHasPosTags(tree: AACTree): boolean { + for (const page of Object.values(tree.pages)) { + for (const row of page.grid) { + for (const btn of row) { + if (btn?.pos && btn.pos !== 'Unknown' && btn.pos !== 'Ignore') { + return true; + } + } + } + } + return false; + } + + /** + * Expand morphological predictions from POS tags on buttons + * + * For each button that has a POS tag (e.g., 'Verb', 'Noun'), use the + * MorphologyEngine to generate inflected word forms and populate the + * button's predictions array. This is done as a pre-processing step + * before calculateWordFormMetrics assigns effort to each form. + */ + private expandMorphologicalPredictions(tree: AACTree, options: MetricsOptions): void { + const locale = options.morphologyLocale || 'en-gb'; + const morph = new MorphologyEngine(locale); + + // Words that should never be POS-inferred (function words, determiners, etc.) + const skipInference = new Set([ + 'a', + 'an', + 'the', + 'to', + 'in', + 'on', + 'at', + 'of', + 'for', + 'and', + 'or', + 'but', + 'not', + 'no', + 'yes', + 'is', + 'am', + 'are', + 'was', + 'were', + 'be', + 'been', + 'being', + 'has', + 'have', + 'had', + 'do', + 'does', + 'did', + 'will', + 'would', + 'could', + 'should', + 'shall', + 'may', + 'might', + 'can', + 'must', + 'with', + 'from', + 'by', + 'up', + 'down', + 'out', + 'off', + 'over', + 'under', + 'again', + 'then', + 'than', + 'so', + 'if', + 'when', + 'where', + 'how', + 'what', + 'who', + 'which', + 'that', + 'this', + 'these', + 'those', + 'here', + 'there', + 'now', + 'very', + 'just', + 'more', + 'also', + 'too', + 'please', + 'thank', + 'hi', + 'hello', + 'bye', + 'goodbye', + 'okay', + 'oh', + 'wow', + 'sorry', + ]); + + for (const page of Object.values(tree.pages)) { + for (const row of page.grid) { + for (const btn of row) { + if (!btn || !btn.label) continue; + + let pos = btn.pos; + + // If no POS tag (or Unknown/Ignore), attempt POS inference. + // Many content words on topic pages lack POS tags even though + // they are clearly nouns (e.g., "bird", "tree", "cloud"). + // Strategy: check irregular tables first for confident POS, + // then fall back to Noun for single-word content labels. + if (!pos || pos === 'Unknown' || pos === 'Ignore') { + const lower = btn.label.toLowerCase(); + + // Skip function words and multi-word labels + if (!skipInference.has(lower) && !lower.includes(' ') && lower.length > 1) { + // Check irregular tables for confident POS assignment + const inferredPOS = morph.inferPOS(lower); + if (inferredPOS) { + pos = inferredPOS; + btn.pos = inferredPOS; + } else { + // Default to Noun for untagged content words. + // This generates plurals (e.g., bird → birds, tree → trees). + pos = 'Noun'; + btn.pos = 'Noun'; + } + } + } + + if (!pos || pos === 'Unknown' || pos === 'Ignore') continue; + + const forms = morph.inflect(btn.label, pos); + if (forms.length > 0) { + const existing = btn.predictions || []; + const merged = new Set([...existing, ...forms]); + btn.predictions = Array.from(merged); + } + } + } + } + } + /** * Calculate metrics for word forms (smart grammar predictions) * @@ -769,14 +939,54 @@ export class MetricsCalculator { const existingLabels = new Map(); buttons.forEach((btn) => existingLabels.set(btn.label.toLowerCase(), btn)); + // Build a map of POS tags from ALL tree buttons, keyed by lowercase label. + // This ensures words on BFS-unreachable pages still contribute POS data. + const treePosMap = new Map(); + const treePredictionsMap = new Map(); + Object.values(tree.pages).forEach((page: AACPage) => { + page.grid.forEach((row: (AACButton | null)[]) => { + row.forEach((btn: AACButton | null) => { + if (!btn || !btn.label) return; + const lower = btn.label.toLowerCase(); + if (btn.pos && btn.pos !== 'Unknown' && btn.pos !== 'Ignore') { + treePosMap.set(lower, btn.pos); + } + if (btn.predictions && btn.predictions.length > 0) { + const existing = treePredictionsMap.get(lower); + if (!existing || btn.predictions.length > existing.length) { + treePredictionsMap.set(lower, btn.predictions); + } + } + }); + }); + }); + + // For metrics buttons that lack POS but have a tree counterpart with POS, + // propagate the POS tag so it's available in the output. + buttons.forEach((btn) => { + const lower = btn.label.toLowerCase(); + if (!btn.pos || btn.pos === 'Unknown' || btn.pos === 'Ignore') { + const treePos = treePosMap.get(lower); + if (treePos) btn.pos = treePos; + } + }); + + // Note: buttons on pages unreachable via BFS from the root page are + // intentionally excluded. If there is no navigation path to a page, + // those buttons are not accessible to the user and should not count + // as available vocabulary. + // Iterate through all pages to find buttons with predictions Object.values(tree.pages).forEach((page: AACPage) => { page.grid.forEach((row: (AACButton | null)[]) => { row.forEach((btn: AACButton | null) => { if (!btn || !btn.predictions || btn.predictions.length === 0) return; - // Find the parent button's metrics - const parentMetrics = buttons.find((b) => b.id === btn.id); + // Find the parent button's metrics (by id first, then by label) + let parentMetrics = buttons.find((b) => b.id === btn.id); + if (!parentMetrics && btn.label) { + parentMetrics = existingLabels.get(btn.label.toLowerCase()); + } if (!parentMetrics) return; // Calculate effort for each word form @@ -901,7 +1111,11 @@ export class MetricsCalculator { // If no block assigned, treat as its own block at the end (fallback) if (blockId === null) { const loop = board.grid.length + (board.grid[0]?.length || 0); - return { steps: rowIndex + colIndex + 1, selections: 1, loopSteps: loop }; + return { + steps: rowIndex + colIndex + 1, + selections: 1, + loopSteps: loop, + }; } const blockConfig = board.scanBlocksConfig?.find((c) => c.id === blockId); diff --git a/src/utilities/analytics/metrics/types.ts b/src/utilities/analytics/metrics/types.ts index 97487f4..da117e9 100644 --- a/src/utilities/analytics/metrics/types.ts +++ b/src/utilities/analytics/metrics/types.ts @@ -26,6 +26,7 @@ export interface ButtonMetrics { is_word_form?: boolean; // True if this is a word form from predictions parent_button_id?: string; // ID of parent button that has these predictions parent_button_label?: string; // Label of parent button + pos?: string; // Part-of-speech tag from gridset (e.g., 'Verb', 'Noun') } /** @@ -128,17 +129,32 @@ export interface MetricsOptions { /** * Whether to include smart grammar word forms in metrics * - * When true (default): Word forms from smart grammar predictions are included + * When true: Word forms from smart grammar predictions are included * in the metrics. If a word exists as both a regular button and a word form, * the version with lower effort is used. * * When false: Smart grammar word forms are excluded from metrics. Only actual * buttons in the tree are analyzed. * - * Only applicable to systems that support smart grammar (e.g., Grid 3). - * Default is true. + * Auto-detected by default: if any button in the tree has a POS tag (e.g., + * from Grid 3's Action.InsertText), smart grammar is enabled automatically. + * For non-Grid-3 formats (TD Snap, TouchChat, OBF), no buttons have POS tags, + * so smart grammar is automatically disabled with zero overhead. + * + * Set explicitly to `true` to force-enable, or `false` to force-disable. */ useSmartGrammar?: boolean; + + /** + * Locale for morphological inflection rules + * + * When provided, the MorphologyEngine will generate inflected word forms + * (e.g., "going", "went" for "go") based on the POS tags extracted from + * the gridset. Defaults to 'en-gb'. + * + * Only used when useSmartGrammar is true. + */ + morphologyLocale?: string; } /** diff --git a/src/utilities/analytics/metrics/vocabulary.ts b/src/utilities/analytics/metrics/vocabulary.ts index c59e123..01adc5a 100644 --- a/src/utilities/analytics/metrics/vocabulary.ts +++ b/src/utilities/analytics/metrics/vocabulary.ts @@ -198,8 +198,18 @@ export class VocabularyAnalyzer { /** * Check if a word is in the board set + * + * Checks both direct button labels and smart grammar word forms + * (morphological inflections stored as predictions). */ hasWord(word: string, metrics: MetricsResult): boolean { - return metrics.buttons.some((b) => b.label.toLowerCase() === word.toLowerCase()); + const lower = word.toLowerCase(); + return metrics.buttons.some((b) => { + if (b.label.toLowerCase() === lower) return true; + if (b.is_word_form && b.parent_button_label) { + if (b.label.toLowerCase() === lower) return true; + } + return false; + }); } } diff --git a/src/utilities/analytics/morphology/engine.ts b/src/utilities/analytics/morphology/engine.ts new file mode 100644 index 0000000..54d40c8 --- /dev/null +++ b/src/utilities/analytics/morphology/engine.ts @@ -0,0 +1,931 @@ +import { MorphRuleSet, MorphRule } from './types'; +import type { Grid3VerbForms } from './grid3VerbsParser'; + +export class MorphologyEngine { + private ruleSet: MorphRuleSet; + private grid3Verbs?: Map; + private cache = new Map(); + + constructor(ruleSetOrLocale: string | MorphRuleSet) { + if (typeof ruleSetOrLocale === 'string') { + this.ruleSet = this.loadBundled(ruleSetOrLocale); + } else { + this.ruleSet = ruleSetOrLocale; + } + } + + static fromGrid3Verbs(verbForms: Grid3VerbForms): MorphologyEngine { + const engine = new MorphologyEngine({ + locale: verbForms.locale, + version: 1, + irregular: {}, + regular: {}, + }); + engine.grid3Verbs = verbForms.verbs; + return engine; + } + + get locale(): string { + return this.ruleSet.locale; + } + + inflect(base: string, pos: string): string[] { + const key = `${base.toLowerCase()}|${pos}`; + const cached = this.cache.get(key); + if (cached) return cached; + + if (this.grid3Verbs) { + const forms = this.grid3Verbs.get(base) || this.grid3Verbs.get(base.toLowerCase()); + if (forms) { + this.cache.set(key, forms); + return forms; + } + } + + const forms = this.computeForms(base, pos); + this.cache.set(key, forms); + return forms; + } + + isFormOf(word: string, base: string, pos: string): boolean { + const forms = this.inflect(base, pos); + const lower = word.toLowerCase(); + return forms.some((f) => f.toLowerCase() === lower); + } + + expandVocabulary( + buttons: Array<{ label: string; pos?: string; predictions?: string[] }> + ): Map { + const result = new Map(); + for (const btn of buttons) { + const pos = btn.pos; + if (!pos || pos === 'Unknown' || pos === 'Ignore') continue; + const forms = this.inflect(btn.label, pos); + if (forms.length > 0) { + result.set(btn.label, forms); + } + } + return result; + } + + inflectWithSlots(base: string, pos: string): Array<{ slot: string; form: string }> { + const lower = base.toLowerCase(); + const result: Array<{ slot: string; form: string }> = []; + const seen = new Set(); + + const irregular = this.ruleSet.irregular[pos]?.[lower]; + const regularSlots = this.ruleSet.regular[pos]; + + if (irregular) { + for (const [slot, value] of Object.entries(irregular)) { + if (slot === 'extra' && Array.isArray(value)) { + value.forEach((v) => { + if (!seen.has(v)) { + seen.add(v); + result.push({ slot, form: v }); + } + }); + } else if (Array.isArray(value)) { + value.forEach((v) => { + if (!seen.has(v)) { + seen.add(v); + result.push({ slot, form: v }); + } + }); + } else { + if (!seen.has(value)) { + seen.add(value); + result.push({ slot, form: value }); + } + } + } + } + + if (!regularSlots) return result; + + for (const [slot, rulesOrAlias] of Object.entries(regularSlots)) { + if (irregular && irregular[slot] !== undefined) continue; + + let rules: MorphRule[]; + if (typeof rulesOrAlias === 'string') { + const aliased = regularSlots[rulesOrAlias]; + if (!aliased || typeof aliased === 'string') continue; + rules = aliased; + } else { + rules = rulesOrAlias; + } + + const form = this.applyRules(lower, rules); + if (form && form !== lower && !seen.has(form)) { + seen.add(form); + result.push({ slot, form }); + } + } + + return result; + } + + private computeForms(base: string, pos: string): string[] { + const lower = base.toLowerCase(); + const forms = new Set(); + + const irregular = this.ruleSet.irregular[pos]?.[lower]; + const regularSlots = this.ruleSet.regular[pos]; + + if (irregular) { + for (const [slot, value] of Object.entries(irregular)) { + if (slot === 'extra' && Array.isArray(value)) { + value.forEach((v) => forms.add(v)); + } else if (Array.isArray(value)) { + value.forEach((v) => forms.add(v)); + } else { + forms.add(value); + } + } + } + + if (!regularSlots) { + return Array.from(forms); + } + + for (const [slot, rulesOrAlias] of Object.entries(regularSlots)) { + if (irregular && irregular[slot] !== undefined) continue; + + let rules: MorphRule[]; + if (typeof rulesOrAlias === 'string') { + const aliased = regularSlots[rulesOrAlias]; + if (!aliased || typeof aliased === 'string') continue; + rules = aliased; + } else { + rules = rulesOrAlias; + } + + const form = this.applyRules(lower, rules); + if (form && form !== lower) { + forms.add(form); + } + } + + return Array.from(forms); + } + + private applyRules(word: string, rules: MorphRule[]): string | undefined { + for (const rule of rules) { + const regex = new RegExp(rule.match, 'i'); + if (regex.test(word)) { + return word.replace(new RegExp(rule.match, 'i'), rule.replace); + } + } + return undefined; + } + + /** + * Infer the most likely POS for a word by checking the irregular tables. + * Returns the POS if found in any irregular table, or null if not found. + * Priority: Verb > Noun > Adjective > Pronoun + */ + inferPOS(word: string): string | null { + const lower = word.toLowerCase(); + for (const pos of ['Verb', 'Noun', 'Adjective', 'Pronoun']) { + if (this.ruleSet.irregular[pos]?.[lower]) { + return pos; + } + } + return null; + } + + private loadBundled(locale: string): MorphRuleSet { + const normalized = locale.toLowerCase().replace('_', '-'); + switch (normalized) { + case 'en-gb': + case 'en-us': + case 'en-au': + case 'en-ca': + case 'en-nz': + case 'en-za': + case 'en': + return builtinEn(); + default: + return { locale, version: 1, irregular: {}, regular: {} }; + } + } +} + +function builtinEn(): MorphRuleSet { + return { + locale: 'en-gb', + version: 1, + irregular: { + Verb: { + be: { + '3sg': 'is', + past: 'was', + pastPart: 'been', + presPart: 'being', + extra: ['am', 'are', 'were'], + }, + have: { + '3sg': 'has', + past: 'had', + pastPart: 'had', + presPart: 'having', + }, + do: { '3sg': 'does', past: 'did', pastPart: 'done', presPart: 'doing' }, + go: { + '3sg': 'goes', + past: 'went', + pastPart: 'gone', + presPart: 'going', + }, + say: { + '3sg': 'says', + past: 'said', + pastPart: 'said', + presPart: 'saying', + }, + get: { + '3sg': 'gets', + past: 'got', + pastPart: 'got', + presPart: 'getting', + }, + make: { + '3sg': 'makes', + past: 'made', + pastPart: 'made', + presPart: 'making', + }, + come: { + '3sg': 'comes', + past: 'came', + pastPart: 'come', + presPart: 'coming', + }, + take: { + '3sg': 'takes', + past: 'took', + pastPart: 'taken', + presPart: 'taking', + }, + know: { + '3sg': 'knows', + past: 'knew', + pastPart: 'known', + presPart: 'knowing', + }, + think: { + '3sg': 'thinks', + past: 'thought', + pastPart: 'thought', + presPart: 'thinking', + }, + see: { + '3sg': 'sees', + past: 'saw', + pastPart: 'seen', + presPart: 'seeing', + }, + give: { + '3sg': 'gives', + past: 'gave', + pastPart: 'given', + presPart: 'giving', + }, + find: { + '3sg': 'finds', + past: 'found', + pastPart: 'found', + presPart: 'finding', + }, + tell: { + '3sg': 'tells', + past: 'told', + pastPart: 'told', + presPart: 'telling', + }, + feel: { + '3sg': 'feels', + past: 'felt', + pastPart: 'felt', + presPart: 'feeling', + }, + run: { + '3sg': 'runs', + past: 'ran', + pastPart: 'run', + presPart: 'running', + }, + fly: { + '3sg': 'flies', + past: 'flew', + pastPart: 'flown', + presPart: 'flying', + }, + try: { + '3sg': 'tries', + past: 'tried', + pastPart: 'tried', + presPart: 'trying', + }, + leave: { + '3sg': 'leaves', + past: 'left', + pastPart: 'left', + presPart: 'leaving', + }, + call: { + '3sg': 'calls', + past: 'called', + pastPart: 'called', + presPart: 'calling', + }, + ask: { + '3sg': 'asks', + past: 'asked', + pastPart: 'asked', + presPart: 'asking', + }, + put: { + '3sg': 'puts', + past: 'put', + pastPart: 'put', + presPart: 'putting', + }, + read: { + '3sg': 'reads', + past: 'read', + pastPart: 'read', + presPart: 'reading', + }, + eat: { + '3sg': 'eats', + past: 'ate', + pastPart: 'eaten', + presPart: 'eating', + }, + drink: { + '3sg': 'drinks', + past: 'drank', + pastPart: 'drunk', + presPart: 'drinking', + }, + sleep: { + '3sg': 'sleeps', + past: 'slept', + pastPart: 'slept', + presPart: 'sleeping', + }, + speak: { + '3sg': 'speaks', + past: 'spoke', + pastPart: 'spoken', + presPart: 'speaking', + }, + write: { + '3sg': 'writes', + past: 'wrote', + pastPart: 'written', + presPart: 'writing', + }, + sit: { + '3sg': 'sits', + past: 'sat', + pastPart: 'sat', + presPart: 'sitting', + }, + stand: { + '3sg': 'stands', + past: 'stood', + pastPart: 'stood', + presPart: 'standing', + }, + fall: { + '3sg': 'falls', + past: 'fell', + pastPart: 'fallen', + presPart: 'falling', + }, + hold: { + '3sg': 'holds', + past: 'held', + pastPart: 'held', + presPart: 'holding', + }, + keep: { + '3sg': 'keeps', + past: 'kept', + pastPart: 'kept', + presPart: 'keeping', + }, + buy: { + '3sg': 'buys', + past: 'bought', + pastPart: 'bought', + presPart: 'buying', + }, + bring: { + '3sg': 'brings', + past: 'brought', + pastPart: 'brought', + presPart: 'bringing', + }, + catch: { + '3sg': 'catches', + past: 'caught', + pastPart: 'caught', + presPart: 'catching', + }, + teach: { + '3sg': 'teaches', + past: 'taught', + pastPart: 'taught', + presPart: 'teaching', + }, + fight: { + '3sg': 'fights', + past: 'fought', + pastPart: 'fought', + presPart: 'fighting', + }, + swim: { + '3sg': 'swims', + past: 'swam', + pastPart: 'swum', + presPart: 'swimming', + }, + sing: { + '3sg': 'sings', + past: 'sang', + pastPart: 'sung', + presPart: 'singing', + }, + draw: { + '3sg': 'draws', + past: 'drew', + pastPart: 'drawn', + presPart: 'drawing', + }, + drive: { + '3sg': 'drives', + past: 'drove', + pastPart: 'driven', + presPart: 'driving', + }, + ride: { + '3sg': 'rides', + past: 'rode', + pastPart: 'ridden', + presPart: 'riding', + }, + grow: { + '3sg': 'grows', + past: 'grew', + pastPart: 'grown', + presPart: 'growing', + }, + throw: { + '3sg': 'throws', + past: 'threw', + pastPart: 'thrown', + presPart: 'throwing', + }, + break: { + '3sg': 'breaks', + past: 'broke', + pastPart: 'broken', + presPart: 'breaking', + }, + wake: { + '3sg': 'wakes', + past: 'woke', + pastPart: 'woken', + presPart: 'waking', + }, + wear: { + '3sg': 'wears', + past: 'wore', + pastPart: 'worn', + presPart: 'wearing', + }, + win: { + '3sg': 'wins', + past: 'won', + pastPart: 'won', + presPart: 'winning', + }, + choose: { + '3sg': 'chooses', + past: 'chose', + pastPart: 'chosen', + presPart: 'choosing', + }, + hide: { + '3sg': 'hides', + past: 'hid', + pastPart: 'hidden', + presPart: 'hiding', + }, + steal: { + '3sg': 'steals', + past: 'stole', + pastPart: 'stolen', + presPart: 'stealing', + }, + begin: { + '3sg': 'begins', + past: 'began', + pastPart: 'begun', + presPart: 'beginning', + }, + ring: { + '3sg': 'rings', + past: 'rang', + pastPart: 'rung', + presPart: 'ringing', + }, + swing: { + '3sg': 'swings', + past: 'swung', + pastPart: 'swung', + presPart: 'swinging', + }, + blow: { + '3sg': 'blows', + past: 'blew', + pastPart: 'blown', + presPart: 'blowing', + }, + show: { + '3sg': 'shows', + past: 'showed', + pastPart: 'shown', + presPart: 'showing', + }, + shut: { + '3sg': 'shuts', + past: 'shut', + pastPart: 'shut', + presPart: 'shutting', + }, + cut: { + '3sg': 'cuts', + past: 'cut', + pastPart: 'cut', + presPart: 'cutting', + }, + hit: { + '3sg': 'hits', + past: 'hit', + pastPart: 'hit', + presPart: 'hitting', + }, + hurt: { + '3sg': 'hurts', + past: 'hurt', + pastPart: 'hurt', + presPart: 'hurting', + }, + let: { + '3sg': 'lets', + past: 'let', + pastPart: 'let', + presPart: 'letting', + }, + set: { + '3sg': 'sets', + past: 'set', + pastPart: 'set', + presPart: 'setting', + }, + cost: { + '3sg': 'costs', + past: 'cost', + pastPart: 'cost', + presPart: 'costing', + }, + send: { + '3sg': 'sends', + past: 'sent', + pastPart: 'sent', + presPart: 'sending', + }, + build: { + '3sg': 'builds', + past: 'built', + pastPart: 'built', + presPart: 'building', + }, + spend: { + '3sg': 'spends', + past: 'spent', + pastPart: 'spent', + presPart: 'spending', + }, + lend: { + '3sg': 'lends', + past: 'lent', + pastPart: 'lent', + presPart: 'lending', + }, + lose: { + '3sg': 'loses', + past: 'lost', + pastPart: 'lost', + presPart: 'losing', + }, + mean: { + '3sg': 'means', + past: 'meant', + pastPart: 'meant', + presPart: 'meaning', + }, + meet: { + '3sg': 'meets', + past: 'met', + pastPart: 'met', + presPart: 'meeting', + }, + pay: { + '3sg': 'pays', + past: 'paid', + pastPart: 'paid', + presPart: 'paying', + }, + sell: { + '3sg': 'sells', + past: 'sold', + pastPart: 'sold', + presPart: 'selling', + }, + hang: { + '3sg': 'hangs', + past: 'hung', + pastPart: 'hung', + presPart: 'hanging', + }, + shine: { + '3sg': 'shines', + past: 'shone', + pastPart: 'shone', + presPart: 'shining', + }, + dig: { + '3sg': 'digs', + past: 'dug', + pastPart: 'dug', + presPart: 'digging', + }, + stick: { + '3sg': 'sticks', + past: 'stuck', + pastPart: 'stuck', + presPart: 'sticking', + }, + spin: { + '3sg': 'spins', + past: 'spun', + pastPart: 'spun', + presPart: 'spinning', + }, + spread: { + '3sg': 'spreads', + past: 'spread', + pastPart: 'spread', + presPart: 'spreading', + }, + bite: { + '3sg': 'bites', + past: 'bit', + pastPart: 'bitten', + presPart: 'biting', + }, + feed: { + '3sg': 'feeds', + past: 'fed', + pastPart: 'fed', + presPart: 'feeding', + }, + lead: { + '3sg': 'leads', + past: 'led', + pastPart: 'led', + presPart: 'leading', + }, + light: { + '3sg': 'lights', + past: 'lit', + pastPart: 'lit', + presPart: 'lighting', + }, + shoot: { + '3sg': 'shoots', + past: 'shot', + pastPart: 'shot', + presPart: 'shooting', + }, + slide: { + '3sg': 'slides', + past: 'slid', + pastPart: 'slid', + presPart: 'sliding', + }, + }, + Noun: { + child: { plural: 'children' }, + person: { plural: 'people' }, + man: { plural: 'men' }, + woman: { plural: 'women' }, + mouse: { plural: 'mice' }, + foot: { plural: 'feet' }, + tooth: { plural: 'teeth' }, + goose: { plural: 'geese' }, + sheep: { plural: 'sheep' }, + fish: { plural: 'fish' }, + deer: { plural: 'deer' }, + ox: { plural: 'oxen' }, + leaf: { plural: 'leaves' }, + loaf: { plural: 'loaves' }, + wolf: { plural: 'wolves' }, + calf: { plural: 'calves' }, + half: { plural: 'halves' }, + knife: { plural: 'knives' }, + life: { plural: 'lives' }, + wife: { plural: 'wives' }, + self: { plural: 'selves' }, + shelf: { plural: 'shelves' }, + elf: { plural: 'elves' }, + thief: { plural: 'thieves' }, + roof: { plural: 'roofs' }, + chief: { plural: 'chiefs' }, + belief: { plural: 'beliefs' }, + proof: { plural: 'proofs' }, + hoof: { plural: 'hooves' }, + scarf: { plural: 'scarves' }, + wharf: { plural: 'wharves' }, + bus: { plural: 'buses' }, + glass: { plural: 'glasses' }, + class: { plural: 'classes' }, + box: { plural: 'boxes' }, + fox: { plural: 'foxes' }, + watch: { plural: 'watches' }, + match: { plural: 'matches' }, + brush: { plural: 'brushes' }, + dish: { plural: 'dishes' }, + wish: { plural: 'wishes' }, + wash: { plural: 'washes' }, + bush: { plural: 'bushes' }, + push: { plural: 'pushes' }, + potato: { plural: 'potatoes' }, + tomato: { plural: 'tomatoes' }, + hero: { plural: 'heroes' }, + echo: { plural: 'echoes' }, + veto: { plural: 'vetoes' }, + mango: { plural: 'mangoes' }, + mosquito: { plural: 'mosquitoes' }, + tornado: { plural: 'tornadoes' }, + volcano: { plural: 'volcanoes' }, + radio: { plural: 'radios' }, + studio: { plural: 'studios' }, + video: { plural: 'videos' }, + piano: { plural: 'pianos' }, + photo: { plural: 'photos' }, + zoo: { plural: 'zoos' }, + bamboo: { plural: 'bamboos' }, + embryo: { plural: 'embryos' }, + ratio: { plural: 'ratios' }, + scenario: { plural: 'scenarios' }, + analysis: { plural: 'analyses' }, + basis: { plural: 'bases' }, + crisis: { plural: 'crises' }, + diagnosis: { plural: 'diagnoses' }, + hypothesis: { plural: 'hypotheses' }, + oasis: { plural: 'oases' }, + parenthesis: { plural: 'parentheses' }, + synthesis: { plural: 'syntheses' }, + thesis: { plural: 'theses' }, + phenomenon: { plural: 'phenomena' }, + criterion: { plural: 'criteria' }, + datum: { plural: 'data' }, + medium: { plural: 'media' }, + curriculum: { plural: 'curricula' }, + bacterium: { plural: 'bacteria' }, + stimulus: { plural: 'stimuli' }, + syllabus: { plural: 'syllabi' }, + focus: { plural: 'foci' }, + nucleus: { plural: 'nuclei' }, + fungus: { plural: 'fungi' }, + cactus: { plural: 'cacti' }, + appendix: { plural: 'appendices' }, + index: { plural: 'indices' }, + matrix: { plural: 'matrices' }, + vertex: { plural: 'vertices' }, + }, + Adjective: { + good: { comparative: 'better', superlative: 'best' }, + bad: { comparative: 'worse', superlative: 'worst' }, + far: { comparative: 'farther', superlative: 'farthest' }, + little: { comparative: 'less', superlative: 'least' }, + much: { comparative: 'more', superlative: 'most' }, + many: { comparative: 'more', superlative: 'most' }, + well: { comparative: 'better', superlative: 'best' }, + old: { + comparative: 'older', + superlative: 'oldest', + extra: ['elder', 'eldest'], + }, + late: { + comparative: 'later', + superlative: 'latest', + extra: ['latter', 'last'], + }, + }, + Pronoun: { + i: { + objective: 'me', + possessive: 'my', + possessivePronoun: 'mine', + }, + you: { + objective: 'you', + possessive: 'your', + possessivePronoun: 'yours', + }, + he: { + objective: 'him', + possessive: 'his', + possessivePronoun: 'his', + }, + she: { + objective: 'her', + possessive: 'her', + possessivePronoun: 'hers', + }, + it: { + objective: 'it', + possessive: 'its', + }, + we: { + objective: 'us', + possessive: 'our', + possessivePronoun: 'ours', + }, + they: { + objective: 'them', + possessive: 'their', + possessivePronoun: 'theirs', + }, + mine: { extra: ['my'] }, + yours: { extra: ['your'] }, + his: { extra: ['him'] }, + hers: { extra: ['her'] }, + ours: { extra: ['our'] }, + theirs: { extra: ['their'] }, + }, + }, + regular: { + Verb: { + '3sg': [ + { match: '(ss|sh|ch|x|z|o)$', replace: '$1es' }, + { match: '([^aeiou])y$', replace: '$1ies' }, + { match: '$', replace: 's' }, + ], + past: [ + { match: '([^aeiou])y$', replace: '$1ied' }, + { match: '([^aeiou][aeiou])([^aeiouwxy])$', replace: '$1$2$2ed' }, + { match: '(.*)e$', replace: '$1ed' }, + { match: '$', replace: 'ed' }, + ], + pastPart: 'past', + presPart: [ + { match: 'ie$', replace: 'ying' }, + { match: '(.*)e$', replace: '$1ing' }, + { match: '([^aeiou][aeiou])([^aeiouwxy])$', replace: '$1$2$2ing' }, + { match: '$', replace: 'ing' }, + ], + }, + Noun: { + plural: [ + { match: '(ss|sh|ch|x|z)$', replace: '$1es' }, + { match: '([^aeiou])y$', replace: '$1ies' }, + { match: 'fe$', replace: 'ves' }, + { match: 'f$', replace: 'ves' }, + { match: '$', replace: 's' }, + ], + }, + Adjective: { + comparative: [ + { match: 'e$', replace: 'r' }, + { match: '([^aeiou])y$', replace: '$1ier' }, + { match: '([^aeiou][aeiou])([^aeiouwxy])$', replace: '$1$2$2er' }, + { match: '$', replace: 'er' }, + ], + superlative: [ + { match: 'e$', replace: 'st' }, + { match: '([^aeiou])y$', replace: '$1iest' }, + { match: '([^aeiou][aeiou])([^aeiouwxy])$', replace: '$1$2$2est' }, + { match: '$', replace: 'est' }, + ], + }, + }, + }; +} diff --git a/src/utilities/analytics/morphology/grid3VerbsParser.ts b/src/utilities/analytics/morphology/grid3VerbsParser.ts new file mode 100644 index 0000000..90b165b --- /dev/null +++ b/src/utilities/analytics/morphology/grid3VerbsParser.ts @@ -0,0 +1,549 @@ +import { XMLParser } from 'fast-xml-parser'; +import AdmZip from 'adm-zip'; +import { join, dirname, basename } from 'path'; +import type { VerbFormWithConditions, Grid3VerbFormsDetailed } from './types'; + +export interface Grid3VerbForms { + locale: string; + verbs: Map; +} + +interface ParsedRuleSet { + id: string; + placeholders: string[]; + participleRules: Map; + conjugationRules: Array<{ value: string; conditions: Map }>; +} + +interface ParsedVerb { + root: string; + ruleId?: string; + placeholderValues: Map; + participleOverrides: Map; + conjugationOverrides: Array<{ + value: string; + conditions: Map; + }>; +} + +export class Grid3VerbsParser { + private parser = new XMLParser({ + ignoreAttributes: false, + ignoreDeclaration: true, + textNodeName: '#text', + }); + + parseXml(xmlContent: string, locale?: string): Grid3VerbForms { + const data = this.parser.parse(xmlContent); + const verbdata = data.verbdata || data.Verbdata; + if (!verbdata) { + return { locale: locale || 'unknown', verbs: new Map() }; + } + + const detectedLocale = verbdata['@_locale'] || verbdata.locale || locale || 'unknown'; + const ruleSets = this.parseRuleSets(verbdata); + const verbs = this.parseVerbs(verbdata); + const result = new Map(); + + for (const verb of verbs) { + const forms = this.generateForms(verb, ruleSets); + if (forms.length > 0) { + result.set(verb.root, forms); + } + } + + return { locale: detectedLocale, verbs: result }; + } + + parseXmlDetailed(xmlContent: string, locale?: string): Grid3VerbFormsDetailed { + const data = this.parser.parse(xmlContent); + const verbdata = data.verbdata || data.Verbdata; + if (!verbdata) { + return { locale: locale || 'unknown', verbs: new Map() }; + } + + const detectedLocale = verbdata['@_locale'] || verbdata.locale || locale || 'unknown'; + const ruleSets = this.parseRuleSets(verbdata); + const verbs = this.parseVerbs(verbdata); + const result = new Map(); + + for (const verb of verbs) { + const forms = this.generateFormsDetailed(verb, ruleSets); + if (forms.length > 0) { + result.set(verb.root, forms); + } + } + + return { locale: detectedLocale, verbs: result }; + } + + /* eslint-disable @typescript-eslint/no-var-requires, @typescript-eslint/no-unsafe-argument */ + parseXmlFileDetailed(filePath: string): Grid3VerbFormsDetailed { + const fs = require('fs'); + const xml = fs.readFileSync(filePath, 'utf-8'); + return this.parseXmlDetailed(xml); + } + /* eslint-enable @typescript-eslint/no-var-requires, @typescript-eslint/no-unsafe-argument */ + + parseZip(zipPath: string): Grid3VerbForms { + const zip = new AdmZip(zipPath); + const entries = zip.getEntries(); + const verbEntry = entries.find((e) => e.entryName.toLowerCase().endsWith('verbs.xml')); + if (!verbEntry) { + const locale = basename(dirname(zipPath)); + return { locale, verbs: new Map() }; + } + const xml = verbEntry.getData().toString('utf-8'); + return this.parseXml(xml); + } + + parseZipDetailed(zipPath: string): Grid3VerbFormsDetailed { + const zip = new AdmZip(zipPath); + const entries = zip.getEntries(); + const verbEntry = entries.find((e) => e.entryName.toLowerCase().endsWith('verbs.xml')); + if (!verbEntry) { + const locale = basename(dirname(zipPath)); + return { locale, verbs: new Map() }; + } + const xml = verbEntry.getData().toString('utf-8'); + return this.parseXmlDetailed(xml); + } + + // eslint-disable-next-line @typescript-eslint/require-await + async parseLocale(locale: string, grid3InstallPath?: string): Promise { + const installPath = grid3InstallPath || this.getDefaultInstallPath(); + if (!installPath) { + return { locale, verbs: new Map() }; + } + const zipPath = join(installPath, 'Locale', locale, 'verbs', 'verbs.zip'); + try { + return this.parseZip(zipPath); + } catch { + return { locale, verbs: new Map() }; + } + } + + async parseInstalledLocales(grid3InstallPath?: string): Promise> { + const installPath = grid3InstallPath || this.getDefaultInstallPath(); + const results = new Map(); + if (!installPath) return results; + + const fs = await import('fs'); + const localeDir = join(installPath, 'Locale'); + let locales: string[]; + try { + locales = fs + .readdirSync(localeDir) + .filter((d) => fs.statSync(join(localeDir, d)).isDirectory()); + } catch { + return results; + } + + for (const locale of locales) { + const verbsZip = join(localeDir, locale, 'verbs', 'verbs.zip'); + try { + fs.accessSync(verbsZip, fs.constants.R_OK); + const forms = this.parseZip(verbsZip); + results.set(locale, forms); + } catch { + // No verbs.zip for this locale + } + } + + return results; + } + + /** + * Parse all locales from a custom directory. + * + * The directory should have the same structure as Grid 3's Locale folder: + * customDir/en-GB/verbs/verbs.zip + * customDir/nb-NO/verbs/verbs.zip + * customDir/de-DE/verbs/verbs.zip + * + * This allows users to supply morphology data copied from any Grid 3 + * installation without needing Grid 3 installed on this machine. + */ + parseCustomDirectory(dirPath: string, detailed?: false): Map; + parseCustomDirectory( + dirPath: string, + detailed: true + ): Map; + // eslint-disable-next-line @typescript-eslint/no-var-requires + parseCustomDirectory( + dirPath: string, + detailed = false + ): Map | Map { + /* eslint-disable @typescript-eslint/no-var-requires, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-return */ + const fs = require('fs'); + let locales: string[]; + try { + locales = fs + .readdirSync(dirPath) + .filter((d: string) => fs.statSync(join(dirPath, d)).isDirectory()); + } catch { + return new Map(); + } + + if (detailed) { + const results = new Map(); + for (const locale of locales) { + const verbsZip = join(dirPath, locale, 'verbs', 'verbs.zip'); + try { + fs.accessSync(verbsZip, fs.constants.R_OK); + results.set(locale, this.parseZipDetailed(verbsZip)); + } catch { + // No verbs.zip for this locale + } + } + return results; + } + + const results = new Map(); + for (const locale of locales) { + const verbsZip = join(dirPath, locale, 'verbs', 'verbs.zip'); + try { + fs.accessSync(verbsZip, fs.constants.R_OK); + results.set(locale, this.parseZip(verbsZip)); + } catch { + // No verbs.zip for this locale + } + } + return results; + /* eslint-enable @typescript-eslint/no-var-requires, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-return */ + } + + private getDefaultInstallPath(): string | null { + if (typeof process === 'undefined' || process.platform !== 'win32') { + return null; + } + const paths = [ + 'C:\\Program Files (x86)\\Smartbox\\Grid 3', + 'C:\\Program Files\\Smartbox\\Grid 3', + ]; + try { + // eslint-disable-next-line @typescript-eslint/no-var-requires + const fs = require('fs'); + for (const p of paths) { + if (fs.existsSync(p)) return p; + } + } catch { + // Grid 3 not installed + } + return null; + } + + /* eslint-disable @typescript-eslint/no-unsafe-argument, @typescript-eslint/no-unsafe-member-access */ + private parseRuleSets(verbdata: any): Map { + const rulesMap = new Map(); + const rulesList = verbdata.verbruleslist?.verbrules; + if (!rulesList) return rulesMap; + + const rulesArr = Array.isArray(rulesList) ? rulesList : [rulesList]; + for (const rules of rulesArr) { + // eslint-disable-next-line @typescript-eslint/no-unsafe-argument + const id = rules.Id || rules.id || `rule_${rulesMap.size}`; + const placeholders: string[] = []; + const ph = rules.placeholders?.placeholder; + if (ph) { + const phArr = Array.isArray(ph) ? ph : [ph]; + for (const p of phArr) { + const val = typeof p === 'string' ? p : p['#text'] || p; + if (val) placeholders.push(val); + } + } + + const participleRules = new Map(); + const pr = rules.participlerules?.participlerule; + if (pr) { + const prArr = Array.isArray(pr) ? pr : [pr]; + for (const rule of prArr) { + const type = rule['@_type'] || rule.type; + const value = rule['@_value'] || rule.value; + if (type && value) { + participleRules.set(type, value); + } + } + } + + const conjugationRules: Array<{ + value: string; + conditions: Map; + }> = []; + const cr = rules.conjugationrules?.conjugationrule; + if (cr) { + const crArr = Array.isArray(cr) ? cr : [cr]; + for (const rule of crArr) { + const value = rule['@_value'] || rule.value; + if (!value) continue; + const conditions = new Map(); + for (const attr of [ + 'time', + 'number', + 'person', + 'aspect', + 'mood', + 'voice', + 'tense', + 'polarity', + ]) { + const v = rule[`@_${attr}`] || rule[attr]; + if (v && v !== '*') { + conditions.set(attr, String(v)); + } + } + conjugationRules.push({ value, conditions }); + } + } + + rulesMap.set(id, { id, placeholders, participleRules, conjugationRules }); + } + + return rulesMap; + } + + /* eslint-disable @typescript-eslint/no-unsafe-argument, @typescript-eslint/no-unsafe-member-access */ + private parseVerbs(verbdata: any): ParsedVerb[] { + const verbs: ParsedVerb[] = []; + const verbsData = verbdata.verbs?.verb; + if (!verbsData) return verbs; + + const arr = Array.isArray(verbsData) ? verbsData : [verbsData]; + for (const v of arr) { + const root = v['@_root'] || v.root; + if (!root) continue; + + const ruleId = v['@_ruleid'] || v.ruleid || undefined; + + const placeholderValues = new Map(); + const rphv = v.ruleplaceholdervalues?.ruleplaceholdervalue; + if (rphv) { + const rphvArr = Array.isArray(rphv) ? rphv : [rphv]; + for (const ph of rphvArr) { + const placeholder = ph['@_placeholder'] || ph.placeholder; + const value = ph['@_value'] || ph.value; + if (placeholder && value) { + placeholderValues.set(placeholder, value); + } + } + } + + const participleOverrides = new Map(); + const parts = v.participles?.participle; + if (parts) { + const partsArr = Array.isArray(parts) ? parts : [parts]; + for (const p of partsArr) { + const type = p['@_type'] || p.type; + const value = p['@_value'] || p.value; + if (type && value) { + participleOverrides.set(type, value); + } + } + } + + const conjugationOverrides: Array<{ + value: string; + conditions: Map; + }> = []; + const conjs = v.conjugations?.conjugation; + if (conjs) { + const conjArr = Array.isArray(conjs) ? conjs : [conjs]; + for (const c of conjArr) { + const value = c['@_value'] || c.value; + if (!value) continue; + const conditions = new Map(); + const parts2 = c.part; + if (parts2) { + const pArr = Array.isArray(parts2) ? parts2 : [parts2]; + for (const p of pArr) { + for (const attr of ['time', 'number', 'person', 'aspect', 'mood', 'voice']) { + const pv = p[`@_${attr}`] || p[attr]; + if (pv && pv !== '*') { + conditions.set(attr, String(pv)); + } + } + } + } + conjugationOverrides.push({ value, conditions }); + } + } + + verbs.push({ + root, + ruleId, + placeholderValues, + participleOverrides, + conjugationOverrides, + }); + } + + return verbs; + } + + private generateForms(verb: ParsedVerb, ruleSets: Map): string[] { + const forms = new Set(); + const resolvedParticiples = new Map(); + + for (const [type, value] of verb.participleOverrides) { + resolvedParticiples.set(type, value); + } + + let appliedRule: ParsedRuleSet | undefined; + + if (verb.ruleId && ruleSets.has(verb.ruleId)) { + appliedRule = ruleSets.get(verb.ruleId)!; + } else if (ruleSets.size > 0) { + appliedRule = ruleSets.values().next().value; + } + + if (appliedRule) { + const context = this.buildContext(verb, resolvedParticiples); + + for (const [type, template] of appliedRule.participleRules) { + if (!resolvedParticiples.has(type)) { + const resolved = this.resolveTemplate(template, context); + if (resolved) { + resolvedParticiples.set(type, resolved); + } + } + } + + const fullContext = this.buildContext(verb, resolvedParticiples); + + for (const conjRule of appliedRule.conjugationRules) { + const resolved = this.resolveTemplate(conjRule.value, fullContext); + this.addIfSingleWord(resolved, forms); + } + } + + for (const [, value] of resolvedParticiples) { + this.addIfSingleWord(value, forms); + } + + for (const conj of verb.conjugationOverrides) { + this.addIfSingleWord(conj.value, forms); + } + + forms.delete(verb.root); + return Array.from(forms); + } + + private generateFormsDetailed( + verb: ParsedVerb, + ruleSets: Map + ): VerbFormWithConditions[] { + const forms = new Map>(); + const resolvedParticiples = new Map(); + + for (const [type, value] of verb.participleOverrides) { + resolvedParticiples.set(type, value); + } + + let appliedRule: ParsedRuleSet | undefined; + + if (verb.ruleId && ruleSets.has(verb.ruleId)) { + appliedRule = ruleSets.get(verb.ruleId)!; + } else if (ruleSets.size > 0) { + appliedRule = ruleSets.values().next().value; + } + + if (appliedRule) { + const context = this.buildContext(verb, resolvedParticiples); + + for (const [type, template] of appliedRule.participleRules) { + if (!resolvedParticiples.has(type)) { + const resolved = this.resolveTemplate(template, context); + if (resolved) { + resolvedParticiples.set(type, resolved); + } + } + } + + const fullContext = this.buildContext(verb, resolvedParticiples); + + for (const conjRule of appliedRule.conjugationRules) { + const resolved = this.resolveTemplate(conjRule.value, fullContext); + if (resolved && !resolved.includes(' ') && resolved !== '-') { + const trimmed = resolved.trim(); + if (trimmed.length > 0 && trimmed !== verb.root) { + const existing = forms.get(trimmed); + if (existing) { + for (const [k, v] of conjRule.conditions) { + existing.set(k, v); + } + } else { + forms.set(trimmed, new Map(conjRule.conditions)); + } + } + } + } + } + + for (const [type, value] of resolvedParticiples) { + if (!value.includes(' ') && value !== '-' && value.trim().length > 0 && value !== verb.root) { + const conditions = forms.get(value) || new Map(); + conditions.set('participleType', type); + forms.set(value, conditions); + } + } + + for (const conj of verb.conjugationOverrides) { + if ( + conj.value && + !conj.value.includes(' ') && + conj.value !== '-' && + conj.value.trim().length > 0 && + conj.value !== verb.root + ) { + const existing = forms.get(conj.value) || new Map(); + for (const [k, v] of conj.conditions) { + existing.set(k, v); + } + forms.set(conj.value, existing); + } + } + + forms.delete(verb.root); + return Array.from(forms.entries()).map(([value, conditions]) => ({ + value, + conditions, + })); + } + + private buildContext( + verb: ParsedVerb, + resolvedParticiples: Map + ): Map { + const context = new Map(); + context.set('{root}', verb.root); + for (const [key, value] of verb.placeholderValues) { + context.set(key, value); + if (!key.startsWith('{')) { + context.set(`{${key}}`, value); + } + } + for (const [type, value] of resolvedParticiples) { + context.set(`{${type}}`, value); + } + return context; + } + + private resolveTemplate(template: string, context: Map): string { + let result = template; + for (const [key, value] of context) { + const keyToReplace = key.startsWith('{') ? key : `{${key}}`; + result = result.split(keyToReplace).join(value); + } + return result; + } + + private addIfSingleWord(form: string, set: Set): void { + if (!form) return; + if (form.includes(' ')) return; + if (form === '-') return; + const trimmed = form.trim(); + if (trimmed.length > 0) { + set.add(trimmed); + } + } +} diff --git a/src/utilities/analytics/morphology/index.ts b/src/utilities/analytics/morphology/index.ts new file mode 100644 index 0000000..4c888a6 --- /dev/null +++ b/src/utilities/analytics/morphology/index.ts @@ -0,0 +1,12 @@ +export { MorphologyEngine } from './engine'; +export { Grid3VerbsParser } from './grid3VerbsParser'; +export { WordFormGenerator } from './wordFormGenerator'; +export type { + MorphRuleSet, + MorphRule, + MorphWordForms, + AstericsWordForm, + VerbFormWithConditions, + Grid3VerbFormsDetailed, +} from './types'; +export type { Grid3VerbForms } from './grid3VerbsParser'; diff --git a/src/utilities/analytics/morphology/types.ts b/src/utilities/analytics/morphology/types.ts new file mode 100644 index 0000000..fcd303a --- /dev/null +++ b/src/utilities/analytics/morphology/types.ts @@ -0,0 +1,43 @@ +export interface MorphRule { + match: string; + replace: string; +} + +export interface MorphRuleSet { + locale: string; + version: number; + irregular: { + [pos: string]: { + [baseWord: string]: { [slot: string]: string | string[] }; + }; + }; + regular: { + [pos: string]: { + [slot: string]: MorphRule[] | string; + }; + }; +} + +export interface MorphWordForms { + base: string; + pos: string; + forms: string[]; +} + +export interface AstericsWordForm { + lang?: string; + tags: string[]; + value: string; + pronunciation?: string; + base?: string; +} + +export interface VerbFormWithConditions { + value: string; + conditions: Map; +} + +export interface Grid3VerbFormsDetailed { + locale: string; + verbs: Map; +} diff --git a/src/utilities/analytics/morphology/wordFormGenerator.ts b/src/utilities/analytics/morphology/wordFormGenerator.ts new file mode 100644 index 0000000..cfe4d6d --- /dev/null +++ b/src/utilities/analytics/morphology/wordFormGenerator.ts @@ -0,0 +1,104 @@ +import { MorphologyEngine } from './engine'; +import { Grid3VerbsParser } from './grid3VerbsParser'; +import type { AstericsWordForm, VerbFormWithConditions } from './types'; + +const SLOT_TAG_MAP: Record = { + '3sg': ['3.PERS'], + past: ['PAST'], + pastPart: ['PAST', 'PARTICIPLE'], + presPart: ['GERUND'], + plural: ['PLURAL'], + comparative: ['COMPARATIVE'], + superlative: ['SUPERLATIVE'], +}; + +const CONDITION_TAG_MAP: Record> = { + person: { first: ['1.PERS'], second: ['2.PERS'], third: ['3.PERS'] }, + number: { singular: [], plural: ['PLURAL'] }, + time: { present: ['PRESENT'], past: ['PAST'], future: ['FUTURE'] }, + aspect: { simple: [], continuous: ['CONTINUOUS'], perfect: ['PERFECT'] }, + mood: { + imperative: ['IMPERATIVE'], + indicative: [], + conditional: ['CONDITIONAL'], + }, + participleType: { + presentparticiple: ['GERUND'], + pastparticiple: ['PAST', 'PARTICIPLE'], + infinitive: ['BASE'], + }, +}; + +export class WordFormGenerator { + generateFromEngineSlots( + base: string, + pos: string, + engine: MorphologyEngine, + lang: string = 'en' + ): AstericsWordForm[] { + const forms = engine.inflectWithSlots(base, pos); + const result: AstericsWordForm[] = [{ lang, tags: ['BASE'], value: base }]; + + for (const { slot, form } of forms) { + const tags = SLOT_TAG_MAP[slot] || [slot.toUpperCase()]; + result.push({ lang, tags, value: form, base }); + } + + return result; + } + + generateFromGrid3Conditions( + base: string, + formsWithConditions: VerbFormWithConditions[], + lang: string = 'en' + ): AstericsWordForm[] { + const result: AstericsWordForm[] = [{ lang, tags: ['BASE'], value: base }]; + + for (const form of formsWithConditions) { + const tags = this.conditionsToTags(form.conditions); + result.push({ lang, tags, value: form.value, base }); + } + + return this.deduplicate(result); + } + + generateFromPos( + base: string, + pos: string, + engine: MorphologyEngine, + grid3Parser: Grid3VerbsParser, + verbsZipPath?: string, + lang: string = 'en' + ): AstericsWordForm[] { + if (verbsZipPath) { + const detailed = grid3Parser.parseZipDetailed(verbsZipPath); + const forms = detailed.verbs.get(base) || detailed.verbs.get(base.toLowerCase()); + if (forms && forms.length > 0) { + return this.generateFromGrid3Conditions(base, forms, lang); + } + } + + return this.generateFromEngineSlots(base, pos, engine, lang); + } + + conditionsToTags(conditions: Map): string[] { + const tags: string[] = []; + for (const [dim, value] of conditions) { + const mapped = CONDITION_TAG_MAP[dim]?.[value]; + if (mapped) { + tags.push(...mapped); + } + } + return tags.length > 0 ? tags : ['UNKNOWN']; + } + + private deduplicate(forms: AstericsWordForm[]): AstericsWordForm[] { + const seen = new Set(); + return forms.filter((f) => { + const key = `${f.value}|${f.tags.sort().join(',')}`; + if (seen.has(key)) return false; + seen.add(key); + return true; + }); + } +} diff --git a/test/assets/grid3/synthetic-verbs.xml b/test/assets/grid3/synthetic-verbs.xml new file mode 100644 index 0000000..5809e09 --- /dev/null +++ b/test/assets/grid3/synthetic-verbs.xml @@ -0,0 +1,78 @@ + + + + + false + + + + first + second + third + + + singular + plural + + + + + + + + + infinitive + presentparticiple + pastparticiple + + + + + Regular + + {root} + + + + + + + + + + + + + + Irregular + + {root} + {pres} + {past} + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/test/cli.comprehensive.test.ts b/test/cli.comprehensive.test.ts index ab96814..7560574 100644 --- a/test/cli.comprehensive.test.ts +++ b/test/cli.comprehensive.test.ts @@ -389,14 +389,14 @@ describe('CLI Comprehensive Tests', () => { } catch (error: any) { expect(error.message).toContain('Command failed'); } - } catch (permissionError) { + } catch (_permissionError) { // If we can't change permissions, skip this test console.log('Skipping permission test - unable to change file permissions'); } finally { // Restore permissions for cleanup try { fs.chmodSync(restrictedFile, 0o644); - } catch (e) { + } catch (_e) { // Ignore cleanup errors } } diff --git a/test/errorHandling.test.ts b/test/errorHandling.test.ts index 23a6de0..c94ed53 100644 --- a/test/errorHandling.test.ts +++ b/test/errorHandling.test.ts @@ -49,14 +49,14 @@ describe('Error Handling', () => { const processor = new DotProcessor(); await expect(processor.loadIntoTree(restrictedFile)).rejects.toThrow(); - } catch (e) { + } catch (_e) { // chmod might not work on all systems, skip this test console.log('Skipping permission test - chmod not supported'); } finally { try { fs.chmodSync(restrictedFile, 0o644); // Restore permissions for cleanup fs.unlinkSync(restrictedFile); - } catch (e) { + } catch (_e) { // Cleanup failed, but that's ok } } @@ -197,14 +197,14 @@ describe('Error Handling', () => { const outputPath = path.join(readOnlyDir, 'output.dot'); await expect(processor.saveFromTree(tree, outputPath)).rejects.toThrow(); - } catch (e) { + } catch (_e) { // chmod might not work on all systems console.log('Skipping read-only directory test - chmod not supported'); } finally { try { fs.chmodSync(readOnlyDir, 0o755); // Restore permissions fs.rmSync(readOnlyDir, { recursive: true, force: true }); - } catch (e) { + } catch (_e) { // Cleanup failed } } diff --git a/test/grid3VerbsParser.test.ts b/test/grid3VerbsParser.test.ts new file mode 100644 index 0000000..54a5dc5 --- /dev/null +++ b/test/grid3VerbsParser.test.ts @@ -0,0 +1,210 @@ +import { Grid3VerbsParser } from '../src/utilities/analytics/morphology/grid3VerbsParser'; +import { join } from 'path'; + +const SYNTHETIC_XML = join(__dirname, 'assets', 'grid3', 'synthetic-verbs.xml'); + +// Users can set GRID3_MORPHOLOGY_DIR to point to their own copy of +// Grid 3's Locale directory (e.g. copied from another machine). +// Example: GRID3_MORPHOLOGY_DIR=/path/to/Grid3/Locale npm test +const MORPHOLOGY_DIR = + process.env.GRID3_MORPHOLOGY_DIR || + (process.platform === 'win32' ? 'C:\\Program Files (x86)\\Smartbox\\Grid 3\\Locale' : ''); + +const parser = new Grid3VerbsParser(); + +function fileExists(p: string): boolean { + try { + // eslint-disable-next-line @typescript-eslint/no-var-requires + const fs = require('fs'); + fs.accessSync(p, fs.constants.R_OK); + return true; + } catch { + return false; + } +} + +function getMorphZip(locale: string): string { + return join(MORPHOLOGY_DIR, locale, 'verbs', 'verbs.zip'); +} + +// eslint-disable-next-line @typescript-eslint/no-var-requires +const fs = require('fs'); + +describe('Grid3VerbsParser - Synthetic XML (always runs)', () => { + test('synthetic fixture exists', () => { + expect(fileExists(SYNTHETIC_XML)).toBe(true); + }); + + describe('parseXml (flat forms)', () => { + let forms: Map; + let locale: string; + + beforeAll(() => { + const xml = fs.readFileSync(SYNTHETIC_XML, 'utf-8'); + const result = parser.parseXml(xml); + locale = result.locale; + forms = result.verbs; + }); + + test('detects locale test-XX', () => { + expect(locale).toBe('test-XX'); + }); + + test('parses 3 verbs', () => { + expect(forms.size).toBe(3); + }); + + test('regular verb walk -> walks, walked, walking', () => { + const walkForms = forms.get('walk'); + expect(walkForms).toBeDefined(); + expect(walkForms).toContain('walks'); + expect(walkForms).toContain('walked'); + expect(walkForms).toContain('walking'); + }); + + test('irregular verb go -> goes, went, going, gone', () => { + const goForms = forms.get('go'); + expect(goForms).toBeDefined(); + expect(goForms).toContain('goes'); + expect(goForms).toContain('went'); + expect(goForms).toContain('going'); + expect(goForms).toContain('gone'); + }); + + test('default-rule verb jump -> jumps, jumped, jumping', () => { + const jumpForms = forms.get('jump'); + expect(jumpForms).toBeDefined(); + expect(jumpForms).toContain('jumps'); + expect(jumpForms).toContain('jumped'); + expect(jumpForms).toContain('jumping'); + }); + + test('no compound forms', () => { + for (const [, wordForms] of forms) { + for (const f of wordForms) { + expect(f).not.toContain(' '); + } + } + }); + }); + + describe('parseXmlDetailed (forms with conditions)', () => { + let detailed: Map }>>; + + beforeAll(() => { + const result = parser.parseXmlFileDetailed(SYNTHETIC_XML); + detailed = result.verbs; + }); + + test('parses 3 verbs with conditions', () => { + expect(detailed.size).toBe(3); + }); + + test('walk has "walks" with person=third, time=present', () => { + const walkForms = detailed.get('walk'); + expect(walkForms).toBeDefined(); + const walksForm = walkForms!.find((f) => f.value === 'walks'); + expect(walksForm).toBeDefined(); + expect(walksForm!.conditions.get('person')).toBe('third'); + expect(walksForm!.conditions.get('time')).toBe('present'); + }); + + test('walk has "walked" with time=past', () => { + const walkForms = detailed.get('walk'); + const walkedForm = walkForms!.find((f) => f.value === 'walked'); + expect(walkedForm).toBeDefined(); + expect(walkedForm!.conditions.get('time')).toBe('past'); + }); + + test('go has "went" with time=past', () => { + const goForms = detailed.get('go'); + const wentForm = goForms!.find((f) => f.value === 'went'); + expect(wentForm).toBeDefined(); + expect(wentForm!.conditions.get('time')).toBe('past'); + }); + + test('go has "gone" with participleType=pastparticiple', () => { + const goForms = detailed.get('go'); + const goneForm = goForms!.find((f) => f.value === 'gone'); + expect(goneForm).toBeDefined(); + expect(goneForm!.conditions.get('participleType')).toBe('pastparticiple'); + }); + + test('go has "goes" with person=third', () => { + const goForms = detailed.get('go'); + const goesForm = goForms!.find((f) => f.value === 'goes'); + expect(goesForm).toBeDefined(); + expect(goesForm!.conditions.get('person')).toBe('third'); + }); + }); + + describe('parseZip with synthetic data', () => { + test('can parse a zip file containing verbs.xml', () => { + // Create a temporary zip with our synthetic XML + // eslint-disable-next-line @typescript-eslint/no-var-requires + const AdmZip = require('adm-zip'); + const tmpDir = join(__dirname, 'assets', 'grid3', '_tmp'); + const zipPath = join(tmpDir, 'verbs.zip'); + + try { + fs.mkdirSync(tmpDir, { recursive: true }); + const zip = new AdmZip(); + zip.addFile('verbs.xml', fs.readFileSync(SYNTHETIC_XML)); + zip.writeZip(zipPath); + + const result = parser.parseZip(zipPath); + expect(result.verbs.size).toBe(3); + expect(result.verbs.get('go')).toContain('went'); + } finally { + try { + fs.unlinkSync(zipPath); + fs.rmdirSync(tmpDir); + } catch { + // cleanup best effort + } + } + }); + }); +}); + +describe('Grid3VerbsParser - External morphology data', () => { + // These tests run when GRID3_MORPHOLOGY_DIR points to a valid + // Grid 3 Locale directory (or a directory someone has copied + // the verbs.zip files into). On CI without Grid 3, these skip. + test('parses en-GB verbs.zip', () => { + const zipPath = getMorphZip('en-GB'); + if (!fileExists(zipPath)) return; + const result = parser.parseZip(zipPath); + expect(result.verbs.size).toBeGreaterThan(100); + }); + + test('en-GB go -> goes, going, gone, went', () => { + const zipPath = getMorphZip('en-GB'); + if (!fileExists(zipPath)) return; + const result = parser.parseZip(zipPath); + const goForms = result.verbs.get('go'); + expect(goForms).toBeDefined(); + expect(goForms).toContain('goes'); + expect(goForms).toContain('going'); + expect(goForms).toContain('gone'); + expect(goForms).toContain('went'); + }); + + test('en-GB detailed has conditions', () => { + const zipPath = getMorphZip('en-GB'); + if (!fileExists(zipPath)) return; + const result = parser.parseZipDetailed(zipPath); + const goForms = result.verbs.get('go'); + expect(goForms).toBeDefined(); + expect(goForms!.length).toBeGreaterThan(0); + const went = goForms!.find((f) => f.value === 'went'); + expect(went).toBeDefined(); + }); + + test('parses nb-NO verbs.zip', () => { + const zipPath = getMorphZip('nb-NO'); + if (!fileExists(zipPath)) return; + const result = parser.parseZip(zipPath); + expect(result.verbs.size).toBeGreaterThan(100); + }); +}); diff --git a/test/history.test.ts b/test/history.test.ts index 95a80e1..5a97942 100644 --- a/test/history.test.ts +++ b/test/history.test.ts @@ -17,7 +17,7 @@ describe('History analytics', () => { afterAll(async () => { try { fs.rmSync(tempDir, { recursive: true, force: true }); - } catch (error) { + } catch (_error) { // Ignore cleanup issues on platforms that briefly lock the file } }); diff --git a/test/morphology.test.ts b/test/morphology.test.ts new file mode 100644 index 0000000..5641604 --- /dev/null +++ b/test/morphology.test.ts @@ -0,0 +1,303 @@ +import { MorphologyEngine } from '../src/utilities/analytics/morphology/engine'; +import { MorphRuleSet } from '../src/utilities/analytics/morphology/types'; + +describe('MorphologyEngine', () => { + describe('built-in English rules', () => { + let engine: MorphologyEngine; + + beforeEach(() => { + engine = new MorphologyEngine('en-gb'); + }); + + describe('irregular verbs', () => { + test('go -> goes, going, gone, went', () => { + const forms = engine.inflect('go', 'Verb'); + expect(forms).toContain('goes'); + expect(forms).toContain('going'); + expect(forms).toContain('gone'); + expect(forms).toContain('went'); + expect(forms).not.toContain('goed'); + }); + + test('be -> is, am, are, was, were, been, being', () => { + const forms = engine.inflect('be', 'Verb'); + expect(forms).toContain('is'); + expect(forms).toContain('am'); + expect(forms).toContain('are'); + expect(forms).toContain('was'); + expect(forms).toContain('were'); + expect(forms).toContain('been'); + expect(forms).toContain('being'); + }); + + test('have -> has, had, having', () => { + const forms = engine.inflect('have', 'Verb'); + expect(forms).toContain('has'); + expect(forms).toContain('had'); + expect(forms).toContain('having'); + }); + + test('do -> does, did, done, doing', () => { + const forms = engine.inflect('do', 'Verb'); + expect(forms).toContain('does'); + expect(forms).toContain('did'); + expect(forms).toContain('done'); + expect(forms).toContain('doing'); + }); + + test('say -> says, said, saying', () => { + const forms = engine.inflect('say', 'Verb'); + expect(forms).toContain('says'); + expect(forms).toContain('said'); + expect(forms).toContain('saying'); + }); + + test('get -> gets, got, getting', () => { + const forms = engine.inflect('get', 'Verb'); + expect(forms).toContain('gets'); + expect(forms).toContain('got'); + expect(forms).toContain('getting'); + }); + + test('take -> takes, took, taken, taking', () => { + const forms = engine.inflect('take', 'Verb'); + expect(forms).toContain('takes'); + expect(forms).toContain('took'); + expect(forms).toContain('taken'); + expect(forms).toContain('taking'); + }); + + test('come -> comes, came, coming', () => { + const forms = engine.inflect('come', 'Verb'); + expect(forms).toContain('comes'); + expect(forms).toContain('came'); + expect(forms).toContain('coming'); + }); + }); + + describe('regular verbs', () => { + test('walk -> walks, walked, walking', () => { + const forms = engine.inflect('walk', 'Verb'); + expect(forms).toContain('walks'); + expect(forms).toContain('walked'); + expect(forms).toContain('walking'); + }); + + test('watch -> watches, watched, watching', () => { + const forms = engine.inflect('watch', 'Verb'); + expect(forms).toContain('watches'); + expect(forms).toContain('watched'); + expect(forms).toContain('watching'); + }); + + test('carry -> carries, carried, carrying', () => { + const forms = engine.inflect('carry', 'Verb'); + expect(forms).toContain('carries'); + expect(forms).toContain('carried'); + expect(forms).toContain('carrying'); + }); + + test('like -> likes, liked, liking', () => { + const forms = engine.inflect('like', 'Verb'); + expect(forms).toContain('likes'); + expect(forms).toContain('liked'); + expect(forms).toContain('liking'); + }); + }); + + describe('irregular nouns', () => { + test('child -> children', () => { + const forms = engine.inflect('child', 'Noun'); + expect(forms).toContain('children'); + }); + + test('person -> people', () => { + const forms = engine.inflect('person', 'Noun'); + expect(forms).toContain('people'); + }); + + test('mouse -> mice', () => { + const forms = engine.inflect('mouse', 'Noun'); + expect(forms).toContain('mice'); + }); + + test('foot -> feet', () => { + const forms = engine.inflect('foot', 'Noun'); + expect(forms).toContain('feet'); + }); + + test('sheep -> sheep (no change)', () => { + const forms = engine.inflect('sheep', 'Noun'); + expect(forms).toContain('sheep'); + expect(forms.length).toBe(1); + }); + }); + + describe('regular nouns', () => { + test('book -> books', () => { + const forms = engine.inflect('book', 'Noun'); + expect(forms).toContain('books'); + }); + + test('thing -> things', () => { + const forms = engine.inflect('thing', 'Noun'); + expect(forms).toContain('things'); + }); + + test('story -> stories', () => { + const forms = engine.inflect('story', 'Noun'); + expect(forms).toContain('stories'); + }); + + test('bus -> buses', () => { + const forms = engine.inflect('bus', 'Noun'); + expect(forms).toContain('buses'); + }); + }); + + describe('adjectives', () => { + test('good -> better, best', () => { + const forms = engine.inflect('good', 'Adjective'); + expect(forms).toContain('better'); + expect(forms).toContain('best'); + }); + + test('bad -> worse, worst', () => { + const forms = engine.inflect('bad', 'Adjective'); + expect(forms).toContain('worse'); + expect(forms).toContain('worst'); + }); + + test('big -> bigger, biggest', () => { + const forms = engine.inflect('big', 'Adjective'); + expect(forms).toContain('bigger'); + expect(forms).toContain('biggest'); + }); + + test('happy -> happier, happiest', () => { + const forms = engine.inflect('happy', 'Adjective'); + expect(forms).toContain('happier'); + expect(forms).toContain('happiest'); + }); + }); + + describe('pronouns', () => { + test('I -> me, my, mine', () => { + const forms = engine.inflect('I', 'Pronoun'); + expect(forms).toContain('me'); + expect(forms).toContain('my'); + expect(forms).toContain('mine'); + }); + + test('they -> them, their, theirs', () => { + const forms = engine.inflect('they', 'Pronoun'); + expect(forms).toContain('them'); + expect(forms).toContain('their'); + expect(forms).toContain('theirs'); + }); + }); + }); + + describe('isFormOf', () => { + let engine: MorphologyEngine; + + beforeEach(() => { + engine = new MorphologyEngine('en-gb'); + }); + + test('detects "went" as form of "go"', () => { + expect(engine.isFormOf('went', 'go', 'Verb')).toBe(true); + }); + + test('detects "going" as form of "go"', () => { + expect(engine.isFormOf('going', 'go', 'Verb')).toBe(true); + }); + + test('detects "children" as form of "child"', () => { + expect(engine.isFormOf('children', 'child', 'Noun')).toBe(true); + }); + + test('does not match unrelated words', () => { + expect(engine.isFormOf('running', 'go', 'Verb')).toBe(false); + }); + + test('case insensitive', () => { + expect(engine.isFormOf('Went', 'Go', 'Verb')).toBe(true); + expect(engine.isFormOf('WENT', 'go', 'Verb')).toBe(true); + }); + }); + + describe('expandVocabulary', () => { + let engine: MorphologyEngine; + + beforeEach(() => { + engine = new MorphologyEngine('en-gb'); + }); + + test('expands verb buttons', () => { + const buttons = [ + { label: 'go', pos: 'Verb' }, + { label: 'book', pos: 'Noun' }, + ]; + const result = engine.expandVocabulary(buttons); + + expect(result.get('go')).toContain('goes'); + expect(result.get('go')).toContain('going'); + expect(result.get('go')).toContain('went'); + expect(result.get('go')).toContain('gone'); + expect(result.get('book')).toContain('books'); + }); + + test('skips Unknown and Ignore POS', () => { + const buttons = [ + { label: 'hello', pos: 'Unknown' }, + { label: 'world', pos: 'Ignore' }, + ]; + const result = engine.expandVocabulary(buttons); + expect(result.has('hello')).toBe(false); + expect(result.has('world')).toBe(false); + }); + + test('skips buttons without POS', () => { + const buttons = [{ label: 'hello' }]; + const result = engine.expandVocabulary(buttons); + expect(result.has('hello')).toBe(false); + }); + }); + + describe('custom rule set', () => { + test('accepts custom MorphRuleSet', () => { + const customRules: MorphRuleSet = { + locale: 'test', + version: 1, + irregular: {}, + regular: { + Verb: { + past: [{ match: '$', replace: 'ed' }], + }, + }, + }; + + const engine = new MorphologyEngine(customRules); + const forms = engine.inflect('walk', 'Verb'); + expect(forms).toContain('walked'); + }); + }); + + describe('unknown locale', () => { + test('returns empty for unsupported locale', () => { + const engine = new MorphologyEngine('xx-xx'); + const forms = engine.inflect('go', 'Verb'); + expect(forms).toEqual([]); + }); + }); + + describe('caching', () => { + test('caches results for same base+pos', () => { + const engine = new MorphologyEngine('en-gb'); + const first = engine.inflect('go', 'Verb'); + const second = engine.inflect('go', 'Verb'); + expect(first).toBe(second); + }); + }); +}); diff --git a/test/processTexts.realworld.test.ts b/test/processTexts.realworld.test.ts index 6895d23..1727dc8 100644 --- a/test/processTexts.realworld.test.ts +++ b/test/processTexts.realworld.test.ts @@ -24,7 +24,7 @@ describe('ProcessTexts with Real-World Data', () => { if (fs.existsSync(tempDir)) { try { fs.rmSync(tempDir, { recursive: true, force: true }); - } catch (error) { + } catch (_error) { // On Windows the file can be locked briefly; ignore cleanup failure in tests } } diff --git a/test/snapProcessor.audio.test.ts b/test/snapProcessor.audio.test.ts index 664c398..b6eed5d 100644 --- a/test/snapProcessor.audio.test.ts +++ b/test/snapProcessor.audio.test.ts @@ -178,7 +178,7 @@ describe('SnapProcessor Audio Support', () => { if (metadata.PunjabiText) { expect(metadata.PunjabiText).toMatch(/[\u0A00-\u0A7F]/); // Gurmukhi script } - } catch (e) { + } catch (_e) { // Metadata might not be JSON } } diff --git a/test/utils/testHelpers.ts b/test/utils/testHelpers.ts index 4b76bbe..50f933d 100644 --- a/test/utils/testHelpers.ts +++ b/test/utils/testHelpers.ts @@ -218,7 +218,7 @@ export class FileSystemHelper { static getFileStats(filePath: string): fs.Stats | null { try { return fs.statSync(filePath); - } catch (error) { + } catch (_error) { return null; } } @@ -228,7 +228,7 @@ export class FileSystemHelper { const content1 = fs.readFileSync(filePath1); const content2 = fs.readFileSync(filePath2); return content1.equals(content2); - } catch (error) { + } catch (_error) { return false; } } diff --git a/test/wordFormGenerator.test.ts b/test/wordFormGenerator.test.ts new file mode 100644 index 0000000..fcd7dfc --- /dev/null +++ b/test/wordFormGenerator.test.ts @@ -0,0 +1,212 @@ +import { WordFormGenerator } from '../src/utilities/analytics/morphology/wordFormGenerator'; +import { MorphologyEngine } from '../src/utilities/analytics/morphology/engine'; +import { Grid3VerbsParser } from '../src/utilities/analytics/morphology/grid3VerbsParser'; +import { join } from 'path'; + +const SYNTHETIC_XML = join(__dirname, 'assets', 'grid3', 'synthetic-verbs.xml'); + +describe('WordFormGenerator', () => { + let generator: WordFormGenerator; + let engine: MorphologyEngine; + + beforeEach(() => { + generator = new WordFormGenerator(); + engine = new MorphologyEngine('en-gb'); + }); + + describe('generateFromEngineSlots', () => { + test('regular verb walk -> BASE + 3.PERS + PAST + GERUND', () => { + const forms = generator.generateFromEngineSlots('walk', 'Verb', engine); + const base = forms.find((f) => f.tags.includes('BASE')); + expect(base).toBeDefined(); + expect(base!.value).toBe('walk'); + + const third = forms.find((f) => f.tags.includes('3.PERS')); + expect(third).toBeDefined(); + expect(third!.value).toBe('walks'); + + const past = forms.find((f) => f.tags.includes('PAST') && !f.tags.includes('PARTICIPLE')); + expect(past).toBeDefined(); + expect(past!.value).toBe('walked'); + + const gerund = forms.find((f) => f.tags.includes('GERUND')); + expect(gerund).toBeDefined(); + expect(gerund!.value).toBe('walking'); + }); + + test('irregular verb go -> correct tagged forms', () => { + const forms = generator.generateFromEngineSlots('go', 'Verb', engine); + + const went = forms.find((f) => f.value === 'went'); + expect(went).toBeDefined(); + expect(went!.tags).toContain('PAST'); + + const goes = forms.find((f) => f.value === 'goes'); + expect(goes).toBeDefined(); + expect(goes!.tags).toContain('3.PERS'); + + const gone = forms.find((f) => f.value === 'gone'); + expect(gone).toBeDefined(); + expect(gone!.tags).toContain('PAST'); + expect(gone!.tags).toContain('PARTICIPLE'); + + const going = forms.find((f) => f.value === 'going'); + expect(going).toBeDefined(); + expect(going!.tags).toContain('GERUND'); + }); + + test('noun book -> BASE + PLURAL', () => { + const forms = generator.generateFromEngineSlots('book', 'Noun', engine); + + const base = forms.find((f) => f.tags.includes('BASE')); + expect(base).toBeDefined(); + expect(base!.value).toBe('book'); + + const plural = forms.find((f) => f.tags.includes('PLURAL')); + expect(plural).toBeDefined(); + expect(plural!.value).toBe('books'); + }); + + test('adjective big -> BASE + COMPARATIVE + SUPERLATIVE', () => { + const forms = generator.generateFromEngineSlots('big', 'Adjective', engine); + + const comp = forms.find((f) => f.tags.includes('COMPARATIVE')); + expect(comp).toBeDefined(); + expect(comp!.value).toBe('bigger'); + + const sup = forms.find((f) => f.tags.includes('SUPERLATIVE')); + expect(sup).toBeDefined(); + expect(sup!.value).toBe('biggest'); + }); + + test('all forms have lang', () => { + const forms = generator.generateFromEngineSlots('walk', 'Verb', engine, 'en'); + for (const f of forms) { + expect(f.lang).toBe('en'); + } + }); + + test('inflected forms have base set', () => { + const forms = generator.generateFromEngineSlots('go', 'Verb', engine, 'en'); + const nonBase = forms.filter((f) => !f.tags.includes('BASE')); + for (const f of nonBase) { + expect(f.base).toBe('go'); + } + }); + }); + + describe('generateFromGrid3Conditions', () => { + test('maps person/time conditions to AsTeRICS tags', () => { + const forms = generator.generateFromGrid3Conditions( + 'walk', + [ + { + value: 'walks', + conditions: new Map([ + ['person', 'third'], + ['time', 'present'], + ]), + }, + { + value: 'walked', + conditions: new Map([['time', 'past']]), + }, + ], + 'en' + ); + + const base = forms.find((f) => f.tags.includes('BASE')); + expect(base).toBeDefined(); + expect(base!.value).toBe('walk'); + + const walks = forms.find((f) => f.value === 'walks'); + expect(walks).toBeDefined(); + expect(walks!.tags).toContain('3.PERS'); + + const walked = forms.find((f) => f.value === 'walked'); + expect(walked).toBeDefined(); + expect(walked!.tags).toContain('PAST'); + }); + + test('maps participleType to correct tags', () => { + const forms = generator.generateFromGrid3Conditions( + 'go', + [ + { + value: 'gone', + conditions: new Map([['participleType', 'pastparticiple']]), + }, + { + value: 'going', + conditions: new Map([['participleType', 'presentparticiple']]), + }, + ], + 'en' + ); + + const gone = forms.find((f) => f.value === 'gone'); + expect(gone).toBeDefined(); + expect(gone!.tags).toContain('PAST'); + expect(gone!.tags).toContain('PARTICIPLE'); + + const going = forms.find((f) => f.value === 'going'); + expect(going).toBeDefined(); + expect(going!.tags).toContain('GERUND'); + }); + + test('deduplicates same value+tags combos', () => { + const forms = generator.generateFromGrid3Conditions( + 'test', + [ + { value: 'tested', conditions: new Map([['time', 'past']]) }, + { value: 'tested', conditions: new Map([['time', 'past']]) }, + ], + 'en' + ); + + const testedForms = forms.filter((f) => f.value === 'tested'); + expect(testedForms.length).toBe(1); + }); + }); + + describe('conditionsToTags', () => { + test('maps person conditions', () => { + const tags = generator.conditionsToTags(new Map([['person', 'first']])); + expect(tags).toContain('1.PERS'); + }); + + test('maps number conditions', () => { + const tags = generator.conditionsToTags(new Map([['number', 'plural']])); + expect(tags).toContain('PLURAL'); + }); + + test('maps time conditions', () => { + const tags = generator.conditionsToTags(new Map([['time', 'past']])); + expect(tags).toContain('PAST'); + }); + + test('returns UNKNOWN for unmapped conditions', () => { + const tags = generator.conditionsToTags(new Map([['unknownDim', 'unknownVal']])); + expect(tags).toContain('UNKNOWN'); + }); + }); + + describe('end-to-end with synthetic XML', () => { + test('walk via synthetic parser produces correct word forms', () => { + const parser = new Grid3VerbsParser(); + // eslint-disable-next-line @typescript-eslint/no-var-requires + const fs = require('fs'); + const xml = fs.readFileSync(SYNTHETIC_XML, 'utf-8'); + const detailed = parser.parseXmlDetailed(xml); + + const walkForms = detailed.verbs.get('walk'); + expect(walkForms).toBeDefined(); + + const astericsForms = generator.generateFromGrid3Conditions('walk', walkForms!, 'en'); + + expect(astericsForms.length).toBeGreaterThan(1); + const base = astericsForms.find((f) => f.tags.includes('BASE')); + expect(base!.value).toBe('walk'); + }); + }); +});