diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml new file mode 100644 index 0000000..0130113 --- /dev/null +++ b/.github/workflows/test.yml @@ -0,0 +1,39 @@ +name: Run Tests + +on: + push: + branches: [main] + pull_request: + +jobs: + test: + name: Test on Node.js ${{ matrix.node-version }} + runs-on: ubuntu-latest + strategy: + fail-fast: false + matrix: + node-version: [18.x, 20.x, 22.x] + + steps: + - name: Checkout repository + uses: actions/checkout@v4 + + - name: Setup Node.js ${{ matrix.node-version }} + uses: actions/setup-node@v4 + with: + node-version: ${{ matrix.node-version }} + cache: "npm" + + - name: Install dependencies + run: npm ci + + - name: Run tests with coverage + run: npm run test:ci + + - name: Upload coverage report + if: matrix.node-version == '20.x' + uses: actions/upload-artifact@v4 + with: + name: coverage-report + path: coverage/ + retention-days: 7 diff --git a/jest.config.js b/jest.config.js new file mode 100644 index 0000000..389b2a9 --- /dev/null +++ b/jest.config.js @@ -0,0 +1,23 @@ +module.exports = { + testEnvironment: "node", + setupFilesAfterEnv: ["/test/setup.js"], + testMatch: ["**/test/**/*.test.js"], + // Skip polling/monitor tests for now - they have complex async timing issues + testPathIgnorePatterns: ["/node_modules/"], + collectCoverageFrom: ["nodes/**/*.js", "!nodes/**/*.html"], + coverageDirectory: "coverage", + coverageReporters: ["text", "text-summary", "html", "lcov"], + // Coverage thresholds - raised to reflect comprehensive test coverage + coverageThreshold: { + global: { + branches: 70, + functions: 75, + lines: 85, + statements: 85, + }, + }, + clearMocks: true, + resetMocks: true, + restoreMocks: true, + testTimeout: 10000, +}; diff --git a/nodes/datalink-poll.js b/nodes/datalink-poll.js index 75775d7..9de4252 100644 --- a/nodes/datalink-poll.js +++ b/nodes/datalink-poll.js @@ -107,15 +107,18 @@ module.exports = function (RED) { }; // Start the polling interval + let intervalId = null; if (node.seqeraConfig && config.dataLinkName && config.dataLinkName.trim() !== "") { const intervalMs = node.pollFrequencySec * 1000; - const intervalId = setInterval(executePoll, intervalMs); + intervalId = setInterval(executePoll, intervalMs); // run once immediately executePoll(); } node.on("close", () => { - clearInterval(intervalId); + if (intervalId) { + clearInterval(intervalId); + } }); } diff --git a/package-lock.json b/package-lock.json index cb9968d..7752840 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,299 +1,3945 @@ { "name": "@seqera/node-red-seqera", - "version": "1.1.0", + "version": "1.4.1", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "@seqera/node-red-seqera", - "version": "1.1.0", + "version": "1.4.1", "license": "Apache-2.0", "dependencies": { "axios": "^1.10.0", "form-data": "^4.0.3" }, + "devDependencies": { + "jest": "^29.7.0", + "nock": "^13.5.0" + }, "engines": { "node": ">=12.0.0" } }, - "node_modules/asynckit": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", - "license": "MIT" + "node_modules/@babel/code-frame": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.27.1.tgz", + "integrity": "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.27.1", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } }, - "node_modules/axios": { - "version": "1.12.0", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.12.0.tgz", - "integrity": "sha512-oXTDccv8PcfjZmPGlWsPSwtOJCZ/b6W5jAMCNcfwJbCzDckwG0jrYJFaWH1yvivfCXjVzV/SPDEhMB3Q+DSurg==", + "node_modules/@babel/compat-data": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.5.tgz", + "integrity": "sha512-6uFXyCayocRbqhZOB+6XcuZbkMNimwfVGFji8CTZnCzOHVGvDqzvitu1re2AU5LROliz7eQPhB8CpAMvnx9EjA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.5.tgz", + "integrity": "sha512-e7jT4DxYvIDLk1ZHmU/m/mB19rex9sv0c2ftBtjSBv+kVM/902eh0fINUzD7UwLLNR+jU585GxUJ8/EBfAM5fw==", + "dev": true, "license": "MIT", "dependencies": { - "follow-redirects": "^1.15.6", - "form-data": "^4.0.4", - "proxy-from-env": "^1.1.0" + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.5", + "@babel/helper-compilation-targets": "^7.27.2", + "@babel/helper-module-transforms": "^7.28.3", + "@babel/helpers": "^7.28.4", + "@babel/parser": "^7.28.5", + "@babel/template": "^7.27.2", + "@babel/traverse": "^7.28.5", + "@babel/types": "^7.28.5", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" } }, - "node_modules/call-bind-apply-helpers": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", - "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "node_modules/@babel/generator": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.5.tgz", + "integrity": "sha512-3EwLFhZ38J4VyIP6WNtt2kUdW9dokXA9Cr4IVIFHuCpZ3H8/YFOl5JjZHisrn1fATPBmKKqXzDFvh9fUwHz6CQ==", + "dev": true, "license": "MIT", "dependencies": { - "es-errors": "^1.3.0", - "function-bind": "^1.1.2" + "@babel/parser": "^7.28.5", + "@babel/types": "^7.28.5", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" } }, - "node_modules/combined-stream": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", - "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "node_modules/@babel/helper-compilation-targets": { + "version": "7.27.2", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.2.tgz", + "integrity": "sha512-2+1thGUUWWjLTYTHZWK1n8Yga0ijBz1XAhUXcKy81rd5g6yh7hGqMp45v7cadSbEHc9G3OTv45SyneRN3ps4DQ==", + "dev": true, "license": "MIT", "dependencies": { - "delayed-stream": "~1.0.0" + "@babel/compat-data": "^7.27.2", + "@babel/helper-validator-option": "^7.27.1", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" }, "engines": { - "node": ">= 0.8" + "node": ">=6.9.0" } }, - "node_modules/delayed-stream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "node_modules/@babel/helper-globals": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", + "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", + "dev": true, "license": "MIT", "engines": { - "node": ">=0.4.0" + "node": ">=6.9.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==", + "node_modules/@babel/helper-module-imports": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.27.1.tgz", + "integrity": "sha512-0gSFWUPNXNopqtIPQvlD5WgXYI5GY2kP2cCvoT8kczjbfcfuIljTbcWrulD1CIPIX2gt1wghbDy08yE1p+/r3w==", + "dev": true, "license": "MIT", "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "es-errors": "^1.3.0", - "gopd": "^1.2.0" + "@babel/traverse": "^7.27.1", + "@babel/types": "^7.27.1" }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" } }, - "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==", + "node_modules/@babel/helper-module-transforms": { + "version": "7.28.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.3.tgz", + "integrity": "sha512-gytXUbs8k2sXS9PnQptz5o0QnpLL51SwASIORY6XaBKF88nsOT0Zw9szLqlSGQDP/4TljBAD5y98p2U1fqkdsw==", + "dev": true, "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.27.1", + "@babel/helper-validator-identifier": "^7.27.1", + "@babel/traverse": "^7.28.3" + }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" } }, - "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==", + "node_modules/@babel/helper-plugin-utils": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.27.1.tgz", + "integrity": "sha512-1gn1Up5YXka3YYAHGKpbideQ5Yjf1tDa9qYcgysz+cNCXukyLl6DjPXhD3VRwSb8c0J9tA4b2+rHEZtc6R0tlw==", + "dev": true, "license": "MIT", "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" } }, - "node_modules/es-object-atoms": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", - "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", + "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.28.4", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.4.tgz", + "integrity": "sha512-HFN59MmQXGHVyYadKLVumYsA9dBFun/ldYxipEjzA4196jpLZd8UjEEBLkbEkvfYreDqJhZxYAWFPtrfhNpj4w==", + "dev": true, "license": "MIT", "dependencies": { - "es-errors": "^1.3.0" + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.4" }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" } }, - "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==", + "node_modules/@babel/parser": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz", + "integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==", + "dev": true, "license": "MIT", "dependencies": { - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" + "@babel/types": "^7.28.5" + }, + "bin": { + "parser": "bin/babel-parser.js" }, "engines": { - "node": ">= 0.4" + "node": ">=6.0.0" } }, - "node_modules/follow-redirects": { - "version": "1.15.9", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", - "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", - "funding": [ - { - "type": "individual", - "url": "https://github.com/sponsors/RubenVerborgh" - } - ], + "node_modules/@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", + "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", + "dev": true, "license": "MIT", - "engines": { - "node": ">=4.0" + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" }, - "peerDependenciesMeta": { - "debug": { - "optional": true - } + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "node_modules/form-data": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.4.tgz", - "integrity": "sha512-KrGhL9Q4zjj0kiUt5OO4Mr/A/jlI2jDYs5eHBpYHPcBEVSiipAvn2Ko2HnPe20rmcuuvMHNdZFp+4IlGTMF0Ow==", + "node_modules/@babel/plugin-syntax-bigint": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", + "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", + "dev": true, "license": "MIT", "dependencies": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.8", - "es-set-tostringtag": "^2.1.0", - "hasown": "^2.0.2", - "mime-types": "^2.1.12" + "@babel/helper-plugin-utils": "^7.8.0" }, - "engines": { - "node": ">= 6" + "peerDependencies": { + "@babel/core": "^7.0.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==", + "node_modules/@babel/plugin-syntax-class-properties": { + "version": "7.12.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", + "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", + "dev": true, "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" + "dependencies": { + "@babel/helper-plugin-utils": "^7.12.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "node_modules/get-intrinsic": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", - "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", + "node_modules/@babel/plugin-syntax-class-static-block": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-static-block/-/plugin-syntax-class-static-block-7.14.5.tgz", + "integrity": "sha512-b+YyPmr6ldyNnM6sqYeMWE+bgJcJpO6yS4QD7ymxgH34GBPNDM/THBh8iunyvKIZztiwLH4CJZ0RxTk9emgpjw==", + "dev": true, "license": "MIT", "dependencies": { - "call-bind-apply-helpers": "^1.0.2", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "function-bind": "^1.1.2", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "math-intrinsics": "^1.1.0" + "@babel/helper-plugin-utils": "^7.14.5" }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "node_modules/get-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", - "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "node_modules/@babel/plugin-syntax-import-attributes": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.27.1.tgz", + "integrity": "sha512-oFT0FrKHgF53f4vOsZGi2Hh3I35PfSmVs4IBFLFj4dnafP+hIWDLg3VyKmUHfLoLHlyxY4C7DGtmHuJgn+IGww==", + "dev": true, "license": "MIT", "dependencies": { - "dunder-proto": "^1.0.1", - "es-object-atoms": "^1.0.0" + "@babel/helper-plugin-utils": "^7.27.1" }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "node_modules/gopd": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", - "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "node_modules/@babel/plugin-syntax-import-meta": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", + "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", + "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.27.1.tgz", + "integrity": "sha512-y8YTNIeKoyhGd9O0Jiyzyyqk8gdjnumGTQPsz0xOZOQ2RmkVJeZ1vmmfIvFEKqucBG6axJGBZDE/7iI5suUI/w==", + "dev": true, "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "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==", + "node_modules/@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", + "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-nullish-coalescing-operator": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", + "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", + "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", + "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", + "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-chaining": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", + "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-private-property-in-object": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-private-property-in-object/-/plugin-syntax-private-property-in-object-7.14.5.tgz", + "integrity": "sha512-0wVnp9dxJ72ZUJDV27ZfbSj6iHLoytYZmh3rFcxNnvsJF3ktkzLDZPy/mA17HGsaQT3/DQsWYX1f1QGWkCoVUg==", + "dev": true, "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "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==", + "node_modules/@babel/plugin-syntax-top-level-await": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", + "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", + "dev": true, "license": "MIT", "dependencies": { - "has-symbols": "^1.0.3" + "@babel/helper-plugin-utils": "^7.14.5" }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "node_modules/hasown": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", - "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "node_modules/@babel/plugin-syntax-typescript": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.27.1.tgz", + "integrity": "sha512-xfYCBMxveHrRMnAWl1ZlPXOZjzkN82THFvLhQhFXFt81Z5HnN+EtUkZhv/zcKpmT3fzmWZB0ywiBrbC3vogbwQ==", + "dev": true, "license": "MIT", "dependencies": { - "function-bind": "^1.1.2" + "@babel/helper-plugin-utils": "^7.27.1" }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "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==", + "node_modules/@babel/template": { + "version": "7.27.2", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.2.tgz", + "integrity": "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==", + "dev": true, "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/parser": "^7.27.2", + "@babel/types": "^7.27.1" + }, "engines": { - "node": ">= 0.4" + "node": ">=6.9.0" } }, - "node_modules/mime-db": { - "version": "1.52.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", - "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "node_modules/@babel/traverse": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.5.tgz", + "integrity": "sha512-TCCj4t55U90khlYkVV/0TfkJkAkUg3jZFA3Neb7unZT8CPok7iiRfaX0F+WnqWqt7OxhOn0uBKXCw4lbL8W0aQ==", + "dev": true, "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.5", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.28.5", + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.5", + "debug": "^4.3.1" + }, "engines": { - "node": ">= 0.6" + "node": ">=6.9.0" } }, - "node_modules/mime-types": { - "version": "2.1.35", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", - "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "node_modules/@babel/types": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz", + "integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==", + "dev": true, "license": "MIT", "dependencies": { - "mime-db": "1.52.0" + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" }, "engines": { - "node": ">= 0.6" + "node": ">=6.9.0" } }, - "node_modules/proxy-from-env": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", - "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", + "node_modules/@bcoe/v8-coverage": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", + "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", + "dev": true, "license": "MIT" + }, + "node_modules/@istanbuljs/load-nyc-config": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", + "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "camelcase": "^5.3.1", + "find-up": "^4.1.0", + "get-package-type": "^0.1.0", + "js-yaml": "^3.13.1", + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/schema": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", + "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/console": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-29.7.0.tgz", + "integrity": "sha512-5Ni4CU7XHQi32IJ398EEP4RrB8eV09sXP2ROqD4bksHrnTree52PsxvX8tpL8LvTZ3pFzXyPbNQReSN41CAhOg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/core": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/core/-/core-29.7.0.tgz", + "integrity": "sha512-n7aeXWKMnGtDA48y8TLWJPJmLmmZ642Ceo78cYWEpiD7FzDgmNDV/GCVRorPABdXLJZ/9wzzgZAlHjXjxDHGsg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/reporters": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-changed-files": "^29.7.0", + "jest-config": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-resolve-dependencies": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "jest-watcher": "^29.7.0", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/environment": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-29.7.0.tgz", + "integrity": "sha512-aQIfHDq33ExsN4jP1NWGXhxgQ/wixs60gDiKO+XVMd8Mn0NWPWgc34ZQDTb2jKaUWQ7MuwoitXAsN2XVXNMpAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-8uMeAMycttpva3P1lBHB8VciS9V0XAr3GymPpipdyQXbBcuhkLQOSe8E/p92RyAdToS6ZD1tFkX+CkhoECE0dQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "expect": "^29.7.0", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/expect-utils/-/expect-utils-29.7.0.tgz", + "integrity": "sha512-GlsNBWiFQFCVi9QVSx7f5AgMeLxe9YCCs5PuP2O2LdjDAA8Jh9eX7lA1Jq/xdXw3Wb3hyvlFNfZIfcRetSzYcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-get-type": "^29.6.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/fake-timers": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-29.7.0.tgz", + "integrity": "sha512-q4DH1Ha4TTFPdxLsqDXK1d3+ioSL7yL5oCMJZgDYm6i+6CygW5E5xVr/D1HdsGxjt1ZWSfUAs9OxSB/BNelWrQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@sinonjs/fake-timers": "^10.0.2", + "@types/node": "*", + "jest-message-util": "^29.7.0", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/globals": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-29.7.0.tgz", + "integrity": "sha512-mpiz3dutLbkW2MNFubUGUEVLkTGiqW6yLVTA+JbP6fI6J5iL9Y0Nlg8k95pcF8ctKwCS7WVxteBs29hhfAotzQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/types": "^29.6.3", + "jest-mock": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/reporters": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-29.7.0.tgz", + "integrity": "sha512-DApq0KJbJOEzAFYjHADNNxAE3KbhxQB1y5Kplb5Waqw6zVbuWatSnMjE5gs8FUgEPmNsnZA3NCWl9NG0ia04Pg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@bcoe/v8-coverage": "^0.2.3", + "@jest/console": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@jridgewell/trace-mapping": "^0.3.18", + "@types/node": "*", + "chalk": "^4.0.0", + "collect-v8-coverage": "^1.0.0", + "exit": "^0.1.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "istanbul-lib-coverage": "^3.0.0", + "istanbul-lib-instrument": "^6.0.0", + "istanbul-lib-report": "^3.0.0", + "istanbul-lib-source-maps": "^4.0.0", + "istanbul-reports": "^3.1.3", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "slash": "^3.0.0", + "string-length": "^4.0.1", + "strip-ansi": "^6.0.0", + "v8-to-istanbul": "^9.0.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/schemas": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.6.3.tgz", + "integrity": "sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@sinclair/typebox": "^0.27.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/source-map": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-29.6.3.tgz", + "integrity": "sha512-MHjT95QuipcPrpLM+8JMSzFx6eHp5Bm+4XeFDJlwsvVBjmKNiIAvasGK2fxz2WbGRlnvqehFbh07MMa7n3YJnw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.18", + "callsites": "^3.0.0", + "graceful-fs": "^4.2.9" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/test-result": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-29.7.0.tgz", + "integrity": "sha512-Fdx+tv6x1zlkJPcWXmMDAG2HBnaR9XPSd5aDWQVsfrZmLVT3lU1cwyxLgRmXR9yrq4NBoEm9BMsfgFzTQAbJYA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "collect-v8-coverage": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/test-sequencer": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-29.7.0.tgz", + "integrity": "sha512-GQwJ5WZVrKnOJuiYiAF52UNUJXgTZx1NHjFSEB0qEMmSZKAkdMoIzw/Cj6x6NF4AvV23AUqDpFzQkN/eYCYTxw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/test-result": "^29.7.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/transform": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-29.7.0.tgz", + "integrity": "sha512-ok/BTPFzFKVMwO5eOHRrvnBVHdRy9IrsrW1GpMaQ9MCnilNLXQKmAX8s1YXDFaai9xJpac2ySzV0YeRRECr2Vw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/types": "^29.6.3", + "@jridgewell/trace-mapping": "^0.3.18", + "babel-plugin-istanbul": "^6.1.1", + "chalk": "^4.0.0", + "convert-source-map": "^2.0.0", + "fast-json-stable-stringify": "^2.1.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "micromatch": "^4.0.4", + "pirates": "^4.0.4", + "slash": "^3.0.0", + "write-file-atomic": "^4.0.2" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/types": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-29.6.3.tgz", + "integrity": "sha512-u3UPsIilWKOM3F9CXtrG8LEJmNxwoCQC/XVj4IKYXvvpx7QIi/Kg1LI5uDmDpKlac62NUtX7eLjRh+jVZcLOzw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^17.0.8", + "chalk": "^4.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", + "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", + "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", + "dev": true, + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@sinclair/typebox": { + "version": "0.27.8", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.27.8.tgz", + "integrity": "sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@sinonjs/commons": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-3.0.1.tgz", + "integrity": "sha512-K3mCHKQ9sVh8o1C9cxkwxaOmXoAMlDxC1mYyHrjqOWEcBjYr76t96zL2zlj5dUGZ3HSw240X1qgH3Mjf1yJWpQ==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "type-detect": "4.0.8" + } + }, + "node_modules/@sinonjs/fake-timers": { + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-10.3.0.tgz", + "integrity": "sha512-V4BG07kuYSUkTCSBHG8G8TNhM+F19jXFWnQtzj+we8DrkpSBCee9Z3Ms8yiGer/dlmhe35/Xdgyo3/0rQKg7YA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@sinonjs/commons": "^3.0.0" + } + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz", + "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz", + "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.2" + } + }, + "node_modules/@types/graceful-fs": { + "version": "4.1.9", + "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.9.tgz", + "integrity": "sha512-olP3sd1qOEe5dXTSaFvQG+02VdRXcdytWLAZsAq1PecU8uqQAhkrnbli7DagjtXKW/Bl7YJbUsa8MPcuc8LHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/istanbul-lib-coverage": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.6.tgz", + "integrity": "sha512-2QF/t/auWm0lsy8XtKVPG19v3sSOQlJe/YHZgfjb/KBBHOGSV+J2q/S671rcq9uTBrLAXmZpqJiaQbMT+zNU1w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/istanbul-lib-report": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.3.tgz", + "integrity": "sha512-NQn7AHQnk/RSLOxrBbGyJM/aVQ+pjj5HCgasFxc0K/KhoATfQ/47AyUl15I2yBUpihjmas+a+VJBOqecrFH+uA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/istanbul-lib-coverage": "*" + } + }, + "node_modules/@types/istanbul-reports": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.4.tgz", + "integrity": "sha512-pk2B1NWalF9toCRu6gjBzR69syFjP4Od8WRAX+0mmf9lAjCRicLOWc+ZrxZHx/0XRjotgkF9t6iaMJ+aXcOdZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/istanbul-lib-report": "*" + } + }, + "node_modules/@types/node": { + "version": "25.0.0", + "resolved": "https://registry.npmjs.org/@types/node/-/node-25.0.0.tgz", + "integrity": "sha512-rl78HwuZlaDIUSeUKkmogkhebA+8K1Hy7tddZuJ3D0xV8pZSfsYGTsliGUol1JPzu9EKnTxPC4L1fiWouStRew==", + "dev": true, + "license": "MIT", + "dependencies": { + "undici-types": "~7.16.0" + } + }, + "node_modules/@types/stack-utils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.3.tgz", + "integrity": "sha512-9aEbYZ3TbYMznPdcdr3SmIrLXwC/AKZXQeCf9Pgao5CKb8CyHuEX5jzWPTkvregvhRJHcpRO6BFoGW9ycaOkYw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/yargs": { + "version": "17.0.35", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.35.tgz", + "integrity": "sha512-qUHkeCyQFxMXg79wQfTtfndEC+N9ZZg76HJftDJp+qH2tV7Gj4OJi7l+PiWwJ+pWtW8GwSmqsDj/oymhrTWXjg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/yargs-parser": "*" + } + }, + "node_modules/@types/yargs-parser": { + "version": "21.0.3", + "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.3.tgz", + "integrity": "sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dev": true, + "license": "ISC", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "license": "MIT", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", + "license": "MIT" + }, + "node_modules/axios": { + "version": "1.12.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.12.0.tgz", + "integrity": "sha512-oXTDccv8PcfjZmPGlWsPSwtOJCZ/b6W5jAMCNcfwJbCzDckwG0jrYJFaWH1yvivfCXjVzV/SPDEhMB3Q+DSurg==", + "license": "MIT", + "dependencies": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.4", + "proxy-from-env": "^1.1.0" + } + }, + "node_modules/babel-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-29.7.0.tgz", + "integrity": "sha512-BrvGY3xZSwEcCzKvKsCi2GgHqDqsYkOP4/by5xCgIwGXQxIEh+8ew3gmrE1y7XRR6LHZIj6yLYnUi/mm2KXKBg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/transform": "^29.7.0", + "@types/babel__core": "^7.1.14", + "babel-plugin-istanbul": "^6.1.1", + "babel-preset-jest": "^29.6.3", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.8.0" + } + }, + "node_modules/babel-plugin-istanbul": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", + "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-instrument": "^5.0.4", + "test-exclude": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-istanbul/node_modules/istanbul-lib-instrument": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.2.1.tgz", + "integrity": "sha512-pzqtp31nLv/XFOzXGuvhCb8qhjmTVo5vjVk19XE4CRlSWz0KoeJ3bw9XsA7nOp9YBf4qHjwBxkDzKcME/J29Yg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.12.3", + "@babel/parser": "^7.14.7", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-jest-hoist": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-29.6.3.tgz", + "integrity": "sha512-ESAc/RJvGTFEzRwOTT4+lNDk/GNHMkKbNzsvT0qKRfDyyYTskxB5rnU2njIDYVxXCBHHEI1c0YwHob3WaYujOg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.3.3", + "@babel/types": "^7.3.3", + "@types/babel__core": "^7.1.14", + "@types/babel__traverse": "^7.0.6" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/babel-preset-current-node-syntax": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.2.0.tgz", + "integrity": "sha512-E/VlAEzRrsLEb2+dv8yp3bo4scof3l9nR4lrld+Iy5NyVqgVYUJnDAmunkhPMisRI32Qc4iRiz425d8vM++2fg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.12.13", + "@babel/plugin-syntax-class-static-block": "^7.14.5", + "@babel/plugin-syntax-import-attributes": "^7.24.7", + "@babel/plugin-syntax-import-meta": "^7.10.4", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-private-property-in-object": "^7.14.5", + "@babel/plugin-syntax-top-level-await": "^7.14.5" + }, + "peerDependencies": { + "@babel/core": "^7.0.0 || ^8.0.0-0" + } + }, + "node_modules/babel-preset-jest": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-29.6.3.tgz", + "integrity": "sha512-0B3bhxR6snWXJZtR/RliHTDPRgn1sNHOR0yVtq/IiQFyuOVjFS+wuio/R4gSNkyYmKmJB4wGZv2NZanmKmTnNA==", + "dev": true, + "license": "MIT", + "dependencies": { + "babel-plugin-jest-hoist": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/baseline-browser-mapping": { + "version": "2.9.6", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.9.6.tgz", + "integrity": "sha512-v9BVVpOTLB59C9E7aSnmIF8h7qRsFpx+A2nugVMTszEOMcfjlZMsXRm4LF23I3Z9AJxc8ANpIvzbzONoX9VJlg==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.js" + } + }, + "node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/browserslist": { + "version": "4.28.1", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.1.tgz", + "integrity": "sha512-ZC5Bd0LgJXgwGqUknZY/vkUQ04r8NXnJZ3yYi4vDmSiZmC/pdSN0NbNRPxZpbtO4uAfDUAFffO8IZoM3Gj8IkA==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "baseline-browser-mapping": "^2.9.0", + "caniuse-lite": "^1.0.30001759", + "electron-to-chromium": "^1.5.263", + "node-releases": "^2.0.27", + "update-browserslist-db": "^1.2.0" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/bser": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", + "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "node-int64": "^0.4.0" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/call-bind-apply-helpers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", + "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001760", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001760.tgz", + "integrity": "sha512-7AAMPcueWELt1p3mi13HR/LHH0TJLT11cnwDJEs3xA4+CK/PLKeO9Kl1oru24htkyUKtkGCvAx4ohB0Ttry8Dw==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/char-regex": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", + "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/ci-info": { + "version": "3.9.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", + "integrity": "sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/sibiraj-s" + } + ], + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/cjs-module-lexer": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.4.3.tgz", + "integrity": "sha512-9z8TZaGM1pfswYeXrUpzPrkx8UnWYdhJclsiYMm6x/w5+nN+8Tf/LnAgfLGQCm59qAOxU8WwHEq2vNwF6i4j+Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/collect-v8-coverage": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.3.tgz", + "integrity": "sha512-1L5aqIkwPfiodaMgQunkF1zRhNqifHBmtbbbxcr6yVxxBnliw4TDOW6NxpO8DJLgJ16OT+Y4ztZqP6p/FtXnAw==", + "dev": true, + "license": "MIT" + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true, + "license": "MIT" + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "license": "MIT", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true, + "license": "MIT" + }, + "node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true, + "license": "MIT" + }, + "node_modules/create-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", + "integrity": "sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "prompts": "^2.0.1" + }, + "bin": { + "create-jest": "bin/create-jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/dedent": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.7.0.tgz", + "integrity": "sha512-HGFtf8yhuhGhqO07SV79tRp+br4MnbdjeVxotpn1QBl30pcLLCQjX5b2295ll0fv8RKDKsmWYrl05usHM9CewQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "babel-plugin-macros": "^3.1.0" + }, + "peerDependenciesMeta": { + "babel-plugin-macros": { + "optional": true + } + } + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", + "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "license": "MIT", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/detect-newline": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", + "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/diff-sequences": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", + "integrity": "sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "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/electron-to-chromium": { + "version": "1.5.267", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.267.tgz", + "integrity": "sha512-0Drusm6MVRXSOJpGbaSVgcQsuB4hEkMpHXaVstcPmhu5LIedxs1xNK/nIxmQIU/RPC0+1/o0AVZfBTkTNJOdUw==", + "dev": true, + "license": "ISC" + }, + "node_modules/emittery": { + "version": "0.13.1", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.13.1.tgz", + "integrity": "sha512-DeWwawk6r5yR9jFgnDKYt4sLS0LmHJJi3ZOnb5/JdbYwj3nW+FxQnHIjhBKz8YLC7oRNPVM9NQ47I3CVx34eqQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sindresorhus/emittery?sponsor=1" + } + }, + "node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true, + "license": "MIT" + }, + "node_modules/error-ex": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.4.tgz", + "integrity": "sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "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==", + "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==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-object-atoms": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", + "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", + "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==", + "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/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", + "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "license": "BSD-2-Clause", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/execa?sponsor=1" + } + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/expect-utils": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fb-watchman": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.2.tgz", + "integrity": "sha512-p5161BqbuCaSnB8jIbzQHOlpgsPmK5rJVDfDKO91Axs5NC1uu3HRQm6wt9cd9/+GtQQIO53JdGXXoyDpTAsgYA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "bser": "2.1.1" + } + }, + "node_modules/fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dev": true, + "license": "MIT", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/follow-redirects": { + "version": "1.15.9", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", + "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "license": "MIT", + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/form-data": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.4.tgz", + "integrity": "sha512-KrGhL9Q4zjj0kiUt5OO4Mr/A/jlI2jDYs5eHBpYHPcBEVSiipAvn2Ko2HnPe20rmcuuvMHNdZFp+4IlGTMF0Ow==", + "license": "MIT", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "es-set-tostringtag": "^2.1.0", + "hasown": "^2.0.2", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true, + "license": "ISC" + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "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==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true, + "license": "ISC", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-intrinsic": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.1.1", + "function-bind": "^1.1.2", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-package-type": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", + "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/gopd": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "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/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "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==", + "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==", + "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==", + "license": "MIT", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true, + "license": "MIT" + }, + "node_modules/human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=10.17.0" + } + }, + "node_modules/import-local": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.2.0.tgz", + "integrity": "sha512-2SPlun1JUPWoM6t3F0dw0FkCF/jWY8kttcY4f599GLTSjh2OCuuhdTkJQsEcZzBqbXZGKMK2OqW1oZsjtf/gQA==", + "dev": true, + "license": "MIT", + "dependencies": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", + "dev": true, + "license": "ISC", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true, + "license": "ISC" + }, + "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-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-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-generator-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", + "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" + }, + "node_modules/istanbul-lib-coverage": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.2.tgz", + "integrity": "sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-6.0.3.tgz", + "integrity": "sha512-Vtgk7L/R2JHyyGW07spoFlB8/lpjiOLTjMdms6AFMraYt3BaJauod/NGrfnVG/y4Ix1JEuMRPDPEj2ua+zz1/Q==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.23.9", + "@babel/parser": "^7.23.9", + "@istanbuljs/schema": "^0.1.3", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-instrument/node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", + "integrity": "sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^4.0.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", + "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-reports": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.2.0.tgz", + "integrity": "sha512-HGYWWS/ehqTV3xN10i23tkPkpH46MLCIMFNCaaKNavAXTF1RkqxawEPtnjnGZ6XKSInBKkiOA5BKS+aZiY3AvA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest/-/jest-29.7.0.tgz", + "integrity": "sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/types": "^29.6.3", + "import-local": "^3.0.2", + "jest-cli": "^29.7.0" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-changed-files": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-29.7.0.tgz", + "integrity": "sha512-fEArFiwf1BpQ+4bXSprcDc3/x4HSzL4al2tozwVpDFpsxALjLYdyiIK4e5Vz66GQJIbXJ82+35PtysofptNX2w==", + "dev": true, + "license": "MIT", + "dependencies": { + "execa": "^5.0.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-circus": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-29.7.0.tgz", + "integrity": "sha512-3E1nCMgipcTkCocFwM90XXQab9bS+GMsjdpmPrlelaxwD93Ad8iVEjX/vvHPdLPnFf+L40u+5+iutRdA1N9myw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "dedent": "^1.0.0", + "is-generator-fn": "^2.0.0", + "jest-each": "^29.7.0", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0", + "pretty-format": "^29.7.0", + "pure-rand": "^6.0.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-cli": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-29.7.0.tgz", + "integrity": "sha512-OVVobw2IubN/GSYsxETi+gOe7Ka59EFMR/twOU3Jb2GnKKeMGJB5SGUUrEz3SFVmJASUdZUzy83sLNNQ2gZslg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "create-jest": "^29.7.0", + "exit": "^0.1.2", + "import-local": "^3.0.2", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "yargs": "^17.3.1" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-config": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-29.7.0.tgz", + "integrity": "sha512-uXbpfeQ7R6TZBqI3/TxCU4q4ttk3u0PJeC+E0zbfSoSjq6bJ7buBPxzQPL0ifrkY4DNu4JUdk0ImlBUYi840eQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/test-sequencer": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-jest": "^29.7.0", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "deepmerge": "^4.2.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-circus": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "micromatch": "^4.0.4", + "parse-json": "^5.2.0", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@types/node": "*", + "ts-node": ">=9.0.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "ts-node": { + "optional": true + } + } + }, + "node_modules/jest-diff": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.7.0.tgz", + "integrity": "sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "diff-sequences": "^29.6.3", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-docblock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-29.7.0.tgz", + "integrity": "sha512-q617Auw3A612guyaFgsbFeYpNP5t2aoUNLwBUbc/0kD1R4t9ixDbyFTHd1nok4epoVFpr7PmeWHrhvuV3XaJ4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "detect-newline": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-29.7.0.tgz", + "integrity": "sha512-gns+Er14+ZrEoC5fhOfYCY1LOHHr0TI+rQUHZS8Ttw2l7gl+80eHc/gFf2Ktkw0+SIACDTeWvpFcv3B04VembQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "jest-util": "^29.7.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-environment-node": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-29.7.0.tgz", + "integrity": "sha512-DOSwCRqXirTOyheM+4d5YZOrWcdu0LNZ87ewUoywbcb2XR4wKgqiG8vNeYwhjFMbEkfju7wx2GYH0P2gevGvFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-haste-map": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-29.7.0.tgz", + "integrity": "sha512-fP8u2pyfqx0K1rGn1R9pyE0/KTn+G7PxktWidOBTqFPLYX0b9ksaMFkhK5vrS3DVun09pckLdlx90QthlW7AmA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/graceful-fs": "^4.1.3", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "micromatch": "^4.0.4", + "walker": "^1.0.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "optionalDependencies": { + "fsevents": "^2.3.2" + } + }, + "node_modules/jest-leak-detector": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-29.7.0.tgz", + "integrity": "sha512-kYA8IJcSYtST2BY9I+SMC32nDpBT3J2NvWJx8+JCuCdl/CR1I4EKUJROiP8XtCcxqgTTBGJNdbB1A8XRKbTetw==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz", + "integrity": "sha512-sBkD+Xi9DtcChsI3L3u0+N0opgPYnCRPtGcQYrgXmR+hmt/fYfWAL0xRXYU8eWOdfuLgBe0YCW3AFtnRLagq/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-message-util": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.7.0.tgz", + "integrity": "sha512-GBEV4GRADeP+qtB2+6u61stea8mGcOT4mCtrYISZwfu9/ISHFJ/5zOMXYbpBE9RsS5+Gb63DW4FgmnKJ79Kf6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^29.6.3", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-mock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-29.7.0.tgz", + "integrity": "sha512-ITOMZn+UkYS4ZFh83xYAOzWStloNzJFO2s8DWrE4lhtGD+AorgnbkiKERe4wQVBydIGPx059g6riW5Btp6Llnw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-pnp-resolver": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.3.tgz", + "integrity": "sha512-+3NpwQEnRoIBtx4fyhblQDPgJI0H1IEIkX7ShLUjPGA7TtUTvI1oiKi3SR4oBR0hQhQR80l4WAe5RrXBwWMA8w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "jest-resolve": "*" + }, + "peerDependenciesMeta": { + "jest-resolve": { + "optional": true + } + } + }, + "node_modules/jest-regex-util": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-29.6.3.tgz", + "integrity": "sha512-KJJBsRCyyLNWCNBOvZyRDnAIfUiRJ8v+hOBQYGn8gDyF3UegwiP4gwRR3/SDa42g1YbVycTidUF3rKjyLFDWbg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-29.7.0.tgz", + "integrity": "sha512-IOVhZSrg+UvVAshDSDtHyFCCBUl/Q3AAJv8iZ6ZjnZ74xzvwuzLXid9IIIPgTnY62SJjfuupMKZsZQRsCvxEgA==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-pnp-resolver": "^1.2.2", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "resolve": "^1.20.0", + "resolve.exports": "^2.0.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve-dependencies": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-29.7.0.tgz", + "integrity": "sha512-un0zD/6qxJ+S0et7WxeI3H5XSe9lTBBR7bOHCHXkKR6luG5mwDDlIzVQ0V5cZCuoTgEdcdwzTghYkTWfubi+nA==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-regex-util": "^29.6.3", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runner": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-29.7.0.tgz", + "integrity": "sha512-fsc4N6cPCAahybGBfTRcq5wFR6fpLznMg47sY5aDpsoejOcVYFb07AHuSnR0liMcPTgBsA3ZJL6kFOjPdoNipQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/environment": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "graceful-fs": "^4.2.9", + "jest-docblock": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-leak-detector": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-resolve": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-util": "^29.7.0", + "jest-watcher": "^29.7.0", + "jest-worker": "^29.7.0", + "p-limit": "^3.1.0", + "source-map-support": "0.5.13" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runtime": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-29.7.0.tgz", + "integrity": "sha512-gUnLjgwdGqW7B4LvOIkbKs9WGbn+QLqRQQ9juC6HndeDiezIwhDP+mhMwHWCEcfQ5RUXa6OPnFF8BJh5xegwwQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/globals": "^29.7.0", + "@jest/source-map": "^29.6.3", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "cjs-module-lexer": "^1.0.0", + "collect-v8-coverage": "^1.0.0", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-mock": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "slash": "^3.0.0", + "strip-bom": "^4.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-29.7.0.tgz", + "integrity": "sha512-Rm0BMWtxBcioHr1/OX5YCP8Uov4riHvKPknOGs804Zg9JGZgmIBkbtlxJC/7Z4msKYVbIJtfU+tKb8xlYNfdkw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@babel/generator": "^7.7.2", + "@babel/plugin-syntax-jsx": "^7.7.2", + "@babel/plugin-syntax-typescript": "^7.7.2", + "@babel/types": "^7.3.3", + "@jest/expect-utils": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0", + "chalk": "^4.0.0", + "expect": "^29.7.0", + "graceful-fs": "^4.2.9", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "natural-compare": "^1.4.0", + "pretty-format": "^29.7.0", + "semver": "^7.5.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/jest-util": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.7.0.tgz", + "integrity": "sha512-z6EbKajIpqGKU56y5KBUgy1dt1ihhQJgWzUlZHArA/+X2ad7Cb5iF+AK1EWVL/Bo7Rz9uurpqw6SiBCefUbCGA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-29.7.0.tgz", + "integrity": "sha512-ZB7wHqaRGVw/9hST/OuFUReG7M8vKeq0/J2egIGLdvjHCmYqGARhzXmtgi+gVeZ5uXFF219aOc3Ls2yLg27tkw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "camelcase": "^6.2.0", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "leven": "^3.1.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate/node_modules/camelcase": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", + "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/jest-watcher": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-29.7.0.tgz", + "integrity": "sha512-49Fg7WXkU3Vl2h6LbLtMQ/HyB6rXSIX7SqvBLQmssRBGN9I0PNvPmAmCWSOY6SOvrjhI/F7/bGAv9RtnsPA03g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "jest-util": "^29.7.0", + "string-length": "^4.0.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-worker": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-29.7.0.tgz", + "integrity": "sha512-eIz2msL/EzL9UFTFFx7jBTkeZfku0yUAyZZZmJ93H2TYEiroIx2PQjEXcwYtYl8zXCxb+PAmA2hLIt/6ZEkPHw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "jest-util": "^29.7.0", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-worker/node_modules/supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "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, + "license": "MIT" + }, + "node_modules/js-yaml": { + "version": "3.14.2", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.2.tgz", + "integrity": "sha512-PMSmkqxr106Xa156c2M265Z+FTrPl+oxd/rgOQy2tijQeK5TxQ43psO1ZCwhVOSdnn+RzkzlRz/eY4BgJBYVpg==", + "dev": true, + "license": "MIT", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsesc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", + "dev": true, + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-stringify-safe": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", + "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==", + "dev": true, + "license": "ISC" + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "license": "MIT", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/leven": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", + "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "dev": true, + "license": "MIT" + }, + "node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/make-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", + "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", + "dev": true, + "license": "MIT", + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/make-dir/node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/makeerror": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", + "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tmpl": "1.0.5" + } + }, + "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==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true, + "license": "MIT" + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "license": "MIT", + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true, + "license": "MIT" + }, + "node_modules/nock": { + "version": "13.5.6", + "resolved": "https://registry.npmjs.org/nock/-/nock-13.5.6.tgz", + "integrity": "sha512-o2zOYiCpzRqSzPj0Zt/dQ/DqZeYoaQ7TUonc/xUPjCGl9WeHpNbxgVvOquXYAaJzI0M9BXV3HTzG0p8IUAbBTQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "debug": "^4.1.0", + "json-stringify-safe": "^5.0.1", + "propagate": "^2.0.0" + }, + "engines": { + "node": ">= 10.13" + } + }, + "node_modules/node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-releases": { + "version": "2.0.27", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz", + "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==", + "dev": true, + "license": "MIT" + }, + "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", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "license": "ISC", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-locate/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/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/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "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/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "dev": true, + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pirates": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.7.tgz", + "integrity": "sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 6" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/prompts": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", + "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "kleur": "^3.0.3", + "sisteransi": "^1.0.5" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/propagate": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/propagate/-/propagate-2.0.1.tgz", + "integrity": "sha512-vGrhOavPSTz4QVNuBNdcNXePNdNMaO1xj9yBeH1ScQPjk/rhg9sSlCXPhMkFuaNNW/syTvYqsnbIJxMBfRbbag==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", + "license": "MIT" + }, + "node_modules/pure-rand": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-6.1.0.tgz", + "integrity": "sha512-bVWawvoZoBYpp6yIoQtQXHZjmz35RSVHnUOTefl8Vcjr8snTPY1wnpSPMWekcFwbxI6gtmT7rSYPFvz71ldiOA==", + "dev": true, + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/dubzzz" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/fast-check" + } + ], + "license": "MIT" + }, + "node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve": { + "version": "1.22.11", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz", + "integrity": "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-core-module": "^2.16.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-cwd": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", + "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve.exports": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-2.0.3.tgz", + "integrity": "sha512-OcXjMsGdhL4XnbShKpAcSqPMzQoYkYyhbEaeSko47MjRP9NfEQMhZkXL1DoFlt9LWQn4YttrdnV6X2OiyzBi+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/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/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", + "dev": true, + "license": "MIT" + }, + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.13", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.13.tgz", + "integrity": "sha512-SHSKFHadjVA5oR4PPqhtAVdcBWwRYVd6g6cAXnIbRiIwc2EhPrTuKUBdSLvlEKyIP3GCf89fltvcZiP9MMFA1w==", + "dev": true, + "license": "MIT", + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/stack-utils": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.6.tgz", + "integrity": "sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "escape-string-regexp": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/string-length": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", + "integrity": "sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-bom": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", + "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-final-newline": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", + "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/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/test-exclude": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", + "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", + "dev": true, + "license": "ISC", + "dependencies": { + "@istanbuljs/schema": "^0.1.2", + "glob": "^7.1.4", + "minimatch": "^3.0.4" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tmpl": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", + "integrity": "sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/type-detect": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", + "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/undici-types": { + "version": "7.16.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz", + "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==", + "dev": true, + "license": "MIT" + }, + "node_modules/update-browserslist-db": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.2.tgz", + "integrity": "sha512-E85pfNzMQ9jpKkA7+TJAi4TJN+tBCuWh5rUcS/sv6cFi+1q9LYDwDI5dpUL0u/73EElyQ8d3TEaeW4sPedBqYA==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.1" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/v8-to-istanbul": { + "version": "9.3.0", + "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-9.3.0.tgz", + "integrity": "sha512-kiGUalWN+rgBJ/1OHZsBtU4rXZOfj/7rKQxULKlIzwzQSvMJUUNgPwJEEh7gU6xEVxC0ahoOBvN2YI8GH6FNgA==", + "dev": true, + "license": "ISC", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.12", + "@types/istanbul-lib-coverage": "^2.0.1", + "convert-source-map": "^2.0.0" + }, + "engines": { + "node": ">=10.12.0" + } + }, + "node_modules/walker": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.8.tgz", + "integrity": "sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "makeerror": "1.0.12" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/write-file-atomic": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-4.0.2.tgz", + "integrity": "sha512-7KxauUdBmSdWnmpaGFg+ppNjKF8uNLry8LyzjauQDOVONfFLNKrKvQOxZ/VuTIcS/gge/YNahf5RIIQWTSarlg==", + "dev": true, + "license": "ISC", + "dependencies": { + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.7" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || >=16.0.0" + } + }, + "node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=10" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "license": "ISC" + }, + "node_modules/yargs": { + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "cliui": "^8.0.1", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.3", + "y18n": "^5.0.5", + "yargs-parser": "^21.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs-parser": { + "version": "21.1.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=12" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } } } } diff --git a/package.json b/package.json index 72c5786..b19cd71 100644 --- a/package.json +++ b/package.json @@ -20,10 +20,20 @@ "workflow", "pipeline" ], + "scripts": { + "test": "jest", + "test:watch": "jest --watch", + "test:coverage": "jest --coverage", + "test:ci": "jest --ci --coverage --maxWorkers=2" + }, "dependencies": { "axios": "^1.10.0", "form-data": "^4.0.3" }, + "devDependencies": { + "jest": "^29.7.0", + "nock": "^13.5.0" + }, "engines": { "node": ">=12.0.0" }, diff --git a/test/helpers/mock-axios.js b/test/helpers/mock-axios.js new file mode 100644 index 0000000..433773c --- /dev/null +++ b/test/helpers/mock-axios.js @@ -0,0 +1,271 @@ +/** + * Axios/nock helper functions for mocking Seqera Platform API + */ +const nock = require("nock"); + +const BASE_URL = "https://api.cloud.seqera.io"; + +/** + * Creates a nock scope for the Seqera API + * @param {string} baseUrl - Override base URL + * @returns {Object} Helper functions for mocking API endpoints + */ +function mockSeqeraAPI(baseUrl = BASE_URL) { + return { + /** + * Mock user-info endpoint (connectivity check) + */ + mockUserInfo: (response = { user: { userName: "testuser", email: "test@example.com" } }, status = 200) => { + return nock(baseUrl) + .get("/user-info") + .matchHeader("authorization", /^Bearer .+/) + .reply(status, response); + }, + + /** + * Mock workflow launch endpoint + */ + mockWorkflowLaunch: (response = { workflowId: "wf-123" }, status = 200) => { + return nock(baseUrl) + .post("/workflow/launch") + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, response); + }, + + /** + * Mock workflow status endpoint + */ + mockWorkflowStatus: (workflowId, workflow = {}, status = 200) => { + const defaultWorkflow = { + id: workflowId, + status: "running", + runName: "test-run", + ...workflow, + }; + return nock(baseUrl) + .get(`/workflow/${workflowId}`) + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { workflow: defaultWorkflow }); + }, + + /** + * Mock workflow launch config endpoint (for resume) + */ + mockWorkflowLaunchConfig: (workflowId, launch = {}, status = 200) => { + const defaultLaunch = { + id: "launch-123", + computeEnv: { id: "ce-123" }, + pipeline: "https://github.com/test/pipeline", + workDir: "s3://bucket/work", + sessionId: "session-123", + ...launch, + }; + return nock(baseUrl) + .get(`/workflow/${workflowId}/launch`) + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { launch: defaultLaunch }); + }, + + /** + * Mock pipelines list endpoint (launchpad search) + */ + mockPipelines: (pipelines = [], status = 200) => { + return nock(baseUrl) + .get("/pipelines") + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { pipelines }); + }, + + /** + * Mock pipeline launch config endpoint + */ + mockPipelineLaunchConfig: (pipelineId, launch = {}, status = 200) => { + const defaultLaunch = { + id: "launch-123", + computeEnv: { id: "ce-123" }, + pipeline: "https://github.com/test/pipeline", + workDir: "s3://bucket/work", + paramsText: "{}", + ...launch, + }; + return nock(baseUrl) + .get(`/pipelines/${pipelineId}/launch`) + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { launch: defaultLaunch }); + }, + + /** + * Mock data-links list endpoint + */ + mockDataLinks: (dataLinks = [], status = 200) => { + return nock(baseUrl) + .get("/data-links") + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { dataLinks }); + }, + + /** + * Mock data-links search endpoint (with trailing slash) + */ + mockDataLinksSearch: (dataLinks = [], status = 200) => { + return nock(baseUrl) + .get("/data-links/") + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { dataLinks }); + }, + + /** + * Mock data-link browse endpoint + */ + mockDataLinkBrowse: (dataLinkId, objects = [], nextPageToken = null, status = 200) => { + return nock(baseUrl) + .get(new RegExp(`/data-links/${dataLinkId}/browse.*`)) + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { objects, nextPageToken }); + }, + + /** + * Mock datasets create endpoint + */ + mockDatasetCreate: (response = { dataset: { id: "ds-123" } }, status = 200) => { + return nock(baseUrl) + .post("/datasets") + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, response); + }, + + /** + * Mock dataset upload endpoint + */ + mockDatasetUpload: (datasetId, status = 200) => { + return nock(baseUrl) + .post(`/datasets/${datasetId}/upload`) + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { version: { id: "v-123" } }); + }, + + /** + * Mock studios create endpoint + */ + mockStudiosCreate: (response = { sessionId: "studio-123" }, status = 200) => { + return nock(baseUrl) + .post("/studios") + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, response); + }, + + /** + * Mock studio status endpoint + */ + mockStudioStatus: (studioId, studio = {}, status = 200) => { + const defaultStudio = { + sessionId: studioId, + status: "running", + ...studio, + }; + return nock(baseUrl) + .get(`/studios/${studioId}`) + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, defaultStudio); + }, + + /** + * Mock organizations list endpoint + */ + mockOrganizations: (organizations = [], status = 200) => { + return nock(baseUrl) + .get("/orgs") + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { organizations }); + }, + + /** + * Mock workspaces list endpoint + */ + mockWorkspaces: (orgId, workspaces = [], status = 200) => { + return nock(baseUrl) + .get(`/orgs/${orgId}/workspaces`) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { workspaces }); + }, + + /** + * Mock workflows list endpoint + */ + mockWorkflowsList: (workflows = [], status = 200) => { + return nock(baseUrl) + .get("/workflow") + .query(true) + .matchHeader("authorization", /^Bearer .+/) + .reply(status, { workflows }); + }, + + /** + * Mock any GET endpoint + */ + mockGet: (path, response, status = 200) => { + return nock(baseUrl).get(path).query(true).reply(status, response); + }, + + /** + * Mock any POST endpoint + */ + mockPost: (path, response, status = 200) => { + return nock(baseUrl).post(path).query(true).reply(status, response); + }, + + /** + * Mock error response + */ + mockError: (method, path, errorCode, errorMessage = "Error") => { + const scope = nock(baseUrl); + const methodFn = method.toLowerCase() === "post" ? scope.post(path) : scope.get(path); + return methodFn.query(true).reply(errorCode, { message: errorMessage }); + }, + + /** + * Mock network error + */ + mockNetworkError: (method, path, errorCode = "ECONNREFUSED") => { + const scope = nock(baseUrl); + const methodFn = method.toLowerCase() === "post" ? scope.post(path) : scope.get(path); + return methodFn.query(true).replyWithError({ code: errorCode }); + }, + }; +} + +/** + * Clean all nock interceptors + */ +function cleanNock() { + nock.cleanAll(); +} + +/** + * Check if all nock interceptors were used + */ +function assertNockDone() { + if (!nock.isDone()) { + const pending = nock.pendingMocks(); + throw new Error(`Pending nock interceptors: ${pending.join(", ")}`); + } +} + +module.exports = { + mockSeqeraAPI, + cleanNock, + assertNockDone, + BASE_URL, +}; diff --git a/test/helpers/mock-node.js b/test/helpers/mock-node.js new file mode 100644 index 0000000..3816b13 --- /dev/null +++ b/test/helpers/mock-node.js @@ -0,0 +1,133 @@ +/** + * Mock Node instance factory + * + * Creates mock node instances with common methods and properties + * for testing Node-RED custom nodes. + */ + +/** + * Creates a mock node instance + * @param {Object} config - Node configuration + * @returns {Object} Mock node instance + */ +function createMockNode(config = {}) { + const eventHandlers = {}; + const contextData = {}; + + const node = { + // Node identity + id: config.id || "test-node-id", + name: config.name || "test-node", + type: config.type || "test-node-type", + z: config.z || "test-flow-id", + + // Seqera config reference (commonly used) + seqeraConfig: config.seqeraConfig || null, + + // Credentials + credentials: config.credentials || {}, + + // Node-RED methods + on: jest.fn((event, handler) => { + eventHandlers[event] = handler; + }), + + send: jest.fn(), + error: jest.fn(), + warn: jest.fn(), + log: jest.fn(), + debug: jest.fn(), + trace: jest.fn(), + status: jest.fn(), + done: jest.fn(), + + // Context storage + context: jest.fn(() => ({ + get: jest.fn((key) => contextData[key]), + set: jest.fn((key, value) => { + contextData[key] = value; + }), + keys: jest.fn(() => Object.keys(contextData)), + })), + + // Test helpers + _eventHandlers: eventHandlers, + _contextData: contextData, + + /** + * Trigger input event handler + * @param {Object} msg - Input message + * @param {Function} send - Send function (defaults to node.send) + * @param {Function} done - Done callback + * @returns {Promise} Resolves when handler completes + */ + _triggerInput: async function (msg, send, done) { + const handler = eventHandlers.input; + if (!handler) { + throw new Error("No input handler registered"); + } + return handler.call(this, msg, send || this.send, done || jest.fn()); + }, + + /** + * Trigger close event handler + * @param {boolean} removed - Whether node is being removed + * @param {Function} done - Done callback + * @returns {Promise} Resolves when handler completes + */ + _triggerClose: async function (removed, done) { + const handler = eventHandlers.close; + if (!handler) { + return; // Close handler is optional + } + // Handle both (done) and (removed, done) signatures + if (handler.length === 1) { + return handler.call(this, done || jest.fn()); + } + return handler.call(this, removed, done || jest.fn()); + }, + + /** + * Get stored context value + */ + _getContext: (key) => contextData[key], + + /** + * Set context value directly (for test setup) + */ + _setContext: (key, value) => { + contextData[key] = value; + }, + }; + + // Copy any additional config properties to node + Object.keys(config).forEach((key) => { + if (!["id", "name", "type", "z", "seqeraConfig", "credentials"].includes(key)) { + node[key] = config[key]; + } + }); + + return node; +} + +/** + * Creates a mock seqera-config node + * @param {Object} overrides - Override default values + * @returns {Object} Mock seqera-config node + */ +function createMockSeqeraConfigNode(overrides = {}) { + return { + id: overrides.id || "seqera-config-id", + name: overrides.name || "test-seqera-config", + type: "seqera-config", + baseUrl: overrides.baseUrl || "https://api.cloud.seqera.io", + workspaceId: overrides.workspaceId || "test-workspace-id", + credentials: { + token: overrides.token || "test-token-123", + ...overrides.credentials, + }, + ...overrides, + }; +} + +module.exports = { createMockNode, createMockSeqeraConfigNode }; diff --git a/test/helpers/mock-red.js b/test/helpers/mock-red.js new file mode 100644 index 0000000..6d6ed6d --- /dev/null +++ b/test/helpers/mock-red.js @@ -0,0 +1,321 @@ +/** + * Mock Node-RED runtime factory + * + * Provides a reusable factory for mocking the RED runtime object + * that Node-RED passes to node modules. + */ + +/** + * Creates a mock RED runtime object + * @param {Object} options - Configuration options + * @param {Object} options.nodes - Pre-registered nodes (id -> node) + * @param {Object} options.credentials - Pre-registered credentials (nodeId -> creds) + * @returns {Object} Mock RED object + */ +function createMockRED(options = {}) { + const nodes = new Map(Object.entries(options.nodes || {})); + const credentials = new Map(Object.entries(options.credentials || {})); + const registeredTypes = new Map(); + const httpEndpoints = { get: new Map(), post: new Map() }; + + const RED = { + nodes: { + /** + * Mock createNode - initializes node properties + */ + createNode: jest.fn((node, config) => { + node.id = config.id || "test-node-id"; + node.name = config.name || ""; + node.type = config.type || "test-node-type"; + node.z = config.z || "test-flow-id"; + + // Set up event handlers storage + node._eventHandlers = {}; + node.on = jest.fn((event, handler) => { + node._eventHandlers[event] = handler; + }); + + // Standard Node-RED node methods + node.send = jest.fn(); + node.error = jest.fn(); + node.warn = jest.fn(); + node.log = jest.fn(); + node.debug = jest.fn(); + node.trace = jest.fn(); + node.status = jest.fn(); + node.done = jest.fn(); + + // Context (flow/global storage) + const contextData = {}; + node._contextData = contextData; + node.context = jest.fn(() => ({ + get: jest.fn((key) => contextData[key]), + set: jest.fn((key, value) => { + contextData[key] = value; + }), + keys: jest.fn(() => Object.keys(contextData)), + })); + + /** + * Test helper: Trigger input event handler + * @param {Object} msg - Input message + * @param {Function} send - Send function (defaults to node.send) + * @param {Function} done - Done callback + * @returns {Promise} Resolves when handler completes + */ + node._triggerInput = async function (msg, send, done) { + const handler = node._eventHandlers.input; + if (!handler) { + throw new Error("No input handler registered"); + } + return handler.call(node, msg, send || node.send, done || jest.fn()); + }; + + /** + * Test helper: Trigger close event handler + * @param {boolean} removed - Whether node is being removed + * @param {Function} done - Done callback + * @returns {Promise} Resolves when handler completes + */ + node._triggerClose = async function (removed, done) { + const handler = node._eventHandlers.close; + if (!handler) { + return; // Close handler is optional + } + // Handle both (done) and (removed, done) signatures + if (handler.length === 1) { + return handler.call(node, done || jest.fn()); + } + return handler.call(node, removed, done || jest.fn()); + }; + + /** + * Test helper: Get stored context value + */ + node._getContext = (key) => contextData[key]; + + /** + * Test helper: Set context value directly (for test setup) + */ + node._setContext = (key, value) => { + contextData[key] = value; + }; + }), + + /** + * Mock registerType - stores node constructor for later retrieval + */ + registerType: jest.fn((type, constructor, opts) => { + registeredTypes.set(type, { constructor, opts }); + }), + + /** + * Mock getNode - returns node by ID from registry + */ + getNode: jest.fn((id) => nodes.get(id) || null), + + /** + * Mock getCredentials - returns credentials for a node + */ + getCredentials: jest.fn((id) => credentials.get(id) || {}), + }, + + util: { + /** + * Mock evaluateNodeProperty - evaluates typed input properties + */ + evaluateNodeProperty: jest.fn((value, type, node, msg) => { + switch (type) { + case "str": + return value; + case "num": + return Number(value); + case "bool": + return value === "true" || value === true; + case "json": + try { + return typeof value === "string" ? JSON.parse(value) : value; + } catch { + return value; + } + case "msg": + return getNestedProperty(msg, value); + case "flow": + return node.context?.().get(value); + case "global": + return node.context?.().get(value); + default: + return value; + } + }), + + /** + * Mock prepareJSONataExpression - returns expression object + */ + prepareJSONataExpression: jest.fn((expr, node) => ({ + _expr: expr, + _node: node, + })), + + /** + * Mock evaluateJSONataExpression - evaluates JSONata (simplified) + */ + evaluateJSONataExpression: jest.fn((prepared, msg, callback) => { + // Simple mock - just return payload by default + // Tests can override this for specific JSONata behavior + try { + callback(null, msg.payload); + } catch (err) { + callback(err, null); + } + }), + + /** + * Clone a message (simplified deep clone) + */ + cloneMessage: jest.fn((msg) => JSON.parse(JSON.stringify(msg))), + }, + + httpAdmin: { + /** + * Mock GET endpoint registration + */ + get: jest.fn((path, ...handlers) => { + const handler = handlers[handlers.length - 1]; + httpEndpoints.get.set(path, handler); + }), + + /** + * Mock POST endpoint registration + */ + post: jest.fn((path, ...handlers) => { + const handler = handlers[handlers.length - 1]; + httpEndpoints.post.set(path, handler); + }), + }, + + /** + * Test helpers for accessing internal state + */ + _testHelpers: { + nodes, + credentials, + registeredTypes, + httpEndpoints, + + /** + * Add a node to the registry + */ + addNode: (id, node) => nodes.set(id, node), + + /** + * Add credentials for a node + */ + addCredentials: (id, creds) => credentials.set(id, creds), + + /** + * Get registered node type + */ + getRegisteredType: (type) => registeredTypes.get(type), + + /** + * Get HTTP endpoint handler + */ + getHttpHandler: (method, path) => httpEndpoints[method.toLowerCase()]?.get(path), + + /** + * Simulate HTTP request to registered endpoint + */ + async simulateHttpRequest(method, path, options = {}) { + const handler = httpEndpoints[method.toLowerCase()]?.get(path); + if (!handler) { + throw new Error(`No handler registered for ${method} ${path}`); + } + + const req = { + params: options.params || {}, + query: options.query || {}, + body: options.body || {}, + ...options.req, + }; + + let responseData = null; + let responseStatus = 200; + const res = { + json: jest.fn((data) => { + responseData = data; + }), + status: jest.fn((code) => { + responseStatus = code; + return res; + }), + send: jest.fn((data) => { + responseData = data; + }), + }; + + await handler(req, res); + + return { data: responseData, status: responseStatus, res }; + }, + }, + }; + + return RED; +} + +/** + * Helper to get nested property from object using dot notation + */ +function getNestedProperty(obj, path) { + if (!path) return obj; + const parts = path.split("."); + let current = obj; + for (const part of parts) { + if (current == null) return undefined; + current = current[part]; + } + return current; +} + +/** + * Simulate an HTTP request to a handler + * @param {Function} handler - The HTTP handler function + * @param {Object} options - Request options + * @returns {Promise} Response object with statusCode and body + */ +async function simulateHttpRequest(handler, options = {}) { + const req = { + params: options.params || {}, + query: options.query || {}, + body: options.body || {}, + ...options.req, + }; + + let responseBody = null; + let statusCode = 200; + + const res = { + statusCode: 200, + body: null, + json: jest.fn((data) => { + res.body = data; + responseBody = data; + }), + status: jest.fn((code) => { + res.statusCode = code; + statusCode = code; + return res; + }), + send: jest.fn((data) => { + res.body = data; + responseBody = data; + }), + }; + + await handler(req, res); + + return { res, statusCode, body: responseBody }; +} + +module.exports = { createMockRED, getNestedProperty, simulateHttpRequest }; diff --git a/test/helpers/test-utils.js b/test/helpers/test-utils.js new file mode 100644 index 0000000..b481911 --- /dev/null +++ b/test/helpers/test-utils.js @@ -0,0 +1,175 @@ +/** + * Shared test utilities + */ + +/** + * Creates a mock Seqera config object (for direct use, not as a node) + * @param {Object} overrides - Override default values + * @returns {Object} Mock seqera config + */ +function createMockSeqeraConfig(overrides = {}) { + return { + id: "seqera-config-id", + baseUrl: "https://api.cloud.seqera.io", + workspaceId: "test-workspace-id", + credentials: { + token: "test-token-123", + ...overrides.credentials, + }, + ...overrides, + }; +} + +/** + * Creates a mock message object + * @param {Object} overrides - Override default values + * @returns {Object} Mock message + */ +function createMockMsg(overrides = {}) { + return { + payload: {}, + _msgid: "test-msg-id-" + Math.random().toString(36).substr(2, 9), + ...overrides, + }; +} + +/** + * Asserts that node status was updated with expected values + * @param {Object} node - Mock node instance + * @param {Object} expectedStatus - Expected status properties + */ +function expectStatusUpdate(node, expectedStatus) { + expect(node.status).toHaveBeenCalledWith(expect.objectContaining(expectedStatus)); +} + +/** + * Asserts that node status was set to error (red) + * @param {Object} node - Mock node instance + */ +function expectErrorStatus(node) { + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + }), + ); +} + +/** + * Asserts that node status was set to success (green) + * @param {Object} node - Mock node instance + */ +function expectSuccessStatus(node) { + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "green", + shape: "dot", + }), + ); +} + +/** + * Asserts that input message properties are passed through to output + * @param {Object} outputMsg - Output message + * @param {Object} inputMsg - Input message + * @param {string[]} excludeKeys - Keys to exclude from comparison + */ +function expectMessagePassthrough(outputMsg, inputMsg, excludeKeys = []) { + const defaultExclude = ["payload", "workflowId", "datasetId", "studioId", "files"]; + const allExclude = [...defaultExclude, ...excludeKeys]; + + Object.keys(inputMsg).forEach((key) => { + if (!allExclude.includes(key)) { + expect(outputMsg[key]).toEqual(inputMsg[key]); + } + }); +} + +/** + * Waits for a condition to be true + * @param {Function} condition - Function returning boolean + * @param {number} timeout - Max wait time in ms + * @param {number} interval - Check interval in ms + * @returns {Promise} Resolves when condition is true + */ +async function waitFor(condition, timeout = 5000, interval = 50) { + const startTime = Date.now(); + while (Date.now() - startTime < timeout) { + if (await condition()) { + return; + } + await new Promise((resolve) => setTimeout(resolve, interval)); + } + throw new Error(`waitFor timed out after ${timeout}ms`); +} + +/** + * Creates a deferred promise for async testing + * @returns {Object} Object with promise, resolve, and reject + */ +function createDeferred() { + let resolve, reject; + const promise = new Promise((res, rej) => { + resolve = res; + reject = rej; + }); + return { promise, resolve, reject }; +} + +/** + * Waits for node.send to be called + * @param {Object} node - Mock node instance + * @param {number} timeout - Max wait time + * @returns {Promise} Resolves with sent message + */ +async function waitForSend(node, timeout = 5000) { + await waitFor(() => node.send.mock.calls.length > 0, timeout); + return node.send.mock.calls[0][0]; +} + +/** + * Extracts all messages sent by a node + * @param {Object} node - Mock node instance + * @returns {Array} Array of sent messages + */ +function getSentMessages(node) { + return node.send.mock.calls.map((call) => call[0]); +} + +/** + * Flushes the promise queue to allow async operations to complete + * This is necessary when testing code that uses axios or other async operations + * with fake timers. + * @param {number} iterations - Number of flush iterations (default 10) + */ +async function flushPromises(iterations = 10) { + for (let i = 0; i < iterations; i++) { + await Promise.resolve(); + } + // Also run any pending jest timers that might have been scheduled + (await jest.runAllTimersAsync?.()) || jest.runAllTicks?.(); +} + +/** + * Advances fake timers and runs any pending promises + * @param {number} ms - Milliseconds to advance + */ +async function advanceTimersAndFlush(ms) { + jest.advanceTimersByTime(ms); + // Flush pending promises (multiple iterations for axios chain) + await flushPromises(); +} + +module.exports = { + createMockSeqeraConfig, + createMockMsg, + expectStatusUpdate, + expectErrorStatus, + expectSuccessStatus, + expectMessagePassthrough, + waitFor, + createDeferred, + waitForSend, + getSentMessages, + advanceTimersAndFlush, + flushPromises, +}; diff --git a/test/integration/http-endpoints.test.js b/test/integration/http-endpoints.test.js new file mode 100644 index 0000000..a15f943 --- /dev/null +++ b/test/integration/http-endpoints.test.js @@ -0,0 +1,343 @@ +/** + * Integration tests for HTTP admin endpoints + * + * Tests the HTTP endpoints registered on RED.httpAdmin + */ +const nock = require("nock"); +const { createMockRED, simulateHttpRequest } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); + +describe("HTTP Admin Endpoints", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + }); + + afterEach(() => { + nock.cleanAll(); + }); + + describe("/seqera-config/connectivity-check", () => { + beforeEach(() => { + // Load config node to register endpoints + require("../../nodes/config")(RED); + }); + + it("should register the endpoint", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/seqera-config/connectivity-check", expect.any(Function)); + }); + + it("should return success on valid token", async () => { + api.mockUserInfo({ user: { userName: "testuser" } }); + + const handler = RED._testHelpers.getHttpHandler("get", "/seqera-config/connectivity-check"); + const { res } = await simulateHttpRequest(handler, { + query: { + baseUrl: BASE_URL, + token: "test-token", + }, + }); + + expect(res.statusCode).toBe(200); + expect(res.body.success).toBe(true); + }); + + it("should return error on invalid token", async () => { + api.mockError("get", "/user-info", 401, "Unauthorized"); + + const handler = RED._testHelpers.getHttpHandler("get", "/seqera-config/connectivity-check"); + const { res } = await simulateHttpRequest(handler, { + query: { + baseUrl: BASE_URL, + token: "invalid-token", + }, + }); + + expect(res.statusCode).toBe(200); + expect(res.body.success).toBe(false); + }); + + it("should return success=false if baseUrl missing", async () => { + const handler = RED._testHelpers.getHttpHandler("get", "/seqera-config/connectivity-check"); + const { res } = await simulateHttpRequest(handler, { + query: { + token: "test-token", + }, + }); + + // Implementation returns 200 with success: false, not 400 + expect(res.statusCode).toBe(200); + expect(res.body.success).toBe(false); + }); + + it("should return success=false if token missing", async () => { + const handler = RED._testHelpers.getHttpHandler("get", "/seqera-config/connectivity-check"); + const { res } = await simulateHttpRequest(handler, { + query: { + baseUrl: BASE_URL, + }, + }); + + expect(res.statusCode).toBe(200); + expect(res.body.success).toBe(false); + expect(res.body.isEmptyToken).toBe(true); + }); + }); + + describe("/seqera-config/workspaces", () => { + beforeEach(() => { + require("../../nodes/config")(RED); + }); + + it("should register the endpoint", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/seqera-config/workspaces", expect.any(Function)); + }); + + it("should return organizations and workspaces", async () => { + api.mockUserInfo({ + user: { + userName: "testuser", + orgs: [ + { + orgId: "org-1", + name: "Test Org", + workspaces: [ + { workspaceId: "ws-1", workspaceName: "Workspace 1" }, + { workspaceId: "ws-2", workspaceName: "Workspace 2" }, + ], + }, + ], + }, + }); + + const handler = RED._testHelpers.getHttpHandler("get", "/seqera-config/workspaces"); + const { res } = await simulateHttpRequest(handler, { + query: { + baseUrl: BASE_URL, + token: "test-token", + }, + }); + + expect(res.statusCode).toBe(200); + // The actual implementation returns user data which contains orgs + expect(res.body).toBeDefined(); + }); + + it("should handle API errors gracefully", async () => { + api.mockError("get", "/user-info", 500, "Server Error"); + + const handler = RED._testHelpers.getHttpHandler("get", "/seqera-config/workspaces"); + const { res } = await simulateHttpRequest(handler, { + query: { + baseUrl: BASE_URL, + token: "test-token", + }, + }); + + // Implementation catches errors and returns 200 with error info + expect(res.statusCode).toBe(200); + }); + }); + + describe("/admin/seqera/pipelines/:nodeId", () => { + beforeEach(() => { + require("../../nodes/workflow-launch")(RED); + }); + + it("should register the endpoint", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/admin/seqera/pipelines/:nodeId", expect.any(Function)); + }); + + it("should return pipelines for autocomplete", async () => { + api.mockPipelines([ + { pipelineId: "1", name: "pipeline-1" }, + { pipelineId: "2", name: "pipeline-2" }, + ]); + + const handler = RED._testHelpers.getHttpHandler("get", "/admin/seqera/pipelines/:nodeId"); + const { res } = await simulateHttpRequest(handler, { + params: { nodeId: "seqera-config-id" }, + query: { search: "pipeline" }, + }); + + expect(res.statusCode).toBe(200); + expect(Array.isArray(res.body)).toBe(true); + }); + + it("should handle node not found with query params fallback", async () => { + api.mockUserInfo({ user: { userName: "testuser" } }); + api.mockPipelines([{ pipelineId: "1", name: "test-pipeline" }]); + + const handler = RED._testHelpers.getHttpHandler("get", "/admin/seqera/pipelines/:nodeId"); + const { res } = await simulateHttpRequest(handler, { + params: { nodeId: "non-existent-node" }, + query: { + search: "test", + baseUrl: BASE_URL, + token: "test-token", + workspaceId: "ws-123", + }, + }); + + expect(res.statusCode).toBe(200); + }); + + it("should filter pipelines by search term", async () => { + api.mockPipelines([ + { pipelineId: "1", name: "nf-core/rnaseq" }, + { pipelineId: "2", name: "nf-core/atacseq" }, + ]); + + const handler = RED._testHelpers.getHttpHandler("get", "/admin/seqera/pipelines/:nodeId"); + const { res } = await simulateHttpRequest(handler, { + params: { nodeId: "seqera-config-id" }, + query: { search: "rnaseq" }, + }); + + expect(res.statusCode).toBe(200); + // The API should filter by search, or return all for client-side filtering + expect(Array.isArray(res.body)).toBe(true); + }); + }); + + describe("/admin/seqera/datalinks/:nodeId", () => { + beforeEach(() => { + require("../../nodes/datalink-list")(RED); + }); + + it("should register the endpoint", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/admin/seqera/datalinks/:nodeId", expect.any(Function)); + }); + + it("should return data links for autocomplete", async () => { + api.mockDataLinksSearch([ + { id: "dl-1", name: "data-link-1", resourceRef: "s3://bucket1" }, + { id: "dl-2", name: "data-link-2", resourceRef: "s3://bucket2" }, + ]); + + const handler = RED._testHelpers.getHttpHandler("get", "/admin/seqera/datalinks/:nodeId"); + const { res } = await simulateHttpRequest(handler, { + params: { nodeId: "seqera-config-id" }, + query: { search: "data" }, + }); + + expect(res.statusCode).toBe(200); + expect(Array.isArray(res.body)).toBe(true); + }); + + it("should handle node not found with query params fallback", async () => { + api.mockUserInfo({ user: { userName: "testuser" } }); + api.mockDataLinksSearch([{ id: "dl-1", name: "test-link", resourceRef: "s3://bucket" }]); + + const handler = RED._testHelpers.getHttpHandler("get", "/admin/seqera/datalinks/:nodeId"); + const { res } = await simulateHttpRequest(handler, { + params: { nodeId: "non-existent-node" }, + query: { + search: "test", + baseUrl: BASE_URL, + token: "test-token", + workspaceId: "ws-123", + }, + }); + + expect(res.statusCode).toBe(200); + }); + + it("should return empty array when no matches", async () => { + api.mockDataLinksSearch([]); + + const handler = RED._testHelpers.getHttpHandler("get", "/admin/seqera/datalinks/:nodeId"); + const { res } = await simulateHttpRequest(handler, { + params: { nodeId: "seqera-config-id" }, + query: { search: "nonexistent" }, + }); + + expect(res.statusCode).toBe(200); + expect(res.body).toHaveLength(0); + }); + }); + + describe("endpoint error handling", () => { + beforeEach(() => { + require("../../nodes/config")(RED); + require("../../nodes/workflow-launch")(RED); + require("../../nodes/datalink-list")(RED); + }); + + it("should handle network errors gracefully", async () => { + nock.cleanAll(); + nock(BASE_URL).get("/user-info").replyWithError("Network error"); + + const handler = RED._testHelpers.getHttpHandler("get", "/seqera-config/connectivity-check"); + const { res } = await simulateHttpRequest(handler, { + query: { + baseUrl: BASE_URL, + token: "test-token", + }, + }); + + expect(res.statusCode).toBe(200); + expect(res.body.success).toBe(false); + }); + + it("should handle missing query parameters gracefully", async () => { + const handler = RED._testHelpers.getHttpHandler("get", "/seqera-config/connectivity-check"); + const { res } = await simulateHttpRequest(handler, { + query: {}, + }); + + // Implementation returns 200 with success: false, not 400 + expect(res.statusCode).toBe(200); + expect(res.body.success).toBe(false); + }); + }); + + describe("cross-endpoint consistency", () => { + beforeEach(() => { + require("../../nodes/config")(RED); + require("../../nodes/workflow-launch")(RED); + require("../../nodes/datalink-list")(RED); + }); + + it("should all endpoints use same authentication pattern", async () => { + // All endpoints should accept baseUrl and token in query params + // when the node doesn't exist + api.mockUserInfo({ user: { userName: "testuser", orgs: [] } }); + api.mockPipelines([]); + api.mockDataLinksSearch([]); + + const endpoints = [ + { path: "/admin/seqera/pipelines/:nodeId", method: "get", params: { nodeId: "new-node" } }, + { path: "/admin/seqera/datalinks/:nodeId", method: "get", params: { nodeId: "new-node" } }, + ]; + + for (const endpoint of endpoints) { + // Re-mock for each call since nock consumes mocks + api.mockUserInfo({ user: { userName: "testuser", orgs: [] } }); + api.mockPipelines([]); + api.mockDataLinksSearch([]); + + const handler = RED._testHelpers.getHttpHandler(endpoint.method, endpoint.path); + const { res } = await simulateHttpRequest(handler, { + params: endpoint.params || {}, + query: { + baseUrl: BASE_URL, + token: "test-token", + workspaceId: "ws-123", + search: "", + }, + }); + + expect(res.statusCode).toBe(200); + } + }); + }); +}); diff --git a/test/setup.js b/test/setup.js new file mode 100644 index 0000000..33d9c2a --- /dev/null +++ b/test/setup.js @@ -0,0 +1,19 @@ +/** + * Global test setup for Jest + */ +const nock = require("nock"); + +// Disable real HTTP requests during tests +beforeAll(() => { + nock.disableNetConnect(); +}); + +// Clean up nock after each test +afterEach(() => { + nock.cleanAll(); +}); + +// Re-enable connections after all tests +afterAll(() => { + nock.enableNetConnect(); +}); diff --git a/test/unit/config.test.js b/test/unit/config.test.js new file mode 100644 index 0000000..0885f0e --- /dev/null +++ b/test/unit/config.test.js @@ -0,0 +1,455 @@ +/** + * Tests for nodes/config.js (seqera-config node) + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { BASE_URL } = require("../helpers/mock-axios"); + +describe("seqera-config node", () => { + let RED; + + beforeEach(() => { + RED = createMockRED(); + // Load the config node module + require("../../nodes/config")(RED); + }); + + describe("node registration", () => { + it("should register seqera-config type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith("seqera-config", expect.any(Function), expect.any(Object)); + }); + + it("should register credentials schema with token", () => { + const registerCall = RED.nodes.registerType.mock.calls.find((call) => call[0] === "seqera-config"); + expect(registerCall[2]).toEqual({ + credentials: { + token: { type: "password" }, + }, + }); + }); + }); + + describe("node initialization", () => { + it("should set baseUrl from config", () => { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-config").constructor; + const node = {}; + RED._testHelpers.addCredentials("test-id", { token: "test-token" }); + + NodeConstructor.call(node, { + id: "test-id", + baseUrl: "https://custom.api.example.com", + workspaceId: "ws-123", + }); + + expect(node.baseUrl).toBe("https://custom.api.example.com"); + }); + + it("should use default baseUrl if not provided", () => { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-config").constructor; + const node = {}; + + NodeConstructor.call(node, { id: "test-id" }); + + expect(node.baseUrl).toBe("https://api.cloud.seqera.io"); + }); + + it("should set workspaceId from config", () => { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-config").constructor; + const node = {}; + + NodeConstructor.call(node, { id: "test-id", workspaceId: "my-workspace-id" }); + + expect(node.workspaceId).toBe("my-workspace-id"); + }); + + it("should set workspaceId to null if not provided", () => { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-config").constructor; + const node = {}; + + NodeConstructor.call(node, { id: "test-id" }); + + expect(node.workspaceId).toBeNull(); + }); + + it("should load credentials", () => { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-config").constructor; + const node = {}; + RED._testHelpers.addCredentials("test-id", { token: "secret-token" }); + + NodeConstructor.call(node, { id: "test-id" }); + + expect(node.credentials).toEqual({ token: "secret-token" }); + }); + }); + + describe("/seqera-config/connectivity-check endpoint", () => { + it("should register GET endpoint", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/seqera-config/connectivity-check", expect.any(Function)); + }); + + it("should return success with user info on valid token", async () => { + nock(BASE_URL) + .get("/user-info") + .matchHeader("authorization", "Bearer valid-token") + .reply(200, { + user: { userName: "testuser", email: "test@example.com" }, + }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, token: "valid-token" }, + }); + + expect(data.success).toBe(true); + expect(data.user).toEqual({ + userName: "testuser", + email: "test@example.com", + }); + }); + + it("should return failure if no token provided", async () => { + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("No API token provided"); + expect(data.isEmptyToken).toBe(true); + }); + + it("should return failure if no baseUrl provided", async () => { + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { token: "some-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("No base URL provided"); + }); + + it("should retrieve stored token if nodeId provided", async () => { + RED._testHelpers.addCredentials("config-node-id", { token: "stored-token" }); + + nock(BASE_URL) + .get("/user-info") + .matchHeader("authorization", "Bearer stored-token") + .reply(200, { + user: { userName: "storeduser", email: "stored@example.com" }, + }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, nodeId: "config-node-id" }, + }); + + expect(data.success).toBe(true); + expect(data.user.userName).toBe("storeduser"); + }); + + it("should return failure if nodeId provided but no stored token", async () => { + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, nodeId: "non-existent-node" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("No stored token found"); + expect(data.isEmptyToken).toBe(true); + }); + + it("should return failure on 401 (invalid token)", async () => { + nock(BASE_URL).get("/user-info").reply(401, { message: "Unauthorized" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, token: "invalid-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("Invalid API token"); + }); + + it("should return failure on 403 (forbidden)", async () => { + nock(BASE_URL).get("/user-info").reply(403, { message: "Forbidden" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, token: "forbidden-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("Invalid API token"); + }); + + it("should return failure on other API errors", async () => { + nock(BASE_URL).get("/user-info").reply(500, { message: "Server Error" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, token: "some-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("API error: 500"); + }); + + it("should return failure on network error (ENOTFOUND)", async () => { + nock(BASE_URL).get("/user-info").replyWithError({ code: "ENOTFOUND" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, token: "some-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toContain("Connection failed"); + }); + + it("should return failure on connection refused", async () => { + nock(BASE_URL).get("/user-info").replyWithError({ code: "ECONNREFUSED" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, token: "some-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toContain("Connection failed"); + }); + + it("should return failure on timeout", async () => { + nock(BASE_URL).get("/user-info").replyWithError({ code: "ETIMEDOUT" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, token: "some-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toContain("timeout"); + }); + + it("should return failure on invalid response (missing user)", async () => { + nock(BASE_URL).get("/user-info").reply(200, { something: "else" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/connectivity-check", { + query: { baseUrl: BASE_URL, token: "some-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("Invalid response from Seqera API"); + }); + }); + + describe("/seqera-config/workspaces endpoint", () => { + it("should register GET endpoint", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/seqera-config/workspaces", expect.any(Function)); + }); + + it("should return organizations with workspaces", async () => { + nock(BASE_URL) + .get("/orgs") + .reply(200, { + organizations: [ + { orgId: 1, name: "org1", fullName: "Organization One" }, + { orgId: 2, name: "org2", fullName: "Organization Two" }, + ], + }); + + nock(BASE_URL) + .get("/orgs/1/workspaces") + .reply(200, { + workspaces: [ + { id: "ws-1", name: "workspace1" }, + { id: "ws-2", name: "workspace2" }, + ], + }); + + nock(BASE_URL) + .get("/orgs/2/workspaces") + .reply(200, { + workspaces: [{ id: "ws-3", name: "workspace3" }], + }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, token: "valid-token" }, + }); + + expect(data.success).toBe(true); + expect(data.organizations).toHaveLength(2); + expect(data.organizations[0].orgName).toBe("org1"); + expect(data.organizations[0].workspaces).toHaveLength(2); + }); + + it("should filter out community org", async () => { + nock(BASE_URL) + .get("/orgs") + .reply(200, { + organizations: [ + { orgId: 1, name: "community", fullName: "Community" }, + { orgId: 2, name: "myorg", fullName: "My Org" }, + ], + }); + + nock(BASE_URL) + .get("/orgs/2/workspaces") + .reply(200, { + workspaces: [{ id: "ws-1", name: "workspace1" }], + }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, token: "valid-token" }, + }); + + expect(data.success).toBe(true); + expect(data.organizations).toHaveLength(1); + expect(data.organizations[0].orgName).toBe("myorg"); + }); + + it("should sort organizations alphabetically", async () => { + nock(BASE_URL) + .get("/orgs") + .reply(200, { + organizations: [ + { orgId: 1, name: "zebra", fullName: "Zebra Org" }, + { orgId: 2, name: "alpha", fullName: "Alpha Org" }, + { orgId: 3, name: "beta", fullName: "Beta Org" }, + ], + }); + + nock(BASE_URL) + .get("/orgs/1/workspaces") + .reply(200, { workspaces: [{ id: "ws-1", name: "ws1" }] }); + nock(BASE_URL) + .get("/orgs/2/workspaces") + .reply(200, { workspaces: [{ id: "ws-2", name: "ws2" }] }); + nock(BASE_URL) + .get("/orgs/3/workspaces") + .reply(200, { workspaces: [{ id: "ws-3", name: "ws3" }] }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, token: "valid-token" }, + }); + + expect(data.organizations[0].orgName).toBe("alpha"); + expect(data.organizations[1].orgName).toBe("beta"); + expect(data.organizations[2].orgName).toBe("zebra"); + }); + + it("should sort workspaces alphabetically within org", async () => { + nock(BASE_URL) + .get("/orgs") + .reply(200, { + organizations: [{ orgId: 1, name: "org1", fullName: "Org One" }], + }); + + nock(BASE_URL) + .get("/orgs/1/workspaces") + .reply(200, { + workspaces: [ + { id: "ws-3", name: "zeta" }, + { id: "ws-1", name: "alpha" }, + { id: "ws-2", name: "beta" }, + ], + }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, token: "valid-token" }, + }); + + expect(data.organizations[0].workspaces[0].name).toBe("alpha"); + expect(data.organizations[0].workspaces[1].name).toBe("beta"); + expect(data.organizations[0].workspaces[2].name).toBe("zeta"); + }); + + it("should exclude orgs with no workspaces", async () => { + nock(BASE_URL) + .get("/orgs") + .reply(200, { + organizations: [ + { orgId: 1, name: "empty-org", fullName: "Empty Org" }, + { orgId: 2, name: "has-workspaces", fullName: "Has Workspaces" }, + ], + }); + + nock(BASE_URL).get("/orgs/1/workspaces").reply(200, { workspaces: [] }); + nock(BASE_URL) + .get("/orgs/2/workspaces") + .reply(200, { workspaces: [{ id: "ws-1", name: "ws1" }] }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, token: "valid-token" }, + }); + + expect(data.organizations).toHaveLength(1); + expect(data.organizations[0].orgName).toBe("has-workspaces"); + }); + + it("should handle workspace fetch failure for single org", async () => { + nock(BASE_URL) + .get("/orgs") + .reply(200, { + organizations: [ + { orgId: 1, name: "failing-org", fullName: "Failing Org" }, + { orgId: 2, name: "working-org", fullName: "Working Org" }, + ], + }); + + nock(BASE_URL).get("/orgs/1/workspaces").reply(500, { message: "Error" }); + nock(BASE_URL) + .get("/orgs/2/workspaces") + .reply(200, { workspaces: [{ id: "ws-1", name: "ws1" }] }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, token: "valid-token" }, + }); + + // Should still return working org + expect(data.success).toBe(true); + expect(data.organizations).toHaveLength(1); + expect(data.organizations[0].orgName).toBe("working-org"); + }); + + it("should return failure if no token provided", async () => { + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("No API token provided"); + }); + + it("should return failure if no baseUrl provided", async () => { + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { token: "some-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("No base URL provided"); + }); + + it("should retrieve stored token if nodeId provided", async () => { + RED._testHelpers.addCredentials("config-node-id", { token: "stored-token" }); + + nock(BASE_URL).get("/orgs").matchHeader("authorization", "Bearer stored-token").reply(200, { organizations: [] }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, nodeId: "config-node-id" }, + }); + + expect(data.success).toBe(true); + }); + + it("should return failure on 401", async () => { + nock(BASE_URL).get("/orgs").reply(401, { message: "Unauthorized" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, token: "invalid-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("Invalid API token"); + }); + + it("should return failure on invalid orgs response", async () => { + nock(BASE_URL).get("/orgs").reply(200, { something: "else" }); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/seqera-config/workspaces", { + query: { baseUrl: BASE_URL, token: "valid-token" }, + }); + + expect(data.success).toBe(false); + expect(data.message).toBe("Invalid organizations response"); + }); + }); +}); diff --git a/test/unit/datalink-list.test.js b/test/unit/datalink-list.test.js new file mode 100644 index 0000000..3dc4d8f --- /dev/null +++ b/test/unit/datalink-list.test.js @@ -0,0 +1,255 @@ +/** + * Tests for nodes/datalink-list.js + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); +const { + createMockMsg, + expectSuccessStatus, + expectErrorStatus, + expectMessagePassthrough, +} = require("../helpers/test-utils"); + +describe("seqera-datalink-list node", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + // Load the datalink-list node module + require("../../nodes/datalink-list")(RED); + }); + + function createNode(configOverrides = {}) { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-datalink-list").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-datalink-list", + seqera: "seqera-config-id", + dataLinkName: "test-data-link", + dataLinkNameType: "str", + basePath: "", + basePathType: "str", + prefix: "", + prefixType: "str", + pattern: "", + patternType: "str", + maxResults: "100", + maxResultsType: "num", + workspaceId: "", + workspaceIdType: "str", + baseUrl: "", + baseUrlType: "str", + depth: "0", + depthType: "num", + returnType: "files", + ...configOverrides, + }; + + NodeConstructor.call(node, config); + return node; + } + + describe("node registration", () => { + it("should register seqera-datalink-list type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith( + "seqera-datalink-list", + expect.any(Function), + expect.any(Object), + ); + }); + + it("should register HTTP endpoint for datalink autocomplete", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/admin/seqera/datalinks/:nodeId", expect.any(Function)); + }); + }); + + describe("basic listing", () => { + it("should list files from data link", async () => { + const node = createNode(); + + api.mockDataLinksSearch([ + { + id: "dl-123", + name: "test-data-link", + resourceRef: "s3://my-bucket", + type: "bucket", + provider: "aws", + }, + ]); + + api.mockDataLinkBrowse("dl-123", [ + { name: "file1.txt", type: "FILE" }, + { name: "file2.csv", type: "FILE" }, + ]); + + const msg = createMockMsg({}); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalled(); + const output = send.mock.calls[0][0]; + expect(output.payload.files).toHaveLength(2); + }); + + it("should set blue ring status during listing", async () => { + const node = createNode(); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link", resourceRef: "s3://bucket" }]); + api.mockDataLinkBrowse("dl-123", []); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "blue", + shape: "ring", + }), + ); + }); + + it("should set green dot status with count on success", async () => { + const node = createNode(); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link", resourceRef: "s3://bucket" }]); + api.mockDataLinkBrowse("dl-123", [ + { name: "file1.txt", type: "FILE" }, + { name: "file2.txt", type: "FILE" }, + { name: "file3.txt", type: "FILE" }, + ]); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "green", + shape: "dot", + text: expect.stringContaining("3 items"), + }), + ); + }); + }); + + describe("output format", () => { + it("should return payload.files with full objects", async () => { + const node = createNode(); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link", resourceRef: "s3://bucket" }]); + api.mockDataLinkBrowse("dl-123", [ + { name: "file1.txt", type: "FILE", size: 100 }, + { name: "file2.txt", type: "FILE", size: 200 }, + ]); + + const msg = createMockMsg({}); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const output = send.mock.calls[0][0]; + expect(output.payload.files[0]).toHaveProperty("name"); + expect(output.payload.files[0]).toHaveProperty("type"); + }); + + it("should return files array with full paths including resourceRef", async () => { + const node = createNode(); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link", resourceRef: "s3://my-bucket" }]); + api.mockDataLinkBrowse("dl-123", [{ name: "path/to/file.txt", type: "FILE" }]); + + const msg = createMockMsg({}); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const output = send.mock.calls[0][0]; + expect(output.files[0]).toBe("s3://my-bucket/path/to/file.txt"); + }); + + it("should include resourceType, resourceRef, provider in payload", async () => { + const node = createNode(); + + api.mockDataLinksSearch([ + { + id: "dl-123", + name: "test-data-link", + resourceRef: "gs://gcp-bucket", + type: "bucket", + provider: "gcp", + }, + ]); + api.mockDataLinkBrowse("dl-123", []); + + const msg = createMockMsg({}); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const output = send.mock.calls[0][0]; + expect(output.payload.resourceRef).toBe("gs://gcp-bucket"); + expect(output.payload.resourceType).toBe("bucket"); + expect(output.payload.provider).toBe("gcp"); + }); + }); + + describe("message passthrough", () => { + it("should preserve input message properties in output", async () => { + const node = createNode(); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link", resourceRef: "s3://bucket" }]); + api.mockDataLinkBrowse("dl-123", []); + + const msg = createMockMsg({ + _context: { flowId: "flow-123" }, + correlationId: "corr-456", + customProp: "custom-value", + }); + + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const output = send.mock.calls[0][0]; + expectMessagePassthrough(output, msg); + }); + }); + + describe("error handling", () => { + it("should set red dot status on error", async () => { + const node = createNode(); + + api.mockError("get", "/data-links/", 500, "Server Error"); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expectErrorStatus(node); + }); + + it("should call node.error with message on failure", async () => { + const node = createNode(); + + api.mockDataLinksSearch([]); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("datalink list failed"), expect.anything()); + }); + }); + + describe("datalink autocomplete endpoint", () => { + it("should use handleDatalinkAutoComplete handler", async () => { + // The endpoint is registered - this is tested in utils.test.js + // Here we just verify the endpoint is set up + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/admin/seqera/datalinks/:nodeId", expect.any(Function)); + }); + }); +}); diff --git a/test/unit/datalink-poll.test.js b/test/unit/datalink-poll.test.js new file mode 100644 index 0000000..bfdc55c --- /dev/null +++ b/test/unit/datalink-poll.test.js @@ -0,0 +1,306 @@ +/** + * Tests for nodes/datalink-poll.js + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); + +describe("seqera-datalink-poll node", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + // Load the datalink-poll node module + require("../../nodes/datalink-poll")(RED); + }); + + afterEach(() => { + nock.cleanAll(); + }); + + function setupDataLinkMocks(files = []) { + api.mockDataLinksSearch([ + { + id: "dl-123", + name: "test-data-link", + resourceRef: "s3://my-bucket", + type: "bucket", + provider: "aws", + }, + ]); + api.mockDataLinkBrowse("dl-123", files); + } + + function createNode(configOverrides = {}) { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-datalink-poll").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-datalink-poll", + seqera: "seqera-config-id", + dataLinkName: "test-data-link", + dataLinkNameType: "str", + basePath: "", + basePathType: "str", + prefix: "", + prefixType: "str", + pattern: "", + patternType: "str", + maxResults: "100", + maxResultsType: "num", + workspaceId: "", + workspaceIdType: "str", + baseUrl: "", + baseUrlType: "str", + depth: "0", + depthType: "num", + returnType: "files", + pollFrequency: "15", + pollUnits: "minutes", + ...configOverrides, + }; + + NodeConstructor.call(node, config); + return node; + } + + // Helper to wait for async operations to complete + async function waitForPolling(timeout = 100) { + await new Promise((resolve) => setTimeout(resolve, timeout)); + } + + describe("node registration", () => { + it("should register seqera-datalink-poll type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith( + "seqera-datalink-poll", + expect.any(Function), + expect.any(Object), + ); + }); + + it("should register HTTP endpoint for datalink autocomplete", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/admin/seqera/datalinks/:nodeId", expect.any(Function)); + }); + }); + + describe("automatic polling initialization", () => { + it("should execute poll immediately on initialization", async () => { + setupDataLinkMocks([{ name: "file1.txt", type: "FILE" }]); + + const node = createNode(); + await waitForPolling(); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "blue", + shape: "ring", + }), + ); + + await node._triggerClose(); + }); + + it("should set green status after successful poll", async () => { + setupDataLinkMocks([{ name: "file1.txt", type: "FILE" }]); + + const node = createNode(); + await waitForPolling(); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "green", + shape: "dot", + }), + ); + + await node._triggerClose(); + }); + + it("should calculate poll frequency correctly - seconds", () => { + // Don't start polling - just check calculation + const node = createNode({ pollFrequency: "30", pollUnits: "seconds", dataLinkName: "" }); + expect(node.pollFrequencySec).toBe(30); + }); + + it("should calculate poll frequency correctly - minutes", () => { + // Don't start polling - just check calculation + const node = createNode({ pollFrequency: "5", pollUnits: "minutes", dataLinkName: "" }); + expect(node.pollFrequencySec).toBe(300); + }); + + it("should calculate poll frequency correctly - hours", () => { + // Don't start polling - just check calculation + const node = createNode({ pollFrequency: "2", pollUnits: "hours", dataLinkName: "" }); + expect(node.pollFrequencySec).toBe(7200); + }); + + it("should calculate poll frequency correctly - days", () => { + // Don't start polling - just check calculation + const node = createNode({ pollFrequency: "1", pollUnits: "days", dataLinkName: "" }); + expect(node.pollFrequencySec).toBe(86400); + }); + }); + + describe("output format", () => { + beforeEach(() => { + // Ensure clean nock state before each test in this block + nock.cleanAll(); + }); + + it("should send all files to output 1", async () => { + setupDataLinkMocks([ + { name: "file1.txt", type: "FILE" }, + { name: "file2.txt", type: "FILE" }, + ]); + + const node = createNode(); + await waitForPolling(); + + // Check output 1 was called + expect(node.send).toHaveBeenCalled(); + const output1Call = node.send.mock.calls.find((call) => call[0] && call[0][0] !== null); + expect(output1Call).toBeDefined(); + expect(output1Call[0][0].payload.files.length).toBeGreaterThanOrEqual(2); + + await node._triggerClose(); + }); + + it("should not send to output 2 on first poll (no previous state)", async () => { + setupDataLinkMocks([ + { name: "file1.txt", type: "FILE" }, + { name: "file2.txt", type: "FILE" }, + ]); + + const node = createNode(); + await waitForPolling(); + + // On first poll, output 2 should be null + expect(node.send.mock.calls[0][0][1]).toBeNull(); + + await node._triggerClose(); + }); + + it("should include nextPoll timestamp in output 1", async () => { + setupDataLinkMocks([{ name: "file.txt", type: "FILE" }]); + + const node = createNode({ pollFrequency: "5", pollUnits: "minutes" }); + await waitForPolling(); + + const output1 = node.send.mock.calls[0][0][0]; + expect(output1.payload.nextPoll).toBeDefined(); + expect(new Date(output1.payload.nextPoll).getTime()).toBeGreaterThan(Date.now()); + + await node._triggerClose(); + }); + + it("should include full paths with resourceRef in files array", async () => { + setupDataLinkMocks([{ name: "path/to/file.txt", type: "FILE" }]); + + const node = createNode(); + await waitForPolling(); + + const output1 = node.send.mock.calls[0][0][0]; + expect(output1.files[0]).toBe("s3://my-bucket/path/to/file.txt"); + + await node._triggerClose(); + }); + + it("should include resourceType, resourceRef, provider in payload", async () => { + setupDataLinkMocks([]); + + const node = createNode(); + await waitForPolling(); + + const output1 = node.send.mock.calls[0][0][0]; + expect(output1.payload.resourceRef).toBe("s3://my-bucket"); + expect(output1.payload.resourceType).toBe("bucket"); + expect(output1.payload.provider).toBe("aws"); + + await node._triggerClose(); + }); + }); + + describe("error handling", () => { + it("should set red dot status on error", async () => { + api.mockError("get", "/data-links", 500, "Server Error"); + + const node = createNode(); + await waitForPolling(); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + shape: "dot", + }), + ); + + await node._triggerClose(); + }); + + it("should call error handler on API failure", async () => { + api.mockError("get", "/data-links", 500, "Server Error"); + + const node = createNode(); + await waitForPolling(); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("datalink poll failed")); + + await node._triggerClose(); + }); + }); + + describe("initialization conditions", () => { + it("should not start polling if no seqeraConfig", async () => { + RED._testHelpers.nodes.delete("seqera-config-id"); + + const node = createNode({ seqera: "non-existent-config" }); + await waitForPolling(); + + // Should not have called send since polling didn't start + expect(node.send).not.toHaveBeenCalled(); + + node._triggerClose(); + }); + + it("should not start polling if dataLinkName is empty", async () => { + const node = createNode({ dataLinkName: "" }); + await waitForPolling(); + + // Should not have called send since polling didn't start + expect(node.send).not.toHaveBeenCalled(); + + node._triggerClose(); + }); + + it("should not start polling if dataLinkName is whitespace only", async () => { + const node = createNode({ dataLinkName: " " }); + await waitForPolling(); + + // Should not have called send since polling didn't start + expect(node.send).not.toHaveBeenCalled(); + + node._triggerClose(); + }); + }); + + describe("cleanup", () => { + it("should handle close without error", async () => { + setupDataLinkMocks([]); + + const node = createNode(); + await waitForPolling(); + + // Close should not throw + await node._triggerClose(); + }); + }); +}); diff --git a/test/unit/datalink-utils.test.js b/test/unit/datalink-utils.test.js new file mode 100644 index 0000000..26089f1 --- /dev/null +++ b/test/unit/datalink-utils.test.js @@ -0,0 +1,519 @@ +/** + * Tests for nodes/datalink-utils.js + */ +const nock = require("nock"); +const { resolveDataLink, listDataLink } = require("../../nodes/datalink-utils"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockNode, createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); +const { createMockMsg } = require("../helpers/test-utils"); + +describe("datalink-utils.js", () => { + describe("resolveDataLink", () => { + let RED; + let node; + let api; + + beforeEach(() => { + RED = createMockRED(); + node = createMockNode({ + seqeraConfig: createMockSeqeraConfigNode(), + }); + api = mockSeqeraAPI(); + }); + + it("should resolve data link by name and return IDs", async () => { + api.mockDataLinksSearch([ + { + id: "dl-123", + name: "my-data-link", + credentials: [{ id: "cred-456" }], + resourceRef: "s3://my-bucket", + type: "bucket", + provider: "aws", + }, + ]); + + const result = await resolveDataLink(RED, node, {}, "my-data-link", { + baseUrl: BASE_URL, + workspaceId: "ws-123", + }); + + expect(result).toEqual({ + dataLinkId: "dl-123", + credentialsId: "cred-456", + resourceRef: "s3://my-bucket", + resourceType: "bucket", + provider: "aws", + }); + }); + + it("should throw error if dataLinkName not provided", async () => { + await expect( + resolveDataLink(RED, node, {}, null, { + baseUrl: BASE_URL, + workspaceId: "ws-123", + }), + ).rejects.toThrow("dataLinkName not provided"); + }); + + it("should throw error if data link not found", async () => { + api.mockDataLinksSearch([]); + + await expect( + resolveDataLink(RED, node, {}, "non-existent", { + baseUrl: BASE_URL, + workspaceId: "ws-123", + }), + ).rejects.toThrow("Could not find Data Link 'non-existent'"); + }); + + it("should throw error if multiple data links match", async () => { + api.mockDataLinksSearch([ + { id: "dl-1", name: "ambiguous" }, + { id: "dl-2", name: "ambiguous-link" }, + ]); + + await expect( + resolveDataLink(RED, node, {}, "ambiguous", { + baseUrl: BASE_URL, + workspaceId: "ws-123", + }), + ).rejects.toThrow("Found more than one Data Link matching 'ambiguous'"); + }); + + it("should include workspaceId in search query", async () => { + const scope = nock(BASE_URL) + .get("/data-links/") + .query((q) => q.workspaceId === "specific-ws" && q.search === "test-link") + .reply(200, { + dataLinks: [{ id: "dl-1", name: "test-link" }], + }); + + await resolveDataLink(RED, node, {}, "test-link", { + baseUrl: BASE_URL, + workspaceId: "specific-ws", + }); + + expect(scope.isDone()).toBe(true); + }); + + it("should handle data link without credentials", async () => { + api.mockDataLinksSearch([ + { + id: "dl-123", + name: "public-link", + credentials: [], + resourceRef: "gs://public-bucket", + type: "bucket", + provider: "gcp", + }, + ]); + + const result = await resolveDataLink(RED, node, {}, "public-link", { + baseUrl: BASE_URL, + }); + + expect(result.credentialsId).toBeUndefined(); + }); + + it("should strip trailing slash from baseUrl", async () => { + const scope = nock(BASE_URL) + .get("/data-links/") + .query(true) + .reply(200, { dataLinks: [{ id: "dl-1", name: "test" }] }); + + await resolveDataLink(RED, node, {}, "test", { + baseUrl: `${BASE_URL}/`, + workspaceId: "ws-123", + }); + + expect(scope.isDone()).toBe(true); + }); + }); + + describe("listDataLink", () => { + let RED; + let node; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + }); + + function createDataLinkNode(overrides = {}) { + const seqeraConfig = createMockSeqeraConfigNode(); + return createMockNode({ + seqeraConfig, + defaultBaseUrl: BASE_URL, + // Required property configs + dataLinkNameProp: "test-data-link", + dataLinkNamePropType: "str", + basePathProp: "", + basePathPropType: "str", + prefixProp: "", + prefixPropType: "str", + patternProp: "", + patternPropType: "str", + maxResultsProp: "100", + maxResultsPropType: "num", + workspaceIdProp: "", + workspaceIdPropType: "str", + baseUrlProp: "", + baseUrlPropType: "str", + depthProp: "0", + depthPropType: "num", + returnType: "all", + ...overrides, + }); + } + + it("should list files from data link", async () => { + node = createDataLinkNode(); + + // Mock resolve data link + api.mockDataLinksSearch([ + { + id: "dl-123", + name: "test-data-link", + credentials: [{ id: "cred-1" }], + resourceRef: "s3://bucket", + type: "bucket", + provider: "aws", + }, + ]); + + // Mock browse + api.mockDataLinkBrowse("dl-123", [ + { name: "file1.txt", type: "FILE", size: 100 }, + { name: "file2.csv", type: "FILE", size: 200 }, + ]); + + const result = await listDataLink(RED, node, {}); + + expect(result.items).toHaveLength(2); + expect(result.files).toEqual(["file1.txt", "file2.csv"]); + expect(result.resourceRef).toBe("s3://bucket"); + expect(result.provider).toBe("aws"); + }); + + it("should throw error if dataLinkName not provided", async () => { + node = createDataLinkNode({ dataLinkNameProp: "" }); + + await expect(listDataLink(RED, node, {})).rejects.toThrow("dataLinkName not provided"); + }); + + it("should respect maxResults limit", async () => { + node = createDataLinkNode({ maxResultsProp: "2", maxResultsPropType: "num" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + api.mockDataLinkBrowse("dl-123", [ + { name: "file1.txt", type: "FILE" }, + { name: "file2.txt", type: "FILE" }, + { name: "file3.txt", type: "FILE" }, + { name: "file4.txt", type: "FILE" }, + ]); + + const result = await listDataLink(RED, node, {}); + + expect(result.items).toHaveLength(2); + }); + + it("should filter by regex pattern", async () => { + node = createDataLinkNode({ patternProp: "\\.csv$", patternPropType: "str" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + api.mockDataLinkBrowse("dl-123", [ + { name: "data.csv", type: "FILE" }, + { name: "data.txt", type: "FILE" }, + { name: "report.csv", type: "FILE" }, + ]); + + const result = await listDataLink(RED, node, {}); + + expect(result.files).toEqual(["data.csv", "report.csv"]); + }); + + it("should filter files only when returnType is 'files'", async () => { + node = createDataLinkNode({ returnType: "files" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + api.mockDataLinkBrowse("dl-123", [ + { name: "file.txt", type: "FILE" }, + { name: "folder", type: "FOLDER" }, + { name: "another.csv", type: "FILE" }, + ]); + + const result = await listDataLink(RED, node, {}); + + expect(result.items).toHaveLength(2); + expect(result.files).toEqual(["file.txt", "another.csv"]); + }); + + it("should filter folders only when returnType is 'folders'", async () => { + node = createDataLinkNode({ returnType: "folders" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + api.mockDataLinkBrowse("dl-123", [ + { name: "file.txt", type: "FILE" }, + { name: "folder1", type: "FOLDER" }, + { name: "folder2", type: "FOLDER" }, + ]); + + const result = await listDataLink(RED, node, {}); + + expect(result.items).toHaveLength(2); + expect(result.files).toEqual(["folder1", "folder2"]); + }); + + it("should handle pagination with nextPageToken", async () => { + node = createDataLinkNode({ maxResultsProp: "100" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + + // First page + nock(BASE_URL) + .get(new RegExp("/data-links/dl-123/browse.*")) + .query(true) + .reply(200, { + objects: [{ name: "file1.txt", type: "FILE" }], + nextPageToken: "page2-token", + }); + + // Second page + nock(BASE_URL) + .get(new RegExp("/data-links/dl-123/browse.*")) + .query((q) => q.nextPageToken === "page2-token") + .reply(200, { + objects: [{ name: "file2.txt", type: "FILE" }], + nextPageToken: null, + }); + + const result = await listDataLink(RED, node, {}); + + expect(result.files).toEqual(["file1.txt", "file2.txt"]); + }); + + it("should recurse into folders when depth > 0", async () => { + node = createDataLinkNode({ depthProp: "1", depthPropType: "num" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + + // Root level + nock(BASE_URL) + .get(/\/data-links\/dl-123\/browse\/?$/) + .query(true) + .reply(200, { + objects: [ + { name: "root-file.txt", type: "FILE" }, + { name: "subfolder", type: "FOLDER" }, + ], + }); + + // Subfolder level + nock(BASE_URL) + .get(/\/data-links\/dl-123\/browse\/subfolder/) + .query(true) + .reply(200, { + objects: [{ name: "nested-file.txt", type: "FILE" }], + }); + + const result = await listDataLink(RED, node, {}); + + expect(result.files).toContain("root-file.txt"); + expect(result.files).toContain("subfolder"); + expect(result.files).toContain("subfolder/nested-file.txt"); + }); + + it("should not recurse when depth is 0", async () => { + node = createDataLinkNode({ depthProp: "0", depthPropType: "num" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + + // Only root should be called + nock(BASE_URL) + .get(/\/data-links\/dl-123\/browse\/?$/) + .query(true) + .reply(200, { + objects: [ + { name: "file.txt", type: "FILE" }, + { name: "folder", type: "FOLDER" }, + ], + }); + + const result = await listDataLink(RED, node, {}); + + // Should have both but not recurse into folder + expect(result.files).toEqual(["file.txt", "folder"]); + }); + + it("should use basePath for starting directory", async () => { + node = createDataLinkNode({ basePathProp: "some/path", basePathPropType: "str" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + + const scope = nock(BASE_URL) + .get(/\/data-links\/dl-123\/browse\/some\/path/) + .query(true) + .reply(200, { objects: [{ name: "file.txt", type: "FILE" }] }); + + await listDataLink(RED, node, {}); + + expect(scope.isDone()).toBe(true); + }); + + it("should include prefix in API search parameter", async () => { + node = createDataLinkNode({ prefixProp: "data_", prefixPropType: "str" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + + const scope = nock(BASE_URL) + .get(new RegExp("/data-links/dl-123/browse.*")) + .query((q) => q.search === "data_") + .reply(200, { objects: [] }); + + await listDataLink(RED, node, {}); + + expect(scope.isDone()).toBe(true); + }); + + it("should warn on invalid regex pattern and continue", async () => { + node = createDataLinkNode({ patternProp: "[invalid(regex", patternPropType: "str" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + api.mockDataLinkBrowse("dl-123", [{ name: "file.txt", type: "FILE" }]); + + const result = await listDataLink(RED, node, {}); + + expect(node.warn).toHaveBeenCalledWith(expect.stringContaining("Invalid regex pattern")); + // Should return all items since regex failed + expect(result.items).toHaveLength(1); + }); + + it("should evaluate properties from message", async () => { + node = createDataLinkNode({ + dataLinkNameProp: "linkName", + dataLinkNamePropType: "msg", + }); + + // Set up RED.util.evaluateNodeProperty to handle msg type + RED.util.evaluateNodeProperty.mockImplementation((value, type, n, msg) => { + if (type === "msg") return msg[value]; + if (type === "str") return value; + if (type === "num") return Number(value); + return value; + }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "dynamic-link" }]); + api.mockDataLinkBrowse("dl-123", [{ name: "file.txt", type: "FILE" }]); + + const msg = createMockMsg({ linkName: "dynamic-link" }); + const result = await listDataLink(RED, node, msg); + + expect(result.items).toHaveLength(1); + }); + + it("should use workspace ID from seqeraConfig when not overridden", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "config-ws-id" }); + node = createDataLinkNode({ seqeraConfig }); + + const searchScope = nock(BASE_URL) + .get("/data-links/") + .query((q) => q.workspaceId === "config-ws-id") + .reply(200, { dataLinks: [{ id: "dl-123", name: "test-data-link" }] }); + + const browseScope = nock(BASE_URL) + .get(new RegExp("/data-links/dl-123/browse.*")) + .query((q) => q.workspaceId === "config-ws-id") + .reply(200, { objects: [] }); + + await listDataLink(RED, node, {}); + + expect(searchScope.isDone()).toBe(true); + expect(browseScope.isDone()).toBe(true); + }); + + it("should stop recursion when maxResults reached", async () => { + node = createDataLinkNode({ + depthProp: "2", + depthPropType: "num", + maxResultsProp: "2", + maxResultsPropType: "num", + }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + + // Root with many items + nock(BASE_URL) + .get(/\/data-links\/dl-123\/browse\/?$/) + .query(true) + .reply(200, { + objects: [ + { name: "file1.txt", type: "FILE" }, + { name: "file2.txt", type: "FILE" }, + { name: "folder", type: "FOLDER" }, + ], + }); + + const result = await listDataLink(RED, node, {}); + + // maxResults=2 limits the returned items + expect(result.items).toHaveLength(2); + }); + + it("should prepend folder path to nested file names", async () => { + node = createDataLinkNode({ depthProp: "1", depthPropType: "num" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + + nock(BASE_URL) + .get(/\/data-links\/dl-123\/browse\/?$/) + .query(true) + .reply(200, { + objects: [{ name: "folder", type: "FOLDER" }], + }); + + nock(BASE_URL) + .get(/\/data-links\/dl-123\/browse\/folder/) + .query(true) + .reply(200, { + objects: [{ name: "nested.txt", type: "FILE" }], + }); + + const result = await listDataLink(RED, node, {}); + + expect(result.files).toContain("folder/nested.txt"); + }); + + it("should use default maxResults of 100", async () => { + node = createDataLinkNode({ maxResultsProp: "", maxResultsPropType: "str" }); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + + // Create 150 items + const objects = Array.from({ length: 150 }, (_, i) => ({ + name: `file${i}.txt`, + type: "FILE", + })); + + api.mockDataLinkBrowse("dl-123", objects); + + const result = await listDataLink(RED, node, {}); + + expect(result.items).toHaveLength(100); + }); + + it("should handle empty browse response", async () => { + node = createDataLinkNode(); + + api.mockDataLinksSearch([{ id: "dl-123", name: "test-data-link" }]); + api.mockDataLinkBrowse("dl-123", []); + + const result = await listDataLink(RED, node, {}); + + expect(result.items).toEqual([]); + expect(result.files).toEqual([]); + }); + }); +}); diff --git a/test/unit/dataset-add.test.js b/test/unit/dataset-add.test.js new file mode 100644 index 0000000..16ed51c --- /dev/null +++ b/test/unit/dataset-add.test.js @@ -0,0 +1,353 @@ +/** + * Tests for nodes/dataset-add.js + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); +const { + createMockMsg, + expectSuccessStatus, + expectErrorStatus, + expectMessagePassthrough, +} = require("../helpers/test-utils"); + +describe("seqera-dataset-add node", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + // Load the dataset-add node module + require("../../nodes/dataset-add")(RED); + }); + + function createNode(configOverrides = {}) { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-dataset-add").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-dataset", + seqera: "seqera-config-id", + datasetName: "test-dataset", + datasetNameType: "str", + fileContents: "payload", + fileContentsType: "msg", + description: "", + descriptionType: "str", + baseUrl: "", + baseUrlType: "str", + workspaceId: "", + workspaceIdType: "str", + fileType: "csv", + hasHeader: false, + ...configOverrides, + }; + + NodeConstructor.call(node, config); + return node; + } + + describe("node registration", () => { + it("should register seqera-dataset-add type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith( + "seqera-dataset-add", + expect.any(Function), + expect.any(Object), + ); + }); + }); + + describe("two-step process", () => { + it("should create dataset then upload file", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ payload: "col1,col2\nval1,val2" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalled(); + expect(send.mock.calls[0][0].datasetId).toBe("ds-123"); + }); + + it("should set blue status during create", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "blue", + shape: "ring", + }), + ); + }); + + it("should set yellow status during upload", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "yellow", + shape: "ring", + }), + ); + }); + + it("should set green status on success", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expectSuccessStatus(node); + }); + }); + + describe("file handling", () => { + it("should convert string to Buffer", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + + const uploadScope = nock(BASE_URL) + .post(`/datasets/ds-123/upload`) + .query(true) + .reply(200, { version: { id: "v-1" } }); + + const msg = createMockMsg({ payload: "string-content" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(uploadScope.isDone()).toBe(true); + }); + + it("should handle Buffer input", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ payload: Buffer.from("buffer-content") }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "green", + }), + ); + }); + + it("should handle JSON object input", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ payload: { col1: "val1", col2: "val2" } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expectSuccessStatus(node); + }); + + it("should use correct MIME type for CSV", async () => { + const node = createNode({ fileType: "csv" }); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + + // FormData upload will have multipart/form-data content-type + // The CSV MIME type is embedded in the form field + const uploadScope = nock(BASE_URL) + .post(`/datasets/ds-123/upload`) + .query(true) + .matchHeader("content-type", /multipart\/form-data/) + .reply(200, { version: { id: "v-1" } }); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(uploadScope.isDone()).toBe(true); + }); + + it("should use correct MIME type for TSV", async () => { + const node = createNode({ fileType: "tsv" }); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + + // FormData upload will have multipart/form-data content-type + // The TSV MIME type is embedded in the form field + const uploadScope = nock(BASE_URL) + .post(`/datasets/ds-123/upload`) + .query(true) + .matchHeader("content-type", /multipart\/form-data/) + .reply(200, { version: { id: "v-1" } }); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(uploadScope.isDone()).toBe(true); + }); + + it("should include header query param if hasHeader=true", async () => { + const node = createNode({ hasHeader: true }); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + + const uploadScope = nock(BASE_URL) + .post(`/datasets/ds-123/upload`) + .query((q) => q.header === "true") + .reply(200, { version: { id: "v-1" } }); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(uploadScope.isDone()).toBe(true); + }); + }); + + describe("output", () => { + it("should return datasetId in output message", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-new-123" } }); + api.mockDatasetUpload("ds-new-123"); + + const msg = createMockMsg({ payload: "data" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send.mock.calls[0][0].datasetId).toBe("ds-new-123"); + }); + + it("should pass through input message properties", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ + payload: "data", + _context: { flowId: "flow-123" }, + correlationId: "corr-456", + }); + + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const output = send.mock.calls[0][0]; + expectMessagePassthrough(output, msg); + }); + }); + + describe("error handling", () => { + it("should error if datasetName not provided", async () => { + const node = createNode({ datasetName: "", datasetNameType: "str" }); + + const msg = createMockMsg({ payload: "data" }); + const done = jest.fn(); + await node._triggerInput(msg, jest.fn(), done); + + expect(done).toHaveBeenCalledWith(expect.any(Error)); + }); + + it("should error if fileContents not provided", async () => { + const node = createNode({ fileContents: "", fileContentsType: "str" }); + + const msg = createMockMsg({}); // No payload + const done = jest.fn(); + await node._triggerInput(msg, jest.fn(), done); + + expect(done).toHaveBeenCalledWith(expect.any(Error)); + }); + + it("should set red dot status on create error", async () => { + const node = createNode(); + + api.mockError("post", "/datasets", 500, "Server Error"); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expectErrorStatus(node); + }); + + it("should set red dot status on upload error", async () => { + const node = createNode(); + + api.mockDatasetCreate({ dataset: { id: "ds-123" } }); + api.mockError("post", "/datasets/ds-123/upload", 500, "Upload Error"); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expectErrorStatus(node); + }); + + it("should error if dataset ID not returned", async () => { + const node = createNode(); + + nock(BASE_URL).post("/datasets").query(true).reply(200, { success: true }); // No dataset ID + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalled(); + }); + }); + + describe("description field", () => { + it("should include description in create request if provided", async () => { + const node = createNode({ description: "My dataset description", descriptionType: "str" }); + + const createScope = nock(BASE_URL) + .post("/datasets", (body) => body.description === "My dataset description") + .query(true) + .reply(200, { dataset: { id: "ds-123" } }); + + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(createScope.isDone()).toBe(true); + }); + + it("should not include description if empty", async () => { + const node = createNode({ description: "", descriptionType: "str" }); + + const createScope = nock(BASE_URL) + .post("/datasets", (body) => body.description === undefined) + .query(true) + .reply(200, { dataset: { id: "ds-123" } }); + + api.mockDatasetUpload("ds-123"); + + const msg = createMockMsg({ payload: "data" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(createScope.isDone()).toBe(true); + }); + }); +}); diff --git a/test/unit/studios-add.test.js b/test/unit/studios-add.test.js new file mode 100644 index 0000000..b57c08e --- /dev/null +++ b/test/unit/studios-add.test.js @@ -0,0 +1,470 @@ +/** + * Tests for nodes/studios-add.js + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); +const { + createMockMsg, + expectSuccessStatus, + expectErrorStatus, + expectMessagePassthrough, +} = require("../helpers/test-utils"); + +describe("seqera-studios-add node", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + // Load the studios-add node module + require("../../nodes/studios-add")(RED); + }); + + function createNode(configOverrides = {}) { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-studios-add").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-studio", + seqera: "seqera-config-id", + studioName: "my-studio", + studioNameType: "str", + description: "", + descriptionType: "str", + containerUri: "cr.seqera.io/studio:latest", + containerUriType: "str", + computeEnvId: "ce-123", + computeEnvIdType: "str", + mountData: "", + mountDataType: "str", + cpu: "2", + cpuType: "num", + memory: "8192", + memoryType: "num", + gpu: "0", + gpuType: "num", + initialCheckpointId: "", + initialCheckpointIdType: "num", + condaEnvironment: "", + condaEnvironmentType: "str", + lifespanHours: "", + lifespanHoursType: "num", + isPrivate: "false", + isPrivateType: "bool", + spot: "false", + spotType: "bool", + autoStart: "true", + autoStartType: "bool", + workspaceId: "", + workspaceIdType: "str", + baseUrl: "", + baseUrlType: "str", + ...configOverrides, + }; + + NodeConstructor.call(node, config); + return node; + } + + describe("node registration", () => { + it("should register seqera-studios-add type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith( + "seqera-studios-add", + expect.any(Function), + expect.any(Object), + ); + }); + }); + + describe("basic creation", () => { + it("should create studio via POST /studios", async () => { + const node = createNode(); + api.mockStudiosCreate({ sessionId: "studio-123" }); + + const msg = createMockMsg({}); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalled(); + expect(send.mock.calls[0][0].studioId).toBe("studio-123"); + }); + + it("should set blue ring status during creation", async () => { + const node = createNode(); + api.mockStudiosCreate({ sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "blue", + shape: "ring", + }), + ); + }); + + it("should set green dot status on success", async () => { + const node = createNode(); + api.mockStudiosCreate({ sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expectSuccessStatus(node); + }); + + it("should return studioId in output message", async () => { + const node = createNode(); + api.mockStudiosCreate({ studio: { sessionId: "studio-new-123" } }); + + const msg = createMockMsg({}); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send.mock.calls[0][0].studioId).toBe("studio-new-123"); + }); + }); + + describe("configuration object", () => { + it("should set cpu, memory, gpu in configuration", async () => { + const node = createNode({ + cpu: "4", + cpuType: "num", + memory: "16384", + memoryType: "num", + gpu: "1", + gpuType: "num", + }); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.configuration.cpu === 4 && body.configuration.memory === 16384 && body.configuration.gpu === 1; + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should use default values if not provided (cpu=2, memory=8192, gpu=0)", async () => { + const node = createNode({ + cpu: "", + cpuType: "str", + memory: "", + memoryType: "str", + gpu: "", + gpuType: "str", + }); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.configuration.cpu === 2 && body.configuration.memory === 8192 && body.configuration.gpu === 0; + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should include condaEnvironment if provided", async () => { + const node = createNode({ + condaEnvironment: "environment.yml", + condaEnvironmentType: "str", + }); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.configuration.condaEnvironment === "environment.yml"; + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should include lifespanHours if provided", async () => { + const node = createNode({ + lifespanHours: "24", + lifespanHoursType: "num", + }); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.configuration.lifespanHours === 24; + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + }); + + describe("mount data links", () => { + it("should resolve data link names to IDs", async () => { + const node = createNode({ + mountData: "my-data-link", + mountDataType: "str", + }); + + // Mock data link resolution + api.mockDataLinksSearch([{ id: "dl-resolved-123", name: "my-data-link" }]); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.configuration.mountData.includes("dl-resolved-123"); + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should handle array input for mountData", async () => { + const node = createNode({ + mountData: "link1,link2", + mountDataType: "str", + }); + + // Mock data link resolutions + nock(BASE_URL) + .get("/data-links/") + .query((q) => q.search === "link1") + .reply(200, { dataLinks: [{ id: "dl-1", name: "link1" }] }); + + nock(BASE_URL) + .get("/data-links/") + .query((q) => q.search === "link2") + .reply(200, { dataLinks: [{ id: "dl-2", name: "link2" }] }); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.configuration.mountData.length === 2; + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should handle newline-separated string input for mountData", async () => { + const node = createNode({ + mountData: "link1\nlink2", + mountDataType: "str", + }); + + nock(BASE_URL) + .get("/data-links/") + .query((q) => q.search === "link1") + .reply(200, { dataLinks: [{ id: "dl-1", name: "link1" }] }); + + nock(BASE_URL) + .get("/data-links/") + .query((q) => q.search === "link2") + .reply(200, { dataLinks: [{ id: "dl-2", name: "link2" }] }); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.configuration.mountData.length === 2; + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + }); + + describe("query parameters", () => { + it("should include workspaceId in query string", async () => { + const node = createNode(); + + const scope = nock(BASE_URL) + .post("/studios") + .query((q) => q.workspaceId === "ws-123") + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should include autoStart=true in query string", async () => { + const node = createNode({ autoStart: true, autoStartType: "bool" }); + + const scope = nock(BASE_URL) + .post("/studios") + .query((q) => q.autoStart === "true") + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should include autoStart=false in query string", async () => { + const node = createNode({ autoStart: false, autoStartType: "bool" }); + + const scope = nock(BASE_URL) + .post("/studios") + .query((q) => q.autoStart === "false") + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + }); + + describe("required fields", () => { + it("should error if studioName not provided", async () => { + const node = createNode({ studioName: "", studioNameType: "str" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("studioName not provided"), expect.anything()); + }); + + it("should error if containerUri not provided", async () => { + const node = createNode({ containerUri: "", containerUriType: "str" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith( + expect.stringContaining("containerUri (dataStudioToolUrl) not provided"), + expect.anything(), + ); + }); + + it("should error if computeEnvId not provided", async () => { + const node = createNode({ computeEnvId: "", computeEnvIdType: "str" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("computeEnvId not provided"), expect.anything()); + }); + }); + + describe("error handling", () => { + it("should set red ring status on error", async () => { + const node = createNode(); + api.mockError("post", "/studios", 500, "Server Error"); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + shape: "ring", + }), + ); + }); + + it("should call node.error with message on failure", async () => { + const node = createNode(); + api.mockError("post", "/studios", 400, "Bad Request"); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("Studios create failed"), expect.anything()); + }); + }); + + describe("message passthrough", () => { + it("should preserve input message properties in output", async () => { + const node = createNode(); + api.mockStudiosCreate({ sessionId: "studio-123" }); + + const msg = createMockMsg({ + _context: { flowId: "flow-123" }, + correlationId: "corr-456", + customProp: "custom-value", + }); + + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const output = send.mock.calls[0][0]; + expectMessagePassthrough(output, msg); + }); + }); + + describe("optional fields", () => { + it("should include description if provided", async () => { + const node = createNode({ + description: "My studio description", + descriptionType: "str", + }); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.description === "My studio description"; + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should set isPrivate and spot flags", async () => { + const node = createNode({ + isPrivate: true, + isPrivateType: "bool", + spot: true, + spotType: "bool", + }); + + const scope = nock(BASE_URL) + .post("/studios", (body) => { + return body.isPrivate === true && body.spot === true; + }) + .query(true) + .reply(200, { sessionId: "studio-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + }); +}); diff --git a/test/unit/studios-monitor.test.js b/test/unit/studios-monitor.test.js new file mode 100644 index 0000000..e25e4ca --- /dev/null +++ b/test/unit/studios-monitor.test.js @@ -0,0 +1,430 @@ +/** + * Tests for nodes/studios-monitor.js + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); +const { createMockMsg, expectMessagePassthrough } = require("../helpers/test-utils"); + +describe("seqera-studios-monitor node", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode(); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + // Load the studios-monitor node module + require("../../nodes/studios-monitor")(RED); + }); + + afterEach(() => { + nock.cleanAll(); + }); + + function createNode(configOverrides = {}) { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-studios-monitor").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-monitor", + seqera: "seqera-config-id", + studioId: "studioId", + studioIdType: "msg", + workspaceId: "", + workspaceIdType: "str", + poll: "5", + pollUnits: "seconds", + keepPolling: false, // Default to false for simpler testing + ...configOverrides, + }; + + NodeConstructor.call(node, config); + return node; + } + + describe("node registration", () => { + it("should register seqera-studios-monitor type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith( + "seqera-studios-monitor", + expect.any(Function), + expect.any(Object), + ); + }); + }); + + describe("basic status fetching", () => { + it("should fetch studio status on input", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "running" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalled(); + }); + + it("should set status color - starting/building (yellow)", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "starting" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "yellow", + shape: "ring", + }), + ); + }); + + it("should set status color - building (yellow)", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "building" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "yellow", + shape: "ring", + }), + ); + }); + + it("should set status color - running (blue)", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "running" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "blue", + shape: "ring", + }), + ); + }); + + it("should set status color - stopped (grey)", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "stopped" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "grey", + shape: "dot", + }), + ); + }); + + it("should set status color - errored (red)", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "errored" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + shape: "dot", + }), + ); + }); + + it("should normalize 'build failed' status", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "build failed" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + shape: "dot", + }), + ); + }); + }); + + describe("multiple outputs", () => { + it("should send to output 1 on every poll", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "running" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[0]).not.toBeNull(); // Output 1 + }); + + it("should send to output 2 on first running status", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "running" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[1]).not.toBeNull(); // Output 2 should fire on transition to running + }); + + it("should NOT send to output 2 when starting (not yet running)", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "starting" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[1]).toBeNull(); // Output 2 should be null (not running yet) + }); + + it("should send to output 3 on termination - stopped", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "stopped" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[2]).not.toBeNull(); // Output 3 should fire + }); + + it("should send to output 3 on termination - errored", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "errored" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[2]).not.toBeNull(); // Output 3 should fire + }); + + it("should send to output 3 on termination - build failed", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "build failed" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[2]).not.toBeNull(); // Output 3 should fire + }); + }); + + describe("state transition detection", () => { + it("should reset previousStatus on new input", async () => { + const node = createNode(); + + // First input - running + api.mockStudioStatus("studio-123", { statusInfo: { status: "running" } }); + + const msg1 = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg1, send, jest.fn()); + + expect(send.mock.calls[0][0][1]).not.toBeNull(); // Output 2 fires + + send.mockClear(); + + // Second input - different studio, should reset state + api.mockStudioStatus("studio-456", { statusInfo: { status: "running" } }); + + const msg2 = createMockMsg({ studioId: "studio-456" }); + await node._triggerInput(msg2, send, jest.fn()); + + expect(send.mock.calls[0][0][1]).not.toBeNull(); // Output 2 fires again (new input resets state) + }); + }); + + describe("polling control", () => { + it("should not start polling when keepPolling=false", async () => { + const node = createNode({ keepPolling: false }); + api.mockStudioStatus("studio-123", { statusInfo: { status: "running" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalledTimes(1); + // No polling interval should be set + expect(node._currentPollMs).toBeUndefined(); + }); + + it("should stop polling when studio reaches terminal state", async () => { + const node = createNode({ keepPolling: true }); + api.mockStudioStatus("studio-123", { statusInfo: { status: "stopped" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalledTimes(1); + // Terminal state - output 3 should fire + const args = send.mock.calls[0][0]; + expect(args[2]).not.toBeNull(); + + // Clean up + await node._triggerClose(); + }); + + it("should clear polling on node close", async () => { + const node = createNode({ keepPolling: true }); + api.mockStudioStatus("studio-123", { statusInfo: { status: "running" } }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + // Close should not throw + await node._triggerClose(); + }); + }); + + describe("poll interval units", () => { + it("should calculate seconds correctly", () => { + // Don't start polling - just create node with no seqera config + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-studios-monitor").constructor; + const node = {}; + const config = { + id: "test-node-id", + seqera: "non-existent", + studioId: "", + studioIdType: "str", + poll: "10", + pollUnits: "seconds", + keepPolling: false, + }; + NodeConstructor.call(node, config); + + // Internal conversion is done in convertToSeconds + // poll: "10", pollUnits: "seconds" -> 10 seconds + expect(node.pollIntervalProp).toBe("10"); + expect(node.pollUnitsProp).toBe("seconds"); + }); + + it("should calculate minutes correctly", () => { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-studios-monitor").constructor; + const node = {}; + const config = { + id: "test-node-id", + seqera: "non-existent", + studioId: "", + studioIdType: "str", + poll: "1", + pollUnits: "minutes", + keepPolling: false, + }; + NodeConstructor.call(node, config); + + expect(node.pollIntervalProp).toBe("1"); + expect(node.pollUnitsProp).toBe("minutes"); + }); + + it("should calculate hours correctly", () => { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-studios-monitor").constructor; + const node = {}; + const config = { + id: "test-node-id", + seqera: "non-existent", + studioId: "", + studioIdType: "str", + poll: "2", + pollUnits: "hours", + keepPolling: false, + }; + NodeConstructor.call(node, config); + + expect(node.pollIntervalProp).toBe("2"); + expect(node.pollUnitsProp).toBe("hours"); + }); + }); + + describe("error handling", () => { + it("should set red dot status on error", async () => { + const node = createNode(); + api.mockError("get", "/studios/studio-123", 500, "Server Error"); + + const msg = createMockMsg({ studioId: "studio-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + shape: "dot", + }), + ); + }); + + it("should error if studioId not provided", async () => { + const node = createNode({ studioId: "", studioIdType: "str" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("studioId not provided"), expect.anything()); + }); + }); + + describe("message passthrough", () => { + it("should preserve input message properties in output", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { statusInfo: { status: "running" } }); + + const msg = createMockMsg({ + studioId: "studio-123", + _context: { flowId: "flow-123" }, + correlationId: "corr-456", + customProp: "custom-value", + }); + + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const outputMsg = send.mock.calls[0][0][0]; + expectMessagePassthrough(outputMsg, msg); + }); + }); + + describe("output payload", () => { + it("should include studio data in payload", async () => { + const node = createNode(); + api.mockStudioStatus("studio-123", { + sessionId: "studio-123", + statusInfo: { status: "running" }, + }); + + const msg = createMockMsg({ studioId: "studio-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const outputMsg = send.mock.calls[0][0][0]; + expect(outputMsg.payload).toBeDefined(); + expect(outputMsg.studioId).toBe("studio-123"); + }); + }); +}); diff --git a/test/unit/utils.test.js b/test/unit/utils.test.js new file mode 100644 index 0000000..fbef3b2 --- /dev/null +++ b/test/unit/utils.test.js @@ -0,0 +1,337 @@ +/** + * Tests for nodes/_utils.js + */ +const nock = require("nock"); +const { buildHeaders, apiCall, handleDatalinkAutoComplete } = require("../../nodes/_utils"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockNode, createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); + +describe("_utils.js", () => { + describe("buildHeaders", () => { + it("should include Bearer token from seqeraConfig", () => { + const node = createMockNode({ + seqeraConfig: createMockSeqeraConfigNode({ token: "my-secret-token" }), + }); + + const headers = buildHeaders(node); + + expect(headers.Authorization).toBe("Bearer my-secret-token"); + }); + + it("should merge extra headers with auth header", () => { + const node = createMockNode({ + seqeraConfig: createMockSeqeraConfigNode({ token: "test-token" }), + }); + + const headers = buildHeaders(node, { + "Content-Type": "application/json", + Accept: "application/json", + }); + + expect(headers.Authorization).toBe("Bearer test-token"); + expect(headers["Content-Type"]).toBe("application/json"); + expect(headers.Accept).toBe("application/json"); + }); + + it("should not modify the extraHeaders object", () => { + const node = createMockNode({ + seqeraConfig: createMockSeqeraConfigNode({ token: "test-token" }), + }); + const extraHeaders = { "Content-Type": "application/json" }; + + buildHeaders(node, extraHeaders); + + expect(extraHeaders.Authorization).toBeUndefined(); + }); + + it("should work with empty extra headers", () => { + const node = createMockNode({ + seqeraConfig: createMockSeqeraConfigNode({ token: "token123" }), + }); + + const headers = buildHeaders(node, {}); + + expect(headers.Authorization).toBe("Bearer token123"); + expect(Object.keys(headers)).toHaveLength(1); + }); + }); + + describe("apiCall", () => { + let node; + let api; + + beforeEach(() => { + node = createMockNode({ + seqeraConfig: createMockSeqeraConfigNode({ token: "test-api-token" }), + }); + api = mockSeqeraAPI(); + }); + + it("should make GET request with auth headers", async () => { + api.mockUserInfo({ user: { userName: "testuser" } }); + + const response = await apiCall(node, "get", `${BASE_URL}/user-info`); + + expect(response.data).toEqual({ user: { userName: "testuser" } }); + }); + + it("should make POST request with auth headers and data", async () => { + api.mockWorkflowLaunch({ workflowId: "wf-abc123" }); + + const response = await apiCall(node, "post", `${BASE_URL}/workflow/launch?workspaceId=ws-123`, { + data: { launch: { pipeline: "test" } }, + }); + + expect(response.data).toEqual({ workflowId: "wf-abc123" }); + }); + + it("should merge provided headers with auth headers", async () => { + const scope = nock(BASE_URL) + .get("/test") + .matchHeader("authorization", "Bearer test-api-token") + .matchHeader("x-custom", "custom-value") + .reply(200, { success: true }); + + await apiCall(node, "get", `${BASE_URL}/test`, { + headers: { "X-Custom": "custom-value" }, + }); + + expect(scope.isDone()).toBe(true); + }); + + it("should call node.warn on API error", async () => { + api.mockError("get", "/fail", 500, "Internal Server Error"); + + await expect(apiCall(node, "get", `${BASE_URL}/fail`)).rejects.toThrow(); + + expect(node.warn).toHaveBeenCalled(); + const warnArg = node.warn.mock.calls[0][0]; + expect(warnArg.message).toContain("Seqera API GET call to"); + expect(warnArg.message).toContain("failed"); + }); + + it("should redact token in error log", async () => { + api.mockError("get", "/fail", 401, "Unauthorized"); + + await expect(apiCall(node, "get", `${BASE_URL}/fail`)).rejects.toThrow(); + + const warnArg = node.warn.mock.calls[0][0]; + expect(warnArg.request.headers.Authorization).toBe("Bearer *********"); + expect(warnArg.request.headers.Authorization).not.toContain("test-api-token"); + }); + + it("should re-throw error after logging", async () => { + api.mockError("get", "/error-endpoint", 404, "Not Found"); + + await expect(apiCall(node, "get", `${BASE_URL}/error-endpoint`)).rejects.toThrow(); + }); + + it("should handle network errors", async () => { + api.mockNetworkError("get", "/network-fail", "ECONNREFUSED"); + + await expect(apiCall(node, "get", `${BASE_URL}/network-fail`)).rejects.toThrow(); + + expect(node.warn).toHaveBeenCalled(); + }); + + it("should include request details in error log", async () => { + api.mockError("post", "/debug", 400, "Bad Request"); + + await expect( + apiCall(node, "post", `${BASE_URL}/debug`, { + data: { test: "data" }, + }), + ).rejects.toThrow(); + + const warnArg = node.warn.mock.calls[0][0]; + expect(warnArg.request.method).toBe("POST"); + expect(warnArg.request.url).toBe(`${BASE_URL}/debug`); + }); + }); + + describe("handleDatalinkAutoComplete", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + }); + + it("should return empty array if no workspaceId", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: null }); + RED._testHelpers.addNode("config-id", seqeraConfig); + + // Directly test the function + const req = { params: { nodeId: "test-node" }, query: {} }; + const res = { json: jest.fn() }; + await handleDatalinkAutoComplete(RED, req, res); + + expect(res.json).toHaveBeenCalledWith([]); + }); + + it("should fetch and format data links", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ + workspaceId: "ws-123", + token: "token-123", + }); + RED._testHelpers.addNode("config-id", seqeraConfig); + + // Create a node that references the config + const node = createMockNode({ seqeraConfig }); + RED._testHelpers.addNode("test-node-id", node); + + api.mockDataLinks([ + { id: "dl-1", name: "data-link-1" }, + { id: "dl-2", name: "data-link-2" }, + ]); + + const req = { + params: { nodeId: "test-node-id" }, + query: { search: "" }, + }; + const res = { json: jest.fn() }; + + await handleDatalinkAutoComplete(RED, req, res); + + expect(res.json).toHaveBeenCalledWith([ + { value: "data-link-1", label: "data-link-1" }, + { value: "data-link-2", label: "data-link-2" }, + ]); + }); + + it("should use search parameter in API call", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ + workspaceId: "ws-123", + token: "token-123", + }); + RED._testHelpers.addNode("config-id", seqeraConfig); + + const node = createMockNode({ seqeraConfig }); + RED._testHelpers.addNode("test-node-id", node); + + const scope = nock(BASE_URL) + .get("/data-links") + .query((query) => query.search === "my-search") + .matchHeader("authorization", /^Bearer .+/) + .reply(200, { dataLinks: [{ id: "dl-1", name: "my-search-result" }] }); + + const req = { + params: { nodeId: "test-node-id" }, + query: { search: "my-search" }, + }; + const res = { json: jest.fn() }; + + await handleDatalinkAutoComplete(RED, req, res); + + expect(scope.isDone()).toBe(true); + expect(res.json).toHaveBeenCalledWith([{ value: "my-search-result", label: "my-search-result" }]); + }); + + it("should handle node not existing (new node case)", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ + workspaceId: "ws-123", + token: "token-123", + }); + RED._testHelpers.addNode("config-id", seqeraConfig); + + api.mockDataLinks([{ id: "dl-1", name: "link-1" }]); + + const req = { + params: { nodeId: "non-existent-node" }, + query: { seqeraConfig: "config-id", search: "" }, + }; + const res = { json: jest.fn() }; + + await handleDatalinkAutoComplete(RED, req, res); + + expect(res.json).toHaveBeenCalledWith([{ value: "link-1", label: "link-1" }]); + }); + + it("should use workspace ID from query params", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ + workspaceId: "default-ws", + token: "token-123", + }); + RED._testHelpers.addNode("config-id", seqeraConfig); + + const node = createMockNode({ seqeraConfig }); + RED._testHelpers.addNode("test-node-id", node); + + const scope = nock(BASE_URL) + .get("/data-links") + .query((query) => query.workspaceId === "override-ws") + .matchHeader("authorization", /^Bearer .+/) + .reply(200, { dataLinks: [] }); + + const req = { + params: { nodeId: "test-node-id" }, + query: { workspaceId: "override-ws", search: "" }, + }; + const res = { json: jest.fn() }; + + await handleDatalinkAutoComplete(RED, req, res); + + expect(scope.isDone()).toBe(true); + }); + + it("should return empty array on API error", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ + workspaceId: "ws-123", + token: "token-123", + }); + RED._testHelpers.addNode("config-id", seqeraConfig); + + const node = createMockNode({ seqeraConfig }); + RED._testHelpers.addNode("test-node-id", node); + + api.mockError("get", "/data-links", 500, "Server Error"); + + const req = { + params: { nodeId: "test-node-id" }, + query: { search: "" }, + }; + const res = { json: jest.fn() }; + + await handleDatalinkAutoComplete(RED, req, res); + + expect(res.json).toHaveBeenCalledWith([]); + }); + + it("should return empty array if no seqeraConfig", async () => { + const req = { + params: { nodeId: "non-existent" }, + query: {}, + }; + const res = { json: jest.fn() }; + + await handleDatalinkAutoComplete(RED, req, res); + + expect(res.json).toHaveBeenCalledWith([]); + }); + + it("should handle empty dataLinks response", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ + workspaceId: "ws-123", + token: "token-123", + }); + RED._testHelpers.addNode("config-id", seqeraConfig); + + const node = createMockNode({ seqeraConfig }); + RED._testHelpers.addNode("test-node-id", node); + + api.mockDataLinks([]); + + const req = { + params: { nodeId: "test-node-id" }, + query: { search: "" }, + }; + const res = { json: jest.fn() }; + + await handleDatalinkAutoComplete(RED, req, res); + + expect(res.json).toHaveBeenCalledWith([]); + }); + }); +}); diff --git a/test/unit/workflow-launch.test.js b/test/unit/workflow-launch.test.js new file mode 100644 index 0000000..401fce9 --- /dev/null +++ b/test/unit/workflow-launch.test.js @@ -0,0 +1,598 @@ +/** + * Tests for nodes/workflow-launch.js + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); +const { + createMockMsg, + expectSuccessStatus, + expectErrorStatus, + expectMessagePassthrough, +} = require("../helpers/test-utils"); + +describe("seqera-workflow-launch node", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode(); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + // Load the workflow-launch node module + require("../../nodes/workflow-launch")(RED); + }); + + function createNode(configOverrides = {}) { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-workflow-launch").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-launch", + seqera: "seqera-config-id", + launchpadName: "", + launchpadNameType: "str", + paramsKey: "", + paramsKeyType: "str", + params: [], + runName: "", + runNameType: "str", + baseUrl: "", + baseUrlType: "str", + workspaceId: "", + workspaceIdType: "str", + sourceWorkspaceId: "", + sourceWorkspaceIdType: "str", + resumeWorkflowId: "", + resumeWorkflowIdType: "str", + ...configOverrides, + }; + + NodeConstructor.call(node, config); + return node; + } + + describe("node registration", () => { + it("should register seqera-workflow-launch type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith( + "seqera-workflow-launch", + expect.any(Function), + expect.any(Object), + ); + }); + + it("should register HTTP endpoint for pipeline autocomplete", () => { + expect(RED.httpAdmin.get).toHaveBeenCalledWith("/admin/seqera/pipelines/:nodeId", expect.any(Function)); + }); + }); + + describe("basic launch", () => { + it("should launch workflow with payload as body", async () => { + const node = createNode(); + api.mockWorkflowLaunch({ workflowId: "wf-launched-123" }); + + const msg = createMockMsg({ + payload: { + launch: { + pipeline: "https://github.com/test/pipeline", + computeEnvId: "ce-123", + }, + }, + }); + + const send = jest.fn(); + const done = jest.fn(); + + await node._triggerInput(msg, send, done); + + expect(send).toHaveBeenCalled(); + const outputMsg = send.mock.calls[0][0]; + expect(outputMsg.workflowId).toBe("wf-launched-123"); + }); + + it("should set blue status during launch", async () => { + const node = createNode(); + api.mockWorkflowLaunch({ workflowId: "wf-123" }); + + const msg = createMockMsg({ payload: { launch: {} } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "blue", + shape: "ring", + }), + ); + }); + + it("should set green status on success", async () => { + const node = createNode(); + api.mockWorkflowLaunch({ workflowId: "wf-123" }); + + const msg = createMockMsg({ payload: { launch: {} } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expectSuccessStatus(node); + }); + + it("should return workflowId in output message", async () => { + const node = createNode(); + api.mockWorkflowLaunch({ workflowId: "wf-new-123" }); + + const msg = createMockMsg({ payload: { launch: {} } }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send.mock.calls[0][0].workflowId).toBe("wf-new-123"); + }); + + it("should pass through input message properties", async () => { + const node = createNode(); + api.mockWorkflowLaunch({ workflowId: "wf-123" }); + + const msg = createMockMsg({ + payload: { launch: {} }, + _context: { flowId: "flow-123" }, + correlationId: "corr-456", + customProp: "custom-value", + }); + + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const outputMsg = send.mock.calls[0][0]; + expectMessagePassthrough(outputMsg, msg); + }); + + it("should include workspaceId in query params", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-from-config" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + const node = createNode(); + + const scope = nock(BASE_URL) + .post("/workflow/launch") + .query((q) => q.workspaceId === "ws-from-config") + .reply(200, { workflowId: "wf-123" }); + + const msg = createMockMsg({ payload: { launch: {} } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + }); + + describe("launchpad name resolution", () => { + it("should fetch pipeline by launchpad name", async () => { + const node = createNode({ + launchpadName: "my-pipeline", + launchpadNameType: "str", + }); + + api.mockPipelines([{ pipelineId: 123, name: "my-pipeline" }]); + api.mockPipelineLaunchConfig(123, { + computeEnv: { id: "ce-456" }, + pipeline: "https://github.com/test/pipeline", + }); + api.mockWorkflowLaunch({ workflowId: "wf-launched" }); + + const msg = createMockMsg({}); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalled(); + expect(send.mock.calls[0][0].workflowId).toBe("wf-launched"); + }); + + it("should use launch config as body from launchpad", async () => { + const node = createNode({ + launchpadName: "test-pipeline", + launchpadNameType: "str", + }); + + api.mockPipelines([{ pipelineId: 456, name: "test-pipeline" }]); + api.mockPipelineLaunchConfig(456, { + computeEnv: { id: "ce-789" }, + pipeline: "https://github.com/org/repo", + paramsText: '{"input": "s3://bucket/data"}', + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + return body.launch && body.launch.computeEnvId === "ce-789"; + }) + .query(true) + .reply(200, { workflowId: "wf-123" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should handle pipeline not found", async () => { + const node = createNode({ + launchpadName: "non-existent", + launchpadNameType: "str", + }); + + api.mockPipelines([]); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("No pipeline found"), expect.anything()); + }); + }); + + describe("parameters merging", () => { + it("should merge paramsObj into launch.paramsText", async () => { + const node = createNode({ + paramsKey: "myParams", + paramsKeyType: "msg", + }); + + RED.util.evaluateNodeProperty.mockImplementation((value, type, n, msg) => { + if (type === "msg" && value === "myParams") return { input: "s3://bucket/data" }; + if (type === "str") return value; + return undefined; + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + const params = JSON.parse(body.launch.paramsText); + return params.input === "s3://bucket/data"; + }) + .query(true) + .reply(200, { workflowId: "wf-123" }); + + const msg = createMockMsg({ + payload: { launch: {} }, + myParams: { input: "s3://bucket/data" }, + }); + + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should merge paramsArray into launch.paramsText", async () => { + const node = createNode({ + params: [ + { name: "outdir", value: "s3://bucket/results", valueType: "str" }, + { name: "threads", value: "8", valueType: "str" }, + ], + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + const params = JSON.parse(body.launch.paramsText); + return params.outdir === "s3://bucket/results" && params.threads === "8"; + }) + .query(true) + .reply(200, { workflowId: "wf-123" }); + + const msg = createMockMsg({ payload: { launch: {} } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should give paramsArray precedence over paramsObj", async () => { + const node = createNode({ + paramsKey: "jsonParams", + paramsKeyType: "msg", + params: [{ name: "input", value: "from-array", valueType: "str" }], + }); + + RED.util.evaluateNodeProperty.mockImplementation((value, type, n, msg) => { + if (type === "msg" && value === "jsonParams") return { input: "from-json" }; + if (type === "str") return value; + return undefined; + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + const params = JSON.parse(body.launch.paramsText); + return params.input === "from-array"; // Array should win + }) + .query(true) + .reply(200, { workflowId: "wf-123" }); + + const msg = createMockMsg({ + payload: { launch: {} }, + jsonParams: { input: "from-json" }, + }); + + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should preserve existing launch.paramsText", async () => { + const node = createNode({ + params: [{ name: "newParam", value: "newValue", valueType: "str" }], + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + const params = JSON.parse(body.launch.paramsText); + return params.existing === "existingValue" && params.newParam === "newValue"; + }) + .query(true) + .reply(200, { workflowId: "wf-123" }); + + const msg = createMockMsg({ + payload: { + launch: { + paramsText: JSON.stringify({ existing: "existingValue" }), + }, + }, + }); + + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + }); + + describe("custom run name", () => { + it("should set launch.runName if provided", async () => { + const node = createNode({ + runName: "my-custom-run", + runNameType: "str", + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + return body.launch.runName === "my-custom-run"; + }) + .query(true) + .reply(200, { workflowId: "wf-123" }); + + const msg = createMockMsg({ payload: { launch: {} } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should not set runName if empty", async () => { + const node = createNode({ runName: "", runNameType: "str" }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + return body.launch.runName === undefined; + }) + .query(true) + .reply(200, { workflowId: "wf-123" }); + + const msg = createMockMsg({ payload: { launch: {} } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + }); + + describe("resume workflow", () => { + it("should fetch workflow details and launch config for resume", async () => { + const node = createNode({ + resumeWorkflowId: "wf-original", + resumeWorkflowIdType: "str", + }); + + // Mock workflow details + api.mockWorkflowStatus("wf-original", { + id: "wf-original", + status: "failed", + commitId: "abc123", + }); + + // Mock workflow launch config + api.mockWorkflowLaunchConfig("wf-original", { + id: "launch-1", + computeEnv: { id: "ce-123" }, + pipeline: "https://github.com/test/pipeline", + workDir: "s3://bucket/work", + sessionId: "session-123", + }); + + api.mockWorkflowLaunch({ workflowId: "wf-resumed" }); + + const msg = createMockMsg({}); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send.mock.calls[0][0].workflowId).toBe("wf-resumed"); + }); + + it("should set resume=true if workflow has commitId", async () => { + const node = createNode({ + resumeWorkflowId: "wf-with-commit", + resumeWorkflowIdType: "str", + }); + + api.mockWorkflowStatus("wf-with-commit", { + id: "wf-with-commit", + commitId: "commit-hash-123", + }); + + api.mockWorkflowLaunchConfig("wf-with-commit", { + id: "launch-1", + computeEnv: { id: "ce-123" }, + pipeline: "test", + workDir: "s3://work", + sessionId: "session-1", + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + return body.launch.resume === true; + }) + .query(true) + .reply(200, { workflowId: "wf-new" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should set resume=false if workflow cancelled early (no commitId)", async () => { + const node = createNode({ + resumeWorkflowId: "wf-cancelled", + resumeWorkflowIdType: "str", + }); + + api.mockWorkflowStatus("wf-cancelled", { + id: "wf-cancelled", + commitId: null, // No commit - was cancelled before any tasks ran + }); + + api.mockWorkflowLaunchConfig("wf-cancelled", { + id: "launch-1", + computeEnv: { id: "ce-123" }, + pipeline: "test", + workDir: "s3://work", + sessionId: "session-1", + // No resumeCommitId or revision either + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + return body.launch.resume === false && body.launch.revision === undefined; + }) + .query(true) + .reply(200, { workflowId: "wf-new" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + + it("should include revision field when resuming with commitId", async () => { + const node = createNode({ + resumeWorkflowId: "wf-resume", + resumeWorkflowIdType: "str", + }); + + api.mockWorkflowStatus("wf-resume", { + id: "wf-resume", + commitId: "abc123def", + }); + + api.mockWorkflowLaunchConfig("wf-resume", { + id: "launch-1", + computeEnv: { id: "ce-123" }, + pipeline: "test", + workDir: "s3://work", + sessionId: "session-1", + }); + + const scope = nock(BASE_URL) + .post("/workflow/launch", (body) => { + return body.launch.revision === "abc123def"; + }) + .query(true) + .reply(200, { workflowId: "wf-new" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(scope.isDone()).toBe(true); + }); + }); + + describe("error handling", () => { + it("should set red status on error", async () => { + const node = createNode(); + api.mockError("post", "/workflow/launch", 500, "Server Error"); + + const msg = createMockMsg({ payload: { launch: {} } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expectErrorStatus(node); + }); + + it("should call node.error with message on failure", async () => { + const node = createNode(); + api.mockError("post", "/workflow/launch", 400, "Bad Request"); + + const msg = createMockMsg({ payload: { launch: {} } }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("failed"), expect.anything()); + }); + + it("should error if no body provided and no launchpadName", async () => { + const node = createNode(); + + // Explicitly set payload to null to trigger the "no body" error + const msg = { _msgid: "test-msg", payload: null, body: null }; + const done = jest.fn(); + await node._triggerInput(msg, jest.fn(), done); + + expect(done).toHaveBeenCalledWith(expect.any(Error)); + }); + }); + + describe("pipeline autocomplete endpoint", () => { + it("should return pipeline autocomplete results", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + const node = createNode(); + RED._testHelpers.addNode("test-node-id", node); + + api.mockPipelines([ + { pipelineId: 1, name: "pipeline-one" }, + { pipelineId: 2, name: "pipeline-two" }, + ]); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/admin/seqera/pipelines/:nodeId", { + params: { nodeId: "test-node-id" }, + query: { search: "" }, + }); + + expect(data).toEqual([ + { value: "pipeline-one", label: "pipeline-one" }, + { value: "pipeline-two", label: "pipeline-two" }, + ]); + }); + + it("should return empty array if no workspaceId", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: null }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + const node = createNode(); + RED._testHelpers.addNode("test-node-id", node); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/admin/seqera/pipelines/:nodeId", { + params: { nodeId: "test-node-id" }, + query: {}, + }); + + expect(data).toEqual([]); + }); + + it("should handle node not existing (new node case)", async () => { + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + api.mockPipelines([{ pipelineId: 1, name: "test-pipeline" }]); + + const { data } = await RED._testHelpers.simulateHttpRequest("get", "/admin/seqera/pipelines/:nodeId", { + params: { nodeId: "non-existent" }, + query: { seqeraConfig: "seqera-config-id", search: "" }, + }); + + expect(data).toEqual([{ value: "test-pipeline", label: "test-pipeline" }]); + }); + }); +}); diff --git a/test/unit/workflow-monitor.test.js b/test/unit/workflow-monitor.test.js new file mode 100644 index 0000000..ae7e0f2 --- /dev/null +++ b/test/unit/workflow-monitor.test.js @@ -0,0 +1,332 @@ +/** + * Tests for nodes/workflow-monitor.js + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); +const { createMockMsg, expectMessagePassthrough } = require("../helpers/test-utils"); + +describe("seqera-workflow-monitor node", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + // Load the workflow-monitor node module + require("../../nodes/workflow-monitor")(RED); + }); + + afterEach(() => { + nock.cleanAll(); + }); + + function createNode(configOverrides = {}) { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-workflow-monitor").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-monitor", + seqera: "seqera-config-id", + workflowId: "workflowId", + workflowIdType: "msg", + workspaceId: "", + workspaceIdType: "str", + poll: "5", + pollType: "num", + keepPolling: false, // Default to false for simpler testing + ...configOverrides, + }; + + NodeConstructor.call(node, config); + return node; + } + + describe("node registration", () => { + it("should register seqera-workflow-monitor type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith( + "seqera-workflow-monitor", + expect.any(Function), + expect.any(Object), + ); + }); + }); + + describe("basic status fetching", () => { + it("should fetch workflow status on input", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "running" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + const done = jest.fn(); + + await node._triggerInput(msg, send, done); + + expect(send).toHaveBeenCalled(); + }); + + it("should set status color based on workflow status - submitted (yellow)", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "submitted" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "yellow", + shape: "ring", + }), + ); + }); + + it("should set status color based on workflow status - running (blue)", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "running" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "blue", + shape: "ring", + }), + ); + }); + + it("should set status color based on workflow status - succeeded (green)", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "succeeded" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "green", + shape: "dot", + }), + ); + }); + + it("should set status color based on workflow status - failed (red)", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "failed" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + shape: "dot", + }), + ); + }); + + it("should set status color based on workflow status - cancelled (grey)", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "cancelled" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "grey", + shape: "dot", + }), + ); + }); + }); + + describe("multiple outputs", () => { + it("should send to output 1 for submitted status", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "submitted" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[0]).not.toBeNull(); // Output 1 + expect(args[1]).toBeNull(); // Output 2 + expect(args[2]).toBeNull(); // Output 3 + }); + + it("should send to output 1 for running status", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "running" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[0]).not.toBeNull(); // Output 1 + expect(args[1]).toBeNull(); // Output 2 + expect(args[2]).toBeNull(); // Output 3 + }); + + it("should send to output 2 for succeeded status", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "succeeded" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[0]).toBeNull(); // Output 1 + expect(args[1]).not.toBeNull(); // Output 2 + expect(args[2]).toBeNull(); // Output 3 + }); + + it("should send to output 3 for failed status", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "failed" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[0]).toBeNull(); // Output 1 + expect(args[1]).toBeNull(); // Output 2 + expect(args[2]).not.toBeNull(); // Output 3 + }); + + it("should send to output 3 for cancelled status", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "cancelled" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const args = send.mock.calls[0][0]; + expect(args[0]).toBeNull(); // Output 1 + expect(args[1]).toBeNull(); // Output 2 + expect(args[2]).not.toBeNull(); // Output 3 + }); + }); + + describe("polling control", () => { + it("should not start polling when keepPolling=false", async () => { + const node = createNode({ keepPolling: false }); + api.mockWorkflowStatus("wf-123", { status: "running" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalledTimes(1); + // Verify no interval was set + expect(node._currentPollMs).toBeUndefined(); + }); + + it("should not continue polling when workflow reaches terminal state", async () => { + const node = createNode({ keepPolling: true }); + api.mockWorkflowStatus("wf-123", { status: "succeeded" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + expect(send).toHaveBeenCalledTimes(1); + // Terminal state means polling should be cleared + const args = send.mock.calls[0][0]; + expect(args[1]).not.toBeNull(); // Succeeded output + + // Clean up any intervals + await node._triggerClose(); + }); + + it("should clear polling on node close", async () => { + const node = createNode({ keepPolling: true }); + api.mockWorkflowStatus("wf-123", { status: "running" }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + // Close should work without error and clear intervals + await node._triggerClose(); + }); + }); + + describe("error handling", () => { + it("should set red dot status on error", async () => { + const node = createNode(); + api.mockError("get", "/workflow/wf-123", 500, "Server Error"); + + const msg = createMockMsg({ workflowId: "wf-123" }); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + shape: "dot", + }), + ); + }); + + it("should error if workflowId not provided", async () => { + const node = createNode({ workflowId: "", workflowIdType: "str" }); + + const msg = createMockMsg({}); + await node._triggerInput(msg, jest.fn(), jest.fn()); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("workflowId not provided"), expect.anything()); + }); + }); + + describe("message passthrough", () => { + it("should preserve input message properties in output", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { status: "running" }); + + const msg = createMockMsg({ + workflowId: "wf-123", + _context: { flowId: "flow-123" }, + correlationId: "corr-456", + customProp: "custom-value", + }); + + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const outputMsg = send.mock.calls[0][0][0]; + expectMessagePassthrough(outputMsg, msg); + }); + }); + + describe("output payload", () => { + it("should include workflow data in payload", async () => { + const node = createNode(); + api.mockWorkflowStatus("wf-123", { + id: "wf-123", + status: "running", + runName: "my-workflow-run", + }); + + const msg = createMockMsg({ workflowId: "wf-123" }); + const send = jest.fn(); + await node._triggerInput(msg, send, jest.fn()); + + const outputMsg = send.mock.calls[0][0][0]; + expect(outputMsg.payload.workflow).toBeDefined(); + expect(outputMsg.workflowId).toBe("wf-123"); + }); + }); +}); diff --git a/test/unit/workflow-poll.test.js b/test/unit/workflow-poll.test.js new file mode 100644 index 0000000..ab2560f --- /dev/null +++ b/test/unit/workflow-poll.test.js @@ -0,0 +1,324 @@ +/** + * Tests for nodes/workflow-poll.js + */ +const nock = require("nock"); +const { createMockRED } = require("../helpers/mock-red"); +const { createMockSeqeraConfigNode } = require("../helpers/mock-node"); +const { mockSeqeraAPI, BASE_URL } = require("../helpers/mock-axios"); + +describe("seqera-workflow-poll node", () => { + let RED; + let api; + + beforeEach(() => { + RED = createMockRED(); + api = mockSeqeraAPI(); + + // Set up the seqera-config node + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: "ws-123" }); + RED._testHelpers.addNode("seqera-config-id", seqeraConfig); + + // Load the workflow-poll node module + require("../../nodes/workflow-poll")(RED); + }); + + afterEach(() => { + nock.cleanAll(); + }); + + function createNode(configOverrides = {}) { + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-workflow-poll").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-poll", + seqera: "seqera-config-id", + search: "", + searchType: "str", + maxResults: "50", + maxResultsType: "num", + workspaceId: "", + workspaceIdType: "str", + pollFrequency: "1", + pollUnits: "minutes", + ...configOverrides, + }; + + NodeConstructor.call(node, config); + return node; + } + + // Helper to wait for async operations to complete + async function waitForPolling(node, timeout = 100) { + // Give time for the immediate poll to complete + await new Promise((resolve) => setTimeout(resolve, timeout)); + } + + describe("node registration", () => { + it("should register seqera-workflow-poll type", () => { + expect(RED.nodes.registerType).toHaveBeenCalledWith( + "seqera-workflow-poll", + expect.any(Function), + expect.any(Object), + ); + }); + }); + + describe("automatic polling initialization", () => { + it("should execute poll immediately on initialization", async () => { + api.mockWorkflowsList([]); + + const node = createNode(); + await waitForPolling(node); + + // Should have set status during polling + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "blue", + shape: "ring", + }), + ); + + // Clean up interval + await node._triggerClose(); + }); + + it("should set green status after successful poll", async () => { + api.mockWorkflowsList([]); + + const node = createNode(); + await waitForPolling(node); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "green", + shape: "dot", + }), + ); + + await node._triggerClose(); + }); + + it("should calculate poll frequency correctly - seconds", () => { + api.mockWorkflowsList([]); + + const node = createNode({ pollFrequency: "30", pollUnits: "seconds" }); + expect(node.pollFrequencySec).toBe(30); + + node._triggerClose(); + }); + + it("should calculate poll frequency correctly - minutes", () => { + api.mockWorkflowsList([]); + + const node = createNode({ pollFrequency: "5", pollUnits: "minutes" }); + expect(node.pollFrequencySec).toBe(300); + + node._triggerClose(); + }); + + it("should calculate poll frequency correctly - hours", () => { + api.mockWorkflowsList([]); + + const node = createNode({ pollFrequency: "2", pollUnits: "hours" }); + expect(node.pollFrequencySec).toBe(7200); + + node._triggerClose(); + }); + + it("should calculate poll frequency correctly - days", () => { + api.mockWorkflowsList([]); + + const node = createNode({ pollFrequency: "1", pollUnits: "days" }); + expect(node.pollFrequencySec).toBe(86400); + + node._triggerClose(); + }); + }); + + describe("output format", () => { + it("should send all workflows to output 1", async () => { + api.mockWorkflowsList([ + { workflow: { id: "wf-1", status: "running" } }, + { workflow: { id: "wf-2", status: "succeeded" } }, + ]); + + const node = createNode(); + await waitForPolling(node); + + // Check output 1 was called with all workflows + const sendCalls = node.send.mock.calls; + const output1Call = sendCalls.find((call) => call[0][0] !== null); + + expect(output1Call).toBeDefined(); + expect(output1Call[0][0].payload.workflows).toHaveLength(2); + expect(output1Call[0][0].workflowIds).toEqual(["wf-1", "wf-2"]); + + await node._triggerClose(); + }); + + it("should include nextPoll timestamp in output", async () => { + api.mockWorkflowsList([{ workflow: { id: "wf-1" } }]); + + const node = createNode({ pollFrequency: "5", pollUnits: "minutes" }); + await waitForPolling(node); + + const output1 = node.send.mock.calls[0][0][0]; + expect(output1.payload.nextPoll).toBeDefined(); + // nextPoll should be in the future + expect(new Date(output1.payload.nextPoll).getTime()).toBeGreaterThan(Date.now()); + + await node._triggerClose(); + }); + + it("should not send to output 2 on first poll (no previous state)", async () => { + api.mockWorkflowsList([ + { workflow: { id: "wf-1", status: "running" } }, + { workflow: { id: "wf-2", status: "running" } }, + ]); + + const node = createNode(); + await waitForPolling(node); + + // On first poll, output 2 should be null for all calls + const output2Calls = node.send.mock.calls.filter((call) => call[0][1] !== null); + expect(output2Calls).toHaveLength(0); + + await node._triggerClose(); + }); + }); + + describe("API parameters", () => { + it("should use search parameter if provided", async () => { + const scope = nock(BASE_URL) + .get("/workflow") + .query((q) => q.search === "my-search-term") + .reply(200, { workflows: [] }); + + const node = createNode({ search: "my-search-term", searchType: "str" }); + await waitForPolling(node); + + expect(scope.isDone()).toBe(true); + + await node._triggerClose(); + }); + + it("should use maxResults parameter", async () => { + const scope = nock(BASE_URL) + .get("/workflow") + .query((q) => q.max === "25") + .reply(200, { workflows: [] }); + + const node = createNode({ maxResults: "25", maxResultsType: "num" }); + await waitForPolling(node); + + expect(scope.isDone()).toBe(true); + + await node._triggerClose(); + }); + + it("should use workspaceId parameter", async () => { + const scope = nock(BASE_URL) + .get("/workflow") + .query((q) => q.workspaceId === "ws-123") + .reply(200, { workflows: [] }); + + const node = createNode(); + await waitForPolling(node); + + expect(scope.isDone()).toBe(true); + + await node._triggerClose(); + }); + + it("should include attributes=minimal in query", async () => { + const scope = nock(BASE_URL) + .get("/workflow") + .query((q) => q.attributes === "minimal") + .reply(200, { workflows: [] }); + + const node = createNode(); + await waitForPolling(node); + + expect(scope.isDone()).toBe(true); + + await node._triggerClose(); + }); + }); + + describe("error handling", () => { + it("should error if no workspaceId provided", async () => { + // Create config without workspaceId + const seqeraConfig = createMockSeqeraConfigNode({ workspaceId: null }); + RED._testHelpers.addNode("seqera-config-no-ws", seqeraConfig); + + const node = createNode({ seqera: "seqera-config-no-ws" }); + await waitForPolling(node); + + expect(node.error).toHaveBeenCalledWith(expect.stringContaining("Workspace ID not provided")); + + await node._triggerClose(); + }); + + it("should set red dot status on API error", async () => { + api.mockError("get", "/workflow", 500, "Server Error"); + + const node = createNode(); + await waitForPolling(node); + + expect(node.status).toHaveBeenCalledWith( + expect.objectContaining({ + fill: "red", + shape: "dot", + }), + ); + + await node._triggerClose(); + }); + + it("should handle empty workflows response", async () => { + api.mockWorkflowsList([]); + + const node = createNode(); + await waitForPolling(node); + + const output1 = node.send.mock.calls[0][0][0]; + expect(output1.payload.workflows).toEqual([]); + expect(output1.workflowIds).toEqual([]); + + await node._triggerClose(); + }); + }); + + describe("cleanup", () => { + it("should clear polling interval on node close", async () => { + api.mockWorkflowsList([]); + + const node = createNode(); + await waitForPolling(node); + + // Trigger close - should not throw + await node._triggerClose(); + }); + + it("should handle close when no config node exists", () => { + // Create node without seqera config reference + const NodeConstructor = RED._testHelpers.getRegisteredType("seqera-workflow-poll").constructor; + const node = {}; + + const config = { + id: "test-node-id", + name: "test-poll", + seqera: "non-existent-config", // This won't resolve + pollFrequency: "1", + pollUnits: "minutes", + }; + + NodeConstructor.call(node, config); + + // Close should not throw even without interval + node._triggerClose(); + }); + }); +});