diff --git a/.git-blame-ignore-revs b/.git-blame-ignore-revs new file mode 100644 index 0000000..0c1c4c9 --- /dev/null +++ b/.git-blame-ignore-revs @@ -0,0 +1,3 @@ +# Prettier and eslint automatic formatting/linting +b79fcf2c596c359151b2955733e0e772feaa2b51 + diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..c657cc3 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,6 @@ +{ + "arrowParens": "avoid", + "endOfLine": "auto", + "singleQuote": true, + "trailingComma": "all" +} diff --git a/README.md b/README.md index bc7a542..ce4f3a2 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,5 @@ [![Build](https://github.com/hsldevcom/digitransit-deployer/workflows/Process%20master%20push/badge.svg)](https://github.com/HSLdevcom/digitransit-deployer/actions) - ## Autodeployer, autorestarter and monitoring for Digitransit deployments We deploy digitransit docker images automatically from Docker Hub. When a Docker image of a kubernetes deployment that we have deployed in our environment is updated at Docker Hub the new image is deployed automatically. @@ -12,16 +11,18 @@ Additionally, some deployments are restarted periodically. ## Env variable configuration These following environmental variables should be added: -* "SLACK_ACCESS_TOKEN" access token used for sending slack messages through a Slack app -* "MONITORING_SLACK_CHANNEL_ID" slack channel id (not the name) for most of the Slack messages -* "ALERT_SLACK_CHANNEL_ID" slack channel id (not the name) for sending messages about image freshness checks -* "DOCKER_USER" docker user that is used for interacting with the Docker API -* "DOCKER_AUTH" docker password that is used for interacting with the Docker API -* "TZ" optional timezone (defaults to "Europe/Helsinki") + +- "SLACK_ACCESS_TOKEN" access token used for sending slack messages through a Slack app +- "MONITORING_SLACK_CHANNEL_ID" slack channel id (not the name) for most of the Slack messages +- "ALERT_SLACK_CHANNEL_ID" slack channel id (not the name) for sending messages about image freshness checks +- "DOCKER_USER" docker user that is used for interacting with the Docker API +- "DOCKER_AUTH" docker password that is used for interacting with the Docker API +- "TZ" optional timezone (defaults to "Europe/Helsinki") ## Prerequisites Deployments should have the following labels defined as deployer uses `app` as an identifier for finding deployments/pods. + ```yaml metadata: name: @@ -39,21 +40,24 @@ spec: Deployer configuration is stored in labels. For example take a look at https://github.com/HSLdevcom/digitransit-kubernetes-deploy/blob/master/roles/aks-apply/files/prod/opentripplanner-hsl-prod.yml where we have labels set as follows: ```yaml - metadata: - labels: - update: "auto" - restartAfterDeployments: "digitransit-ui-hsl-v3" - restartDelay: "5" +metadata: + labels: + update: 'auto' + restartAfterDeployments: 'digitransit-ui-hsl-v3' + restartDelay: '5' ``` ### update: "auto" + Automatic image updates are enabled for deployment. ### restartAfterDeployments: "opentripplanner-data-con-hsl" + Restart this deployment when deployment opentripplanner-data-con-hsl is restarted. It is possible to add multiple dependencies by separating them with underscore ### restartDelay: "1" + Wait at minimum 1 minute before restarting this deployment (because of dependant deployment has restarted) ## Cron style autorestarter configuration @@ -61,19 +65,21 @@ Wait at minimum 1 minute before restarting this deployment (because of dependant Labels are also used for the periodic (cron style) restarts. These labels can coexist with the labels required for the auto deployments. Example label use below: ```yaml - metadata: - labels: - update: "auto" - restartAfterDeployments: "digitransit-ui-hsl-v3" - restartDelay: "1" - restartAt: "04.30" - restartLimitInterval: "240" +metadata: + labels: + update: 'auto' + restartAfterDeployments: 'digitransit-ui-hsl-v3' + restartDelay: '1' + restartAt: '04.30' + restartLimitInterval: '240' ``` ### restartAt: "04.30" + Restarts deployment at 04:30. Attempts to restart deployment stop after deployment has been successfully restarted or an hour has passed. It is possible to define multiple restart points by separating them with underscores, for example "restartAt": "13.00_18.50". ### restartLimitInterval: "240" + Optional label that defines in minutes how long time has to be since the last restart for a restart to trigger at the time defined in "restartAt" label. If "restartLimitInterval" is not defined, the default value will be 1080 minutes (18 hours). ## Deployment image freshness monitoring @@ -83,15 +89,17 @@ Optionally, it can be checked that an image has been updated within the last 12 Example config: ```yaml - metadata: - labels: - update: "auto" - checkImageFreshnessAt: "09.00" - imageFreshnessTitle: "Service_X" +metadata: + labels: + update: 'auto' + checkImageFreshnessAt: '09.00' + imageFreshnessTitle: 'Service_X' ``` ### checkImageFreshnessAt: "04.30" + Label that defines when the check is done in `hh.mm`. If multiple deployments have identical check times, the checks and messaging are grouped together. ### imageFreshnessTitle: "240" + The title which is used for the deployment when messaging about image freshness over slack. diff --git a/eslint.config.js b/eslint.config.js new file mode 100644 index 0000000..e7ca13c --- /dev/null +++ b/eslint.config.js @@ -0,0 +1,8 @@ +import globals from 'globals'; +import pluginJs from '@eslint/js'; + +/** @type {import('eslint').Linter.Config[]} */ +export default [ + { languageOptions: { globals: globals.node } }, + pluginJs.configs.recommended, +]; diff --git a/package-lock.json b/package-lock.json index 0996e6c..4c211e9 100644 --- a/package-lock.json +++ b/package-lock.json @@ -14,9 +14,12 @@ "axios": "^1.7.9" }, "devDependencies": { + "@eslint/js": "^9.18.0", "chai": "^5.1.2", + "eslint": "^9.18.0", + "globals": "^15.14.0", "mocha": "^11.0.1", - "standard": "^17.1.2" + "prettier": "^3.4.2" } }, "node_modules/@dagrejs/graphlib": { @@ -29,20 +32,37 @@ } }, "node_modules/@eslint-community/eslint-utils": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", - "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.1.tgz", + "integrity": "sha512-s3O3waFUrMV8P/XaF/+ZTp1X9XBZW1a4B97ZnjQF2KYWaFD2A8KyFBsrsfSjEmjn3RGWAIuvlneuZm3CUK3jbA==", "dev": true, + "license": "MIT", "dependencies": { - "eslint-visitor-keys": "^3.3.0" + "eslint-visitor-keys": "^3.4.3" }, "engines": { "node": "^12.22.0 || ^14.17.0 || >=16.0.0" }, + "funding": { + "url": "https://opencollective.com/eslint" + }, "peerDependencies": { "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" } }, + "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, "node_modules/@eslint-community/regexpp": { "version": "4.12.1", "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz", @@ -53,17 +73,58 @@ "node": "^12.0.0 || ^14.0.0 || >=16.0.0" } }, + "node_modules/@eslint/config-array": { + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.19.1.tgz", + "integrity": "sha512-fo6Mtm5mWyKjA/Chy1BYTdn5mGJoDNjC7C64ug20ADsRDGrA85bN3uK3MaKbeRkRuuIEAR5N33Jr1pbm411/PA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/object-schema": "^2.1.5", + "debug": "^4.3.1", + "minimatch": "^3.1.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/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==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/@eslint/core": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.10.0.tgz", + "integrity": "sha512-gFHJ+xBOo4G3WRlR1e/3G8A6/KZAH6zcE/hkLRCZTi/B9avAG365QhFA8uOGzTMqgTghpn7/fSnscW++dpMSAw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@types/json-schema": "^7.0.15" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, "node_modules/@eslint/eslintrc": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", - "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.2.0.tgz", + "integrity": "sha512-grOjVNN8P3hjJn/eIETF1wwd12DdnwFDoyceUJLYYdkpbwq3nLi+4fqrTAONx7XDALqlL220wC/RHSC/QTI/0w==", "dev": true, "license": "MIT", "dependencies": { "ajv": "^6.12.4", "debug": "^4.3.2", - "espree": "^9.6.0", - "globals": "^13.19.0", + "espree": "^10.0.1", + "globals": "^14.0.0", "ignore": "^5.2.0", "import-fresh": "^3.2.1", "js-yaml": "^4.1.0", @@ -71,12 +132,25 @@ "strip-json-comments": "^3.1.1" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" } }, + "node_modules/@eslint/eslintrc/node_modules/globals": { + "version": "14.0.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", + "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/@eslint/eslintrc/node_modules/minimatch": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", @@ -91,42 +165,75 @@ } }, "node_modules/@eslint/js": { - "version": "8.57.1", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.1.tgz", - "integrity": "sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==", + "version": "9.18.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.18.0.tgz", + "integrity": "sha512-fK6L7rxcq6/z+AaQMtiFTkvbHkBLNlwyRxHpKawP0x3u9+NC6MQTnFW+AdpwC6gfHTW0051cokQgtTN2FqlxQA==", "dev": true, "license": "MIT", "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/object-schema": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.5.tgz", + "integrity": "sha512-o0bhxnL89h5Bae5T318nFoFzGy+YE5i/gGkoPAgkmTVdRKTiv3p8JHevPiPaMwoloKfEiiaHlawCqaZMqRm+XQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "node_modules/@humanwhocodes/config-array": { - "version": "0.13.0", - "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.13.0.tgz", - "integrity": "sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw==", - "deprecated": "Use @eslint/config-array instead", + "node_modules/@eslint/plugin-kit": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.5.tgz", + "integrity": "sha512-lB05FkqEdUg2AA0xEbUz0SnkXT1LcCTa438W4IWTUh4hdOnVbQyOJ81OrDXsJk/LSiJHubgGEFoR5EHq1NsH1A==", "dev": true, "license": "Apache-2.0", "dependencies": { - "@humanwhocodes/object-schema": "^2.0.3", - "debug": "^4.3.1", - "minimatch": "^3.0.5" + "@eslint/core": "^0.10.0", + "levn": "^0.4.1" }, "engines": { - "node": ">=10.10.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "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==", + "node_modules/@humanfs/core": { + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.1.tgz", + "integrity": "sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==", "dev": true, - "license": "ISC", + "license": "Apache-2.0", + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node": { + "version": "0.16.6", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.6.tgz", + "integrity": "sha512-YuI2ZHQL78Q5HbhDiBA1X4LmYdXCKCMQIfw0pw7piHJwyREFebJUvrQN4cMssyES6x+vfUbx1CIpaQUKYdQZOw==", + "dev": true, + "license": "Apache-2.0", "dependencies": { - "brace-expansion": "^1.1.7" + "@humanfs/core": "^0.19.1", + "@humanwhocodes/retry": "^0.3.0" }, "engines": { - "node": "*" + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node/node_modules/@humanwhocodes/retry": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.1.tgz", + "integrity": "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" } }, "node_modules/@humanwhocodes/module-importer": { @@ -134,6 +241,7 @@ "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", "dev": true, + "license": "Apache-2.0", "engines": { "node": ">=12.22" }, @@ -142,13 +250,19 @@ "url": "https://github.com/sponsors/nzakas" } }, - "node_modules/@humanwhocodes/object-schema": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", - "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", - "deprecated": "Use @eslint/object-schema instead", + "node_modules/@humanwhocodes/retry": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.1.tgz", + "integrity": "sha512-c7hNEllBlenFTHBky65mhq8WD2kbN9Q6gk0bTk8lSBvc554jpXSkST1iePudpt7+A/AQvuHs9EMqjHDXMY1lrA==", "dev": true, - "license": "BSD-3-Clause" + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } }, "node_modules/@isaacs/cliui": { "version": "8.0.2", @@ -330,41 +444,6 @@ "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.0.tgz", "integrity": "sha512-336iVw3rtn2BUK7ORdIAHTyxHGRIHVReokCR3XjbckJMK7ms8FysBfhLR8IXnAgy7T0PTPNBWKiH514FOW/WSg==" }, - "node_modules/@nodelib/fs.scandir": { - "version": "2.1.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", - "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", - "dev": true, - "dependencies": { - "@nodelib/fs.stat": "2.0.5", - "run-parallel": "^1.1.9" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nodelib/fs.stat": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", - "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", - "dev": true, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nodelib/fs.walk": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", - "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", - "dev": true, - "dependencies": { - "@nodelib/fs.scandir": "2.1.5", - "fastq": "^1.6.0" - }, - "engines": { - "node": ">= 8" - } - }, "node_modules/@pkgjs/parseargs": { "version": "0.11.0", "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", @@ -375,10 +454,10 @@ "node": ">=14" } }, - "node_modules/@rtsao/scc": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@rtsao/scc/-/scc-1.1.0.tgz", - "integrity": "sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==", + "node_modules/@types/estree": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.6.tgz", + "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==", "dev": true, "license": "MIT" }, @@ -387,10 +466,10 @@ "resolved": "https://registry.npmjs.org/@types/js-yaml/-/js-yaml-4.0.5.tgz", "integrity": "sha512-FhpRzf927MNQdRZP0J5DLIdTXhjLYzeUTmLAu69mnVksLH9CJY3IuSeEgbKUki7GQZm0WqDkGzyxju2EZGD2wA==" }, - "node_modules/@types/json5": { - "version": "0.0.29", - "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", - "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==", + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", "dev": true, "license": "MIT" }, @@ -432,13 +511,6 @@ "minipass": "^4.0.0" } }, - "node_modules/@ungap/structured-clone": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.1.tgz", - "integrity": "sha512-fEzPV3hSkSMltkw152tJKNARhOupqbH96MZWyRjNaYZOMIzbrTeQDG+MTc6Mr2pgzFQzFxAfmhGDNP5QK++2ZA==", - "dev": true, - "license": "ISC" - }, "node_modules/acorn": { "version": "8.14.0", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.0.tgz", @@ -529,163 +601,6 @@ "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" }, - "node_modules/array-buffer-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.2.tgz", - "integrity": "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "is-array-buffer": "^3.0.5" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array-includes": { - "version": "3.1.8", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.8.tgz", - "integrity": "sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.4", - "is-string": "^1.0.7" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.findlast": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/array.prototype.findlast/-/array.prototype.findlast-1.2.5.tgz", - "integrity": "sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.findlastindex": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.5.tgz", - "integrity": "sha512-zfETvRFA8o7EiNn++N5f/kaCw221hrpGsDmcpndVupkPzEc1Wuf3VgC0qby1BbHs7f5DVYjgtEU2LLh5bqeGfQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.flat": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.3.tgz", - "integrity": "sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.flatmap": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.3.tgz", - "integrity": "sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.tosorted": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/array.prototype.tosorted/-/array.prototype.tosorted-1.1.4.tgz", - "integrity": "sha512-p6Fx8B7b7ZhL/gmUsAy0D15WhvDccw3mnGNbZpi3pmeJdxtWsj2jEaI4Y6oo3XiHfzuSgPwKc04MYt6KgvC/wA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.3", - "es-errors": "^1.3.0", - "es-shim-unscopables": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/arraybuffer.prototype.slice": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.4.tgz", - "integrity": "sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "array-buffer-byte-length": "^1.0.1", - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "is-array-buffer": "^3.0.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/assertion-error": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz", @@ -702,22 +617,6 @@ "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", "license": "MIT" }, - "node_modules/available-typed-arrays": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", - "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "possible-typed-array-names": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/axios": { "version": "1.7.9", "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.9.tgz", @@ -770,79 +669,6 @@ "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", "dev": true }, - "node_modules/builtins": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/builtins/-/builtins-5.1.0.tgz", - "integrity": "sha512-SW9lzGTLvWTP1AY8xeAMZimqDrIaSdLQUcVr9DMef51niJ022Ri87SwRRKYm4A6iHfkPaiVUu/Duw2Wc4J7kKg==", - "dev": true, - "license": "MIT", - "dependencies": { - "semver": "^7.0.0" - } - }, - "node_modules/builtins/node_modules/semver": { - "version": "7.6.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", - "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", - "dev": true, - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/call-bind": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.8.tgz", - "integrity": "sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.0", - "es-define-property": "^1.0.0", - "get-intrinsic": "^1.2.4", - "set-function-length": "^1.2.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/call-bind-apply-helpers": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.1.tgz", - "integrity": "sha512-BhYE+WDaywFg2TBWYNXAE+8B1ATnThNBqXHP5nQu0jWJdVvY2hvkpyB3qOmtmDePiS5/BDQ8wASEWGMWRG148g==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "function-bind": "^1.1.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/call-bound": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.3.tgz", - "integrity": "sha512-YTd+6wGlNlPxSuri7Y6X8tY2dmm12UMH66RpKMhiX6rsk5wXXnYgbUcOt8kiS31/AjfoTOvCsE+w8nZQLQnzHA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/callsites": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", @@ -1001,9 +827,10 @@ "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" }, "node_modules/cross-spawn": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", - "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "license": "MIT", "dependencies": { "path-key": "^3.1.0", "shebang-command": "^2.0.0", @@ -1013,60 +840,6 @@ "node": ">= 8" } }, - "node_modules/data-view-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.2.tgz", - "integrity": "sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/data-view-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.2.tgz", - "integrity": "sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/inspect-js" - } - }, - "node_modules/data-view-byte-offset": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.1.tgz", - "integrity": "sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/debug": { "version": "4.4.0", "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", @@ -1114,42 +887,6 @@ "dev": true, "license": "MIT" }, - "node_modules/define-data-property": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", - "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-define-property": "^1.0.0", - "es-errors": "^1.3.0", - "gopd": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/define-properties": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", - "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.0.1", - "has-property-descriptors": "^1.0.0", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/delayed-stream": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", @@ -1169,33 +906,6 @@ "node": ">=0.3.1" } }, - "node_modules/doctrine": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", - "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", - "dev": true, - "dependencies": { - "esutils": "^2.0.2" - }, - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/dunder-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", - "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "es-errors": "^1.3.0", - "gopd": "^1.2.0" - }, - "engines": { - "node": ">= 0.4" - } - }, "node_modules/eastasianwidth": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", @@ -1207,185 +917,6 @@ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" }, - "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==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-arrayish": "^0.2.1" - } - }, - "node_modules/es-abstract": { - "version": "1.23.8", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.23.8.tgz", - "integrity": "sha512-lfab8IzDn6EpI1ibZakcgS6WsfEBiB+43cuJo+wgylx1xKXf+Sp+YR3vFuQwC/u3sxYwV8Cxe3B0DpVUu/WiJQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "array-buffer-byte-length": "^1.0.2", - "arraybuffer.prototype.slice": "^1.0.4", - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "data-view-buffer": "^1.0.2", - "data-view-byte-length": "^1.0.2", - "data-view-byte-offset": "^1.0.1", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "es-set-tostringtag": "^2.0.3", - "es-to-primitive": "^1.3.0", - "function.prototype.name": "^1.1.8", - "get-intrinsic": "^1.2.6", - "get-symbol-description": "^1.1.0", - "globalthis": "^1.0.4", - "gopd": "^1.2.0", - "has-property-descriptors": "^1.0.2", - "has-proto": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "internal-slot": "^1.1.0", - "is-array-buffer": "^3.0.5", - "is-callable": "^1.2.7", - "is-data-view": "^1.0.2", - "is-regex": "^1.2.1", - "is-shared-array-buffer": "^1.0.4", - "is-string": "^1.1.1", - "is-typed-array": "^1.1.15", - "is-weakref": "^1.1.0", - "math-intrinsics": "^1.1.0", - "object-inspect": "^1.13.3", - "object-keys": "^1.1.1", - "object.assign": "^4.1.7", - "own-keys": "^1.0.0", - "regexp.prototype.flags": "^1.5.3", - "safe-array-concat": "^1.1.3", - "safe-push-apply": "^1.0.0", - "safe-regex-test": "^1.1.0", - "string.prototype.trim": "^1.2.10", - "string.prototype.trimend": "^1.0.9", - "string.prototype.trimstart": "^1.0.8", - "typed-array-buffer": "^1.0.3", - "typed-array-byte-length": "^1.0.3", - "typed-array-byte-offset": "^1.0.4", - "typed-array-length": "^1.0.7", - "unbox-primitive": "^1.1.0", - "which-typed-array": "^1.1.18" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/es-define-property": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", - "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-errors": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", - "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-iterator-helpers": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.2.1.tgz", - "integrity": "sha512-uDn+FE1yrDzyC0pCo961B2IHbdM8y/ACZsKD4dG6WqrjV53BADjwa7D+1aom2rsNVfLyDgU/eigvlJGJ08OQ4w==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.6", - "es-errors": "^1.3.0", - "es-set-tostringtag": "^2.0.3", - "function-bind": "^1.1.2", - "get-intrinsic": "^1.2.6", - "globalthis": "^1.0.4", - "gopd": "^1.2.0", - "has-property-descriptors": "^1.0.2", - "has-proto": "^1.2.0", - "has-symbols": "^1.1.0", - "internal-slot": "^1.1.0", - "iterator.prototype": "^1.1.4", - "safe-array-concat": "^1.1.3" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-object-atoms": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.0.0.tgz", - "integrity": "sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-set-tostringtag": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", - "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-shim-unscopables": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.0.2.tgz", - "integrity": "sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw==", - "dev": true, - "license": "MIT", - "dependencies": { - "hasown": "^2.0.0" - } - }, - "node_modules/es-to-primitive": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.3.0.tgz", - "integrity": "sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-callable": "^1.2.7", - "is-date-object": "^1.0.5", - "is-symbol": "^1.0.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/escalade": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", @@ -1408,429 +939,69 @@ } }, "node_modules/eslint": { - "version": "8.57.1", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.1.tgz", - "integrity": "sha512-ypowyDxpVSYpkXr9WPv2PAZCtNip1Mv5KTW0SCurXv/9iOpcrH9PaqUElksqEB6pChqHGDRCFTyrZlGhnLNGiA==", - "deprecated": "This version is no longer supported. Please see https://eslint.org/version-support for other options.", + "version": "9.18.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.18.0.tgz", + "integrity": "sha512-+waTfRWQlSbpt3KWE+CjrPPYnbq9kfZIYUqapc0uBXyjTp8aYXZDsUH16m39Ryq3NjAVP4tjuF7KaukeqoCoaA==", "dev": true, "license": "MIT", "dependencies": { "@eslint-community/eslint-utils": "^4.2.0", - "@eslint-community/regexpp": "^4.6.1", - "@eslint/eslintrc": "^2.1.4", - "@eslint/js": "8.57.1", - "@humanwhocodes/config-array": "^0.13.0", + "@eslint-community/regexpp": "^4.12.1", + "@eslint/config-array": "^0.19.0", + "@eslint/core": "^0.10.0", + "@eslint/eslintrc": "^3.2.0", + "@eslint/js": "9.18.0", + "@eslint/plugin-kit": "^0.2.5", + "@humanfs/node": "^0.16.6", "@humanwhocodes/module-importer": "^1.0.1", - "@nodelib/fs.walk": "^1.2.8", - "@ungap/structured-clone": "^1.2.0", + "@humanwhocodes/retry": "^0.4.1", + "@types/estree": "^1.0.6", + "@types/json-schema": "^7.0.15", "ajv": "^6.12.4", "chalk": "^4.0.0", - "cross-spawn": "^7.0.2", + "cross-spawn": "^7.0.6", "debug": "^4.3.2", - "doctrine": "^3.0.0", "escape-string-regexp": "^4.0.0", - "eslint-scope": "^7.2.2", - "eslint-visitor-keys": "^3.4.3", - "espree": "^9.6.1", - "esquery": "^1.4.2", + "eslint-scope": "^8.2.0", + "eslint-visitor-keys": "^4.2.0", + "espree": "^10.3.0", + "esquery": "^1.5.0", "esutils": "^2.0.2", "fast-deep-equal": "^3.1.3", - "file-entry-cache": "^6.0.1", + "file-entry-cache": "^8.0.0", "find-up": "^5.0.0", "glob-parent": "^6.0.2", - "globals": "^13.19.0", - "graphemer": "^1.4.0", "ignore": "^5.2.0", "imurmurhash": "^0.1.4", "is-glob": "^4.0.0", - "is-path-inside": "^3.0.3", - "js-yaml": "^4.1.0", "json-stable-stringify-without-jsonify": "^1.0.1", - "levn": "^0.4.1", "lodash.merge": "^4.6.2", "minimatch": "^3.1.2", "natural-compare": "^1.4.0", - "optionator": "^0.9.3", - "strip-ansi": "^6.0.1", - "text-table": "^0.2.0" + "optionator": "^0.9.3" }, "bin": { "eslint": "bin/eslint.js" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/eslint-config-standard": { - "version": "17.1.0", - "resolved": "https://registry.npmjs.org/eslint-config-standard/-/eslint-config-standard-17.1.0.tgz", - "integrity": "sha512-IwHwmaBNtDK4zDHQukFDW5u/aTb8+meQWZvNFWkiGmbWjD6bqyuSSBxxXKkCftCUzc1zwCH2m/baCNDLGmuO5Q==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "engines": { - "node": ">=12.0.0" + "url": "https://eslint.org/donate" }, "peerDependencies": { - "eslint": "^8.0.1", - "eslint-plugin-import": "^2.25.2", - "eslint-plugin-n": "^15.0.0 || ^16.0.0 ", - "eslint-plugin-promise": "^6.0.0" - } - }, - "node_modules/eslint-config-standard-jsx": { - "version": "11.0.0", - "resolved": "https://registry.npmjs.org/eslint-config-standard-jsx/-/eslint-config-standard-jsx-11.0.0.tgz", - "integrity": "sha512-+1EV/R0JxEK1L0NGolAr8Iktm3Rgotx3BKwgaX+eAuSX8D952LULKtjgZD3F+e6SvibONnhLwoTi9DPxN5LvvQ==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "peerDependencies": { - "eslint": "^8.8.0", - "eslint-plugin-react": "^7.28.0" - } - }, - "node_modules/eslint-import-resolver-node": { - "version": "0.3.9", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz", - "integrity": "sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==", - "dev": true, - "license": "MIT", - "dependencies": { - "debug": "^3.2.7", - "is-core-module": "^2.13.0", - "resolve": "^1.22.4" - } - }, - "node_modules/eslint-import-resolver-node/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-module-utils": { - "version": "2.12.0", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.12.0.tgz", - "integrity": "sha512-wALZ0HFoytlyh/1+4wuZ9FJCD/leWHQzzrxJ8+rebyReSLk7LApMyd3WJaLVoN+D5+WIdJyDK1c6JnE65V4Zyg==", - "dev": true, - "license": "MIT", - "dependencies": { - "debug": "^3.2.7" - }, - "engines": { - "node": ">=4" + "jiti": "*" }, "peerDependenciesMeta": { - "eslint": { + "jiti": { "optional": true } } }, - "node_modules/eslint-module-utils/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-plugin-es": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-es/-/eslint-plugin-es-4.1.0.tgz", - "integrity": "sha512-GILhQTnjYE2WorX5Jyi5i4dz5ALWxBIdQECVQavL6s7cI76IZTDWleTHkxz/QT3kvcs2QlGHvKLYsSlPOlPXnQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "eslint-utils": "^2.0.0", - "regexpp": "^3.0.0" - }, - "engines": { - "node": ">=8.10.0" - }, - "funding": { - "url": "https://github.com/sponsors/mysticatea" - }, - "peerDependencies": { - "eslint": ">=4.19.1" - } - }, - "node_modules/eslint-plugin-es/node_modules/eslint-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", - "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", - "dev": true, - "license": "MIT", - "dependencies": { - "eslint-visitor-keys": "^1.1.0" - }, - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/mysticatea" - } - }, - "node_modules/eslint-plugin-es/node_modules/eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=4" - } - }, - "node_modules/eslint-plugin-import": { - "version": "2.31.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.31.0.tgz", - "integrity": "sha512-ixmkI62Rbc2/w8Vfxyh1jQRTdRTF52VxwRVHl/ykPAmqG+Nb7/kNn+byLP0LxPgI7zWA16Jt82SybJInmMia3A==", - "dev": true, - "license": "MIT", - "dependencies": { - "@rtsao/scc": "^1.1.0", - "array-includes": "^3.1.8", - "array.prototype.findlastindex": "^1.2.5", - "array.prototype.flat": "^1.3.2", - "array.prototype.flatmap": "^1.3.2", - "debug": "^3.2.7", - "doctrine": "^2.1.0", - "eslint-import-resolver-node": "^0.3.9", - "eslint-module-utils": "^2.12.0", - "hasown": "^2.0.2", - "is-core-module": "^2.15.1", - "is-glob": "^4.0.3", - "minimatch": "^3.1.2", - "object.fromentries": "^2.0.8", - "object.groupby": "^1.0.3", - "object.values": "^1.2.0", - "semver": "^6.3.1", - "string.prototype.trimend": "^1.0.8", - "tsconfig-paths": "^3.15.0" - }, - "engines": { - "node": ">=4" - }, - "peerDependencies": { - "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 || ^9" - } - }, - "node_modules/eslint-plugin-import/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-plugin-import/node_modules/doctrine": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", - "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "esutils": "^2.0.2" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/eslint-plugin-import/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/eslint-plugin-n": { - "version": "15.7.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-n/-/eslint-plugin-n-15.7.0.tgz", - "integrity": "sha512-jDex9s7D/Qial8AGVIHq4W7NswpUD5DPDL2RH8Lzd9EloWUuvUkHfv4FRLMipH5q2UtyurorBkPeNi1wVWNh3Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "builtins": "^5.0.1", - "eslint-plugin-es": "^4.1.0", - "eslint-utils": "^3.0.0", - "ignore": "^5.1.1", - "is-core-module": "^2.11.0", - "minimatch": "^3.1.2", - "resolve": "^1.22.1", - "semver": "^7.3.8" - }, - "engines": { - "node": ">=12.22.0" - }, - "funding": { - "url": "https://github.com/sponsors/mysticatea" - }, - "peerDependencies": { - "eslint": ">=7.0.0" - } - }, - "node_modules/eslint-plugin-n/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/eslint-plugin-n/node_modules/semver": { - "version": "7.6.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", - "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", - "dev": true, - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/eslint-plugin-promise": { - "version": "6.6.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-6.6.0.tgz", - "integrity": "sha512-57Zzfw8G6+Gq7axm2Pdo3gW/Rx3h9Yywgn61uE/3elTCOePEHVrn2i5CdfBwA1BLK0Q0WqctICIUSqXZW/VprQ==", - "dev": true, - "license": "ISC", - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - }, - "peerDependencies": { - "eslint": "^7.0.0 || ^8.0.0 || ^9.0.0" - } - }, - "node_modules/eslint-plugin-react": { - "version": "7.37.3", - "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.37.3.tgz", - "integrity": "sha512-DomWuTQPFYZwF/7c9W2fkKkStqZmBd3uugfqBYLdkZ3Hii23WzZuOLUskGxB8qkSKqftxEeGL1TB2kMhrce0jA==", - "dev": true, - "license": "MIT", - "dependencies": { - "array-includes": "^3.1.8", - "array.prototype.findlast": "^1.2.5", - "array.prototype.flatmap": "^1.3.3", - "array.prototype.tosorted": "^1.1.4", - "doctrine": "^2.1.0", - "es-iterator-helpers": "^1.2.1", - "estraverse": "^5.3.0", - "hasown": "^2.0.2", - "jsx-ast-utils": "^2.4.1 || ^3.0.0", - "minimatch": "^3.1.2", - "object.entries": "^1.1.8", - "object.fromentries": "^2.0.8", - "object.values": "^1.2.1", - "prop-types": "^15.8.1", - "resolve": "^2.0.0-next.5", - "semver": "^6.3.1", - "string.prototype.matchall": "^4.0.12", - "string.prototype.repeat": "^1.0.0" - }, - "engines": { - "node": ">=4" - }, - "peerDependencies": { - "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8 || ^9.7" - } - }, - "node_modules/eslint-plugin-react/node_modules/doctrine": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", - "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", - "dev": true, - "dependencies": { - "esutils": "^2.0.2" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/eslint-plugin-react/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, - "dependencies": { - "brace-expansion": "^1.1.7" - }, - "engines": { - "node": "*" - } - }, - "node_modules/eslint-plugin-react/node_modules/resolve": { - "version": "2.0.0-next.5", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-2.0.0-next.5.tgz", - "integrity": "sha512-U7WjGVG9sH8tvjW5SmGbQuui75FiyjAX72HX15DwBBwF9dNiQZRQAg9nnPhYy+TUnE0+VcrttuvNI8oSxZcocA==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-core-module": "^2.13.0", - "path-parse": "^1.0.7", - "supports-preserve-symlinks-flag": "^1.0.0" - }, - "bin": { - "resolve": "bin/resolve" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/eslint-scope": { - "version": "7.2.2", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", - "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.2.0.tgz", + "integrity": "sha512-PHlWUfG6lvPc3yvP5A4PNyBL1W8fkDUccmI21JUu/+GKZBoH/W5u6usENXUrWFRsyoW5ACUjFGgAFQp5gUlb/A==", "dev": true, "license": "BSD-2-Clause", "dependencies": { @@ -1838,49 +1009,20 @@ "estraverse": "^5.2.0" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" } }, - "node_modules/eslint-utils": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz", - "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==", - "dev": true, - "license": "MIT", - "dependencies": { - "eslint-visitor-keys": "^2.0.0" - }, - "engines": { - "node": "^10.0.0 || ^12.0.0 || >= 14.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/mysticatea" - }, - "peerDependencies": { - "eslint": ">=5" - } - }, - "node_modules/eslint-utils/node_modules/eslint-visitor-keys": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", - "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", - "dev": true, - "license": "Apache-2.0", - "engines": { - "node": ">=10" - } - }, "node_modules/eslint-visitor-keys": { - "version": "3.4.3", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", - "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", "dev": true, "license": "Apache-2.0", "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" @@ -1891,6 +1033,7 @@ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", "dev": true, + "license": "ISC", "dependencies": { "is-glob": "^4.0.3" }, @@ -1903,6 +1046,7 @@ "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" }, @@ -1911,28 +1055,29 @@ } }, "node_modules/espree": { - "version": "9.6.1", - "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", - "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.3.0.tgz", + "integrity": "sha512-0QYC8b24HWY8zjRnDTL6RiHfDbAWn63qb4LMj1Z4b076A4une81+z03Kg7l7mn/48PUTqoLptSXez8oknU8Clg==", "dev": true, "license": "BSD-2-Clause", "dependencies": { - "acorn": "^8.9.0", + "acorn": "^8.14.0", "acorn-jsx": "^5.3.2", - "eslint-visitor-keys": "^3.4.1" + "eslint-visitor-keys": "^4.2.0" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" } }, "node_modules/esquery": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", - "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", "dev": true, + "license": "BSD-3-Clause", "dependencies": { "estraverse": "^5.1.0" }, @@ -1958,6 +1103,7 @@ "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", "dev": true, + "license": "BSD-2-Clause", "engines": { "node": ">=4.0" } @@ -1967,6 +1113,7 @@ "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", "dev": true, + "license": "BSD-2-Clause", "engines": { "node": ">=0.10.0" } @@ -1992,25 +1139,17 @@ "dev": true, "license": "MIT" }, - "node_modules/fastq": { - "version": "1.15.0", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.15.0.tgz", - "integrity": "sha512-wBrocU2LCXXa+lWBt8RoIRD89Fi8OdABODa/kEnyeyjS5aZO5/GNvI5sEINADqP/h8M29UHTHUb53sUu5Ihqdw==", - "dev": true, - "dependencies": { - "reusify": "^1.0.4" - } - }, "node_modules/file-entry-cache": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", - "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", "dev": true, + "license": "MIT", "dependencies": { - "flat-cache": "^3.0.4" + "flat-cache": "^4.0.0" }, "engines": { - "node": "^10.12.0 || >=12.0.0" + "node": ">=16.0.0" } }, "node_modules/fill-range": { @@ -2051,23 +1190,25 @@ } }, "node_modules/flat-cache": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", - "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", + "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", "dev": true, + "license": "MIT", "dependencies": { - "flatted": "^3.1.0", - "rimraf": "^3.0.2" + "flatted": "^3.2.9", + "keyv": "^4.5.4" }, "engines": { - "node": "^10.12.0 || >=12.0.0" + "node": ">=16" } }, "node_modules/flatted": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.7.tgz", - "integrity": "sha512-5nqDSxl8nn5BSNxyR3n4I6eDmbolI6WT+QqR547RwxQapgjQBmtktdP+HTBb/a/zLsbzERTONyUB5pefh5TtjQ==", - "dev": true + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.2.tgz", + "integrity": "sha512-AiwGJM8YcNOaobumgtng+6NHuOqC3A7MixFeDafM3X9cIUM+xUXoS5Vfgf+OihAYe20fxqNM9yPBXJzRtZ/4eA==", + "dev": true, + "license": "ISC" }, "node_modules/follow-redirects": { "version": "1.15.9", @@ -2089,16 +1230,6 @@ } } }, - "node_modules/for-each": { - "version": "0.3.3", - "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", - "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-callable": "^1.1.3" - } - }, "node_modules/foreground-child": { "version": "3.3.0", "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.0.tgz", @@ -2148,47 +1279,6 @@ "node": "^8.16.0 || ^10.6.0 || >=11.0.0" } }, - "node_modules/function-bind": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", - "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "dev": true, - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/function.prototype.name": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.8.tgz", - "integrity": "sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "functions-have-names": "^1.2.3", - "hasown": "^2.0.2", - "is-callable": "^1.2.7" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/functions-have-names": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", - "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", - "dev": true, - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/get-caller-file": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", @@ -2198,76 +1288,6 @@ "node": "6.* || 8.* || >= 10.*" } }, - "node_modules/get-intrinsic": { - "version": "1.2.6", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.6.tgz", - "integrity": "sha512-qxsEs+9A+u85HhllWJJFicJfPDhRmjzoYdl64aMWW9yRIJmSyxdn8IEkuIM530/7T+lv0TIHd8L6Q/ra0tEoeA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "dunder-proto": "^1.0.0", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "function-bind": "^1.1.2", - "gopd": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "math-intrinsics": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/get-proto": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.0.tgz", - "integrity": "sha512-TtLgOcKaF1nMP2ijJnITkE4nRhbpshHhmzKiuhmSniiwWzovoqwqQ8rNuhf0mXJOqIY5iU+QkUe0CkJYrLsG9w==", - "dev": true, - "license": "MIT", - "dependencies": { - "dunder-proto": "^1.0.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/get-stdin": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-8.0.0.tgz", - "integrity": "sha512-sY22aA6xchAzprjyqmSEQv4UbAAzRN0L2dQB0NlN5acTTK9Don6nhoc3eAbUnpZiCANAMfd/+40kVdKfFygohg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/get-symbol-description": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.1.0.tgz", - "integrity": "sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/glob": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", @@ -2311,78 +1331,18 @@ } }, "node_modules/globals": { - "version": "13.24.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", - "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "version": "15.14.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-15.14.0.tgz", + "integrity": "sha512-OkToC372DtlQeje9/zHIo5CT8lRP/FUgEOKBEhU4e0abL7J7CD24fD9ohiLN5hagG/kWCYj4K5oaxxtj2Z0Dig==", "dev": true, "license": "MIT", - "dependencies": { - "type-fest": "^0.20.2" - }, "engines": { - "node": ">=8" + "node": ">=18" }, "funding": { "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/globalthis": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.4.tgz", - "integrity": "sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-properties": "^1.2.1", - "gopd": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/gopd": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", - "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/graceful-fs": { - "version": "4.2.11", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", - "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", - "dev": true, - "license": "ISC" - }, - "node_modules/graphemer": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", - "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", - "dev": true, - "license": "MIT" - }, - "node_modules/has-bigints": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.1.0.tgz", - "integrity": "sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", @@ -2392,77 +1352,6 @@ "node": ">=8" } }, - "node_modules/has-property-descriptors": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", - "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-define-property": "^1.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-proto": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.2.0.tgz", - "integrity": "sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "dunder-proto": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-symbols": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", - "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-tostringtag": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", - "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-symbols": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/hasown": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", - "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "function-bind": "^1.1.2" - }, - "engines": { - "node": ">= 0.4" - } - }, "node_modules/he": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", @@ -2504,6 +1393,7 @@ "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.8.19" } @@ -2522,78 +1412,6 @@ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" }, - "node_modules/internal-slot": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.1.0.tgz", - "integrity": "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "hasown": "^2.0.2", - "side-channel": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/is-array-buffer": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.5.tgz", - "integrity": "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-arrayish": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", - "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", - "dev": true, - "license": "MIT" - }, - "node_modules/is-async-function": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.0.0.tgz", - "integrity": "sha512-Y1JXKrfykRJGdlDwdKlLpLyMIiWqWvuSd17TvZk68PLAOGOoF4Xyav1z0Xhoi+gCYjZVeC5SI+hYFOfvXmGRCA==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-tostringtag": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-bigint": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.1.0.tgz", - "integrity": "sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-bigints": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-binary-path": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", @@ -2603,88 +1421,7 @@ "binary-extensions": "^2.0.0" }, "engines": { - "node": ">=8" - } - }, - "node_modules/is-boolean-object": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.2.1.tgz", - "integrity": "sha512-l9qO6eFlUETHtuihLcYOaLKByJ1f+N4kthcU9YjHy3N+B3hWv0y/2Nd0mu/7lTFnRQHTrSdXF50HQ3bl5fEnng==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-callable": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", - "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "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-data-view": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.2.tgz", - "integrity": "sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "is-typed-array": "^1.1.13" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-date-object": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.1.0.tgz", - "integrity": "sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">=8" } }, "node_modules/is-extglob": { @@ -2696,22 +1433,6 @@ "node": ">=0.10.0" } }, - "node_modules/is-finalizationregistry": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.1.1.tgz", - "integrity": "sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-fullwidth-code-point": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", @@ -2720,22 +1441,6 @@ "node": ">=8" } }, - "node_modules/is-generator-function": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz", - "integrity": "sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-tostringtag": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-glob": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", @@ -2748,19 +1453,6 @@ "node": ">=0.10.0" } }, - "node_modules/is-map": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", - "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", @@ -2770,32 +1462,6 @@ "node": ">=0.12.0" } }, - "node_modules/is-number-object": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.1.1.tgz", - "integrity": "sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-path-inside": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", - "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", - "dev": true, - "engines": { - "node": ">=8" - } - }, "node_modules/is-plain-obj": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-2.1.0.tgz", @@ -2805,105 +1471,6 @@ "node": ">=8" } }, - "node_modules/is-regex": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.2.1.tgz", - "integrity": "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-set": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", - "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-shared-array-buffer": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.4.tgz", - "integrity": "sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-string": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.1.1.tgz", - "integrity": "sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-symbol": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.1.1.tgz", - "integrity": "sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "has-symbols": "^1.1.0", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-typed-array": { - "version": "1.1.15", - "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.15.tgz", - "integrity": "sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "which-typed-array": "^1.1.16" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-unicode-supported": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", @@ -2916,59 +1483,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-weakmap": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", - "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakref": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.1.0.tgz", - "integrity": "sha512-SXM8Nwyys6nT5WP6pltOwKytLV7FqQ4UiibxVmW+EIosHcmCqkkjViTb5SNssDlkCiEYRP1/pdWUKVvZBmsR2Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakset": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.4.tgz", - "integrity": "sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", - "dev": true, - "license": "MIT" - }, "node_modules/isexe": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", @@ -2982,24 +1496,6 @@ "ws": "*" } }, - "node_modules/iterator.prototype": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/iterator.prototype/-/iterator.prototype-1.1.5.tgz", - "integrity": "sha512-H0dkQoCa3b2VEeKQBOxFph+JAbcrQdE7KC0UkqwpLmv2EC4P41QXP+rqo9wYodACiG5/WM5s9oDApTU8utwj9g==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.1.4", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.6", - "get-proto": "^1.0.0", - "has-symbols": "^1.1.0", - "set-function-name": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, "node_modules/jackspeak": { "version": "3.4.3", "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz", @@ -3024,12 +1520,6 @@ "url": "https://github.com/sponsors/panva" } }, - "node_modules/js-tokens": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", - "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", - "dev": true - }, "node_modules/js-yaml": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", @@ -3050,10 +1540,10 @@ "node": ">= 10.16.0" } }, - "node_modules/json-parse-better-errors": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", - "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", "dev": true, "license": "MIT" }, @@ -3068,20 +1558,8 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", - "dev": true - }, - "node_modules/json5": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", - "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", "dev": true, - "license": "MIT", - "dependencies": { - "minimist": "^1.2.0" - }, - "bin": { - "json5": "lib/cli.js" - } + "license": "MIT" }, "node_modules/jsonpath-plus": { "version": "10.2.0", @@ -3101,17 +1579,14 @@ "node": ">=18.0.0" } }, - "node_modules/jsx-ast-utils": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-3.3.3.tgz", - "integrity": "sha512-fYQHZTZ8jSfmWZ0iyzfwiU4WDX4HpHbMCZ3gPlWYiCl3BoeOTsqKBqnTVfH2rYT7eP5c3sVbeSPHnnJOaTrWiw==", + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", "dev": true, + "license": "MIT", "dependencies": { - "array-includes": "^3.1.5", - "object.assign": "^4.1.3" - }, - "engines": { - "node": ">=4.0" + "json-buffer": "3.0.1" } }, "node_modules/levn": { @@ -3128,33 +1603,6 @@ "node": ">= 0.8.0" } }, - "node_modules/load-json-file": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-5.3.0.tgz", - "integrity": "sha512-cJGP40Jc/VXUsp8/OrnyKyTZ1y6v/dphm3bioS+RrKXjK2BB6wHUd6JptZEFDGgGahMT+InnZO5i1Ei9mpC8Bw==", - "dev": true, - "license": "MIT", - "dependencies": { - "graceful-fs": "^4.1.15", - "parse-json": "^4.0.0", - "pify": "^4.0.1", - "strip-bom": "^3.0.0", - "type-fest": "^0.3.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/load-json-file/node_modules/type-fest": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.3.1.tgz", - "integrity": "sha512-cUGJnCdr4STbePCgqNFbpVNCepa+kAVohJs1sLhxzdH+gnEoOd8VhbYa7pD3zZYGiURWM2xzEII3fQcRizDkYQ==", - "dev": true, - "license": "(MIT OR CC0-1.0)", - "engines": { - "node": ">=6" - } - }, "node_modules/locate-path": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", @@ -3174,7 +1622,8 @@ "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/log-symbols": { "version": "4.1.0", @@ -3192,18 +1641,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/loose-envify": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", - "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", - "dev": true, - "dependencies": { - "js-tokens": "^3.0.0 || ^4.0.0" - }, - "bin": { - "loose-envify": "cli.js" - } - }, "node_modules/loupe": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.1.2.tgz", @@ -3217,16 +1654,6 @@ "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==", "license": "ISC" }, - "node_modules/math-intrinsics": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", - "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, "node_modules/mime-db": { "version": "1.52.0", "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", @@ -3271,16 +1698,6 @@ "balanced-match": "^1.0.0" } }, - "node_modules/minimist": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", - "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", - "dev": true, - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/minipass": { "version": "4.2.8", "resolved": "https://registry.npmjs.org/minipass/-/minipass-4.2.8.tgz", @@ -3489,7 +1906,8 @@ "version": "1.4.0", "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", - "dev": true + "dev": true, + "license": "MIT" }, "node_modules/node-fetch": { "version": "2.7.0", @@ -3505,148 +1923,28 @@ "peerDependencies": { "encoding": "^0.1.0" }, - "peerDependenciesMeta": { - "encoding": { - "optional": true - } - } - }, - "node_modules/normalize-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", - "dev": true, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/oauth4webapi": { - "version": "3.1.4", - "resolved": "https://registry.npmjs.org/oauth4webapi/-/oauth4webapi-3.1.4.tgz", - "integrity": "sha512-eVfN3nZNbok2s/ROifO0UAc5G8nRoLSbrcKJ09OqmucgnhXEfdIQOR4gq1eJH1rN3gV7rNw62bDEgftsgFtBEg==", - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/panva" - } - }, - "node_modules/object-assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", - "dev": true, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/object-inspect": { - "version": "1.13.3", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.3.tgz", - "integrity": "sha512-kDCGIbxkDSXE3euJZZXzc6to7fCrKHNI/hSRQnRuQ+BWjFNzZwiFF8fj/6o2t2G9/jTj8PSIYTfCLelLZEeRpA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object-keys": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", - "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", - "dev": true, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object.assign": { - "version": "4.1.7", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.7.tgz", - "integrity": "sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0", - "has-symbols": "^1.1.0", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object.entries": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.8.tgz", - "integrity": "sha512-cmopxi8VwRIAw/fkijJohSfpef5PdN0pMQJN6VC/ZKvn0LIknWD8KtgY6KlQdEc4tIjcQ3HxSMmnvtzIscdaYQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object.fromentries": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", - "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "peerDependenciesMeta": { + "encoding": { + "optional": true + } } }, - "node_modules/object.groupby": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.3.tgz", - "integrity": "sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==", + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2" - }, "engines": { - "node": ">= 0.4" + "node": ">=0.10.0" } }, - "node_modules/object.values": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.1.tgz", - "integrity": "sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA==", - "dev": true, + "node_modules/oauth4webapi": { + "version": "3.1.4", + "resolved": "https://registry.npmjs.org/oauth4webapi/-/oauth4webapi-3.1.4.tgz", + "integrity": "sha512-eVfN3nZNbok2s/ROifO0UAc5G8nRoLSbrcKJ09OqmucgnhXEfdIQOR4gq1eJH1rN3gV7rNw62bDEgftsgFtBEg==", "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, "funding": { - "url": "https://github.com/sponsors/ljharb" + "url": "https://github.com/sponsors/panva" } }, "node_modules/once": { @@ -3688,24 +1986,6 @@ "node": ">= 0.8.0" } }, - "node_modules/own-keys": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/own-keys/-/own-keys-1.0.1.tgz", - "integrity": "sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "get-intrinsic": "^1.2.6", - "object-keys": "^1.1.1", - "safe-push-apply": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/p-limit": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", @@ -3736,16 +2016,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/p-try": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", - "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, "node_modules/package-json-from-dist": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", @@ -3765,20 +2035,6 @@ "node": ">=6" } }, - "node_modules/parse-json": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", - "integrity": "sha512-aOIos8bujGN93/8Ox/jPLh7RwVnPEysynVFE+fQZyg6jKELEHwzgKdLRFHUgXJL6kylijVSBC4BvN9OmsB48Rw==", - "dev": true, - "license": "MIT", - "dependencies": { - "error-ex": "^1.3.1", - "json-parse-better-errors": "^1.0.1" - }, - "engines": { - "node": ">=4" - } - }, "node_modules/path-exists": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", @@ -3804,13 +2060,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", @@ -3858,106 +2107,6 @@ "url": "https://github.com/sponsors/jonschlinkert" } }, - "node_modules/pify": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", - "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/pkg-conf": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/pkg-conf/-/pkg-conf-3.1.0.tgz", - "integrity": "sha512-m0OTbR/5VPNPqO1ph6Fqbj7Hv6QU7gR/tQW40ZqrL1rjgCU85W6C1bJn0BItuJqnR98PWzw7Z8hHeChD1WrgdQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "find-up": "^3.0.0", - "load-json-file": "^5.2.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/pkg-conf/node_modules/find-up": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", - "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", - "dev": true, - "license": "MIT", - "dependencies": { - "locate-path": "^3.0.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/pkg-conf/node_modules/locate-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", - "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", - "dev": true, - "license": "MIT", - "dependencies": { - "p-locate": "^3.0.0", - "path-exists": "^3.0.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/pkg-conf/node_modules/p-limit": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", - "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", - "dev": true, - "license": "MIT", - "dependencies": { - "p-try": "^2.0.0" - }, - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/pkg-conf/node_modules/p-locate": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", - "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "p-limit": "^2.0.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/pkg-conf/node_modules/path-exists": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", - "integrity": "sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=4" - } - }, - "node_modules/possible-typed-array-names": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", - "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, "node_modules/prelude-ls": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", @@ -3968,15 +2117,20 @@ "node": ">= 0.8.0" } }, - "node_modules/prop-types": { - "version": "15.8.1", - "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz", - "integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==", + "node_modules/prettier": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.2.tgz", + "integrity": "sha512-e9MewbtFo+Fevyuxn/4rrcDAaq0IYxPGLvObpQjiZBMAzB9IGmzlnG9RZy3FFas+eBMu2vA0CszMeduow5dIuQ==", "dev": true, - "dependencies": { - "loose-envify": "^1.4.0", - "object-assign": "^4.1.1", - "react-is": "^16.13.1" + "license": "MIT", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" } }, "node_modules/proxy-from-env": { @@ -3995,26 +2149,6 @@ "node": ">=6" } }, - "node_modules/queue-microtask": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", - "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ] - }, "node_modules/randombytes": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", @@ -4025,12 +2159,6 @@ "safe-buffer": "^5.1.0" } }, - "node_modules/react-is": { - "version": "16.13.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", - "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", - "dev": true - }, "node_modules/readdirp": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", @@ -4043,61 +2171,6 @@ "node": ">=8.10.0" } }, - "node_modules/reflect.getprototypeof": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.9.tgz", - "integrity": "sha512-r0Ay04Snci87djAsI4U+WNRcSw5S4pOH7qFjd/veA5gC7TbqESR3tcj28ia95L/fYUDw11JKP7uqUKUAfVvV5Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "dunder-proto": "^1.0.1", - "es-abstract": "^1.23.6", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "gopd": "^1.2.0", - "which-builtin-type": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/regexp.prototype.flags": { - "version": "1.5.3", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.3.tgz", - "integrity": "sha512-vqlC04+RQoFalODCbCumG2xIOvapzVMHwsyIGM/SIE8fRhFFsXeH8/QQ+s0T0kDAhKc4k30s73/0ydkHQz6HlQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-errors": "^1.3.0", - "set-function-name": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/regexpp": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", - "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/mysticatea" - } - }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", @@ -4107,27 +2180,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-from": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", @@ -4138,16 +2190,6 @@ "node": ">=4" } }, - "node_modules/reusify": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", - "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", - "dev": true, - "engines": { - "iojs": ">=1.0.0", - "node": ">=0.10.0" - } - }, "node_modules/rfc4648": { "version": "1.5.2", "resolved": "https://registry.npmjs.org/rfc4648/-/rfc4648-1.5.2.tgz", @@ -4167,49 +2209,6 @@ "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/run-parallel": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", - "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "dependencies": { - "queue-microtask": "^1.2.2" - } - }, - "node_modules/safe-array-concat": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.3.tgz", - "integrity": "sha512-AURm5f0jYEOydBj7VQlVvDrjeFgthDdEF5H1dP+6mNpoXOMo1quQqJ4wvJDyRZ9+pO3kGWoOdmV08cSv2aJV6Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "has-symbols": "^1.1.0", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">=0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/safe-buffer": { "version": "5.2.1", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", @@ -4231,93 +2230,14 @@ ], "license": "MIT" }, - "node_modules/safe-push-apply": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-push-apply/-/safe-push-apply-1.0.0.tgz", - "integrity": "sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/safe-regex-test": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.1.0.tgz", - "integrity": "sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-regex": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "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_modules/serialize-javascript": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.2.tgz", - "integrity": "sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "randombytes": "^2.1.0" - } - }, - "node_modules/set-function-length": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", - "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "function-bind": "^1.1.2", - "get-intrinsic": "^1.2.4", - "gopd": "^1.0.1", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/set-function-name": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", - "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", + "integrity": "sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==", "dev": true, - "license": "MIT", + "license": "BSD-3-Clause", "dependencies": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "functions-have-names": "^1.2.3", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" + "randombytes": "^2.1.0" } }, "node_modules/shebang-command": { @@ -4339,82 +2259,6 @@ "node": ">=8" } }, - "node_modules/side-channel": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", - "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.3", - "side-channel-list": "^1.0.0", - "side-channel-map": "^1.0.1", - "side-channel-weakmap": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-list": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", - "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-map": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", - "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-weakmap": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", - "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3", - "side-channel-map": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/signal-exit": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", @@ -4427,74 +2271,6 @@ "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/standard": { - "version": "17.1.2", - "resolved": "https://registry.npmjs.org/standard/-/standard-17.1.2.tgz", - "integrity": "sha512-WLm12WoXveKkvnPnPnaFUUHuOB2cUdAsJ4AiGHL2G0UNMrcRAWY2WriQaV8IQ3oRmYr0AWUbLNr94ekYFAHOrA==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "dependencies": { - "eslint": "^8.41.0", - "eslint-config-standard": "17.1.0", - "eslint-config-standard-jsx": "^11.0.0", - "eslint-plugin-import": "^2.27.5", - "eslint-plugin-n": "^15.7.0", - "eslint-plugin-promise": "^6.1.1", - "eslint-plugin-react": "^7.36.1", - "standard-engine": "^15.1.0", - "version-guard": "^1.1.1" - }, - "bin": { - "standard": "bin/cmd.cjs" - }, - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - } - }, - "node_modules/standard-engine": { - "version": "15.1.0", - "resolved": "https://registry.npmjs.org/standard-engine/-/standard-engine-15.1.0.tgz", - "integrity": "sha512-VHysfoyxFu/ukT+9v49d4BRXIokFRZuH3z1VRxzFArZdjSCFpro6rEIU3ji7e4AoAtuSfKBkiOmsrDqKW5ZSRw==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "dependencies": { - "get-stdin": "^8.0.0", - "minimist": "^1.2.6", - "pkg-conf": "^3.1.0", - "xdg-basedir": "^4.0.0" - }, - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - } - }, "node_modules/stream-buffers": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/stream-buffers/-/stream-buffers-3.0.2.tgz", @@ -4531,104 +2307,6 @@ "node": ">=8" } }, - "node_modules/string.prototype.matchall": { - "version": "4.0.12", - "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.12.tgz", - "integrity": "sha512-6CC9uyBL+/48dYizRf7H7VAYCMCNTBeM78x/VTUe9bFEaxBepPJDa1Ow99LqI/1yF7kuy7Q3cQsYMrcjGUcskA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.6", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.6", - "gopd": "^1.2.0", - "has-symbols": "^1.1.0", - "internal-slot": "^1.1.0", - "regexp.prototype.flags": "^1.5.3", - "set-function-name": "^2.0.2", - "side-channel": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.repeat": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/string.prototype.repeat/-/string.prototype.repeat-1.0.0.tgz", - "integrity": "sha512-0u/TldDbKD8bFCQ/4f5+mNRrXwZ8hg2w7ZR8wa16e8z9XpePWl3eGEcUD0OXpEH/VJH/2G3gjUtR3ZOiBe2S/w==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" - } - }, - "node_modules/string.prototype.trim": { - "version": "1.2.10", - "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.10.tgz", - "integrity": "sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-data-property": "^1.1.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-object-atoms": "^1.0.0", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimend": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.9.tgz", - "integrity": "sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimstart": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", - "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/strip-ansi": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", @@ -4653,16 +2331,6 @@ "node": ">=8" } }, - "node_modules/strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=4" - } - }, "node_modules/strip-json-comments": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", @@ -4690,19 +2358,6 @@ "url": "https://github.com/chalk/supports-color?sponsor=1" } }, - "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/tar": { "version": "7.4.3", "resolved": "https://registry.npmjs.org/tar/-/tar-7.4.3.tgz", @@ -4729,12 +2384,6 @@ "node": ">=16 || 14 >=14.17" } }, - "node_modules/text-table": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", - "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", - "dev": true - }, "node_modules/tmp-promise": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/tmp-promise/-/tmp-promise-3.0.3.tgz", @@ -4772,19 +2421,6 @@ "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", "license": "MIT" }, - "node_modules/tsconfig-paths": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz", - "integrity": "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/json5": "^0.0.29", - "json5": "^1.0.2", - "minimist": "^1.2.6", - "strip-bom": "^3.0.0" - } - }, "node_modules/type-check": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", @@ -4798,116 +2434,6 @@ "node": ">= 0.8.0" } }, - "node_modules/type-fest": { - "version": "0.20.2", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", - "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", - "dev": true, - "license": "(MIT OR CC0-1.0)", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/typed-array-buffer": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.3.tgz", - "integrity": "sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-typed-array": "^1.1.14" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/typed-array-byte-length": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.3.tgz", - "integrity": "sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.14" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-byte-offset": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.4.tgz", - "integrity": "sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.15", - "reflect.getprototypeof": "^1.0.9" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-length": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.7.tgz", - "integrity": "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "is-typed-array": "^1.1.13", - "possible-typed-array-names": "^1.0.0", - "reflect.getprototypeof": "^1.0.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/unbox-primitive": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.1.0.tgz", - "integrity": "sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-bigints": "^1.0.2", - "has-symbols": "^1.1.0", - "which-boxed-primitive": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/undici-types": { "version": "6.20.0", "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.20.0.tgz", @@ -4924,16 +2450,6 @@ "punycode": "^2.1.0" } }, - "node_modules/version-guard": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/version-guard/-/version-guard-1.1.3.tgz", - "integrity": "sha512-JwPr6erhX53EWH/HCSzfy1tTFrtPXUe927wdM1jqBBeYp1OM+qPHjWbsvv6pIBduqdgxxS+ScfG7S28pzyr2DQ==", - "dev": true, - "license": "0BSD", - "engines": { - "node": ">=0.10.48" - } - }, "node_modules/webidl-conversions": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", @@ -4964,94 +2480,6 @@ "node": ">= 8" } }, - "node_modules/which-boxed-primitive": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.1.1.tgz", - "integrity": "sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-bigint": "^1.1.0", - "is-boolean-object": "^1.2.1", - "is-number-object": "^1.1.1", - "is-string": "^1.1.1", - "is-symbol": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-builtin-type": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.2.1.tgz", - "integrity": "sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "function.prototype.name": "^1.1.6", - "has-tostringtag": "^1.0.2", - "is-async-function": "^2.0.0", - "is-date-object": "^1.1.0", - "is-finalizationregistry": "^1.1.0", - "is-generator-function": "^1.0.10", - "is-regex": "^1.2.1", - "is-weakref": "^1.0.2", - "isarray": "^2.0.5", - "which-boxed-primitive": "^1.1.0", - "which-collection": "^1.0.2", - "which-typed-array": "^1.1.16" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-collection": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", - "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-map": "^2.0.3", - "is-set": "^2.0.3", - "is-weakmap": "^2.0.2", - "is-weakset": "^2.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-typed-array": { - "version": "1.1.18", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.18.tgz", - "integrity": "sha512-qEcY+KJYlWyLH9vNbsr6/5j59AXk5ni5aakf8ldzBvGde6Iz4sxZGkJyWSAueTG7QhOvNRYb1lDdFmL5Td0QKA==", - "dev": true, - "license": "MIT", - "dependencies": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/word-wrap": { "version": "1.2.5", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", @@ -5130,16 +2558,6 @@ } } }, - "node_modules/xdg-basedir": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", - "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=8" - } - }, "node_modules/y18n": { "version": "5.0.8", "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", diff --git a/package.json b/package.json index 395ef04..71c78a0 100644 --- a/package.json +++ b/package.json @@ -6,7 +6,9 @@ "scripts": { "start": "node src/index.js", "build": "docker build -t hsldevcom/digitransit-deployer .", - "test": "node_modules/.bin/standard && TZ=Europe/Helsinki node ./node_modules/mocha/bin/mocha" + "format": "prettier --write .", + "lint": "eslint .", + "test": "npm run format && npm run lint && TZ=Europe/Helsinki node ./node_modules/mocha/bin/mocha" }, "author": "", "license": "(AGPL-3.0 OR EUPL-1.2)", @@ -17,8 +19,11 @@ "axios": "^1.7.9" }, "devDependencies": { + "@eslint/js": "^9.18.0", "chai": "^5.1.2", + "eslint": "^9.18.0", + "globals": "^15.14.0", "mocha": "^11.0.1", - "standard": "^17.1.2" + "prettier": "^3.4.2" } } diff --git a/src/cron-deployment-restarter.js b/src/cron-deployment-restarter.js index fbac2f0..0639af4 100644 --- a/src/cron-deployment-restarter.js +++ b/src/cron-deployment-restarter.js @@ -1,4 +1,4 @@ -import { build, isSubGraphStable } from './graph.js' +import { build, isSubGraphStable } from './graph.js'; /* * Automatically restarts deployments if they have a restartAt label defined. @@ -13,57 +13,68 @@ import { build, isSubGraphStable } from './graph.js' */ const getDateObject = ([hour, minute]) => { - const dateObject = new Date() - dateObject.setHours(parseInt(hour)) - dateObject.setMinutes(parseInt(minute)) - return dateObject -} + const dateObject = new Date(); + dateObject.setHours(parseInt(hour)); + dateObject.setMinutes(parseInt(minute)); + return dateObject; +}; export default { command: (deployments, context) => { - const deploymentGraph = build(deployments) - const nowEpoch = new Date().getTime() - let attemptedRestart = false - deployments.filter((deployment) => deployment.metadata.labels.restartAt) + const deploymentGraph = build(deployments); + const nowEpoch = new Date().getTime(); + let attemptedRestart = false; + deployments + .filter(deployment => deployment.metadata.labels.restartAt) .forEach(deployment => { - const deploymentDateEpoch = deployment.version - const deploymentLabels = deployment.metadata.labels - const deploymentId = deploymentLabels.app + const deploymentDateEpoch = deployment.version; + const deploymentLabels = deployment.metadata.labels; + const deploymentId = deploymentLabels.app; const restartIntervalMins = - parseInt(deploymentLabels.restartLimitInterval) || 60 * 18 + parseInt(deploymentLabels.restartLimitInterval) || 60 * 18; deploymentLabels.restartAt .split('_') - .filter((time) => /\S/.test(time)) // remove elements that consists of just whitespace + .filter(time => /\S/.test(time)) // remove elements that consists of just whitespace .forEach(restartTime => { if (!attemptedRestart) { - const trimmedTime = restartTime.replace(/\s/g, '') - const timeArray = trimmedTime.split('.') - const nextHour = parseInt(timeArray[0]) + 1 + const trimmedTime = restartTime.replace(/\s/g, ''); + const timeArray = trimmedTime.split('.'); + const nextHour = parseInt(timeArray[0]) + 1; - const cronDate = getDateObject(timeArray) + const cronDate = getDateObject(timeArray); // One hour later - const cronDateUpperLimit = getDateObject([nextHour, timeArray[1]]) + const cronDateUpperLimit = getDateObject([ + nextHour, + timeArray[1], + ]); - if (nowEpoch - deploymentDateEpoch >= restartIntervalMins * 60 * 1000 && + if ( + nowEpoch - deploymentDateEpoch >= + restartIntervalMins * 60 * 1000 && nowEpoch >= cronDate.getTime() && - nowEpoch <= cronDateUpperLimit.getTime()) { + nowEpoch <= cronDateUpperLimit.getTime() + ) { if (isSubGraphStable(deploymentGraph, deploymentId)) { - console.log('Restarting deployment %s', deploymentId) - context.kubernetes.restartDeployment(deploymentId) - .then((r) => { - console.log('Restart called: %s', JSON.stringify(r)) + console.log('Restarting deployment %s', deploymentId); + context.kubernetes + .restartDeployment(deploymentId) + .then(r => { + console.log('Restart called: %s', JSON.stringify(r)); }) - .catch((err) => console.log(err)) - attemptedRestart = true + .catch(err => console.log(err)); + attemptedRestart = true; } else { - console.log('Delaying restart for %s (subgraph not stable)', deploymentId) + console.log( + 'Delaying restart for %s (subgraph not stable)', + deploymentId, + ); } } else { - console.log('No need to update %s', deploymentId) + console.log('No need to update %s', deploymentId); } } - }) - }) - } -} + }); + }); + }, +}; diff --git a/src/dep-deployment-restarter.js b/src/dep-deployment-restarter.js index 3d4a308..3a630ce 100644 --- a/src/dep-deployment-restarter.js +++ b/src/dep-deployment-restarter.js @@ -1,6 +1,6 @@ -import graphlib from '@dagrejs/graphlib' -import { build, isSubGraphStable, deploymentsNeedingRestart } from './graph.js' -import { postMonitoringSlackMessage } from './util.js' +import graphlib from '@dagrejs/graphlib'; +import { build, isSubGraphStable, deploymentsNeedingRestart } from './graph.js'; +import { postMonitoringSlackMessage } from './util.js'; /* * Automatically restarts dependand deployments in controlled manner. This is @@ -29,36 +29,45 @@ import { postMonitoringSlackMessage } from './util.js' */ export default { command: (deployments, context) => { - const NOW = new Date().getTime() + const NOW = new Date().getTime(); - const deploymentGraph = build(deployments) + const deploymentGraph = build(deployments); if (graphlib.alg.findCycles(deploymentGraph).length > 0) { - console.log('Bummer! Graph has cycle, %s', deploymentGraph.toJSON()) - postMonitoringSlackMessage('Deployments are configured to restart each other in a cycle.') + console.log('Bummer! Graph has cycle, %s', deploymentGraph.toJSON()); + postMonitoringSlackMessage( + 'Deployments are configured to restart each other in a cycle.', + ); } else { - deploymentsNeedingRestart(deploymentGraph).filter(({ from, value }) => { - console.log('deployment %s needs restart', from) - // check that enough time has passed after all depedency restarts - for (const deploymentId of deploymentGraph.successors(from)) { - const deployment = deploymentGraph.node(deploymentId) - console.log('checking %s %s', NOW, deployment.version) - if (NOW <= deployment.version + value.delay) { - return false + deploymentsNeedingRestart(deploymentGraph) + .filter(({ from, value }) => { + console.log('deployment %s needs restart', from); + // check that enough time has passed after all depedency restarts + for (const deploymentId of deploymentGraph.successors(from)) { + const deployment = deploymentGraph.node(deploymentId); + console.log('checking %s %s', NOW, deployment.version); + if (NOW <= deployment.version + value.delay) { + return false; + } } - } - return true - }).filter(({ from }) => { - if (!isSubGraphStable(deploymentGraph, from)) { - console.log('Sub Graph for %s is not stable, delaying restart', from) - return false - } - return true - }).forEach(({ from }) => { - console.log('Restarting deployment %s', from) - context.kubernetes.restartDeployment(from) - .then((e) => console.log('Restarted: %s', JSON.stringify(e))) - .catch((err) => console.log(err)) - }) + return true; + }) + .filter(({ from }) => { + if (!isSubGraphStable(deploymentGraph, from)) { + console.log( + 'Sub Graph for %s is not stable, delaying restart', + from, + ); + return false; + } + return true; + }) + .forEach(({ from }) => { + console.log('Restarting deployment %s', from); + context.kubernetes + .restartDeployment(from) + .then(e => console.log('Restarted: %s', JSON.stringify(e))) + .catch(err => console.log(err)); + }); } - } -} + }, +}; diff --git a/src/dockerRepo.js b/src/dockerRepo.js index eaee10d..33d30a6 100644 --- a/src/dockerRepo.js +++ b/src/dockerRepo.js @@ -1,45 +1,50 @@ const loginRequest = `{ "username": "${process.env.DOCKER_USER}", "password": "${process.env.DOCKER_AUTH}" -}` +}`; export default { getImageDate: function (repoAndRef) { - const [image, tag] = repoAndRef.split(':') - const [namespace, repository] = image.split('/') - const url = `https://hub.docker.com/v2/namespaces/${namespace}/repositories/${repository}/tags/${tag || 'latest'}` - const tokenUrl = 'https://hub.docker.com/v2/users/login/' + const [image, tag] = repoAndRef.split(':'); + const [namespace, repository] = image.split('/'); + const url = `https://hub.docker.com/v2/namespaces/${namespace}/repositories/${repository}/tags/${tag || 'latest'}`; + const tokenUrl = 'https://hub.docker.com/v2/users/login/'; return fetch(tokenUrl, { method: 'POST', headers: { - 'Content-Type': 'application/json' + 'Content-Type': 'application/json', }, - body: loginRequest - }).then(res => { - if (res.ok) { - return res.json() - } else { - console.log('failed to get access token from docker hub') - } - }).then(body => { - const token = body?.token - return fetch(url, { - headers: { - 'Content-Type': 'application/json', - Authorization: `JWT ${token}` + body: loginRequest, + }) + .then(res => { + if (res.ok) { + return res.json(); + } else { + console.log('failed to get access token from docker hub'); } }) - }).then(res => { - if (res.ok) { - return res.json() - } else { - console.log(`failed to fetch data for ${repoAndRef} from docker hub`) - } - }).then(data => { - return data && Date.parse(data.last_updated) - }).catch(err => { - console.log(err) - }) - } -} + .then(body => { + const token = body?.token; + return fetch(url, { + headers: { + 'Content-Type': 'application/json', + Authorization: `JWT ${token}`, + }, + }); + }) + .then(res => { + if (res.ok) { + return res.json(); + } else { + console.log(`failed to fetch data for ${repoAndRef} from docker hub`); + } + }) + .then(data => { + return data && Date.parse(data.last_updated); + }) + .catch(err => { + console.log(err); + }); + }, +}; diff --git a/src/graph.js b/src/graph.js index f61f14c..af874a9 100644 --- a/src/graph.js +++ b/src/graph.js @@ -1,105 +1,129 @@ -import { Graph } from '@dagrejs/graphlib' -import { postMonitoringSlackMessage } from './util.js' +import { Graph } from '@dagrejs/graphlib'; +import { postMonitoringSlackMessage } from './util.js'; -function addDepEdges (graph, deployment, deployments) { - const deploymentLabels = deployment.metadata.labels +function addDepEdges(graph, deployment, deployments) { + const deploymentLabels = deployment.metadata.labels; const dependencies = deploymentLabels.restartAfterDeployments .split('_') - .filter((unfilteredDeployment) => /\S/.test(unfilteredDeployment)) // remove elements that consists of just whitespace - const delay = (deploymentLabels.restartDelay || 5) * 60 * 1000 - const deploymentName = deploymentLabels.app + .filter(unfilteredDeployment => /\S/.test(unfilteredDeployment)); // remove elements that consists of just whitespace + const delay = (deploymentLabels.restartDelay || 5) * 60 * 1000; + const deploymentName = deploymentLabels.app; dependencies.forEach(dependency => { - if (deployments.filter(deploymentInstance => (deploymentInstance.metadata.labels.app === dependency)).length > 0) { - graph.setEdge(deploymentName, dependency, { delay }) + if ( + deployments.filter( + deploymentInstance => + deploymentInstance.metadata.labels.app === dependency, + ).length > 0 + ) { + graph.setEdge(deploymentName, dependency, { delay }); } else { - console.log(`${dependency} does not exist but is defined as a dependency for a deployment`) - postMonitoringSlackMessage(`${dependency} does not exist but is defined as a dependency for a deployment`) + console.log( + `${dependency} does not exist but is defined as a dependency for a deployment`, + ); + postMonitoringSlackMessage( + `${dependency} does not exist but is defined as a dependency for a deployment`, + ); } - }) + }); } -function needsRestart (graph, from, to, edge) { +function needsRestart(graph, from, to, edge) { // needs restart if deployment time is smaller than dependency time + delay - const deploymentTime = graph.node(from).version - const dependencyTime = graph.node(to).version - const needsStart = dependencyTime + edge.delay > deploymentTime - return needsStart + const deploymentTime = graph.node(from).version; + const dependencyTime = graph.node(to).version; + const needsStart = dependencyTime + edge.delay > deploymentTime; + return needsStart; } -const deploymentIsStable = (deployment) => - deployment.status.replicas > 0 && deployment.status.readyReplicas === deployment.status.replicas && - deployment.status.updatedReplicas === deployment.status.replicas && deployment.status.availableReplicas === deployment.status.replicas +const deploymentIsStable = deployment => + deployment.status.replicas > 0 && + deployment.status.readyReplicas === deployment.status.replicas && + deployment.status.updatedReplicas === deployment.status.replicas && + deployment.status.availableReplicas === deployment.status.replicas; -export function hasPendingDependentRestarts (graph, deploymentId) { +export function hasPendingDependentRestarts(graph, deploymentId) { // has pending dependent restart if the deploymentId or any vertex that deployment // has Path to has pending restarts - const outEdges = graph.outEdges(deploymentId) + const outEdges = graph.outEdges(deploymentId); for (const edge of outEdges) { - const edgeData = graph.edge(edge) - const dependency = edge.w - console.log('next checking dependency %s %s %s', deploymentId, dependency, edgeData) + const edgeData = graph.edge(edge); + const dependency = edge.w; + console.log( + 'next checking dependency %s %s %s', + deploymentId, + dependency, + edgeData, + ); if (needsRestart(graph, deploymentId, dependency, edgeData)) { - return true + return true; } if (hasPendingDependentRestarts(graph, dependency)) { - return true + return true; } } - return false + return false; } -export function build (deployments) { - const graph = new Graph({ directed: true }) +export function build(deployments) { + const graph = new Graph({ directed: true }); deployments.forEach(deployment => { - graph.setNode(deployment.metadata.labels.app, deployment) - }) + graph.setNode(deployment.metadata.labels.app, deployment); + }); deployments.forEach(deployment => { if (deployment.metadata.labels.restartAfterDeployments) { - addDepEdges(graph, deployment, deployments) + addDepEdges(graph, deployment, deployments); } - }) - return graph + }); + return graph; } -export function isSubGraphStable (graph, vertexId) { +export function isSubGraphStable(graph, vertexId) { // sub graph is stable if all vertexes accessible from the vertex for (const nodeId of graph.successors(vertexId)) { - const deployment = graph.node(nodeId) - if (!deploymentIsStable(deployment)) return false + const deployment = graph.node(nodeId); + if (!deploymentIsStable(deployment)) return false; } - return true + return true; } -export function deploymentsNeedingRestart (graph) { - const deployments = [] +export function deploymentsNeedingRestart(graph) { + const deployments = []; for (const edge of graph.edges()) { - const from = edge.v - const to = edge.w - const value = graph.edge(edge) + const from = edge.v; + const to = edge.w; + const value = graph.edge(edge); if (needsRestart(graph, from, to, value)) { - deployments.push({ from, to, value }) + deployments.push({ from, to, value }); } } - return deployments + return deployments; } -export function deploymentsNeedingImageFreshnessCheck (graph, currentDate) { - const deployments = [] +export function deploymentsNeedingImageFreshnessCheck(graph, currentDate) { + const deployments = []; for (const node of graph.nodes()) { - const deployment = graph.node(node) - const checkTime = deployment.metadata.labels.checkImageFreshnessAt + const deployment = graph.node(node); + const checkTime = deployment.metadata.labels.checkImageFreshnessAt; if (checkTime) { // time format is hh.mm - const checkTimeParts = checkTime.split('.') - const checkDate = new Date(currentDate.getFullYear(), currentDate.getMonth(), currentDate.getDate(), checkTimeParts[0], checkTimeParts[1]) - const timeDifferenceSeconds = Math.round((currentDate.getTime() - checkDate.getTime()) / 1000) + const checkTimeParts = checkTime.split('.'); + const checkDate = new Date( + currentDate.getFullYear(), + currentDate.getMonth(), + currentDate.getDate(), + checkTimeParts[0], + checkTimeParts[1], + ); + const timeDifferenceSeconds = Math.round( + (currentDate.getTime() - checkDate.getTime()) / 1000, + ); // Between 0 and 5 minutes since the checkTime, this is to avoid duplicate checks if (timeDifferenceSeconds >= 0 && timeDifferenceSeconds <= 5 * 60) { - deployments.push(deployment) + deployments.push(deployment); } } } - return deployments + return deployments; } diff --git a/src/image-deployer.js b/src/image-deployer.js index 67eea31..0e8fb51 100644 --- a/src/image-deployer.js +++ b/src/image-deployer.js @@ -1,63 +1,80 @@ -import { build, isSubGraphStable } from './graph.js' +import { build, isSubGraphStable } from './graph.js'; -const logError = (e) => { - console.error('Error occurred %s', JSON.stringify(e)) -} +const logError = e => { + console.error('Error occurred %s', JSON.stringify(e)); +}; -const COOL_OFF_PERIOD = 60 * 60 * 1000 // 1 hour +const COOL_OFF_PERIOD = 60 * 60 * 1000; // 1 hour /* * Automatically deploys new container versions for deployments that are tagged with update: "auto" and have imagePullPolicy configured as Always. */ export default { command: function (deployments, context) { - const deploymentGraph = build(deployments) + const deploymentGraph = build(deployments); - const NOW = new Date().getTime() + const NOW = new Date().getTime(); - deployments.filter((deployment) => deployment.metadata.labels.update === 'auto' && - deployment.spec.template.spec.containers[0].imagePullPolicy === 'Always') + deployments + .filter( + deployment => + deployment.metadata.labels.update === 'auto' && + deployment.spec.template.spec.containers[0].imagePullPolicy === + 'Always', + ) .forEach(deployment => { - const deploymentLabels = deployment.metadata.labels - const deploymentId = deploymentLabels.app - const dependencies = [] - dependencies.push(deployment.spec.template.spec.containers[0].image) - const promises = [] + const deploymentLabels = deployment.metadata.labels; + const deploymentId = deploymentLabels.app; + const dependencies = []; + dependencies.push(deployment.spec.template.spec.containers[0].image); + const promises = []; for (let i = 0; i < dependencies.length; i++) { - const dependency = dependencies[i] - promises.push(new Promise((resolve) => { - context.dockerRepo.getImageDate(dependency).then(repoImageDate => { - const deploymentDate = deployment.version - if (repoImageDate && repoImageDate > deploymentDate) { - if (NOW > deploymentDate + COOL_OFF_PERIOD) { - if (isSubGraphStable(deploymentGraph, deploymentId)) { - resolve('restart') + const dependency = dependencies[i]; + promises.push( + new Promise(resolve => { + context.dockerRepo + .getImageDate(dependency) + .then(repoImageDate => { + const deploymentDate = deployment.version; + if (repoImageDate && repoImageDate > deploymentDate) { + if (NOW > deploymentDate + COOL_OFF_PERIOD) { + if (isSubGraphStable(deploymentGraph, deploymentId)) { + resolve('restart'); + } else { + console.log( + 'Delaying restart for %s (subgraph not stable)', + deploymentId, + ); + resolve(null); + } + } else { + console.log( + 'Delaying restart for %s (cool off period)', + deploymentId, + ); + resolve(null); + } } else { - console.log('Delaying restart for %s (subgraph not stable)', deploymentId) - resolve(null) + console.log('No need to update %s', deploymentId); + resolve(null); } - } else { - console.log('Delaying restart for %s (cool off period)', deploymentId) - resolve(null) - } - } else { - console.log('No need to update %s', deploymentId) - resolve(null) - } - }).catch((err) => { - logError(err) - resolve(null) - }) - })) + }) + .catch(err => { + logError(err); + resolve(null); + }); + }), + ); } - Promise.all(promises).then((values) => { + Promise.all(promises).then(values => { if (values.indexOf('restart') >= 0) { - console.log('Restarting deployment %s', deploymentId) - context.kubernetes.restartDeployment(deploymentId) - .then((r) => console.log('Restart called: %s', JSON.stringify(r))) - .catch((err) => console.log(err)) + console.log('Restarting deployment %s', deploymentId); + context.kubernetes + .restartDeployment(deploymentId) + .then(r => console.log('Restart called: %s', JSON.stringify(r))) + .catch(err => console.log(err)); } - }) - }) - } -} + }); + }); + }, +}; diff --git a/src/image-freshness-monitor.js b/src/image-freshness-monitor.js index 9fb24dd..a70b284 100644 --- a/src/image-freshness-monitor.js +++ b/src/image-freshness-monitor.js @@ -1,5 +1,5 @@ -import { build, deploymentsNeedingImageFreshnessCheck } from './graph.js' -import { postAlertSlackMessage } from './util.js' +import { build, deploymentsNeedingImageFreshnessCheck } from './graph.js'; +import { postAlertSlackMessage } from './util.js'; /* * Automatically checks that the image + tag combination used by the deployment @@ -13,41 +13,61 @@ import { postAlertSlackMessage } from './util.js' */ export default { command: (deployments, context) => { - console.log('Checking for a need to do image freshness checks') - const deploymentGraph = build(deployments) - const now = new Date() - const deploymentsNeedingCheck = deploymentsNeedingImageFreshnessCheck(deploymentGraph, now) + console.log('Checking for a need to do image freshness checks'); + const deploymentGraph = build(deployments); + const now = new Date(); + const deploymentsNeedingCheck = deploymentsNeedingImageFreshnessCheck( + deploymentGraph, + now, + ); if (deploymentsNeedingCheck.length === 0) { - console.log('Found no deployments that need an image freshness check') - return + console.log('Found no deployments that need an image freshness check'); + return; } - const promises = [] + const promises = []; deploymentsNeedingCheck.forEach(deployment => { - const deploymentId = deployment.metadata.labels.app - const deploymentTitle = deployment.metadata.labels.imageFreshnessTitle || deploymentId - const image = deployment.spec.template.spec.containers[0].image - console.log(`Deployment ${deploymentId} needs image freshness check`) - promises.push(new Promise((resolve) => { - context.dockerRepo.getImageDate(image).then(repoImageDate => { - // check that image is older than 12 hours old - if (repoImageDate && repoImageDate < now.getTime() - 12 * 60 * 60 * 1000) { - console.log('%s image has not been updated within the last 12 hours', deploymentId) - resolve(deploymentTitle) - } else { - console.log('%s image has been updated within the last 12 hours', deploymentId) - resolve(null) - } - }).catch((err) => { - console.log(err) - resolve(null) - }) - })) - }) - Promise.all(promises).then((values) => { - const deploymentsWithOldImages = values.filter(value => value != null) + const deploymentId = deployment.metadata.labels.app; + const deploymentTitle = + deployment.metadata.labels.imageFreshnessTitle || deploymentId; + const image = deployment.spec.template.spec.containers[0].image; + console.log(`Deployment ${deploymentId} needs image freshness check`); + promises.push( + new Promise(resolve => { + context.dockerRepo + .getImageDate(image) + .then(repoImageDate => { + // check that image is older than 12 hours old + if ( + repoImageDate && + repoImageDate < now.getTime() - 12 * 60 * 60 * 1000 + ) { + console.log( + '%s image has not been updated within the last 12 hours', + deploymentId, + ); + resolve(deploymentTitle); + } else { + console.log( + '%s image has been updated within the last 12 hours', + deploymentId, + ); + resolve(null); + } + }) + .catch(err => { + console.log(err); + resolve(null); + }); + }), + ); + }); + Promise.all(promises).then(values => { + const deploymentsWithOldImages = values.filter(value => value != null); if (deploymentsWithOldImages.length > 0) { - postAlertSlackMessage(`:boom: These have not been updated within the last 12 hours: ${deploymentsWithOldImages.join(', ')} :boom:`) + postAlertSlackMessage( + `:boom: These have not been updated within the last 12 hours: ${deploymentsWithOldImages.join(', ')} :boom:`, + ); } - }) - } -} + }); + }, +}; diff --git a/src/index.js b/src/index.js index 9d750ec..9639aa7 100644 --- a/src/index.js +++ b/src/index.js @@ -1,38 +1,44 @@ -import kubernetes from './kubernetes.js' -import dockerRepo from './dockerRepo.js' -import imageDeployer from './image-deployer.js' -import depDeploymentRestarter from './dep-deployment-restarter.js' -import cronDeploymentRestarter from './cron-deployment-restarter.js' -import imageFreshnessMonitor from './image-freshness-monitor.js' +import kubernetes from './kubernetes.js'; +import dockerRepo from './dockerRepo.js'; +import imageDeployer from './image-deployer.js'; +import depDeploymentRestarter from './dep-deployment-restarter.js'; +import cronDeploymentRestarter from './cron-deployment-restarter.js'; +import imageFreshnessMonitor from './image-freshness-monitor.js'; -const CHECK_INTERVAL = 5 * 60 * 1000 +const CHECK_INTERVAL = 5 * 60 * 1000; -const actions = [imageDeployer, depDeploymentRestarter, cronDeploymentRestarter, imageFreshnessMonitor] +const actions = [ + imageDeployer, + depDeploymentRestarter, + cronDeploymentRestarter, + imageFreshnessMonitor, +]; const logError = (name, e) => { - console.log('%s: Error occurred %s', name, e) -} + console.log('%s: Error occurred %s', name, e); +}; const checkDeployments = () => { - console.log('Retrieving deployment configuration from kubernetes') + console.log('Retrieving deployment configuration from kubernetes'); const context = { kubernetes, - dockerRepo - } + dockerRepo, + }; - kubernetes.getDeployments().then(deployments => { - actions.forEach( - (action) => { + kubernetes + .getDeployments() + .then(deployments => { + actions.forEach(action => { try { - action.command(deployments, context) + action.command(deployments, context); } catch (e) { - logError(action.name, e) + logError(action.name, e); } - }) - }) - .catch((err) => console.log("Couldn't get deployments: " + err)) -} + }); + }) + .catch(err => console.log("Couldn't get deployments: " + err)); +}; -checkDeployments() -setInterval(checkDeployments, CHECK_INTERVAL) +checkDeployments(); +setInterval(checkDeployments, CHECK_INTERVAL); diff --git a/src/kubernetes.js b/src/kubernetes.js index 07613cb..f0237bf 100644 --- a/src/kubernetes.js +++ b/src/kubernetes.js @@ -1,71 +1,94 @@ -import * as k8s from '@kubernetes/client-node' +import * as k8s from '@kubernetes/client-node'; -const kubeconfig = new k8s.KubeConfig() -kubeconfig.loadFromCluster() +const kubeconfig = new k8s.KubeConfig(); +kubeconfig.loadFromCluster(); const getDeployments = () => { return new Promise((resolve, reject) => { - const appsApi = kubeconfig.makeApiClient(k8s.AppsV1Api) - appsApi.listNamespacedDeployment({ namespace: 'default' }).then(deploymentResponse => { - const promises = [] - const deployments = deploymentResponse.items - for (const deployment of deployments) { - promises.push(new Promise(resolve => { - const deploymentName = deployment.metadata.name - const labelSelector = `app=${deploymentName}` - console.log(`Fetching replica sets for deployment: ${deploymentName}`) - appsApi.listNamespacedReplicaSet({ namespace: 'default', labelSelector }).then(replicaSetResponse => { - const replicaSets = replicaSetResponse.items - let newestReplicaSetTime - for (const replicaSet of replicaSets) { - const creationTime = replicaSet.metadata.creationTimestamp - if (!newestReplicaSetTime || creationTime > newestReplicaSetTime) { - newestReplicaSetTime = creationTime - } - } - resolve({ ...deployment, version: newestReplicaSetTime ? newestReplicaSetTime.getTime() : null }) - }).catch(err => { - reject(err) - }) - })) - } - Promise.all(promises).then(values => { - // some helm installed deployment didn't have the creation time for a replica set for some reason - resolve(values.filter(value => value.version != null)) - }) - }) - .catch((err) => { - reject(err) + const appsApi = kubeconfig.makeApiClient(k8s.AppsV1Api); + appsApi + .listNamespacedDeployment({ namespace: 'default' }) + .then(deploymentResponse => { + const promises = []; + const deployments = deploymentResponse.items; + for (const deployment of deployments) { + promises.push( + new Promise(resolve => { + const deploymentName = deployment.metadata.name; + const labelSelector = `app=${deploymentName}`; + console.log( + `Fetching replica sets for deployment: ${deploymentName}`, + ); + appsApi + .listNamespacedReplicaSet({ + namespace: 'default', + labelSelector, + }) + .then(replicaSetResponse => { + const replicaSets = replicaSetResponse.items; + let newestReplicaSetTime; + for (const replicaSet of replicaSets) { + const creationTime = replicaSet.metadata.creationTimestamp; + if ( + !newestReplicaSetTime || + creationTime > newestReplicaSetTime + ) { + newestReplicaSetTime = creationTime; + } + } + resolve({ + ...deployment, + version: newestReplicaSetTime + ? newestReplicaSetTime.getTime() + : null, + }); + }) + .catch(err => { + reject(err); + }); + }), + ); + } + Promise.all(promises).then(values => { + // some helm installed deployment didn't have the creation time for a replica set for some reason + resolve(values.filter(value => value.version != null)); + }); }) - }) -} + .catch(err => { + reject(err); + }); + }); +}; // Restart by modifying deployment annotation (the same annotation is edited when kubectl rollout // restart deployment is run). // It is important the deployment name and label "app" have same values -const restartDeployment = (name) => { +const restartDeployment = name => { return new Promise((resolve, reject) => { - const appsApi = kubeconfig.makeApiClient(k8s.AppsV1Api) - const now = new Date() + const appsApi = kubeconfig.makeApiClient(k8s.AppsV1Api); + const now = new Date(); const patch = [ { op: 'replace', path: '/spec/template/metadata/annotations', value: { - 'kubectl.kubernetes.io/restartedAt': now.toISOString() - } - } - ] + 'kubectl.kubernetes.io/restartedAt': now.toISOString(), + }, + }, + ]; - appsApi.patchNamespacedDeployment({ name, namespace: 'default', body: patch }).then(response => { - resolve(name) - }).catch(err => { - reject(err) - }) - }) -} + appsApi + .patchNamespacedDeployment({ name, namespace: 'default', body: patch }) + .then(() => { + resolve(name); + }) + .catch(err => { + reject(err); + }); + }); +}; export default { getDeployments, - restartDeployment -} + restartDeployment, +}; diff --git a/src/util.js b/src/util.js index c4b034c..a27c0be 100644 --- a/src/util.js +++ b/src/util.js @@ -1,43 +1,52 @@ -import axios from 'axios' +import axios from 'axios'; -const MONITORING_CHANNEL_ID = process.env.MONITORING_SLACK_CHANNEL_ID -const MONITORING_USERNAME = `Configuration checker ${process.env.ENVIRONMENT_TYPE}` +const MONITORING_CHANNEL_ID = process.env.MONITORING_SLACK_CHANNEL_ID; +const MONITORING_USERNAME = `Configuration checker ${process.env.ENVIRONMENT_TYPE}`; -const ALERT_CHANNEL_ID = process.env.ALERT_SLACK_CHANNEL_ID -const ALERT_USERNAME = `Image freshness monitor ${process.env.ENVIRONMENT_TYPE}` +const ALERT_CHANNEL_ID = process.env.ALERT_SLACK_CHANNEL_ID; +const ALERT_USERNAME = `Image freshness monitor ${process.env.ENVIRONMENT_TYPE}`; const headers = { Authorization: `Bearer ${process.env.SLACK_ACCESS_TOKEN}`, 'Content-Type': 'application/json', - Accept: '*/*' -} + Accept: '*/*', +}; -function postSlackMessage (text, username, channel) { +function postSlackMessage(text, username, channel) { if (!process.env.SLACK_ACCESS_TOKEN) { - console.log('Not sending to slack: ' + text) + console.log('Not sending to slack: ' + text); } - axios.post('https://slack.com/api/chat.postMessage', { - channel, - text, - username - }, { headers }) + axios + .post( + 'https://slack.com/api/chat.postMessage', + { + channel, + text, + username, + }, + { headers }, + ) .then(response => { if (response.status !== 200) { - console.log(`Slack message was not sent successfully. Response: ${response}`) + console.log( + `Slack message was not sent successfully. Response: ${response}`, + ); } else { - console.log(`Sent to slack: ${text}`) + console.log(`Sent to slack: ${text}`); } }) .catch(error => { - console.log(`Something went wrong when trying to send message to Slack:\n${error}`) - }) + console.log( + `Something went wrong when trying to send message to Slack:\n${error}`, + ); + }); } -export function postMonitoringSlackMessage (text) { - postSlackMessage(text, MONITORING_USERNAME, MONITORING_CHANNEL_ID) +export function postMonitoringSlackMessage(text) { + postSlackMessage(text, MONITORING_USERNAME, MONITORING_CHANNEL_ID); } -export function postAlertSlackMessage (text) { - postSlackMessage(text, ALERT_USERNAME, ALERT_CHANNEL_ID) +export function postAlertSlackMessage(text) { + postSlackMessage(text, ALERT_USERNAME, ALERT_CHANNEL_ID); } diff --git a/test/cron-service-restarter-test.js b/test/cron-service-restarter-test.js index daf07a7..7a02819 100644 --- a/test/cron-service-restarter-test.js +++ b/test/cron-service-restarter-test.js @@ -1,182 +1,257 @@ -import { describe, it } from 'mocha' -import { assert, expect } from 'chai' -import cronRestarter from './../src/cron-deployment-restarter.js' +import { describe, it } from 'mocha'; +import { assert, expect } from 'chai'; +import cronRestarter from './../src/cron-deployment-restarter.js'; const appConfig = (id, version, labels, stable) => ({ metadata: { labels: { app: id, - ...labels - } + ...labels, + }, }, status: { replicas: 1, readyReplicas: stable ? 1 : 0, updatedReplicas: stable ? 1 : 0, - availableReplicas: stable ? 1 : 0 + availableReplicas: stable ? 1 : 0, }, - version -}) + version, +}); const failIfRestart = { kubernetes: { restartDeployment: function () { - assert(false, 'deployment restart was called when it should not have') - } - } -} + assert(false, 'deployment restart was called when it should not have'); + }, + }, +}; const countRestarts = () => { - let count = 0 - let deployment = null + let count = 0; + let deployment = null; return { kubernetes: { restartDeployment: function (id) { - deployment = id - count += 1 - return Promise.resolve('restarted!') - } + deployment = id; + count += 1; + return Promise.resolve('restarted!'); + }, }, - get: () => (count), - deployment: () => (deployment) - } -} + get: () => count, + deployment: () => deployment, + }; +}; -const NOW = new Date().getTime() +const NOW = new Date().getTime(); -const minutes = (m) => (m * 60 * 1000) +const minutes = m => m * 60 * 1000; describe('cron-deployment-restarter', function () { it('no apps should restart when no restartAt defined', () => { - const testApps = [ - appConfig('app1', NOW, {}, true) - ] - cronRestarter.command(testApps, failIfRestart) - }) + const testApps = [appConfig('app1', NOW, {}, true)]; + cronRestarter.command(testApps, failIfRestart); + }); it('no apps should restart when restartAt is after current time', () => { - const restartAt = new Date(NOW + minutes(5)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW + minutes(5)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(23 * 60), { restartAt: restartAtString }, true) - ] - cronRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app1', + NOW - minutes(23 * 60), + { restartAt: restartAtString }, + true, + ), + ]; + cronRestarter.command(testApps, failIfRestart); + }); it('no apps should restart when last restart within default limit interval', () => { - const restartAt = new Date(NOW - minutes(5)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW - minutes(5)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(15 * 60), { restartAt: restartAtString }, true) - ] - cronRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app1', + NOW - minutes(15 * 60), + { restartAt: restartAtString }, + true, + ), + ]; + cronRestarter.command(testApps, failIfRestart); + }); it('no apps should restart when last restart within user set limit interval', () => { - const restartAt = new Date(NOW - minutes(5)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW - minutes(5)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(110), { restartAt: restartAtString, restartLimitInterval: '120' }, true) - ] - cronRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app1', + NOW - minutes(110), + { restartAt: restartAtString, restartLimitInterval: '120' }, + true, + ), + ]; + cronRestarter.command(testApps, failIfRestart); + }); it('no apps should restart when it has been over hour since when the deployment was supposed to restart', () => { - const restartAt = new Date(NOW - minutes(350)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW - minutes(350)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(600), { restartAt: restartAtString, restartLimitInterval: '120' }, true) - ] - cronRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app1', + NOW - minutes(600), + { restartAt: restartAtString, restartLimitInterval: '120' }, + true, + ), + ]; + cronRestarter.command(testApps, failIfRestart); + }); it('app should be restarted when deployment is not stable', () => { - const restartAt = new Date(NOW - minutes(42)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW - minutes(42)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(150), { restartAt: restartAtString, restartLimitInterval: '120' }, false) - ] - const counter = countRestarts() - cronRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app1') - }) + appConfig( + 'app1', + NOW - minutes(150), + { restartAt: restartAtString, restartLimitInterval: '120' }, + false, + ), + ]; + const counter = countRestarts(); + cronRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app1'); + }); it('stable app with last restart outside of the user set limit interval should restart', () => { - const restartAt = new Date(NOW - minutes(1)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW - minutes(1)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(150), { restartAt: restartAtString, restartLimitInterval: '120' }, true), - appConfig('app2', NOW - minutes(100), { restartAt: restartAtString, restartLimitInterval: '120' }, true) - ] - const counter = countRestarts() - cronRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app1') - }) + appConfig( + 'app1', + NOW - minutes(150), + { restartAt: restartAtString, restartLimitInterval: '120' }, + true, + ), + appConfig( + 'app2', + NOW - minutes(100), + { restartAt: restartAtString, restartLimitInterval: '120' }, + true, + ), + ]; + const counter = countRestarts(); + cronRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app1'); + }); it('unstable app with last restart outside of the user set limit interval should restart', () => { - const restartAt = new Date(NOW - minutes(1)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW - minutes(1)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(150), { restartAt: restartAtString, restartLimitInterval: '120' }, false), - appConfig('app2', NOW - minutes(100), { restartAt: restartAtString, restartLimitInterval: '120' }, true) - ] - const counter = countRestarts() - cronRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app1') - }) + appConfig( + 'app1', + NOW - minutes(150), + { restartAt: restartAtString, restartLimitInterval: '120' }, + false, + ), + appConfig( + 'app2', + NOW - minutes(100), + { restartAt: restartAtString, restartLimitInterval: '120' }, + true, + ), + ]; + const counter = countRestarts(); + cronRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app1'); + }); it('stable app with last restart outside of the default limit interval should restart', () => { - const restartAt = new Date(NOW - minutes(1)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW - minutes(1)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(23 * 60), { restartAt: restartAtString }, true) - ] - const counter = countRestarts() - cronRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app1') - }) + appConfig( + 'app1', + NOW - minutes(23 * 60), + { restartAt: restartAtString }, + true, + ), + ]; + const counter = countRestarts(); + cronRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app1'); + }); it('stable app with restartAt less than 60 mins before should restart', () => { - const restartAt = new Date(NOW - minutes(55)) - const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes() + const restartAt = new Date(NOW - minutes(55)); + const restartAtString = restartAt.getHours() + '.' + restartAt.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(23 * 60), { restartAt: restartAtString }, true) - ] - const counter = countRestarts() - cronRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app1') - }) + appConfig( + 'app1', + NOW - minutes(23 * 60), + { restartAt: restartAtString }, + true, + ), + ]; + const counter = countRestarts(); + cronRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app1'); + }); it('stable app with one restartAt time less than 60 mins before should restart once', () => { - const restartAtFirst = new Date(NOW - minutes(55)) - const restartAtSecond = new Date(NOW - minutes(350)) - const restartAtString = restartAtFirst.getHours() + '.' + restartAtFirst.getMinutes() + ', ' + - restartAtSecond.getHours() + '.' + restartAtSecond.getMinutes() + const restartAtFirst = new Date(NOW - minutes(55)); + const restartAtSecond = new Date(NOW - minutes(350)); + const restartAtString = + restartAtFirst.getHours() + + '.' + + restartAtFirst.getMinutes() + + ', ' + + restartAtSecond.getHours() + + '.' + + restartAtSecond.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(23 * 60), { restartAt: restartAtString }, true) - ] - const counter = countRestarts() - cronRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app1') - }) + appConfig( + 'app1', + NOW - minutes(23 * 60), + { restartAt: restartAtString }, + true, + ), + ]; + const counter = countRestarts(); + cronRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app1'); + }); it('stable app with two restartAt time less than 60 mins before should restart once', () => { - const restartAtFirst = new Date(NOW - minutes(25)) - const restartAtSecond = new Date(NOW - minutes(55)) - const restartAtString = restartAtFirst.getHours() + '.' + restartAtFirst.getMinutes() + ', ' + - restartAtSecond.getHours() + '.' + restartAtSecond.getMinutes() + const restartAtFirst = new Date(NOW - minutes(25)); + const restartAtSecond = new Date(NOW - minutes(55)); + const restartAtString = + restartAtFirst.getHours() + + '.' + + restartAtFirst.getMinutes() + + ', ' + + restartAtSecond.getHours() + + '.' + + restartAtSecond.getMinutes(); const testApps = [ - appConfig('app1', NOW - minutes(23 * 60), { restartAt: restartAtString }, true) - ] - const counter = countRestarts() - cronRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app1') - }) -}) + appConfig( + 'app1', + NOW - minutes(23 * 60), + { restartAt: restartAtString }, + true, + ), + ]; + const counter = countRestarts(); + cronRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app1'); + }); +}); diff --git a/test/dep-service-restarter-test.js b/test/dep-service-restarter-test.js index c9cfc05..bfad886 100644 --- a/test/dep-service-restarter-test.js +++ b/test/dep-service-restarter-test.js @@ -1,133 +1,178 @@ -import { assert, expect } from 'chai' -import { describe, it } from 'mocha' -import deploymentRestarter from './../src/dep-deployment-restarter.js' +import { assert, expect } from 'chai'; +import { describe, it } from 'mocha'; +import deploymentRestarter from './../src/dep-deployment-restarter.js'; const appConfig = (id, version, labels, stable) => ({ metadata: { labels: { app: id, - ...labels - } + ...labels, + }, }, status: { replicas: 1, readyReplicas: stable ? 1 : 0, updatedReplicas: stable ? 1 : 0, - availableReplicas: stable ? 1 : 0 + availableReplicas: stable ? 1 : 0, }, - version -}) + version, +}); const failIfRestart = { kubernetes: { restartDeployment: function () { - assert(false, 'deployment restart was called when it should not have') - } - } -} + assert(false, 'deployment restart was called when it should not have'); + }, + }, +}; const countRestarts = () => { - let count = 0 - let deployment = null + let count = 0; + let deployment = null; return { kubernetes: { restartDeployment: function (id) { - deployment = id - count += 1 - return Promise.resolve('restarted!') - } + deployment = id; + count += 1; + return Promise.resolve('restarted!'); + }, }, - get: () => (count), - deployment: () => (deployment) - } -} + get: () => count, + deployment: () => deployment, + }; +}; -const NOW = new Date().getTime() +const NOW = new Date().getTime(); -const minutes = (m) => (m * 60 * 1000) +const minutes = m => m * 60 * 1000; describe('dep-deployment-restarter', function () { it('no apps should restart when restartDelay has not passed for only dependency', () => { const testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW - 1, { restartAfterDeployments: 'app1', restartDelay: '5' }, true) - ] - deploymentRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app2', + NOW - 1, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + ]; + deploymentRestarter.command(testApps, failIfRestart); + }); it('no apps should restart when restartDelay has not passed for some dependency', () => { const testApps = [ appConfig('app1', NOW - minutes(4), {}, true), appConfig('app2', NOW - minutes(5), {}, true), - appConfig('app3', NOW - minutes(5), { restartAfterDeployments: 'app1_app2', restartDelay: '5' }, true) - ] - deploymentRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app3', + NOW - minutes(5), + { restartAfterDeployments: 'app1_app2', restartDelay: '5' }, + true, + ), + ]; + deploymentRestarter.command(testApps, failIfRestart); + }); it('stable app should be restarted when restartDelay has passed for only dependency', () => { const testApps = [ appConfig('app1', NOW - minutes(1), {}, true), - appConfig('app2', NOW - minutes(1), { restartAfterDeployments: 'app1', restartDelay: '1' }, true) - ] - const counter = countRestarts() - deploymentRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app2') - }) + appConfig( + 'app2', + NOW - minutes(1), + { restartAfterDeployments: 'app1', restartDelay: '1' }, + true, + ), + ]; + const counter = countRestarts(); + deploymentRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app2'); + }); it('stable app should not be restarted when restartDelay has not passed for only dependency', () => { const testApps = [ appConfig('app1', NOW - minutes(0), {}, true), - appConfig('app2', NOW - minutes(1), { restartAfterDeployments: 'app1', restartDelay: '1' }, true) - ] - deploymentRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app2', + NOW - minutes(1), + { restartAfterDeployments: 'app1', restartDelay: '1' }, + true, + ), + ]; + deploymentRestarter.command(testApps, failIfRestart); + }); it('stable app should be restarted when restartDelay has passed for every dependency', () => { const testApps = [ appConfig('app1', NOW - minutes(10), {}, true), appConfig('app2', NOW - minutes(5), {}, true), - appConfig('app3', NOW - minutes(5), { restartAfterDeployments: 'app1_app2', restartDelay: '5' }, true) - ] - const counter = countRestarts() - deploymentRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app3') - }) + appConfig( + 'app3', + NOW - minutes(5), + { restartAfterDeployments: 'app1_app2', restartDelay: '5' }, + true, + ), + ]; + const counter = countRestarts(); + deploymentRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app3'); + }); it('stable app should not be restarted if it has been restarted already', () => { const testApps = [ appConfig('app1', NOW - minutes(5), {}, true), - appConfig('app2', NOW, { restartAfterDeployments: 'app1', restartDelay: '5' }, true) - ] - deploymentRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app2', + NOW, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + ]; + deploymentRestarter.command(testApps, failIfRestart); + }); it('no apps should restart when single dependency is not stable', () => { const testApps = [ appConfig('app1', NOW - minutes(5), {}, false), - appConfig('app2', NOW - minutes(5), { restartAfterDeployments: 'app1', restartDelay: '5' }, true) - ] - deploymentRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app2', + NOW - minutes(5), + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + ]; + deploymentRestarter.command(testApps, failIfRestart); + }); it('no apps should restart when any dependency is not stable', () => { const testApps = [ appConfig('app1', NOW - minutes(5), {}, true), appConfig('app2', NOW - minutes(5), {}, false), - appConfig('app3', NOW - minutes(5), { restartAfterDeployments: 'app1_app2', restartDelay: '5' }, true) - ] - deploymentRestarter.command(testApps, failIfRestart) - }) + appConfig( + 'app3', + NOW - minutes(5), + { restartAfterDeployments: 'app1_app2', restartDelay: '5' }, + true, + ), + ]; + deploymentRestarter.command(testApps, failIfRestart); + }); it('incorrect dependency should be ignored and restart should be called because of valid dependency', () => { const testApps = [ appConfig('app1', NOW - minutes(1), {}, true), - appConfig('app2', NOW - minutes(1), { restartAfterDeployments: 'app3_app1', restartDelay: '1' }, true) - ] - const counter = countRestarts() - deploymentRestarter.command(testApps, counter) - expect(counter.get()).to.be.equal(1) - expect(counter.deployment()).to.be.equal('app2') - }) -}) + appConfig( + 'app2', + NOW - minutes(1), + { restartAfterDeployments: 'app3_app1', restartDelay: '1' }, + true, + ), + ]; + const counter = countRestarts(); + deploymentRestarter.command(testApps, counter); + expect(counter.get()).to.be.equal(1); + expect(counter.deployment()).to.be.equal('app2'); + }); +}); diff --git a/test/graph-test.js b/test/graph-test.js index 9427959..7b137b5 100644 --- a/test/graph-test.js +++ b/test/graph-test.js @@ -1,133 +1,229 @@ -import { expect } from 'chai' -import { describe, it } from 'mocha' -import graphlib from '@dagrejs/graphlib' -import { build, isSubGraphStable, hasPendingDependentRestarts, deploymentsNeedingRestart, deploymentsNeedingImageFreshnessCheck } from './../src/graph.js' +import { expect } from 'chai'; +import { describe, it } from 'mocha'; +import graphlib from '@dagrejs/graphlib'; +import { + build, + isSubGraphStable, + hasPendingDependentRestarts, + deploymentsNeedingRestart, + deploymentsNeedingImageFreshnessCheck, +} from './../src/graph.js'; -const NOW = new Date().getTime() +const NOW = new Date().getTime(); const appConfig = (id, version, labels, stable) => ({ metadata: { labels: { app: id, - ...labels - } + ...labels, + }, }, status: { replicas: 1, readyReplicas: stable ? 1 : 0, updatedReplicas: stable ? 1 : 0, - availableReplicas: stable ? 1 : 0 + availableReplicas: stable ? 1 : 0, }, - version -}) + version, +}); describe('graph-builder', function () { it('the built graph should not have cycle if so configured', () => { const testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW, { restartAfterDeployments: 'app1', restartDelay: '5' }, true) - ] - const deploymentGraph = build(testApps) - expect(graphlib.alg.findCycles(deploymentGraph).length).to.equal(0) - }) + appConfig( + 'app2', + NOW, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + ]; + const deploymentGraph = build(testApps); + expect(graphlib.alg.findCycles(deploymentGraph).length).to.equal(0); + }); it('the built graph should have cycle if so configured', () => { const testApps = [ - appConfig('app1', NOW, { restartAfterDeployments: 'app2', restartDelay: '5' }, true), - appConfig('app2', NOW, { restartAfterDeployments: 'app1', restartDelay: '5' }, true) - ] - const deploymentGraph = build(testApps) - expect(graphlib.alg.findCycles(deploymentGraph).length).to.equal(1) - }) + appConfig( + 'app1', + NOW, + { restartAfterDeployments: 'app2', restartDelay: '5' }, + true, + ), + appConfig( + 'app2', + NOW, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + ]; + const deploymentGraph = build(testApps); + expect(graphlib.alg.findCycles(deploymentGraph).length).to.equal(1); + }); it('Sub Graph should be reported stable when it is', () => { const testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW, { restartAfterDeployments: 'app1', restartDelay: '5' }, true), - appConfig('app3', NOW, { restartAfterDeployments: 'app2', restartDelay: '5' }, true) - ] - const deploymentGraph = build(testApps) - expect(isSubGraphStable(deploymentGraph, 'app1')).to.equal(true) - expect(isSubGraphStable(deploymentGraph, 'app2')).to.equal(true) - expect(isSubGraphStable(deploymentGraph, 'app3')).to.equal(true) - }) + appConfig( + 'app2', + NOW, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + appConfig( + 'app3', + NOW, + { restartAfterDeployments: 'app2', restartDelay: '5' }, + true, + ), + ]; + const deploymentGraph = build(testApps); + expect(isSubGraphStable(deploymentGraph, 'app1')).to.equal(true); + expect(isSubGraphStable(deploymentGraph, 'app2')).to.equal(true); + expect(isSubGraphStable(deploymentGraph, 'app3')).to.equal(true); + }); it('Sub Graph should be reported unstable when it is', () => { const testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW, { restartAfterDeployments: 'app1', restartDelay: '5' }, false), - appConfig('app3', NOW, { restartAfterDeployments: 'app2', restartDelay: '5' }, true) - ] - const deploymentGraph = build(testApps) - expect(isSubGraphStable(deploymentGraph, 'app1')).to.equal(true) - expect(isSubGraphStable(deploymentGraph, 'app2')).to.equal(true) - expect(isSubGraphStable(deploymentGraph, 'app3')).to.equal(false) - }) + appConfig( + 'app2', + NOW, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + false, + ), + appConfig( + 'app3', + NOW, + { restartAfterDeployments: 'app2', restartDelay: '5' }, + true, + ), + ]; + const deploymentGraph = build(testApps); + expect(isSubGraphStable(deploymentGraph, 'app1')).to.equal(true); + expect(isSubGraphStable(deploymentGraph, 'app2')).to.equal(true); + expect(isSubGraphStable(deploymentGraph, 'app3')).to.equal(false); + }); it('Graph should tell us if there are pending dependent restarts upstream', () => { const testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW + 59999, { restartAfterDeployments: 'app1', restartDelay: '1' }, true), - appConfig('app3', NOW + 120000, { restartAfterDeployments: 'app2', restartDelay: '1' }, true) - ] - const deploymentGraph = build(testApps) - expect(hasPendingDependentRestarts(deploymentGraph, 'app1')).to.equal(false) - expect(hasPendingDependentRestarts(deploymentGraph, 'app2')).to.equal(true) - expect(hasPendingDependentRestarts(deploymentGraph, 'app3')).to.equal(true) - }) + appConfig( + 'app2', + NOW + 59999, + { restartAfterDeployments: 'app1', restartDelay: '1' }, + true, + ), + appConfig( + 'app3', + NOW + 120000, + { restartAfterDeployments: 'app2', restartDelay: '1' }, + true, + ), + ]; + const deploymentGraph = build(testApps); + expect(hasPendingDependentRestarts(deploymentGraph, 'app1')).to.equal( + false, + ); + expect(hasPendingDependentRestarts(deploymentGraph, 'app2')).to.equal(true); + expect(hasPendingDependentRestarts(deploymentGraph, 'app3')).to.equal(true); + }); it('Graph should tell us if there are no pending dependent restarts upstream', () => { const testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW + 60000, { restartAfterDeployments: 'app1', restartDelay: '1' }, true), - appConfig('app3', NOW + 120000, { restartAfterDeployments: 'app2', restartDelay: '1' }, true) - ] - const deploymentGraph = build(testApps) - expect(hasPendingDependentRestarts(deploymentGraph, 'app1')).to.equal(false) - expect(hasPendingDependentRestarts(deploymentGraph, 'app2')).to.equal(false) - expect(hasPendingDependentRestarts(deploymentGraph, 'app3')).to.equal(false) - }) + appConfig( + 'app2', + NOW + 60000, + { restartAfterDeployments: 'app1', restartDelay: '1' }, + true, + ), + appConfig( + 'app3', + NOW + 120000, + { restartAfterDeployments: 'app2', restartDelay: '1' }, + true, + ), + ]; + const deploymentGraph = build(testApps); + expect(hasPendingDependentRestarts(deploymentGraph, 'app1')).to.equal( + false, + ); + expect(hasPendingDependentRestarts(deploymentGraph, 'app2')).to.equal( + false, + ); + expect(hasPendingDependentRestarts(deploymentGraph, 'app3')).to.equal( + false, + ); + }); it('Graph should return deployments needing restart', () => { // restart delay passed app2 started > 1 minute before dependency let testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW - 60001, { restartAfterDeployments: 'app1', restartDelay: '1' }, true) // started > 1 minute before - ] - let deploymentGraph = build(testApps) - expect(deploymentsNeedingRestart(deploymentGraph).length).to.equal(1) + appConfig( + 'app2', + NOW - 60001, + { restartAfterDeployments: 'app1', restartDelay: '1' }, + true, + ), // started > 1 minute before + ]; + let deploymentGraph = build(testApps); + expect(deploymentsNeedingRestart(deploymentGraph).length).to.equal(1); // restart delay passed app2 and app3 started > 1 minute before dependency testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW - 60001, { restartAfterDeployments: 'app1', restartDelay: '1' }, true), - appConfig('app3', NOW - 120001, { restartAfterDeployments: 'app1', restartDelay: '2' }, true) - ] - deploymentGraph = build(testApps) - expect(deploymentsNeedingRestart(deploymentGraph).length).to.equal(2) + appConfig( + 'app2', + NOW - 60001, + { restartAfterDeployments: 'app1', restartDelay: '1' }, + true, + ), + appConfig( + 'app3', + NOW - 120001, + { restartAfterDeployments: 'app1', restartDelay: '2' }, + true, + ), + ]; + deploymentGraph = build(testApps); + expect(deploymentsNeedingRestart(deploymentGraph).length).to.equal(2); testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW - 60001, { restartAfterDeployments: 'app1', restartDelay: '1' }, true) - ] - deploymentGraph = build(testApps) - expect(deploymentsNeedingRestart(deploymentGraph).length).to.equal(1) - }) + appConfig( + 'app2', + NOW - 60001, + { restartAfterDeployments: 'app1', restartDelay: '1' }, + true, + ), + ]; + deploymentGraph = build(testApps); + expect(deploymentsNeedingRestart(deploymentGraph).length).to.equal(1); + }); it('Graph should return deployments needing image freshness check', () => { // app2 should be checked between 09:00 and 09:05 const testApps = [ appConfig('app1', NOW, {}, true), - appConfig('app2', NOW, { checkImageFreshnessAt: '09.00' }, true) - ] - const deploymentGraph = build(testApps) - const currentDate = new Date('2025-01-01T09:01:00') - expect(deploymentsNeedingImageFreshnessCheck(deploymentGraph, currentDate).length).to.equal(1) + appConfig('app2', NOW, { checkImageFreshnessAt: '09.00' }, true), + ]; + const deploymentGraph = build(testApps); + const currentDate = new Date('2025-01-01T09:01:00'); + expect( + deploymentsNeedingImageFreshnessCheck(deploymentGraph, currentDate) + .length, + ).to.equal(1); - const beforeDate = new Date('2025-01-01T08:59:00') - expect(deploymentsNeedingImageFreshnessCheck(deploymentGraph, beforeDate).length).to.equal(0) + const beforeDate = new Date('2025-01-01T08:59:00'); + expect( + deploymentsNeedingImageFreshnessCheck(deploymentGraph, beforeDate).length, + ).to.equal(0); - const afterDate = new Date('2025-01-01T09:06:00') - expect(deploymentsNeedingImageFreshnessCheck(deploymentGraph, afterDate).length).to.equal(0) - }) -}) + const afterDate = new Date('2025-01-01T09:06:00'); + expect( + deploymentsNeedingImageFreshnessCheck(deploymentGraph, afterDate).length, + ).to.equal(0); + }); +}); diff --git a/test/image-deployer-test.js b/test/image-deployer-test.js index 279d6bd..d29d4d1 100644 --- a/test/image-deployer-test.js +++ b/test/image-deployer-test.js @@ -1,6 +1,6 @@ -import { assert, expect } from 'chai' -import { describe, it } from 'mocha' -import deployer from './../src/image-deployer.js' +import { assert, expect } from 'chai'; +import { describe, it } from 'mocha'; +import deployer from './../src/image-deployer.js'; const appConfig = (id, version, labels, stable) => ({ spec: { @@ -8,96 +8,111 @@ const appConfig = (id, version, labels, stable) => ({ spec: { containers: [ { - imagePullPolicy: 'Always' - } - ] - } - } + imagePullPolicy: 'Always', + }, + ], + }, + }, }, metadata: { labels: { app: id, - ...labels - } + ...labels, + }, }, status: { replicas: 1, readyReplicas: stable ? 1 : 0, updatedReplicas: stable ? 1 : 0, - availableReplicas: stable ? 1 : 0 + availableReplicas: stable ? 1 : 0, }, version, - container: { docker: { forcePullImage: true } } -}) + container: { docker: { forcePullImage: true } }, +}); -const failIfRestart = (repoDate) => ({ +const failIfRestart = repoDate => ({ kubernetes: { restartDeployment: function () { - assert(false, 'deployment restart was called when it should not have') - } + assert(false, 'deployment restart was called when it should not have'); + }, }, dockerRepo: { getImageDate: function () { - return Promise.resolve(repoDate.getTime()) - } - } -}) + return Promise.resolve(repoDate.getTime()); + }, + }, +}); -const countRestarts = (repoDate) => { - let count = 0 +const countRestarts = repoDate => { + let count = 0; return { kubernetes: { restartDeployment: function () { - count += 1 - return Promise.resolve('restarted!') - } + count += 1; + return Promise.resolve('restarted!'); + }, }, dockerRepo: { getImageDate: function () { - return Promise.resolve(repoDate.getTime()) - } + return Promise.resolve(repoDate.getTime()); + }, }, - get: () => (count) - } -} + get: () => count, + }; +}; -const NOW = new Date().getTime() +const NOW = new Date().getTime(); describe('image-deployer', function () { - it('image should update when subgraph is stable and cool off period has ended', (done) => { + it('image should update when subgraph is stable and cool off period has ended', done => { const testApps = [ appConfig('app1', NOW - 60 * 60 * 1000, { update: 'auto' }, true), - appConfig('app2', NOW, { restartAfterDeployments: 'app1', restartDelay: '5' }, true) - ] - const counter = countRestarts(new Date(NOW + 1)) - deployer.command(testApps, counter) + appConfig( + 'app2', + NOW, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + ]; + const counter = countRestarts(new Date(NOW + 1)); + deployer.command(testApps, counter); setTimeout(function () { - expect(counter.get()).to.be.equal(1) - done() - }, 10) - }) + expect(counter.get()).to.be.equal(1); + done(); + }, 10); + }); - it('no image should be deployed when subgraph is not stable', (done) => { + it('no image should be deployed when subgraph is not stable', done => { const testApps = [ appConfig('app1', NOW, { update: 'auto' }, false), - appConfig('app2', NOW, { restartAfterDeployments: 'app1', restartDelay: '5' }, true) - ] + appConfig( + 'app2', + NOW, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + ]; setTimeout(function () { - deployer.command(testApps, failIfRestart(new Date(NOW + 1))) - done() - }, 10) - }) + deployer.command(testApps, failIfRestart(new Date(NOW + 1))); + done(); + }, 10); + }); - it('image should not update when the running version is newer', (done) => { + it('image should not update when the running version is newer', done => { const testApps = [ appConfig('app1', NOW, { update: 'auto' }, true), - appConfig('app2', NOW, { restartAfterDeployments: 'app1', restartDelay: '5' }, true) - ] - const counter = countRestarts(new Date(NOW - 1)) - deployer.command(testApps, counter) + appConfig( + 'app2', + NOW, + { restartAfterDeployments: 'app1', restartDelay: '5' }, + true, + ), + ]; + const counter = countRestarts(new Date(NOW - 1)); + deployer.command(testApps, counter); setTimeout(function () { - expect(counter.get()).to.be.equal(0) - done() - }, 10) - }) -}) + expect(counter.get()).to.be.equal(0); + done(); + }, 10); + }); +});