From de871490d76835cd10bedb3aa4c446937796ec6f Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Mon, 3 Jun 2024 21:55:29 -0600 Subject: [PATCH 01/13] feat: add check server command --- .eslintignore | 1 + .eslintrc.json | 3 + .github/workflows/onPushToMain.yml | 56 + .github/workflows/onRelease.yml | 18 + .github/workflows/test.yml | 23 + .gitignore | 14 + .mocharc.json | 15 + .prettierrc.json | 1 + .vscode/launch.json | 20 + bin/dev.cmd | 3 + bin/dev.js | 6 + bin/run.cmd | 3 + bin/run.js | 5 + package.json | 74 + pnpm-lock.yaml | 5071 ++++++++++++++++++++++++++++ readme.md | 397 ++- src/commands/authenticate.ts | 29 + src/commands/check-server.ts | 30 + src/commands/hello/greet.ts | 29 + src/commands/hello/index.ts | 34 + src/commands/hello/world.ts | 19 + src/index.ts | 1 + test/commands/check-server.test.ts | 14 + test/commands/hello/index.test.ts | 9 + test/commands/hello/world.test.ts | 9 + test/tsconfig.json | 9 + tsconfig.json | 15 + 27 files changed, 5907 insertions(+), 1 deletion(-) create mode 100644 .eslintignore create mode 100644 .eslintrc.json create mode 100644 .github/workflows/onPushToMain.yml create mode 100644 .github/workflows/onRelease.yml create mode 100644 .github/workflows/test.yml create mode 100644 .gitignore create mode 100644 .mocharc.json create mode 100644 .prettierrc.json create mode 100644 .vscode/launch.json create mode 100644 bin/dev.cmd create mode 100755 bin/dev.js create mode 100644 bin/run.cmd create mode 100755 bin/run.js create mode 100644 package.json create mode 100644 pnpm-lock.yaml create mode 100644 src/commands/authenticate.ts create mode 100644 src/commands/check-server.ts create mode 100644 src/commands/hello/greet.ts create mode 100644 src/commands/hello/index.ts create mode 100644 src/commands/hello/world.ts create mode 100644 src/index.ts create mode 100644 test/commands/check-server.test.ts create mode 100644 test/commands/hello/index.test.ts create mode 100644 test/commands/hello/world.test.ts create mode 100644 test/tsconfig.json create mode 100644 tsconfig.json diff --git a/.eslintignore b/.eslintignore new file mode 100644 index 0000000..9b1c8b1 --- /dev/null +++ b/.eslintignore @@ -0,0 +1 @@ +/dist diff --git a/.eslintrc.json b/.eslintrc.json new file mode 100644 index 0000000..1dfcfc4 --- /dev/null +++ b/.eslintrc.json @@ -0,0 +1,3 @@ +{ + "extends": ["oclif", "oclif-typescript", "prettier"] +} diff --git a/.github/workflows/onPushToMain.yml b/.github/workflows/onPushToMain.yml new file mode 100644 index 0000000..de7dda1 --- /dev/null +++ b/.github/workflows/onPushToMain.yml @@ -0,0 +1,56 @@ +# test +name: version, tag and github release + +on: + push: + branches: [main] + +jobs: + release: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 + - name: Check if version already exists + id: version-check + run: | + package_version=$(node -p "require('./package.json').version") + exists=$(gh api repos/${{ github.repository }}/releases/tags/v$package_version >/dev/null 2>&1 && echo "true" || echo "") + + if [ -n "$exists" ]; + then + echo "Version v$package_version already exists" + echo "::warning file=package.json,line=1::Version v$package_version already exists - no release will be created. If you want to create a new release, please update the version in package.json and push again." + echo "skipped=true" >> $GITHUB_OUTPUT + else + echo "Version v$package_version does not exist. Creating release..." + echo "skipped=false" >> $GITHUB_OUTPUT + echo "tag=v$package_version" >> $GITHUB_OUTPUT + fi + env: + GH_TOKEN: ${{ secrets.GH_TOKEN }} + - name: Setup git + if: ${{ steps.version-check.outputs.skipped == 'false' }} + run: | + git config --global user.email ${{ secrets.GH_EMAIL }} + git config --global user.name ${{ secrets.GH_USERNAME }} + - name: Generate oclif README + if: ${{ steps.version-check.outputs.skipped == 'false' }} + id: oclif-readme + run: | + pnpm install + pnpm exec oclif readme + if [ -n "$(git status --porcelain)" ]; then + git add . + git commit -am "chore: update README.md" + git push -u origin ${{ github.ref_name }} + fi + - name: Create Github Release + uses: ncipollo/release-action@2c591bcc8ecdcd2db72b97d6147f871fcd833ba5 + if: ${{ steps.version-check.outputs.skipped == 'false' }} + with: + name: ${{ steps.version-check.outputs.tag }} + tag: ${{ steps.version-check.outputs.tag }} + commit: ${{ github.ref_name }} + token: ${{ secrets.GH_TOKEN }} + skipIfReleaseExists: true diff --git a/.github/workflows/onRelease.yml b/.github/workflows/onRelease.yml new file mode 100644 index 0000000..ff94d00 --- /dev/null +++ b/.github/workflows/onRelease.yml @@ -0,0 +1,18 @@ +name: publish + +on: + release: + types: [released] + +jobs: + publish: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 + with: + node-version: latest + - run: pnpm install + - uses: JS-DevTools/npm-publish@19c28f1ef146469e409470805ea4279d47c3d35c + with: + token: ${{ secrets.NPM_TOKEN }} diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml new file mode 100644 index 0000000..848357d --- /dev/null +++ b/.github/workflows/test.yml @@ -0,0 +1,23 @@ +name: tests +on: + push: + branches-ignore: [main] + workflow_dispatch: + +jobs: + unit-tests: + strategy: + matrix: + os: ['ubuntu-latest', 'windows-latest'] + node_version: [lts/-1, lts/*, latest] + fail-fast: false + runs-on: ${{ matrix.os }} + steps: + - uses: actions/checkout@v4 + - uses: actions/setup-node@v4 + with: + node-version: ${{ matrix.node_version }} + cache: pnpm + - run: pnpm install + - run: pnpm run build + - run: pnpm run test diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..f4f8ff0 --- /dev/null +++ b/.gitignore @@ -0,0 +1,14 @@ +*-debug.log +*-error.log +**/.DS_Store +/.idea +/dist +/tmp +/node_modules +oclif.manifest.json + + +yarn.lock +package-lock.json + + diff --git a/.mocharc.json b/.mocharc.json new file mode 100644 index 0000000..fa25f20 --- /dev/null +++ b/.mocharc.json @@ -0,0 +1,15 @@ +{ + "require": [ + "ts-node/register" + ], + "watch-extensions": [ + "ts" + ], + "recursive": true, + "reporter": "spec", + "timeout": 60000, + "node-option": [ + "loader=ts-node/esm", + "experimental-specifier-resolution=node" + ] +} diff --git a/.prettierrc.json b/.prettierrc.json new file mode 100644 index 0000000..6314335 --- /dev/null +++ b/.prettierrc.json @@ -0,0 +1 @@ +"@oclif/prettier-config" diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 0000000..92c98a1 --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,20 @@ +{ + "version": "0.2.0", + "configurations": [ + { + "type": "node", + "request": "attach", + "name": "Attach", + "port": 9229, + "skipFiles": ["/**"] + }, + { + "type": "node", + "request": "launch", + "name": "Execute Command", + "skipFiles": ["/**"], + "program": "${workspaceFolder}/bin/dev", + "args": ["hello", "world"] + } + ] +} diff --git a/bin/dev.cmd b/bin/dev.cmd new file mode 100644 index 0000000..cec553b --- /dev/null +++ b/bin/dev.cmd @@ -0,0 +1,3 @@ +@echo off + +node --loader ts-node/esm --no-warnings=ExperimentalWarning "%~dp0\dev" %* diff --git a/bin/dev.js b/bin/dev.js new file mode 100755 index 0000000..f9166e5 --- /dev/null +++ b/bin/dev.js @@ -0,0 +1,6 @@ +#!/usr/bin/env -S node --loader ts-node/esm --no-warnings=ExperimentalWarning + +// eslint-disable-next-line n/shebang +import {execute} from '@oclif/core' + +await execute({development: true, dir: import.meta.url}) diff --git a/bin/run.cmd b/bin/run.cmd new file mode 100644 index 0000000..968fc30 --- /dev/null +++ b/bin/run.cmd @@ -0,0 +1,3 @@ +@echo off + +node "%~dp0\run" %* diff --git a/bin/run.js b/bin/run.js new file mode 100755 index 0000000..dd50271 --- /dev/null +++ b/bin/run.js @@ -0,0 +1,5 @@ +#!/usr/bin/env node + +import {execute} from '@oclif/core' + +await execute({dir: import.meta.url}) diff --git a/package.json b/package.json new file mode 100644 index 0000000..099d238 --- /dev/null +++ b/package.json @@ -0,0 +1,74 @@ +{ + "name": "dokploy", + "description": "A CLI to manage dokploy server remotely", + "version": "0.0.0", + "author": "Mauricio Siu", + "bin": { + "dokploy": "./bin/run.js" + }, + "bugs": "https://github.com/Dokploy/cli/issues", + "dependencies": { + "@oclif/core": "^3", + "@oclif/plugin-help": "^6", + "@oclif/plugin-plugins": "^5", + "axios": "^1.7.2" + }, + "devDependencies": { + "@oclif/prettier-config": "^0.2.1", + "@oclif/test": "^4", + "@types/chai": "^4", + "@types/mocha": "^10", + "@types/node": "^18", + "chai": "^4", + "eslint": "^8", + "eslint-config-oclif": "^5", + "eslint-config-oclif-typescript": "^3", + "eslint-config-prettier": "^9", + "mocha": "^10", + "oclif": "^4", + "shx": "^0.3.3", + "ts-node": "^10", + "typescript": "^5" + }, + "engines": { + "node": ">=18.0.0" + }, + "files": [ + "/bin", + "/dist", + "/oclif.manifest.json" + ], + "homepage": "https://github.com/Dokploy/cli", + "keywords": [ + "oclif" + ], + "license": "MIT", + "main": "dist/index.js", + "type": "module", + "oclif": { + "bin": "dokploy", + "dirname": "dokploy", + "commands": "./dist/commands", + "plugins": [ + "@oclif/plugin-help", + "@oclif/plugin-plugins" + ], + "topicSeparator": " ", + "topics": { + "hello": { + "description": "Say hello to the world and others" + } + } + }, + "repository": "Dokploy/cli", + "scripts": { + "build": "shx rm -rf dist && tsc -b", + "lint": "eslint . --ext .ts", + "postpack": "shx rm -f oclif.manifest.json", + "posttest": "pnpm run lint", + "prepack": "oclif manifest && oclif readme", + "test": "mocha --forbid-only \"test/**/*.test.ts\"", + "version": "oclif readme && git add README.md" + }, + "types": "dist/index.d.ts" +} diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml new file mode 100644 index 0000000..6453d7b --- /dev/null +++ b/pnpm-lock.yaml @@ -0,0 +1,5071 @@ +lockfileVersion: '6.0' + +settings: + autoInstallPeers: true + excludeLinksFromLockfile: false + +dependencies: + '@oclif/core': + specifier: ^3 + version: 3.26.9 + '@oclif/plugin-help': + specifier: ^6 + version: 6.1.0(typescript@5.4.5) + '@oclif/plugin-plugins': + specifier: ^5 + version: 5.2.1(typescript@5.4.5) + axios: + specifier: ^1.7.2 + version: 1.7.2 + +devDependencies: + '@oclif/prettier-config': + specifier: ^0.2.1 + version: 0.2.1 + '@oclif/test': + specifier: ^4 + version: 4.0.3(@oclif/core@3.26.9) + '@types/chai': + specifier: ^4 + version: 4.3.16 + '@types/mocha': + specifier: ^10 + version: 10.0.6 + '@types/node': + specifier: ^18 + version: 18.19.34 + chai: + specifier: ^4 + version: 4.4.1 + eslint: + specifier: ^8 + version: 8.57.0 + eslint-config-oclif: + specifier: ^5 + version: 5.2.0(eslint@8.57.0) + eslint-config-oclif-typescript: + specifier: ^3 + version: 3.1.7(eslint@8.57.0)(typescript@5.4.5) + eslint-config-prettier: + specifier: ^9 + version: 9.1.0(eslint@8.57.0) + mocha: + specifier: ^10 + version: 10.4.0 + oclif: + specifier: ^4 + version: 4.12.3(typescript@5.4.5) + shx: + specifier: ^0.3.3 + version: 0.3.4 + ts-node: + specifier: ^10 + version: 10.9.2(@types/node@18.19.34)(typescript@5.4.5) + typescript: + specifier: ^5 + version: 5.4.5 + +packages: + + /@aws-crypto/crc32@3.0.0: + resolution: {integrity: sha512-IzSgsrxUcsrejQbPVilIKy16kAT52EwB6zSaI+M3xxIhKh5+aldEyvI+z6erM7TCLB2BJsFrtHjp6/4/sr+3dA==} + dependencies: + '@aws-crypto/util': 3.0.0 + '@aws-sdk/types': 3.577.0 + tslib: 1.14.1 + dev: true + + /@aws-crypto/crc32c@3.0.0: + resolution: {integrity: sha512-ENNPPManmnVJ4BTXlOjAgD7URidbAznURqD0KvfREyc4o20DPYdEldU1f5cQ7Jbj0CJJSPaMIk/9ZshdB3210w==} + dependencies: + '@aws-crypto/util': 3.0.0 + '@aws-sdk/types': 3.577.0 + tslib: 1.14.1 + dev: true + + /@aws-crypto/ie11-detection@3.0.0: + resolution: {integrity: sha512-341lBBkiY1DfDNKai/wXM3aujNBkXR7tq1URPQDL9wi3AUbI80NR74uF1TXHMm7po1AcnFk8iu2S2IeU/+/A+Q==} + dependencies: + tslib: 1.14.1 + dev: true + + /@aws-crypto/sha1-browser@3.0.0: + resolution: {integrity: sha512-NJth5c997GLHs6nOYTzFKTbYdMNA6/1XlKVgnZoaZcQ7z7UJlOgj2JdbHE8tiYLS3fzXNCguct77SPGat2raSw==} + dependencies: + '@aws-crypto/ie11-detection': 3.0.0 + '@aws-crypto/supports-web-crypto': 3.0.0 + '@aws-crypto/util': 3.0.0 + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-locate-window': 3.568.0 + '@aws-sdk/util-utf8-browser': 3.259.0 + tslib: 1.14.1 + dev: true + + /@aws-crypto/sha256-browser@3.0.0: + resolution: {integrity: sha512-8VLmW2B+gjFbU5uMeqtQM6Nj0/F1bro80xQXCW6CQBWgosFWXTx77aeOF5CAIAmbOK64SdMBJdNr6J41yP5mvQ==} + dependencies: + '@aws-crypto/ie11-detection': 3.0.0 + '@aws-crypto/sha256-js': 3.0.0 + '@aws-crypto/supports-web-crypto': 3.0.0 + '@aws-crypto/util': 3.0.0 + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-locate-window': 3.568.0 + '@aws-sdk/util-utf8-browser': 3.259.0 + tslib: 1.14.1 + dev: true + + /@aws-crypto/sha256-js@3.0.0: + resolution: {integrity: sha512-PnNN7os0+yd1XvXAy23CFOmTbMaDxgxXtTKHybrJ39Y8kGzBATgBFibWJKH6BhytLI/Zyszs87xCOBNyBig6vQ==} + dependencies: + '@aws-crypto/util': 3.0.0 + '@aws-sdk/types': 3.577.0 + tslib: 1.14.1 + dev: true + + /@aws-crypto/supports-web-crypto@3.0.0: + resolution: {integrity: sha512-06hBdMwUAb2WFTuGG73LSC0wfPu93xWwo5vL2et9eymgmu3Id5vFAHBbajVWiGhPO37qcsdCap/FqXvJGJWPIg==} + dependencies: + tslib: 1.14.1 + dev: true + + /@aws-crypto/util@3.0.0: + resolution: {integrity: sha512-2OJlpeJpCR48CC8r+uKVChzs9Iungj9wkZrl8Z041DWEWvyIHILYKCPNzJghKsivj+S3mLo6BVc7mBNzdxA46w==} + dependencies: + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-utf8-browser': 3.259.0 + tslib: 1.14.1 + dev: true + + /@aws-sdk/client-cloudfront@3.588.0: + resolution: {integrity: sha512-C9qik7uz1jfjXwhLzwSO2uy5Ze/IkuuMuqcFyjXNnRgpXv1TC0Buv2r1JyYYxnpxV4YvLWk/+kG5UCv1p2jjxw==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-crypto/sha256-browser': 3.0.0 + '@aws-crypto/sha256-js': 3.0.0 + '@aws-sdk/client-sso-oidc': 3.588.0(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/client-sts': 3.588.0 + '@aws-sdk/core': 3.588.0 + '@aws-sdk/credential-provider-node': 3.588.0(@aws-sdk/client-sso-oidc@3.588.0)(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/middleware-host-header': 3.577.0 + '@aws-sdk/middleware-logger': 3.577.0 + '@aws-sdk/middleware-recursion-detection': 3.577.0 + '@aws-sdk/middleware-user-agent': 3.587.0 + '@aws-sdk/region-config-resolver': 3.587.0 + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-endpoints': 3.587.0 + '@aws-sdk/util-user-agent-browser': 3.577.0 + '@aws-sdk/util-user-agent-node': 3.587.0 + '@aws-sdk/xml-builder': 3.575.0 + '@smithy/config-resolver': 3.0.1 + '@smithy/core': 2.1.1 + '@smithy/fetch-http-handler': 3.0.1 + '@smithy/hash-node': 3.0.0 + '@smithy/invalid-dependency': 3.0.0 + '@smithy/middleware-content-length': 3.0.0 + '@smithy/middleware-endpoint': 3.0.1 + '@smithy/middleware-retry': 3.0.3 + '@smithy/middleware-serde': 3.0.0 + '@smithy/middleware-stack': 3.0.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/node-http-handler': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/url-parser': 3.0.0 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.3 + '@smithy/util-defaults-mode-node': 3.0.3 + '@smithy/util-endpoints': 2.0.1 + '@smithy/util-middleware': 3.0.0 + '@smithy/util-retry': 3.0.0 + '@smithy/util-stream': 3.0.1 + '@smithy/util-utf8': 3.0.0 + '@smithy/util-waiter': 3.0.0 + tslib: 2.6.2 + transitivePeerDependencies: + - aws-crt + dev: true + + /@aws-sdk/client-s3@3.588.0: + resolution: {integrity: sha512-MyJs3sbgRtVOdT2xxdg/CmLk+t+dMg26nfEZucBFeJKFAHfTA74sjef9y+GQ2xFUNq+kqG1CnP8JGMiGx2ht0w==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-crypto/sha1-browser': 3.0.0 + '@aws-crypto/sha256-browser': 3.0.0 + '@aws-crypto/sha256-js': 3.0.0 + '@aws-sdk/client-sso-oidc': 3.588.0(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/client-sts': 3.588.0 + '@aws-sdk/core': 3.588.0 + '@aws-sdk/credential-provider-node': 3.588.0(@aws-sdk/client-sso-oidc@3.588.0)(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/middleware-bucket-endpoint': 3.587.0 + '@aws-sdk/middleware-expect-continue': 3.577.0 + '@aws-sdk/middleware-flexible-checksums': 3.587.0 + '@aws-sdk/middleware-host-header': 3.577.0 + '@aws-sdk/middleware-location-constraint': 3.577.0 + '@aws-sdk/middleware-logger': 3.577.0 + '@aws-sdk/middleware-recursion-detection': 3.577.0 + '@aws-sdk/middleware-sdk-s3': 3.587.0 + '@aws-sdk/middleware-signing': 3.587.0 + '@aws-sdk/middleware-ssec': 3.577.0 + '@aws-sdk/middleware-user-agent': 3.587.0 + '@aws-sdk/region-config-resolver': 3.587.0 + '@aws-sdk/signature-v4-multi-region': 3.587.0 + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-endpoints': 3.587.0 + '@aws-sdk/util-user-agent-browser': 3.577.0 + '@aws-sdk/util-user-agent-node': 3.587.0 + '@aws-sdk/xml-builder': 3.575.0 + '@smithy/config-resolver': 3.0.1 + '@smithy/core': 2.1.1 + '@smithy/eventstream-serde-browser': 3.0.0 + '@smithy/eventstream-serde-config-resolver': 3.0.0 + '@smithy/eventstream-serde-node': 3.0.0 + '@smithy/fetch-http-handler': 3.0.1 + '@smithy/hash-blob-browser': 3.0.0 + '@smithy/hash-node': 3.0.0 + '@smithy/hash-stream-node': 3.0.0 + '@smithy/invalid-dependency': 3.0.0 + '@smithy/md5-js': 3.0.0 + '@smithy/middleware-content-length': 3.0.0 + '@smithy/middleware-endpoint': 3.0.1 + '@smithy/middleware-retry': 3.0.3 + '@smithy/middleware-serde': 3.0.0 + '@smithy/middleware-stack': 3.0.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/node-http-handler': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/url-parser': 3.0.0 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.3 + '@smithy/util-defaults-mode-node': 3.0.3 + '@smithy/util-endpoints': 2.0.1 + '@smithy/util-retry': 3.0.0 + '@smithy/util-stream': 3.0.1 + '@smithy/util-utf8': 3.0.0 + '@smithy/util-waiter': 3.0.0 + tslib: 2.6.2 + transitivePeerDependencies: + - aws-crt + dev: true + + /@aws-sdk/client-sso-oidc@3.588.0(@aws-sdk/client-sts@3.588.0): + resolution: {integrity: sha512-CTbgtLSg0y2jIOtESuQKkRIqRe/FQmKuyzFWc+Qy6yGcbk1Pyusfz2BC+GGwpYU+1BlBBSNnLQHpx3XY87+aSA==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-crypto/sha256-browser': 3.0.0 + '@aws-crypto/sha256-js': 3.0.0 + '@aws-sdk/client-sts': 3.588.0 + '@aws-sdk/core': 3.588.0 + '@aws-sdk/credential-provider-node': 3.588.0(@aws-sdk/client-sso-oidc@3.588.0)(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/middleware-host-header': 3.577.0 + '@aws-sdk/middleware-logger': 3.577.0 + '@aws-sdk/middleware-recursion-detection': 3.577.0 + '@aws-sdk/middleware-user-agent': 3.587.0 + '@aws-sdk/region-config-resolver': 3.587.0 + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-endpoints': 3.587.0 + '@aws-sdk/util-user-agent-browser': 3.577.0 + '@aws-sdk/util-user-agent-node': 3.587.0 + '@smithy/config-resolver': 3.0.1 + '@smithy/core': 2.1.1 + '@smithy/fetch-http-handler': 3.0.1 + '@smithy/hash-node': 3.0.0 + '@smithy/invalid-dependency': 3.0.0 + '@smithy/middleware-content-length': 3.0.0 + '@smithy/middleware-endpoint': 3.0.1 + '@smithy/middleware-retry': 3.0.3 + '@smithy/middleware-serde': 3.0.0 + '@smithy/middleware-stack': 3.0.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/node-http-handler': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/url-parser': 3.0.0 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.3 + '@smithy/util-defaults-mode-node': 3.0.3 + '@smithy/util-endpoints': 2.0.1 + '@smithy/util-middleware': 3.0.0 + '@smithy/util-retry': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + transitivePeerDependencies: + - '@aws-sdk/client-sts' + - aws-crt + dev: true + + /@aws-sdk/client-sso@3.588.0: + resolution: {integrity: sha512-zKS+xUkBLfwjbh77ZjtRUoG/vR/fyDteSE6rOAzwlmHQL8p+QUX+zNUNvCInvPi62zGBhEwXOvzs8zvnT4NzfQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-crypto/sha256-browser': 3.0.0 + '@aws-crypto/sha256-js': 3.0.0 + '@aws-sdk/core': 3.588.0 + '@aws-sdk/middleware-host-header': 3.577.0 + '@aws-sdk/middleware-logger': 3.577.0 + '@aws-sdk/middleware-recursion-detection': 3.577.0 + '@aws-sdk/middleware-user-agent': 3.587.0 + '@aws-sdk/region-config-resolver': 3.587.0 + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-endpoints': 3.587.0 + '@aws-sdk/util-user-agent-browser': 3.577.0 + '@aws-sdk/util-user-agent-node': 3.587.0 + '@smithy/config-resolver': 3.0.1 + '@smithy/core': 2.1.1 + '@smithy/fetch-http-handler': 3.0.1 + '@smithy/hash-node': 3.0.0 + '@smithy/invalid-dependency': 3.0.0 + '@smithy/middleware-content-length': 3.0.0 + '@smithy/middleware-endpoint': 3.0.1 + '@smithy/middleware-retry': 3.0.3 + '@smithy/middleware-serde': 3.0.0 + '@smithy/middleware-stack': 3.0.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/node-http-handler': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/url-parser': 3.0.0 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.3 + '@smithy/util-defaults-mode-node': 3.0.3 + '@smithy/util-endpoints': 2.0.1 + '@smithy/util-middleware': 3.0.0 + '@smithy/util-retry': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + transitivePeerDependencies: + - aws-crt + dev: true + + /@aws-sdk/client-sts@3.588.0: + resolution: {integrity: sha512-UIMjcUikgG9NIENQxSyJNTHMD8TaTfK6Jjf1iuZSyQRyTrcGy0/xcDxrmwZQFAPkOPUf6w9KqydLkMLcYOBdPQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-crypto/sha256-browser': 3.0.0 + '@aws-crypto/sha256-js': 3.0.0 + '@aws-sdk/client-sso-oidc': 3.588.0(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/core': 3.588.0 + '@aws-sdk/credential-provider-node': 3.588.0(@aws-sdk/client-sso-oidc@3.588.0)(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/middleware-host-header': 3.577.0 + '@aws-sdk/middleware-logger': 3.577.0 + '@aws-sdk/middleware-recursion-detection': 3.577.0 + '@aws-sdk/middleware-user-agent': 3.587.0 + '@aws-sdk/region-config-resolver': 3.587.0 + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-endpoints': 3.587.0 + '@aws-sdk/util-user-agent-browser': 3.577.0 + '@aws-sdk/util-user-agent-node': 3.587.0 + '@smithy/config-resolver': 3.0.1 + '@smithy/core': 2.1.1 + '@smithy/fetch-http-handler': 3.0.1 + '@smithy/hash-node': 3.0.0 + '@smithy/invalid-dependency': 3.0.0 + '@smithy/middleware-content-length': 3.0.0 + '@smithy/middleware-endpoint': 3.0.1 + '@smithy/middleware-retry': 3.0.3 + '@smithy/middleware-serde': 3.0.0 + '@smithy/middleware-stack': 3.0.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/node-http-handler': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/url-parser': 3.0.0 + '@smithy/util-base64': 3.0.0 + '@smithy/util-body-length-browser': 3.0.0 + '@smithy/util-body-length-node': 3.0.0 + '@smithy/util-defaults-mode-browser': 3.0.3 + '@smithy/util-defaults-mode-node': 3.0.3 + '@smithy/util-endpoints': 2.0.1 + '@smithy/util-middleware': 3.0.0 + '@smithy/util-retry': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + transitivePeerDependencies: + - aws-crt + dev: true + + /@aws-sdk/core@3.588.0: + resolution: {integrity: sha512-O1c2+9ce46Z+iiid+W3iC1IvPbfIo5ev9CBi54GdNB9SaI8/3+f8MJcux0D6c9toCF0ArMersN/gp8ek57e9uQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/core': 2.1.1 + '@smithy/protocol-http': 4.0.0 + '@smithy/signature-v4': 3.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + fast-xml-parser: 4.2.5 + tslib: 2.6.2 + dev: true + + /@aws-sdk/credential-provider-env@3.587.0: + resolution: {integrity: sha512-Hyg/5KFECIk2k5o8wnVEiniV86yVkhn5kzITUydmNGCkXdBFHMHRx6hleQ1bqwJHbBskyu8nbYamzcwymmGwmw==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/property-provider': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/credential-provider-http@3.587.0: + resolution: {integrity: sha512-Su1SRWVRCuR1e32oxX3C1V4c5hpPN20WYcRfdcr2wXwHqSvys5DrnmuCC+JoEnS/zt3adUJhPliTqpfKgSdMrA==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/fetch-http-handler': 3.0.1 + '@smithy/node-http-handler': 3.0.0 + '@smithy/property-provider': 3.1.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/util-stream': 3.0.1 + tslib: 2.6.2 + dev: true + + /@aws-sdk/credential-provider-ini@3.588.0(@aws-sdk/client-sso-oidc@3.588.0)(@aws-sdk/client-sts@3.588.0): + resolution: {integrity: sha512-tP/YmEKvYpmp7pCR2OuhoOhAOtm6BbZ1hbeG9Sw9RFZi55dbGPHqMmfvvzHFAGsJ20z4/oDS+UnHaWVhRnV82w==} + engines: {node: '>=16.0.0'} + peerDependencies: + '@aws-sdk/client-sts': ^3.588.0 + dependencies: + '@aws-sdk/client-sts': 3.588.0 + '@aws-sdk/credential-provider-env': 3.587.0 + '@aws-sdk/credential-provider-http': 3.587.0 + '@aws-sdk/credential-provider-process': 3.587.0 + '@aws-sdk/credential-provider-sso': 3.588.0(@aws-sdk/client-sso-oidc@3.588.0) + '@aws-sdk/credential-provider-web-identity': 3.587.0(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/types': 3.577.0 + '@smithy/credential-provider-imds': 3.1.0 + '@smithy/property-provider': 3.1.0 + '@smithy/shared-ini-file-loader': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + transitivePeerDependencies: + - '@aws-sdk/client-sso-oidc' + - aws-crt + dev: true + + /@aws-sdk/credential-provider-node@3.588.0(@aws-sdk/client-sso-oidc@3.588.0)(@aws-sdk/client-sts@3.588.0): + resolution: {integrity: sha512-8s4Ruo6q1YIrj8AZKBiUQG42051ytochDMSqdVOEZGxskfvmt2XALyi5SsWd0Ve3zR95zi+EtRBNPn2EU8sQpA==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/credential-provider-env': 3.587.0 + '@aws-sdk/credential-provider-http': 3.587.0 + '@aws-sdk/credential-provider-ini': 3.588.0(@aws-sdk/client-sso-oidc@3.588.0)(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/credential-provider-process': 3.587.0 + '@aws-sdk/credential-provider-sso': 3.588.0(@aws-sdk/client-sso-oidc@3.588.0) + '@aws-sdk/credential-provider-web-identity': 3.587.0(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/types': 3.577.0 + '@smithy/credential-provider-imds': 3.1.0 + '@smithy/property-provider': 3.1.0 + '@smithy/shared-ini-file-loader': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + transitivePeerDependencies: + - '@aws-sdk/client-sso-oidc' + - '@aws-sdk/client-sts' + - aws-crt + dev: true + + /@aws-sdk/credential-provider-process@3.587.0: + resolution: {integrity: sha512-V4xT3iCqkF8uL6QC4gqBJg/2asd/damswP1h9HCfqTllmPWzImS+8WD3VjgTLw5b0KbTy+ZdUhKc0wDnyzkzxg==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/property-provider': 3.1.0 + '@smithy/shared-ini-file-loader': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/credential-provider-sso@3.588.0(@aws-sdk/client-sso-oidc@3.588.0): + resolution: {integrity: sha512-1GstMCyFzenVeppK7hWazMvo3P1DXKP70XkXAjH8H2ELBVg5X8Zt043cnQ7CMt4XjCV+ettHAtc9kz/gJTkDNQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/client-sso': 3.588.0 + '@aws-sdk/token-providers': 3.587.0(@aws-sdk/client-sso-oidc@3.588.0) + '@aws-sdk/types': 3.577.0 + '@smithy/property-provider': 3.1.0 + '@smithy/shared-ini-file-loader': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + transitivePeerDependencies: + - '@aws-sdk/client-sso-oidc' + - aws-crt + dev: true + + /@aws-sdk/credential-provider-web-identity@3.587.0(@aws-sdk/client-sts@3.588.0): + resolution: {integrity: sha512-XqIx/I2PG7kyuw3WjAP9wKlxy8IvFJwB8asOFT1xPFoVfZYKIogjG9oLP5YiRtfvDkWIztHmg5MlVv3HdJDGRw==} + engines: {node: '>=16.0.0'} + peerDependencies: + '@aws-sdk/client-sts': ^3.587.0 + dependencies: + '@aws-sdk/client-sts': 3.588.0 + '@aws-sdk/types': 3.577.0 + '@smithy/property-provider': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-bucket-endpoint@3.587.0: + resolution: {integrity: sha512-HkFXLPl8pr6BH/Q0JpOESqEKL0ZK3sk7aSZ1S6GE4RXET7H5R94THULXqQFZzD48gZcyFooO/yNKZTqrZFaWKg==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-arn-parser': 3.568.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/types': 3.0.0 + '@smithy/util-config-provider': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-expect-continue@3.577.0: + resolution: {integrity: sha512-6dPp8Tv4F0of4un5IAyG6q++GrRrNQQ4P2NAMB1W0VO4JoEu1C8GievbbDLi88TFIFmtKpnHB0ODCzwnoe8JsA==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-flexible-checksums@3.587.0: + resolution: {integrity: sha512-URMwp/budDvKhIvZ4a6zIBfFTun/iDlPWXqsGKYjEtHt8jz27OSjCZtDtIeqW4WTBdKL8KZgQcl+DdaE5M1qiQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-crypto/crc32': 3.0.0 + '@aws-crypto/crc32c': 3.0.0 + '@aws-sdk/types': 3.577.0 + '@smithy/is-array-buffer': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/types': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-host-header@3.577.0: + resolution: {integrity: sha512-9ca5MJz455CODIVXs0/sWmJm7t3QO4EUa1zf8pE8grLpzf0J94bz/skDWm37Pli13T3WaAQBHCTiH2gUVfCsWg==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-location-constraint@3.577.0: + resolution: {integrity: sha512-DKPTD2D2s+t2QUo/IXYtVa/6Un8GZ+phSTBkyBNx2kfZz4Kwavhl/JJzSqTV3GfCXkVdFu7CrjoX7BZ6qWeTUA==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-logger@3.577.0: + resolution: {integrity: sha512-aPFGpGjTZcJYk+24bg7jT4XdIp42mFXSuPt49lw5KygefLyJM/sB0bKKqPYYivW0rcuZ9brQ58eZUNthrzYAvg==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-recursion-detection@3.577.0: + resolution: {integrity: sha512-pn3ZVEd2iobKJlR3H+bDilHjgRnNrQ6HMmK9ZzZw89Ckn3Dcbv48xOv4RJvu0aU8SDLl/SNCxppKjeLDTPGBNA==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-sdk-s3@3.587.0: + resolution: {integrity: sha512-vtXTGEiw1E9Fax4LmcU2Z208gbrC8ShrdsSLmGcRPpu5NPOGBFBSDG5sy5EDNClrFxIl/Le8coQnD0EDBtx+uQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-arn-parser': 3.568.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/signature-v4': 3.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/util-config-provider': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-signing@3.587.0: + resolution: {integrity: sha512-tiZaTDj4RvhXGRAlncFn7CSEfL3iNPO67WSaxAq+Ls5j1VgczPhu5262cWONNoMgth3nXR1hhLC4ITSl/a6AzA==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/property-provider': 3.1.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/signature-v4': 3.0.0 + '@smithy/types': 3.0.0 + '@smithy/util-middleware': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-ssec@3.577.0: + resolution: {integrity: sha512-i2BPJR+rp8xmRVIGc0h1kDRFcM2J9GnClqqpc+NLSjmYadlcg4mPklisz9HzwFVcRPJ5XcGf3U4BYs5G8+iTyg==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/middleware-user-agent@3.587.0: + resolution: {integrity: sha512-SyDomN+IOrygLucziG7/nOHkjUXES5oH5T7p8AboO8oakMQJdnudNXiYWTicQWO52R51U6CR27rcMPTGeMedYA==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@aws-sdk/util-endpoints': 3.587.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/region-config-resolver@3.587.0: + resolution: {integrity: sha512-93I7IPZtulZQoRK+O20IJ4a1syWwYPzoO2gc3v+/GNZflZPV3QJXuVbIm0pxBsu0n/mzKGUKqSOLPIaN098HcQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/types': 3.0.0 + '@smithy/util-config-provider': 3.0.0 + '@smithy/util-middleware': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/signature-v4-multi-region@3.587.0: + resolution: {integrity: sha512-TR9+ZSjdXvXUz54ayHcCihhcvxI9W7102J1OK6MrLgBlPE7uRhAx42BR9L5lLJ86Xj3LuqPWf//o9d/zR9WVIg==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/middleware-sdk-s3': 3.587.0 + '@aws-sdk/types': 3.577.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/signature-v4': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/token-providers@3.587.0(@aws-sdk/client-sso-oidc@3.588.0): + resolution: {integrity: sha512-ULqhbnLy1hmJNRcukANBWJmum3BbjXnurLPSFXoGdV0llXYlG55SzIla2VYqdveQEEjmsBuTZdFvXAtNpmS5Zg==} + engines: {node: '>=16.0.0'} + peerDependencies: + '@aws-sdk/client-sso-oidc': ^3.587.0 + dependencies: + '@aws-sdk/client-sso-oidc': 3.588.0(@aws-sdk/client-sts@3.588.0) + '@aws-sdk/types': 3.577.0 + '@smithy/property-provider': 3.1.0 + '@smithy/shared-ini-file-loader': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/types@3.577.0: + resolution: {integrity: sha512-FT2JZES3wBKN/alfmhlo+3ZOq/XJ0C7QOZcDNrpKjB0kqYoKjhVKZ/Hx6ArR0czkKfHzBBEs6y40ebIHx2nSmA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/util-arn-parser@3.568.0: + resolution: {integrity: sha512-XUKJWWo+KOB7fbnPP0+g/o5Ulku/X53t7i/h+sPHr5xxYTJJ9CYnbToo95mzxe7xWvkLrsNtJ8L+MnNn9INs2w==} + engines: {node: '>=16.0.0'} + dependencies: + tslib: 2.6.2 + dev: true + + /@aws-sdk/util-endpoints@3.587.0: + resolution: {integrity: sha512-8I1HG6Em8wQWqKcRW6m358mqebRVNpL8XrrEoT4In7xqkKkmYtHRNVYP6lcmiQh5pZ/c/FXu8dSchuFIWyEtqQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/types': 3.0.0 + '@smithy/util-endpoints': 2.0.1 + tslib: 2.6.2 + dev: true + + /@aws-sdk/util-locate-window@3.568.0: + resolution: {integrity: sha512-3nh4TINkXYr+H41QaPelCceEB2FXP3fxp93YZXB/kqJvX0U9j0N0Uk45gvsjmEPzG8XxkPEeLIfT2I1M7A6Lig==} + engines: {node: '>=16.0.0'} + dependencies: + tslib: 2.6.2 + dev: true + + /@aws-sdk/util-user-agent-browser@3.577.0: + resolution: {integrity: sha512-zEAzHgR6HWpZOH7xFgeJLc6/CzMcx4nxeQolZxVZoB5pPaJd3CjyRhZN0xXeZB0XIRCWmb4yJBgyiugXLNMkLA==} + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/types': 3.0.0 + bowser: 2.11.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/util-user-agent-node@3.587.0: + resolution: {integrity: sha512-Pnl+DUe/bvnbEEDHP3iVJrOtE3HbFJBPgsD6vJ+ml/+IYk1Eq49jEG+EHZdNTPz3SDG0kbp2+7u41MKYJHR/iQ==} + engines: {node: '>=16.0.0'} + peerDependencies: + aws-crt: '>=1.0.0' + peerDependenciesMeta: + aws-crt: + optional: true + dependencies: + '@aws-sdk/types': 3.577.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@aws-sdk/util-utf8-browser@3.259.0: + resolution: {integrity: sha512-UvFa/vR+e19XookZF8RzFZBrw2EUkQWxiBW0yYQAhvk3C+QVGl0H3ouca8LDBlBfQKXwmW3huo/59H8rwb1wJw==} + dependencies: + tslib: 2.6.2 + dev: true + + /@aws-sdk/xml-builder@3.575.0: + resolution: {integrity: sha512-cWgAwmbFYNCFzPwxL705+lWps0F3ZvOckufd2KKoEZUmtpVw9/txUXNrPySUXSmRTSRhoatIMABNfStWR043bQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@babel/code-frame@7.24.6: + resolution: {integrity: sha512-ZJhac6FkEd1yhG2AHOmfcXG4ceoLltoCVJjN5XsWN9BifBQr+cHJbWi0h68HZuSORq+3WtJ2z0hwF2NG1b5kcA==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/highlight': 7.24.6 + picocolors: 1.0.1 + + /@babel/helper-validator-identifier@7.24.6: + resolution: {integrity: sha512-4yA7s865JHaqUdRbnaxarZREuPTHrjpDT+pXoAZ1yhyo6uFnIEpS8VMu16siFOHDpZNKYv5BObhsB//ycbICyw==} + engines: {node: '>=6.9.0'} + + /@babel/highlight@7.24.6: + resolution: {integrity: sha512-2YnuOp4HAk2BsBrJJvYCbItHx0zWscI1C3zgWkz+wDyD9I7GIVrfnLyrR4Y1VR+7p+chAEcrgRQYZAGIKMV7vQ==} + engines: {node: '>=6.9.0'} + dependencies: + '@babel/helper-validator-identifier': 7.24.6 + chalk: 2.4.2 + js-tokens: 4.0.0 + picocolors: 1.0.1 + + /@cspotcode/source-map-support@0.8.1: + resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==} + engines: {node: '>=12'} + dependencies: + '@jridgewell/trace-mapping': 0.3.9 + dev: true + + /@eslint-community/eslint-utils@4.4.0(eslint@8.57.0): + resolution: {integrity: sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + eslint: ^6.0.0 || ^7.0.0 || >=8.0.0 + dependencies: + eslint: 8.57.0 + eslint-visitor-keys: 3.4.3 + dev: true + + /@eslint-community/regexpp@4.10.1: + resolution: {integrity: sha512-Zm2NGpWELsQAD1xsJzGQpYfvICSsFkEpU0jxBjfdC6uNEWXcHnfs9hScFWtXVDVl+rBQJGrl4g1vcKIejpH9dA==} + engines: {node: ^12.0.0 || ^14.0.0 || >=16.0.0} + dev: true + + /@eslint/eslintrc@2.1.4: + resolution: {integrity: sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dependencies: + ajv: 6.12.6 + debug: 4.3.5(supports-color@9.4.0) + espree: 9.6.1 + globals: 13.24.0 + ignore: 5.3.1 + import-fresh: 3.3.0 + js-yaml: 4.1.0 + minimatch: 3.1.2 + strip-json-comments: 3.1.1 + transitivePeerDependencies: + - supports-color + dev: true + + /@eslint/js@8.57.0: + resolution: {integrity: sha512-Ys+3g2TaW7gADOJzPt83SJtCDhMjndcDMFVQ/Tj9iA1BfJzFKD9mAUXT3OenpuPHbI6P/myECxRJrofUsDx/5g==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dev: true + + /@humanwhocodes/config-array@0.11.14: + resolution: {integrity: sha512-3T8LkOmg45BV5FICb15QQMsyUSWrQ8AygVfC7ZG32zOalnqrilm018ZVCw0eapXux8FtA33q8PSRSstjee3jSg==} + engines: {node: '>=10.10.0'} + dependencies: + '@humanwhocodes/object-schema': 2.0.3 + debug: 4.3.5(supports-color@9.4.0) + minimatch: 3.1.2 + transitivePeerDependencies: + - supports-color + dev: true + + /@humanwhocodes/module-importer@1.0.1: + resolution: {integrity: sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==} + engines: {node: '>=12.22'} + dev: true + + /@humanwhocodes/object-schema@2.0.3: + resolution: {integrity: sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==} + dev: true + + /@inquirer/confirm@3.1.9: + resolution: {integrity: sha512-UF09aejxCi4Xqm6N/jJAiFXArXfi9al52AFaSD+2uIHnhZGtd1d6lIGTRMPouVSJxbGEi+HkOWSYaiEY/+szUw==} + engines: {node: '>=18'} + dependencies: + '@inquirer/core': 8.2.2 + '@inquirer/type': 1.3.3 + dev: true + + /@inquirer/core@8.2.2: + resolution: {integrity: sha512-K8SuNX45jEFlX3EBJpu9B+S2TISzMPGXZIuJ9ME924SqbdW6Pt6fIkKvXg7mOEOKJ4WxpQsxj0UTfcL/A434Ww==} + engines: {node: '>=18'} + dependencies: + '@inquirer/figures': 1.0.3 + '@inquirer/type': 1.3.3 + '@types/mute-stream': 0.0.4 + '@types/node': 20.14.1 + '@types/wrap-ansi': 3.0.0 + ansi-escapes: 4.3.2 + chalk: 4.1.2 + cli-spinners: 2.9.2 + cli-width: 4.1.0 + mute-stream: 1.0.0 + signal-exit: 4.1.0 + strip-ansi: 6.0.1 + wrap-ansi: 6.2.0 + dev: true + + /@inquirer/figures@1.0.3: + resolution: {integrity: sha512-ErXXzENMH5pJt5/ssXV0DfWUZqly8nGzf0UcBV9xTnP+KyffE2mqyxIMBrZ8ijQck2nU0TQm40EQB53YreyWHw==} + engines: {node: '>=18'} + dev: true + + /@inquirer/input@2.1.9: + resolution: {integrity: sha512-1xTCHmIe48x9CG1+8glAHrVVdH+QfYhzgBUbgyoVpp5NovnXgRcjSn/SNulepxf9Ol8HDq3gzw3ZCAUr+h1Eyg==} + engines: {node: '>=18'} + dependencies: + '@inquirer/core': 8.2.2 + '@inquirer/type': 1.3.3 + dev: true + + /@inquirer/select@2.3.5: + resolution: {integrity: sha512-IyBj8oEtmdF2Gx4FJTPtEya37MD6s0KATKsHqgmls0lK7EQbhYSq9GQlcFq6cBsYe/cgQ0Fg2cCqYYPi/d/fxQ==} + engines: {node: '>=18'} + dependencies: + '@inquirer/core': 8.2.2 + '@inquirer/figures': 1.0.3 + '@inquirer/type': 1.3.3 + ansi-escapes: 4.3.2 + chalk: 4.1.2 + dev: true + + /@inquirer/type@1.3.3: + resolution: {integrity: sha512-xTUt0NulylX27/zMx04ZYar/kr1raaiFTVvQ5feljQsiAgdm0WPj4S73/ye0fbslh+15QrIuDvfCXTek7pMY5A==} + engines: {node: '>=18'} + dev: true + + /@jridgewell/resolve-uri@3.1.2: + resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==} + engines: {node: '>=6.0.0'} + dev: true + + /@jridgewell/sourcemap-codec@1.4.15: + resolution: {integrity: sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==} + dev: true + + /@jridgewell/trace-mapping@0.3.9: + resolution: {integrity: sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==} + dependencies: + '@jridgewell/resolve-uri': 3.1.2 + '@jridgewell/sourcemap-codec': 1.4.15 + dev: true + + /@nodelib/fs.scandir@2.1.5: + resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} + engines: {node: '>= 8'} + dependencies: + '@nodelib/fs.stat': 2.0.5 + run-parallel: 1.2.0 + + /@nodelib/fs.stat@2.0.5: + resolution: {integrity: sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==} + engines: {node: '>= 8'} + + /@nodelib/fs.walk@1.2.8: + resolution: {integrity: sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==} + engines: {node: '>= 8'} + dependencies: + '@nodelib/fs.scandir': 2.1.5 + fastq: 1.17.1 + + /@oclif/core@3.26.9: + resolution: {integrity: sha512-yB5Yxd62DsHqqCK/60L8IiGpTRIU4J+fzCqfbPRiIYE5+agfN63kppaM+TbqyMBdsnt/PQOnYD8Bhs1quUr6fg==} + engines: {node: '>=18.0.0'} + dependencies: + '@types/cli-progress': 3.11.5 + ansi-escapes: 4.3.2 + ansi-styles: 4.3.0 + cardinal: 2.1.1 + chalk: 4.1.2 + clean-stack: 3.0.1 + cli-progress: 3.12.0 + color: 4.2.3 + debug: 4.3.5(supports-color@8.1.1) + ejs: 3.1.10 + get-package-type: 0.1.0 + globby: 11.1.0 + hyperlinker: 1.0.0 + indent-string: 4.0.0 + is-wsl: 2.2.0 + js-yaml: 3.14.1 + minimatch: 9.0.4 + natural-orderby: 2.0.3 + object-treeify: 1.1.33 + password-prompt: 1.1.3 + slice-ansi: 4.0.0 + string-width: 4.2.3 + strip-ansi: 6.0.1 + supports-color: 8.1.1 + supports-hyperlinks: 2.3.0 + widest-line: 3.1.0 + wordwrap: 1.0.0 + wrap-ansi: 7.0.0 + + /@oclif/core@4.0.0-beta.13(typescript@5.4.5): + resolution: {integrity: sha512-ug8CZUCJphgetSZVgd4HQgyewlYVGGG1LIeFXGxjgYsjZ/f5I3nSCj7xpAMEDqjVD/lwmSujtVwa7tvEgLGICw==} + engines: {node: '>=18.0.0'} + dependencies: + ansi-escapes: 4.3.2 + ansis: 3.2.0 + clean-stack: 3.0.1 + cli-spinners: 2.9.2 + cosmiconfig: 9.0.0(typescript@5.4.5) + debug: 4.3.5(supports-color@9.4.0) + ejs: 3.1.10 + get-package-type: 0.1.0 + globby: 11.1.0 + indent-string: 4.0.0 + is-wsl: 2.2.0 + minimatch: 9.0.4 + string-width: 4.2.3 + supports-color: 9.4.0 + widest-line: 3.1.0 + wordwrap: 1.0.0 + wrap-ansi: 7.0.0 + transitivePeerDependencies: + - typescript + dev: false + + /@oclif/core@4.0.0-beta.15(typescript@5.4.5): + resolution: {integrity: sha512-pmtZk8IR4xqPrlmUpjZcg/dxI2OtMXpZERd9rmPUGsS9eta7gFBKFUXqxfL5Ey/0jL8sJmsbmHsSADml0ntMOg==} + engines: {node: '>=18.0.0'} + dependencies: + ansi-escapes: 4.3.2 + ansis: 3.2.0 + clean-stack: 3.0.1 + cli-spinners: 2.9.2 + cosmiconfig: 9.0.0(typescript@5.4.5) + debug: 4.3.5(supports-color@8.1.1) + ejs: 3.1.10 + get-package-type: 0.1.0 + globby: 11.1.0 + indent-string: 4.0.0 + is-wsl: 2.2.0 + minimatch: 9.0.4 + string-width: 4.2.3 + supports-color: 8.1.1 + widest-line: 3.1.0 + wordwrap: 1.0.0 + wrap-ansi: 7.0.0 + transitivePeerDependencies: + - typescript + + /@oclif/plugin-help@6.1.0(typescript@5.4.5): + resolution: {integrity: sha512-U+LNKKaZBroHqoSGIpYbQmkTl8aW0NJcO9tr091LJOZ74iqEDiupfBTCMuklW9uTZdWFYApqTmshDuB/I41jxQ==} + engines: {node: '>=18.0.0'} + dependencies: + '@oclif/core': 4.0.0-beta.15(typescript@5.4.5) + transitivePeerDependencies: + - typescript + + /@oclif/plugin-not-found@3.2.1(typescript@5.4.5): + resolution: {integrity: sha512-Y2FyWLEziTrgdQwd4LtZcYv3VOMeDhW0pW6hbE0u3fJdxs8f3JLfOHJPd8FPorVt2ldsaj4zcXUWA6vMQKeKbQ==} + engines: {node: '>=18.0.0'} + dependencies: + '@inquirer/confirm': 3.1.9 + '@oclif/core': 4.0.0-beta.15(typescript@5.4.5) + ansis: 3.2.0 + fast-levenshtein: 3.0.0 + transitivePeerDependencies: + - typescript + dev: true + + /@oclif/plugin-plugins@5.2.1(typescript@5.4.5): + resolution: {integrity: sha512-Qk0+7KIYK6cQz116ZyY25XRwniKuAAGKTvY2J91CDfyri0RYUOSDNOzz0m18IG2M51Tgiv8UNY74A3PIhQaEcw==} + engines: {node: '>=18.0.0'} + dependencies: + '@oclif/core': 4.0.0-beta.13(typescript@5.4.5) + ansis: 3.2.0 + debug: 4.3.5(supports-color@9.4.0) + npm: 10.8.1 + npm-package-arg: 11.0.2 + npm-run-path: 5.3.0 + object-treeify: 4.0.1 + semver: 7.6.2 + validate-npm-package-name: 5.0.1 + which: 4.0.0 + yarn: 1.22.22 + transitivePeerDependencies: + - supports-color + - typescript + dev: false + + /@oclif/plugin-warn-if-update-available@3.1.3(typescript@5.4.5): + resolution: {integrity: sha512-xdnGWWYIhzZkkXh3sIeX8b0dwc83A7N95GvtiLdiY5veXgwNRXLf+3dNn8p/kAFKlhdh0pIQGeD7jmRYu74Q6Q==} + engines: {node: '>=18.0.0'} + dependencies: + '@oclif/core': 4.0.0-beta.15(typescript@5.4.5) + ansis: 3.2.0 + debug: 4.3.5(supports-color@9.4.0) + http-call: 5.3.0 + lodash: 4.17.21 + transitivePeerDependencies: + - supports-color + - typescript + dev: true + + /@oclif/prettier-config@0.2.1: + resolution: {integrity: sha512-XB8kwQj8zynXjIIWRm+6gO/r8Qft2xKtwBMSmq1JRqtA6TpwpqECqiu8LosBCyg2JBXuUy2lU23/L98KIR7FrQ==} + dev: true + + /@oclif/test@4.0.3(@oclif/core@3.26.9): + resolution: {integrity: sha512-LxcRYVFTUHoOW2Koo1lmbEwl/4HRFIdNWXuUY1/PHEawjwLvp3xwVe2rOWGqYD+vlHr+TYUw2QDQc8e2vUTDrw==} + engines: {node: '>=18.0.0'} + peerDependencies: + '@oclif/core': '>= 3.0.0' + dependencies: + '@oclif/core': 3.26.9 + ansis: 3.2.0 + debug: 4.3.5(supports-color@9.4.0) + transitivePeerDependencies: + - supports-color + dev: true + + /@sindresorhus/is@5.6.0: + resolution: {integrity: sha512-TV7t8GKYaJWsn00tFDqBw8+Uqmr8A0fRU1tvTQhyZzGv0sJCGRQL3JGMI3ucuKo3XIZdUP+Lx7/gh2t3lewy7g==} + engines: {node: '>=14.16'} + dev: true + + /@smithy/abort-controller@3.0.0: + resolution: {integrity: sha512-p6GlFGBt9K4MYLu72YuJ523NVR4A8oHlC5M2JO6OmQqN8kAc/uh1JqLE+FizTokrSJGg0CSvC+BrsmGzKtsZKA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/chunked-blob-reader-native@3.0.0: + resolution: {integrity: sha512-VDkpCYW+peSuM4zJip5WDfqvg2Mo/e8yxOv3VF1m11y7B8KKMKVFtmZWDe36Fvk8rGuWrPZHHXZ7rR7uM5yWyg==} + dependencies: + '@smithy/util-base64': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/chunked-blob-reader@3.0.0: + resolution: {integrity: sha512-sbnURCwjF0gSToGlsBiAmd1lRCmSn72nu9axfJu5lIx6RUEgHu6GwTMbqCdhQSi0Pumcm5vFxsi9XWXb2mTaoA==} + dependencies: + tslib: 2.6.2 + dev: true + + /@smithy/config-resolver@3.0.1: + resolution: {integrity: sha512-hbkYJc20SBDz2qqLzttjI/EqXemtmWk0ooRznLsiXp3066KQRTvuKHa7U4jCZCJq6Dozqvy0R1/vNESC9inPJg==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/node-config-provider': 3.1.0 + '@smithy/types': 3.0.0 + '@smithy/util-config-provider': 3.0.0 + '@smithy/util-middleware': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/core@2.1.1: + resolution: {integrity: sha512-0vbIwwUcg0FMhTVJgMhbsRSAFL0rwduy/OQz7Xq1pJXJOyaGv+PGjj1iGawRlzBUPA5BkJv7S6q+YU2U8gk/WA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/middleware-endpoint': 3.0.1 + '@smithy/middleware-retry': 3.0.3 + '@smithy/middleware-serde': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/util-middleware': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/credential-provider-imds@3.1.0: + resolution: {integrity: sha512-q4A4d38v8pYYmseu/jTS3Z5I3zXlEOe5Obi+EJreVKgSVyWUHOd7/yaVCinC60QG4MRyCs98tcxBH1IMC0bu7Q==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/node-config-provider': 3.1.0 + '@smithy/property-provider': 3.1.0 + '@smithy/types': 3.0.0 + '@smithy/url-parser': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/eventstream-codec@3.0.0: + resolution: {integrity: sha512-PUtyEA0Oik50SaEFCZ0WPVtF9tz/teze2fDptW6WRXl+RrEenH8UbEjudOz8iakiMl3lE3lCVqYf2Y+znL8QFQ==} + dependencies: + '@aws-crypto/crc32': 3.0.0 + '@smithy/types': 3.0.0 + '@smithy/util-hex-encoding': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/eventstream-serde-browser@3.0.0: + resolution: {integrity: sha512-NB7AFiPN4NxP/YCAnrvYR18z2/ZsiHiF7VtG30gshO9GbFrIb1rC8ep4NGpJSWrz6P64uhPXeo4M0UsCLnZKqw==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/eventstream-serde-universal': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/eventstream-serde-config-resolver@3.0.0: + resolution: {integrity: sha512-RUQG3vQ3LX7peqqHAbmayhgrF5aTilPnazinaSGF1P0+tgM3vvIRWPHmlLIz2qFqB9LqFIxditxc8O2Z6psrRw==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/eventstream-serde-node@3.0.0: + resolution: {integrity: sha512-baRPdMBDMBExZXIUAoPGm/hntixjt/VFpU6+VmCyiYJYzRHRxoaI1MN+5XE+hIS8AJ2GCHLMFEIOLzq9xx1EgQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/eventstream-serde-universal': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/eventstream-serde-universal@3.0.0: + resolution: {integrity: sha512-HNFfShmotWGeAoW4ujP8meV9BZavcpmerDbPIjkJbxKbN8RsUcpRQ/2OyIxWNxXNH2GWCAxuSB7ynmIGJlQ3Dw==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/eventstream-codec': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/fetch-http-handler@3.0.1: + resolution: {integrity: sha512-uaH74i5BDj+rBwoQaXioKpI0SHBJFtOVwzrCpxZxphOW0ki5jhj7dXvDMYM2IJem8TpdFvS2iC08sjOblfFGFg==} + dependencies: + '@smithy/protocol-http': 4.0.0 + '@smithy/querystring-builder': 3.0.0 + '@smithy/types': 3.0.0 + '@smithy/util-base64': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/hash-blob-browser@3.0.0: + resolution: {integrity: sha512-/Wbpdg+bwJvW7lxR/zpWAc1/x/YkcqguuF2bAzkJrvXriZu1vm8r+PUdE4syiVwQg7PPR2dXpi3CLBb9qRDaVQ==} + dependencies: + '@smithy/chunked-blob-reader': 3.0.0 + '@smithy/chunked-blob-reader-native': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/hash-node@3.0.0: + resolution: {integrity: sha512-84qXstNemP3XS5jcof0el6+bDfjzuvhJPQTEfro3lgtbCtKgzPm3MgiS6ehXVPjeQ5+JS0HqmTz8f/RYfzHVxw==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + '@smithy/util-buffer-from': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/hash-stream-node@3.0.0: + resolution: {integrity: sha512-J0i7de+EgXDEGITD4fxzmMX8CyCNETTIRXlxjMiNUvvu76Xn3GJ31wQR85ynlPk2wI1lqoknAFJaD1fiNDlbIA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/invalid-dependency@3.0.0: + resolution: {integrity: sha512-F6wBBaEFgJzj0s4KUlliIGPmqXemwP6EavgvDqYwCH40O5Xr2iMHvS8todmGVZtuJCorBkXsYLyTu4PuizVq5g==} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/is-array-buffer@3.0.0: + resolution: {integrity: sha512-+Fsu6Q6C4RSJiy81Y8eApjEB5gVtM+oFKTffg+jSuwtvomJJrhUJBu2zS8wjXSgH/g1MKEWrzyChTBe6clb5FQ==} + engines: {node: '>=16.0.0'} + dependencies: + tslib: 2.6.2 + dev: true + + /@smithy/md5-js@3.0.0: + resolution: {integrity: sha512-Tm0vrrVzjlD+6RCQTx7D3Ls58S3FUH1ZCtU1MIh/qQmaOo1H9lMN2as6CikcEwgattnA9SURSdoJJ27xMcEfMA==} + dependencies: + '@smithy/types': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/middleware-content-length@3.0.0: + resolution: {integrity: sha512-3C4s4d/iGobgCtk2tnWW6+zSTOBg1PRAm2vtWZLdriwTroFbbWNSr3lcyzHdrQHnEXYCC5K52EbpfodaIUY8sg==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/protocol-http': 4.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/middleware-endpoint@3.0.1: + resolution: {integrity: sha512-lQ/UOdGD4KM5kLZiAl0q8Qy3dPbynvAXKAdXnYlrA1OpaUwr+neSsVokDZpY6ZVb5Yx8jnus29uv6XWpM9P4SQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/middleware-serde': 3.0.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/shared-ini-file-loader': 3.1.0 + '@smithy/types': 3.0.0 + '@smithy/url-parser': 3.0.0 + '@smithy/util-middleware': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/middleware-retry@3.0.3: + resolution: {integrity: sha512-Wve1qzJb83VEU/6q+/I0cQdAkDnuzELC6IvIBwDzUEiGpKqXgX1v10FUuZGbRS6Ov/P+HHthcAoHOJZQvZNAkA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/node-config-provider': 3.1.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/service-error-classification': 3.0.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + '@smithy/util-middleware': 3.0.0 + '@smithy/util-retry': 3.0.0 + tslib: 2.6.2 + uuid: 9.0.1 + dev: true + + /@smithy/middleware-serde@3.0.0: + resolution: {integrity: sha512-I1vKG1foI+oPgG9r7IMY1S+xBnmAn1ISqployvqkwHoSb8VPsngHDTOgYGYBonuOKndaWRUGJZrKYYLB+Ane6w==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/middleware-stack@3.0.0: + resolution: {integrity: sha512-+H0jmyfAyHRFXm6wunskuNAqtj7yfmwFB6Fp37enytp2q047/Od9xetEaUbluyImOlGnGpaVGaVfjwawSr+i6Q==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/node-config-provider@3.1.0: + resolution: {integrity: sha512-ngfB8QItUfTFTfHMvKuc2g1W60V1urIgZHqD1JNFZC2tTWXahqf2XvKXqcBS7yZqR7GqkQQZy11y/lNOUWzq7Q==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/property-provider': 3.1.0 + '@smithy/shared-ini-file-loader': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/node-http-handler@3.0.0: + resolution: {integrity: sha512-3trD4r7NOMygwLbUJo4eodyQuypAWr7uvPnebNJ9a70dQhVn+US8j/lCnvoJS6BXfZeF7PkkkI0DemVJw+n+eQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/abort-controller': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/querystring-builder': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/property-provider@3.1.0: + resolution: {integrity: sha512-Tj3+oVhqdZgemjCiWjFlADfhvLF4C/uKDuKo7/tlEsRQ9+3emCreR2xndj970QSRSsiCEU8hZW3/8JQu+n5w4Q==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/protocol-http@4.0.0: + resolution: {integrity: sha512-qOQZOEI2XLWRWBO9AgIYuHuqjZ2csyr8/IlgFDHDNuIgLAMRx2Bl8ck5U5D6Vh9DPdoaVpuzwWMa0xcdL4O/AQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/querystring-builder@3.0.0: + resolution: {integrity: sha512-bW8Fi0NzyfkE0TmQphDXr1AmBDbK01cA4C1Z7ggwMAU5RDz5AAv/KmoRwzQAS0kxXNf/D2ALTEgwK0U2c4LtRg==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + '@smithy/util-uri-escape': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/querystring-parser@3.0.0: + resolution: {integrity: sha512-UzHwthk0UEccV4dHzPySnBy34AWw3V9lIqUTxmozQ+wPDAO9csCWMfOLe7V9A2agNYy7xE+Pb0S6K/J23JSzfQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/service-error-classification@3.0.0: + resolution: {integrity: sha512-3BsBtOUt2Gsnc3X23ew+r2M71WwtpHfEDGhHYHSDg6q1t8FrWh15jT25DLajFV1H+PpxAJ6gqe9yYeRUsmSdFA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + dev: true + + /@smithy/shared-ini-file-loader@3.1.0: + resolution: {integrity: sha512-dAM7wSX0NR3qTNyGVN/nwwpEDzfV9T/3AN2eABExWmda5VqZKSsjlINqomO5hjQWGv+IIkoXfs3u2vGSNz8+Rg==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/signature-v4@3.0.0: + resolution: {integrity: sha512-kXFOkNX+BQHe2qnLxpMEaCRGap9J6tUGLzc3A9jdn+nD4JdMwCKTJ+zFwQ20GkY+mAXGatyTw3HcoUlR39HwmA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/is-array-buffer': 3.0.0 + '@smithy/types': 3.0.0 + '@smithy/util-hex-encoding': 3.0.0 + '@smithy/util-middleware': 3.0.0 + '@smithy/util-uri-escape': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/smithy-client@3.1.1: + resolution: {integrity: sha512-tj4Ku7MpzZR8cmVuPcSbrLFVxmptWktmJMwST/uIEq4sarabEdF8CbmQdYB7uJ/X51Qq2EYwnRsoS7hdR4B7rA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/middleware-endpoint': 3.0.1 + '@smithy/middleware-stack': 3.0.0 + '@smithy/protocol-http': 4.0.0 + '@smithy/types': 3.0.0 + '@smithy/util-stream': 3.0.1 + tslib: 2.6.2 + dev: true + + /@smithy/types@3.0.0: + resolution: {integrity: sha512-VvWuQk2RKFuOr98gFhjca7fkBS+xLLURT8bUjk5XQoV0ZLm7WPwWPPY3/AwzTLuUBDeoKDCthfe1AsTUWaSEhw==} + engines: {node: '>=16.0.0'} + dependencies: + tslib: 2.6.2 + dev: true + + /@smithy/url-parser@3.0.0: + resolution: {integrity: sha512-2XLazFgUu+YOGHtWihB3FSLAfCUajVfNBXGGYjOaVKjLAuAxx3pSBY3hBgLzIgB17haf59gOG3imKqTy8mcrjw==} + dependencies: + '@smithy/querystring-parser': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-base64@3.0.0: + resolution: {integrity: sha512-Kxvoh5Qtt0CDsfajiZOCpJxgtPHXOKwmM+Zy4waD43UoEMA+qPxxa98aE/7ZhdnBFZFXMOiBR5xbcaMhLtznQQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/util-buffer-from': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-body-length-browser@3.0.0: + resolution: {integrity: sha512-cbjJs2A1mLYmqmyVl80uoLTJhAcfzMOyPgjwAYusWKMdLeNtzmMz9YxNl3/jRLoxSS3wkqkf0jwNdtXWtyEBaQ==} + dependencies: + tslib: 2.6.2 + dev: true + + /@smithy/util-body-length-node@3.0.0: + resolution: {integrity: sha512-Tj7pZ4bUloNUP6PzwhN7K386tmSmEET9QtQg0TgdNOnxhZvCssHji+oZTUIuzxECRfG8rdm2PMw2WCFs6eIYkA==} + engines: {node: '>=16.0.0'} + dependencies: + tslib: 2.6.2 + dev: true + + /@smithy/util-buffer-from@3.0.0: + resolution: {integrity: sha512-aEOHCgq5RWFbP+UDPvPot26EJHjOC+bRgse5A8V3FSShqd5E5UN4qc7zkwsvJPPAVsf73QwYcHN1/gt/rtLwQA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/is-array-buffer': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-config-provider@3.0.0: + resolution: {integrity: sha512-pbjk4s0fwq3Di/ANL+rCvJMKM5bzAQdE5S/6RL5NXgMExFAi6UgQMPOm5yPaIWPpr+EOXKXRonJ3FoxKf4mCJQ==} + engines: {node: '>=16.0.0'} + dependencies: + tslib: 2.6.2 + dev: true + + /@smithy/util-defaults-mode-browser@3.0.3: + resolution: {integrity: sha512-3DFON2bvXJAukJe+qFgPV/rorG7ZD3m4gjCXHD1V5z/tgKQp5MCTCLntrd686tX6tj8Uli3lefWXJudNg5WmCA==} + engines: {node: '>= 10.0.0'} + dependencies: + '@smithy/property-provider': 3.1.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + bowser: 2.11.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-defaults-mode-node@3.0.3: + resolution: {integrity: sha512-D0b8GJXecT00baoSQ3Iieu3k3mZ7GY8w1zmg8pdogYrGvWJeLcIclqk2gbkG4K0DaBGWrO6v6r20iwIFfDYrmA==} + engines: {node: '>= 10.0.0'} + dependencies: + '@smithy/config-resolver': 3.0.1 + '@smithy/credential-provider-imds': 3.1.0 + '@smithy/node-config-provider': 3.1.0 + '@smithy/property-provider': 3.1.0 + '@smithy/smithy-client': 3.1.1 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-endpoints@2.0.1: + resolution: {integrity: sha512-ZRT0VCOnKlVohfoABMc8lWeQo/JEFuPWctfNRXgTHbyOVssMOLYFUNWukxxiHRGVAhV+n3c0kPW+zUqckjVPEA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/node-config-provider': 3.1.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-hex-encoding@3.0.0: + resolution: {integrity: sha512-eFndh1WEK5YMUYvy3lPlVmYY/fZcQE1D8oSf41Id2vCeIkKJXPcYDCZD+4+xViI6b1XSd7tE+s5AmXzz5ilabQ==} + engines: {node: '>=16.0.0'} + dependencies: + tslib: 2.6.2 + dev: true + + /@smithy/util-middleware@3.0.0: + resolution: {integrity: sha512-q5ITdOnV2pXHSVDnKWrwgSNTDBAMHLptFE07ua/5Ty5WJ11bvr0vk2a7agu7qRhrCFRQlno5u3CneU5EELK+DQ==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-retry@3.0.0: + resolution: {integrity: sha512-nK99bvJiziGv/UOKJlDvFF45F00WgPLKVIGUfAK+mDhzVN2hb/S33uW2Tlhg5PVBoqY7tDVqL0zmu4OxAHgo9g==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/service-error-classification': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-stream@3.0.1: + resolution: {integrity: sha512-7F7VNNhAsfMRA8I986YdOY5fE0/T1/ZjFF6OLsqkvQVNP3vZ/szYDfGCyphb7ioA09r32K/0qbSFfNFU68aSzA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/fetch-http-handler': 3.0.1 + '@smithy/node-http-handler': 3.0.0 + '@smithy/types': 3.0.0 + '@smithy/util-base64': 3.0.0 + '@smithy/util-buffer-from': 3.0.0 + '@smithy/util-hex-encoding': 3.0.0 + '@smithy/util-utf8': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-uri-escape@3.0.0: + resolution: {integrity: sha512-LqR7qYLgZTD7nWLBecUi4aqolw8Mhza9ArpNEQ881MJJIU2sE5iHCK6TdyqqzcDLy0OPe10IY4T8ctVdtynubg==} + engines: {node: '>=16.0.0'} + dependencies: + tslib: 2.6.2 + dev: true + + /@smithy/util-utf8@3.0.0: + resolution: {integrity: sha512-rUeT12bxFnplYDe815GXbq/oixEGHfRFFtcTF3YdDi/JaENIM6aSYYLJydG83UNzLXeRI5K8abYd/8Sp/QM0kA==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/util-buffer-from': 3.0.0 + tslib: 2.6.2 + dev: true + + /@smithy/util-waiter@3.0.0: + resolution: {integrity: sha512-+fEXJxGDLCoqRKVSmo0auGxaqbiCo+8oph+4auefYjaNxjOLKSY2MxVQfRzo65PaZv4fr+5lWg+au7vSuJJ/zw==} + engines: {node: '>=16.0.0'} + dependencies: + '@smithy/abort-controller': 3.0.0 + '@smithy/types': 3.0.0 + tslib: 2.6.2 + dev: true + + /@szmarczak/http-timer@5.0.1: + resolution: {integrity: sha512-+PmQX0PiAYPMeVYe237LJAYvOMYW1j2rH5YROyS3b4CTVJum34HfRvKvAzozHAQG0TnHNdUfY9nCeUyRAs//cw==} + engines: {node: '>=14.16'} + dependencies: + defer-to-connect: 2.0.1 + dev: true + + /@tsconfig/node10@1.0.11: + resolution: {integrity: sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==} + dev: true + + /@tsconfig/node12@1.0.11: + resolution: {integrity: sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==} + dev: true + + /@tsconfig/node14@1.0.3: + resolution: {integrity: sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==} + dev: true + + /@tsconfig/node16@1.0.4: + resolution: {integrity: sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==} + dev: true + + /@types/chai@4.3.16: + resolution: {integrity: sha512-PatH4iOdyh3MyWtmHVFXLWCCIhUbopaltqddG9BzB+gMIzee2MJrvd+jouii9Z3wzQJruGWAm7WOMjgfG8hQlQ==} + dev: true + + /@types/cli-progress@3.11.5: + resolution: {integrity: sha512-D4PbNRbviKyppS5ivBGyFO29POlySLmA2HyUFE4p5QGazAMM3CwkKWcvTl8gvElSuxRh6FPKL8XmidX873ou4g==} + dependencies: + '@types/node': 18.19.34 + + /@types/http-cache-semantics@4.0.4: + resolution: {integrity: sha512-1m0bIFVc7eJWyve9S0RnuRgcQqF/Xd5QsUZAZeQFr1Q3/p9JWoQQEqmVy+DPTNpGXwhgIetAoYF8JSc33q29QA==} + dev: true + + /@types/json-schema@7.0.15: + resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} + dev: true + + /@types/json5@0.0.29: + resolution: {integrity: sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==} + dev: true + + /@types/mocha@10.0.6: + resolution: {integrity: sha512-dJvrYWxP/UcXm36Qn36fxhUKu8A/xMRXVT2cliFF1Z7UA9liG5Psj3ezNSZw+5puH2czDXRLcXQxf8JbJt0ejg==} + dev: true + + /@types/mute-stream@0.0.4: + resolution: {integrity: sha512-CPM9nzrCPPJHQNA9keH9CVkVI+WR5kMa+7XEs5jcGQ0VoAGnLv242w8lIVgwAEfmE4oufJRaTc9PNLQl0ioAow==} + dependencies: + '@types/node': 18.19.34 + dev: true + + /@types/node@18.19.34: + resolution: {integrity: sha512-eXF4pfBNV5DAMKGbI02NnDtWrQ40hAN558/2vvS4gMpMIxaf6JmD7YjnZbq0Q9TDSSkKBamime8ewRoomHdt4g==} + dependencies: + undici-types: 5.26.5 + + /@types/node@20.14.1: + resolution: {integrity: sha512-T2MzSGEu+ysB/FkWfqmhV3PLyQlowdptmmgD20C6QxsS8Fmv5SjpZ1ayXaEC0S21/h5UJ9iA6W/5vSNU5l00OA==} + dependencies: + undici-types: 5.26.5 + dev: true + + /@types/normalize-package-data@2.4.4: + resolution: {integrity: sha512-37i+OaWTh9qeK4LSHPsyRC7NahnGotNuZvjLSgcPzblpHB3rrCJxAOgI5gCdKm7coonsaX1Of0ILiTcnZjbfxA==} + dev: true + + /@types/semver@7.5.8: + resolution: {integrity: sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==} + dev: true + + /@types/wrap-ansi@3.0.0: + resolution: {integrity: sha512-ltIpx+kM7g/MLRZfkbL7EsCEjfzCcScLpkg37eXEtx5kmrAKBkTJwd1GIAjDSL8wTpM6Hzn5YO4pSb91BEwu1g==} + dev: true + + /@typescript-eslint/eslint-plugin@6.21.0(@typescript-eslint/parser@6.21.0)(eslint@8.57.0)(typescript@5.4.5): + resolution: {integrity: sha512-oy9+hTPCUFpngkEZUSzbf9MxI65wbKFoQYsgPdILTfbUldp5ovUuphZVe4i30emU9M/kP+T64Di0mxl7dSw3MA==} + engines: {node: ^16.0.0 || >=18.0.0} + peerDependencies: + '@typescript-eslint/parser': ^6.0.0 || ^6.0.0-alpha + eslint: ^7.0.0 || ^8.0.0 + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@eslint-community/regexpp': 4.10.1 + '@typescript-eslint/parser': 6.21.0(eslint@8.57.0)(typescript@5.4.5) + '@typescript-eslint/scope-manager': 6.21.0 + '@typescript-eslint/type-utils': 6.21.0(eslint@8.57.0)(typescript@5.4.5) + '@typescript-eslint/utils': 6.21.0(eslint@8.57.0)(typescript@5.4.5) + '@typescript-eslint/visitor-keys': 6.21.0 + debug: 4.3.5(supports-color@9.4.0) + eslint: 8.57.0 + graphemer: 1.4.0 + ignore: 5.3.1 + natural-compare: 1.4.0 + semver: 7.6.2 + ts-api-utils: 1.3.0(typescript@5.4.5) + typescript: 5.4.5 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/parser@6.21.0(eslint@8.57.0)(typescript@5.4.5): + resolution: {integrity: sha512-tbsV1jPne5CkFQCgPBcDOt30ItF7aJoZL997JSF7MhGQqOeT3svWRYxiqlfA5RUdlHN6Fi+EI9bxqbdyAUZjYQ==} + engines: {node: ^16.0.0 || >=18.0.0} + peerDependencies: + eslint: ^7.0.0 || ^8.0.0 + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@typescript-eslint/scope-manager': 6.21.0 + '@typescript-eslint/types': 6.21.0 + '@typescript-eslint/typescript-estree': 6.21.0(typescript@5.4.5) + '@typescript-eslint/visitor-keys': 6.21.0 + debug: 4.3.5(supports-color@9.4.0) + eslint: 8.57.0 + typescript: 5.4.5 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/scope-manager@6.21.0: + resolution: {integrity: sha512-OwLUIWZJry80O99zvqXVEioyniJMa+d2GrqpUTqi5/v5D5rOrppJVBPa0yKCblcigC0/aYAzxxqQ1B+DS2RYsg==} + engines: {node: ^16.0.0 || >=18.0.0} + dependencies: + '@typescript-eslint/types': 6.21.0 + '@typescript-eslint/visitor-keys': 6.21.0 + dev: true + + /@typescript-eslint/scope-manager@7.12.0: + resolution: {integrity: sha512-itF1pTnN6F3unPak+kutH9raIkL3lhH1YRPGgt7QQOh43DQKVJXmWkpb+vpc/TiDHs6RSd9CTbDsc/Y+Ygq7kg==} + engines: {node: ^18.18.0 || >=20.0.0} + dependencies: + '@typescript-eslint/types': 7.12.0 + '@typescript-eslint/visitor-keys': 7.12.0 + dev: true + + /@typescript-eslint/type-utils@6.21.0(eslint@8.57.0)(typescript@5.4.5): + resolution: {integrity: sha512-rZQI7wHfao8qMX3Rd3xqeYSMCL3SoiSQLBATSiVKARdFGCYSRvmViieZjqc58jKgs8Y8i9YvVVhRbHSTA4VBag==} + engines: {node: ^16.0.0 || >=18.0.0} + peerDependencies: + eslint: ^7.0.0 || ^8.0.0 + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@typescript-eslint/typescript-estree': 6.21.0(typescript@5.4.5) + '@typescript-eslint/utils': 6.21.0(eslint@8.57.0)(typescript@5.4.5) + debug: 4.3.5(supports-color@9.4.0) + eslint: 8.57.0 + ts-api-utils: 1.3.0(typescript@5.4.5) + typescript: 5.4.5 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/types@6.21.0: + resolution: {integrity: sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==} + engines: {node: ^16.0.0 || >=18.0.0} + dev: true + + /@typescript-eslint/types@7.12.0: + resolution: {integrity: sha512-o+0Te6eWp2ppKY3mLCU+YA9pVJxhUJE15FV7kxuD9jgwIAa+w/ycGJBMrYDTpVGUM/tgpa9SeMOugSabWFq7bg==} + engines: {node: ^18.18.0 || >=20.0.0} + dev: true + + /@typescript-eslint/typescript-estree@6.21.0(typescript@5.4.5): + resolution: {integrity: sha512-6npJTkZcO+y2/kr+z0hc4HwNfrrP4kNYh57ek7yCNlrBjWQ1Y0OS7jiZTkgumrvkX5HkEKXFZkkdFNkaW2wmUQ==} + engines: {node: ^16.0.0 || >=18.0.0} + peerDependencies: + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@typescript-eslint/types': 6.21.0 + '@typescript-eslint/visitor-keys': 6.21.0 + debug: 4.3.5(supports-color@9.4.0) + globby: 11.1.0 + is-glob: 4.0.3 + minimatch: 9.0.3 + semver: 7.6.2 + ts-api-utils: 1.3.0(typescript@5.4.5) + typescript: 5.4.5 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/typescript-estree@7.12.0(typescript@5.4.5): + resolution: {integrity: sha512-5bwqLsWBULv1h6pn7cMW5dXX/Y2amRqLaKqsASVwbBHMZSnHqE/HN4vT4fE0aFsiwxYvr98kqOWh1a8ZKXalCQ==} + engines: {node: ^18.18.0 || >=20.0.0} + peerDependencies: + typescript: '*' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + '@typescript-eslint/types': 7.12.0 + '@typescript-eslint/visitor-keys': 7.12.0 + debug: 4.3.5(supports-color@9.4.0) + globby: 11.1.0 + is-glob: 4.0.3 + minimatch: 9.0.4 + semver: 7.6.2 + ts-api-utils: 1.3.0(typescript@5.4.5) + typescript: 5.4.5 + transitivePeerDependencies: + - supports-color + dev: true + + /@typescript-eslint/utils@6.21.0(eslint@8.57.0)(typescript@5.4.5): + resolution: {integrity: sha512-NfWVaC8HP9T8cbKQxHcsJBY5YE1O33+jpMwN45qzWWaPDZgLIbo12toGMWnmhvCpd3sIxkpDw3Wv1B3dYrbDQQ==} + engines: {node: ^16.0.0 || >=18.0.0} + peerDependencies: + eslint: ^7.0.0 || ^8.0.0 + dependencies: + '@eslint-community/eslint-utils': 4.4.0(eslint@8.57.0) + '@types/json-schema': 7.0.15 + '@types/semver': 7.5.8 + '@typescript-eslint/scope-manager': 6.21.0 + '@typescript-eslint/types': 6.21.0 + '@typescript-eslint/typescript-estree': 6.21.0(typescript@5.4.5) + eslint: 8.57.0 + semver: 7.6.2 + transitivePeerDependencies: + - supports-color + - typescript + dev: true + + /@typescript-eslint/utils@7.12.0(eslint@8.57.0)(typescript@5.4.5): + resolution: {integrity: sha512-Y6hhwxwDx41HNpjuYswYp6gDbkiZ8Hin9Bf5aJQn1bpTs3afYY4GX+MPYxma8jtoIV2GRwTM/UJm/2uGCVv+DQ==} + engines: {node: ^18.18.0 || >=20.0.0} + peerDependencies: + eslint: ^8.56.0 + dependencies: + '@eslint-community/eslint-utils': 4.4.0(eslint@8.57.0) + '@typescript-eslint/scope-manager': 7.12.0 + '@typescript-eslint/types': 7.12.0 + '@typescript-eslint/typescript-estree': 7.12.0(typescript@5.4.5) + eslint: 8.57.0 + transitivePeerDependencies: + - supports-color + - typescript + dev: true + + /@typescript-eslint/visitor-keys@6.21.0: + resolution: {integrity: sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==} + engines: {node: ^16.0.0 || >=18.0.0} + dependencies: + '@typescript-eslint/types': 6.21.0 + eslint-visitor-keys: 3.4.3 + dev: true + + /@typescript-eslint/visitor-keys@7.12.0: + resolution: {integrity: sha512-uZk7DevrQLL3vSnfFl5bj4sL75qC9D6EdjemIdbtkuUmIheWpuiiylSY01JxJE7+zGrOWDZrp1WxOuDntvKrHQ==} + engines: {node: ^18.18.0 || >=20.0.0} + dependencies: + '@typescript-eslint/types': 7.12.0 + eslint-visitor-keys: 3.4.3 + dev: true + + /@ungap/structured-clone@1.2.0: + resolution: {integrity: sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==} + dev: true + + /acorn-jsx@5.3.2(acorn@8.11.3): + resolution: {integrity: sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==} + peerDependencies: + acorn: ^6.0.0 || ^7.0.0 || ^8.0.0 + dependencies: + acorn: 8.11.3 + dev: true + + /acorn-walk@8.3.2: + resolution: {integrity: sha512-cjkyv4OtNCIeqhHrfS81QWXoCBPExR/J62oyEqepVw8WaQeSqpW2uhuLPh1m9eWhDuOo/jUXVTlifvesOWp/4A==} + engines: {node: '>=0.4.0'} + dev: true + + /acorn@8.11.3: + resolution: {integrity: sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==} + engines: {node: '>=0.4.0'} + hasBin: true + dev: true + + /ajv@6.12.6: + resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} + dependencies: + fast-deep-equal: 3.1.3 + fast-json-stable-stringify: 2.1.0 + json-schema-traverse: 0.4.1 + uri-js: 4.4.1 + dev: true + + /ansi-colors@4.1.1: + resolution: {integrity: sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==} + engines: {node: '>=6'} + dev: true + + /ansi-escapes@4.3.2: + resolution: {integrity: sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==} + engines: {node: '>=8'} + dependencies: + type-fest: 0.21.3 + + /ansi-regex@5.0.1: + resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} + engines: {node: '>=8'} + + /ansi-styles@3.2.1: + resolution: {integrity: sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==} + engines: {node: '>=4'} + dependencies: + color-convert: 1.9.3 + + /ansi-styles@4.3.0: + resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} + engines: {node: '>=8'} + dependencies: + color-convert: 2.0.1 + + /ansicolors@0.3.2: + resolution: {integrity: sha512-QXu7BPrP29VllRxH8GwB7x5iX5qWKAAMLqKQGWTeLWVlNHNOpVMJ91dsxQAIWXpjuW5wqvxu3Jd/nRjrJ+0pqg==} + + /ansis@3.2.0: + resolution: {integrity: sha512-Yk3BkHH9U7oPyCN3gL5Tc7CpahG/+UFv/6UG03C311Vy9lzRmA5uoxDTpU9CO3rGHL6KzJz/pdDeXZCZ5Mu/Sg==} + engines: {node: '>=15'} + + /anymatch@3.1.3: + resolution: {integrity: sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==} + engines: {node: '>= 8'} + dependencies: + normalize-path: 3.0.0 + picomatch: 2.3.1 + dev: true + + /arg@4.1.3: + resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==} + dev: true + + /argparse@1.0.10: + resolution: {integrity: sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==} + dependencies: + sprintf-js: 1.0.3 + + /argparse@2.0.1: + resolution: {integrity: sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==} + + /array-buffer-byte-length@1.0.1: + resolution: {integrity: sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + is-array-buffer: 3.0.4 + dev: true + + /array-includes@3.1.8: + resolution: {integrity: sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + es-object-atoms: 1.0.0 + get-intrinsic: 1.2.4 + is-string: 1.0.7 + dev: true + + /array-union@2.1.0: + resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} + engines: {node: '>=8'} + + /array.prototype.findlastindex@1.2.5: + resolution: {integrity: sha512-zfETvRFA8o7EiNn++N5f/kaCw221hrpGsDmcpndVupkPzEc1Wuf3VgC0qby1BbHs7f5DVYjgtEU2LLh5bqeGfQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + es-errors: 1.3.0 + es-object-atoms: 1.0.0 + es-shim-unscopables: 1.0.2 + dev: true + + /array.prototype.flat@1.3.2: + resolution: {integrity: sha512-djYB+Zx2vLewY8RWlNCUdHjDXs2XOgm602S9E7P/UpHgfeHL00cRiIF+IN/G/aUJ7kGPb6yO/ErDI5V2s8iycA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + es-shim-unscopables: 1.0.2 + dev: true + + /array.prototype.flatmap@1.3.2: + resolution: {integrity: sha512-Ewyx0c9PmpcsByhSW4r+9zDU7sGjFc86qf/kKtuSCRdhfbk0SNLLkaT5qvcHnRGgc5NP/ly/y+qkXkqONX54CQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + es-shim-unscopables: 1.0.2 + dev: true + + /arraybuffer.prototype.slice@1.0.3: + resolution: {integrity: sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==} + engines: {node: '>= 0.4'} + dependencies: + array-buffer-byte-length: 1.0.1 + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + es-errors: 1.3.0 + get-intrinsic: 1.2.4 + is-array-buffer: 3.0.4 + is-shared-array-buffer: 1.0.3 + dev: true + + /assertion-error@1.1.0: + resolution: {integrity: sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==} + dev: true + + /astral-regex@2.0.0: + resolution: {integrity: sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==} + engines: {node: '>=8'} + + /async-retry@1.3.3: + resolution: {integrity: sha512-wfr/jstw9xNi/0teMHrRW7dsz3Lt5ARhYNZ2ewpadnhaIp5mbALhOAP+EAdsC7t4Z6wqsDVv9+W6gm1Dk9mEyw==} + dependencies: + retry: 0.13.1 + dev: true + + /async@3.2.5: + resolution: {integrity: sha512-baNZyqaaLhyLVKm/DlvdW051MSgO6b8eVfIezl9E5PqWxFgzLm/wQntEW4zOytVburDEr0JlALEpdOFwvErLsg==} + + /asynckit@0.4.0: + resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==} + dev: false + + /available-typed-arrays@1.0.7: + resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==} + engines: {node: '>= 0.4'} + dependencies: + possible-typed-array-names: 1.0.0 + dev: true + + /axios@1.7.2: + resolution: {integrity: sha512-2A8QhOMrbomlDuiLeK9XibIBzuHeRcqqNOHp0Cyp5EoJ1IFDh+XZH3A6BkXtv0K4gFGCI0Y4BM7B1wOEi0Rmgw==} + dependencies: + follow-redirects: 1.15.6 + form-data: 4.0.0 + proxy-from-env: 1.1.0 + transitivePeerDependencies: + - debug + dev: false + + /balanced-match@1.0.2: + resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} + + /binary-extensions@2.3.0: + resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} + engines: {node: '>=8'} + dev: true + + /bowser@2.11.0: + resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} + dev: true + + /brace-expansion@1.1.11: + resolution: {integrity: sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==} + dependencies: + balanced-match: 1.0.2 + concat-map: 0.0.1 + + /brace-expansion@2.0.1: + resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} + dependencies: + balanced-match: 1.0.2 + + /braces@3.0.3: + resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==} + engines: {node: '>=8'} + dependencies: + fill-range: 7.1.1 + + /browser-stdout@1.3.1: + resolution: {integrity: sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==} + dev: true + + /builtin-modules@3.3.0: + resolution: {integrity: sha512-zhaCDicdLuWN5UbN5IMnFqNMhNfo919sH85y2/ea+5Yg9TsTkeZxpL+JLbp6cgYFS4sRLp3YV4S6yDuqVWHYOw==} + engines: {node: '>=6'} + dev: true + + /builtins@5.1.0: + resolution: {integrity: sha512-SW9lzGTLvWTP1AY8xeAMZimqDrIaSdLQUcVr9DMef51niJ022Ri87SwRRKYm4A6iHfkPaiVUu/Duw2Wc4J7kKg==} + dependencies: + semver: 7.6.2 + dev: true + + /cacheable-lookup@7.0.0: + resolution: {integrity: sha512-+qJyx4xiKra8mZrcwhjMRMUhD5NR1R8esPkzIYxX96JiecFoxAXFuz/GpR3+ev4PE1WamHip78wV0vcmPQtp8w==} + engines: {node: '>=14.16'} + dev: true + + /cacheable-request@10.2.14: + resolution: {integrity: sha512-zkDT5WAF4hSSoUgyfg5tFIxz8XQK+25W/TLVojJTMKBaxevLBBtLxgqguAuVQB8PVW79FVjHcU+GJ9tVbDZ9mQ==} + engines: {node: '>=14.16'} + dependencies: + '@types/http-cache-semantics': 4.0.4 + get-stream: 6.0.1 + http-cache-semantics: 4.1.1 + keyv: 4.5.4 + mimic-response: 4.0.0 + normalize-url: 8.0.1 + responselike: 3.0.0 + dev: true + + /call-bind@1.0.7: + resolution: {integrity: sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==} + engines: {node: '>= 0.4'} + dependencies: + es-define-property: 1.0.0 + es-errors: 1.3.0 + function-bind: 1.1.2 + get-intrinsic: 1.2.4 + set-function-length: 1.2.2 + dev: true + + /callsites@3.1.0: + resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} + engines: {node: '>=6'} + + /camel-case@4.1.2: + resolution: {integrity: sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw==} + dependencies: + pascal-case: 3.1.2 + tslib: 2.6.2 + dev: true + + /camelcase@6.3.0: + resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} + engines: {node: '>=10'} + dev: true + + /capital-case@1.0.4: + resolution: {integrity: sha512-ds37W8CytHgwnhGGTi88pcPyR15qoNkOpYwmMMfnWqqWgESapLqvDx6huFjQ5vqWSn2Z06173XNA7LtMOeUh1A==} + dependencies: + no-case: 3.0.4 + tslib: 2.6.2 + upper-case-first: 2.0.2 + dev: true + + /cardinal@2.1.1: + resolution: {integrity: sha512-JSr5eOgoEymtYHBjNWyjrMqet9Am2miJhlfKNdqLp6zoeAh0KN5dRAcxlecj5mAJrmQomgiOBj35xHLrFjqBpw==} + hasBin: true + dependencies: + ansicolors: 0.3.2 + redeyed: 2.1.1 + + /chai@4.4.1: + resolution: {integrity: sha512-13sOfMv2+DWduEU+/xbun3LScLoqN17nBeTLUsmDfKdoiC1fr0n9PU4guu4AhRcOVFk/sW8LyZWHuhWtQZiF+g==} + engines: {node: '>=4'} + dependencies: + assertion-error: 1.1.0 + check-error: 1.0.3 + deep-eql: 4.1.3 + get-func-name: 2.0.2 + loupe: 2.3.7 + pathval: 1.1.1 + type-detect: 4.0.8 + dev: true + + /chalk@2.4.2: + resolution: {integrity: sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==} + engines: {node: '>=4'} + dependencies: + ansi-styles: 3.2.1 + escape-string-regexp: 1.0.5 + supports-color: 5.5.0 + + /chalk@4.1.2: + resolution: {integrity: sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==} + engines: {node: '>=10'} + dependencies: + ansi-styles: 4.3.0 + supports-color: 7.2.0 + + /change-case@4.1.2: + resolution: {integrity: sha512-bSxY2ws9OtviILG1EiY5K7NNxkqg/JnRnFxLtKQ96JaviiIxi7djMrSd0ECT9AC+lttClmYwKw53BWpOMblo7A==} + dependencies: + camel-case: 4.1.2 + capital-case: 1.0.4 + constant-case: 3.0.4 + dot-case: 3.0.4 + header-case: 2.0.4 + no-case: 3.0.4 + param-case: 3.0.4 + pascal-case: 3.1.2 + path-case: 3.0.4 + sentence-case: 3.0.4 + snake-case: 3.0.4 + tslib: 2.6.2 + dev: true + + /check-error@1.0.3: + resolution: {integrity: sha512-iKEoDYaRmd1mxM90a2OEfWhjsjPpYPuQ+lMYsoxB126+t8fw7ySEO48nmDg5COTjxDI65/Y2OWpeEHk3ZOe8zg==} + dependencies: + get-func-name: 2.0.2 + dev: true + + /chokidar@3.5.3: + resolution: {integrity: sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==} + engines: {node: '>= 8.10.0'} + dependencies: + anymatch: 3.1.3 + braces: 3.0.3 + glob-parent: 5.1.2 + is-binary-path: 2.1.0 + is-glob: 4.0.3 + normalize-path: 3.0.0 + readdirp: 3.6.0 + optionalDependencies: + fsevents: 2.3.3 + dev: true + + /ci-info@3.9.0: + resolution: {integrity: sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==} + engines: {node: '>=8'} + dev: true + + /clean-regexp@1.0.0: + resolution: {integrity: sha512-GfisEZEJvzKrmGWkvfhgzcz/BllN1USeqD2V6tg14OAOgaCD2Z/PUEuxnAZ/nPvmaHRG7a8y77p1T/IRQ4D1Hw==} + engines: {node: '>=4'} + dependencies: + escape-string-regexp: 1.0.5 + dev: true + + /clean-stack@3.0.1: + resolution: {integrity: sha512-lR9wNiMRcVQjSB3a7xXGLuz4cr4wJuuXlaAEbRutGowQTmlp7R72/DOgN21e8jdwblMWl9UOJMJXarX94pzKdg==} + engines: {node: '>=10'} + dependencies: + escape-string-regexp: 4.0.0 + + /cli-progress@3.12.0: + resolution: {integrity: sha512-tRkV3HJ1ASwm19THiiLIXLO7Im7wlTuKnvkYaTkyoAPefqjNg7W7DHKUlGRxy9vxDvbyCYQkQozvptuMkGCg8A==} + engines: {node: '>=4'} + dependencies: + string-width: 4.2.3 + + /cli-spinners@2.9.2: + resolution: {integrity: sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==} + engines: {node: '>=6'} + + /cli-width@4.1.0: + resolution: {integrity: sha512-ouuZd4/dm2Sw5Gmqy6bGyNNNe1qt9RpmxveLSO7KcgsTnU7RXfsw+/bukWGo1abgBiMAic068rclZsO4IWmmxQ==} + engines: {node: '>= 12'} + dev: true + + /cliui@7.0.4: + resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==} + dependencies: + string-width: 4.2.3 + strip-ansi: 6.0.1 + wrap-ansi: 7.0.0 + dev: true + + /color-convert@1.9.3: + resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} + dependencies: + color-name: 1.1.3 + + /color-convert@2.0.1: + resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} + engines: {node: '>=7.0.0'} + dependencies: + color-name: 1.1.4 + + /color-name@1.1.3: + resolution: {integrity: sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==} + + /color-name@1.1.4: + resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} + + /color-string@1.9.1: + resolution: {integrity: sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==} + dependencies: + color-name: 1.1.4 + simple-swizzle: 0.2.2 + + /color@4.2.3: + resolution: {integrity: sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A==} + engines: {node: '>=12.5.0'} + dependencies: + color-convert: 2.0.1 + color-string: 1.9.1 + + /combined-stream@1.0.8: + resolution: {integrity: sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==} + engines: {node: '>= 0.8'} + dependencies: + delayed-stream: 1.0.0 + dev: false + + /concat-map@0.0.1: + resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} + + /confusing-browser-globals@1.0.11: + resolution: {integrity: sha512-JsPKdmh8ZkmnHxDk55FZ1TqVLvEQTvoByJZRN9jzI0UjxK/QgAmsphz7PGtqgPieQZ/CQcHWXCR7ATDNhGe+YA==} + dev: true + + /constant-case@3.0.4: + resolution: {integrity: sha512-I2hSBi7Vvs7BEuJDr5dDHfzb/Ruj3FyvFyh7KLilAjNQw3Be+xgqUBA2W6scVEcL0hL1dwPRtIqEPVUCKkSsyQ==} + dependencies: + no-case: 3.0.4 + tslib: 2.6.2 + upper-case: 2.0.2 + dev: true + + /content-type@1.0.5: + resolution: {integrity: sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==} + engines: {node: '>= 0.6'} + dev: true + + /cosmiconfig@9.0.0(typescript@5.4.5): + resolution: {integrity: sha512-itvL5h8RETACmOTFc4UfIyB2RfEHi71Ax6E/PivVxq9NseKbOWpeyHEOIbmAw1rs8Ak0VursQNww7lf7YtUwzg==} + engines: {node: '>=14'} + peerDependencies: + typescript: '>=4.9.5' + peerDependenciesMeta: + typescript: + optional: true + dependencies: + env-paths: 2.2.1 + import-fresh: 3.3.0 + js-yaml: 4.1.0 + parse-json: 5.2.0 + typescript: 5.4.5 + + /create-require@1.1.1: + resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==} + dev: true + + /cross-spawn@7.0.3: + resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} + engines: {node: '>= 8'} + dependencies: + path-key: 3.1.1 + shebang-command: 2.0.0 + which: 2.0.2 + + /data-view-buffer@1.0.1: + resolution: {integrity: sha512-0lht7OugA5x3iJLOWFhWK/5ehONdprk0ISXqVFn/NFrDu+cuc8iADFrGQz5BnRK7LLU3JmkbXSxaqX+/mXYtUA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + es-errors: 1.3.0 + is-data-view: 1.0.1 + dev: true + + /data-view-byte-length@1.0.1: + resolution: {integrity: sha512-4J7wRJD3ABAzr8wP+OcIcqq2dlUKp4DVflx++hs5h5ZKydWMI6/D/fAot+yh6g2tHh8fLFTvNOaVN357NvSrOQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + es-errors: 1.3.0 + is-data-view: 1.0.1 + dev: true + + /data-view-byte-offset@1.0.0: + resolution: {integrity: sha512-t/Ygsytq+R995EJ5PZlD4Cu56sWa8InXySaViRzw9apusqsOO2bQP+SbYzAhR0pFKoB+43lYy8rWban9JSuXnA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + es-errors: 1.3.0 + is-data-view: 1.0.1 + dev: true + + /debug@3.2.7: + resolution: {integrity: sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + dependencies: + ms: 2.1.3 + dev: true + + /debug@4.3.4(supports-color@8.1.1): + resolution: {integrity: sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==} + engines: {node: '>=6.0'} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + dependencies: + ms: 2.1.2 + supports-color: 8.1.1 + dev: true + + /debug@4.3.5(supports-color@8.1.1): + resolution: {integrity: sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==} + engines: {node: '>=6.0'} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + dependencies: + ms: 2.1.2 + supports-color: 8.1.1 + + /debug@4.3.5(supports-color@9.4.0): + resolution: {integrity: sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==} + engines: {node: '>=6.0'} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + dependencies: + ms: 2.1.2 + supports-color: 9.4.0 + + /decamelize@4.0.0: + resolution: {integrity: sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==} + engines: {node: '>=10'} + dev: true + + /decompress-response@6.0.0: + resolution: {integrity: sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==} + engines: {node: '>=10'} + dependencies: + mimic-response: 3.1.0 + dev: true + + /deep-eql@4.1.3: + resolution: {integrity: sha512-WaEtAOpRA1MQ0eohqZjpGD8zdI0Ovsm8mmFhaDN8dvDZzyoUMcYDnf5Y6iu7HTXxf8JDS23qWa4a+hKCDyOPzw==} + engines: {node: '>=6'} + dependencies: + type-detect: 4.0.8 + dev: true + + /deep-is@0.1.4: + resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} + dev: true + + /defer-to-connect@2.0.1: + resolution: {integrity: sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg==} + engines: {node: '>=10'} + dev: true + + /define-data-property@1.1.4: + resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} + engines: {node: '>= 0.4'} + dependencies: + es-define-property: 1.0.0 + es-errors: 1.3.0 + gopd: 1.0.1 + dev: true + + /define-properties@1.2.1: + resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==} + engines: {node: '>= 0.4'} + dependencies: + define-data-property: 1.1.4 + has-property-descriptors: 1.0.2 + object-keys: 1.1.1 + dev: true + + /delayed-stream@1.0.0: + resolution: {integrity: sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==} + engines: {node: '>=0.4.0'} + dev: false + + /detect-indent@7.0.1: + resolution: {integrity: sha512-Mc7QhQ8s+cLrnUfU/Ji94vG/r8M26m8f++vyres4ZoojaRDpZ1eSIh/EpzLNwlWuvzSZ3UbDFspjFvTDXe6e/g==} + engines: {node: '>=12.20'} + dev: true + + /detect-newline@4.0.1: + resolution: {integrity: sha512-qE3Veg1YXzGHQhlA6jzebZN2qVf6NX+A7m7qlhCGG30dJixrAQhYOsJjsnBjJkCSmuOPpCk30145fr8FV0bzog==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dev: true + + /diff@4.0.2: + resolution: {integrity: sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==} + engines: {node: '>=0.3.1'} + dev: true + + /diff@5.0.0: + resolution: {integrity: sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==} + engines: {node: '>=0.3.1'} + dev: true + + /dir-glob@3.0.1: + resolution: {integrity: sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==} + engines: {node: '>=8'} + dependencies: + path-type: 4.0.0 + + /doctrine@2.1.0: + resolution: {integrity: sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==} + engines: {node: '>=0.10.0'} + dependencies: + esutils: 2.0.3 + dev: true + + /doctrine@3.0.0: + resolution: {integrity: sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==} + engines: {node: '>=6.0.0'} + dependencies: + esutils: 2.0.3 + dev: true + + /dot-case@3.0.4: + resolution: {integrity: sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==} + dependencies: + no-case: 3.0.4 + tslib: 2.6.2 + dev: true + + /ejs@3.1.10: + resolution: {integrity: sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA==} + engines: {node: '>=0.10.0'} + hasBin: true + dependencies: + jake: 10.9.1 + + /emoji-regex@8.0.0: + resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} + + /enhanced-resolve@5.16.1: + resolution: {integrity: sha512-4U5pNsuDl0EhuZpq46M5xPslstkviJuhrdobaRDBk2Jy2KO37FDAJl4lb2KlNabxT0m4MTK2UHNrsAcphE8nyw==} + engines: {node: '>=10.13.0'} + dependencies: + graceful-fs: 4.2.11 + tapable: 2.2.1 + dev: true + + /env-paths@2.2.1: + resolution: {integrity: sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==} + engines: {node: '>=6'} + + /error-ex@1.3.2: + resolution: {integrity: sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==} + dependencies: + is-arrayish: 0.2.1 + + /es-abstract@1.23.3: + resolution: {integrity: sha512-e+HfNH61Bj1X9/jLc5v1owaLYuHdeHHSQlkhCBiTK8rBvKaULl/beGMxwrMXjpYrv4pz22BlY570vVePA2ho4A==} + engines: {node: '>= 0.4'} + dependencies: + array-buffer-byte-length: 1.0.1 + arraybuffer.prototype.slice: 1.0.3 + available-typed-arrays: 1.0.7 + call-bind: 1.0.7 + data-view-buffer: 1.0.1 + data-view-byte-length: 1.0.1 + data-view-byte-offset: 1.0.0 + es-define-property: 1.0.0 + es-errors: 1.3.0 + es-object-atoms: 1.0.0 + es-set-tostringtag: 2.0.3 + es-to-primitive: 1.2.1 + function.prototype.name: 1.1.6 + get-intrinsic: 1.2.4 + get-symbol-description: 1.0.2 + globalthis: 1.0.4 + gopd: 1.0.1 + has-property-descriptors: 1.0.2 + has-proto: 1.0.3 + has-symbols: 1.0.3 + hasown: 2.0.2 + internal-slot: 1.0.7 + is-array-buffer: 3.0.4 + is-callable: 1.2.7 + is-data-view: 1.0.1 + is-negative-zero: 2.0.3 + is-regex: 1.1.4 + is-shared-array-buffer: 1.0.3 + is-string: 1.0.7 + is-typed-array: 1.1.13 + is-weakref: 1.0.2 + object-inspect: 1.13.1 + object-keys: 1.1.1 + object.assign: 4.1.5 + regexp.prototype.flags: 1.5.2 + safe-array-concat: 1.1.2 + safe-regex-test: 1.0.3 + string.prototype.trim: 1.2.9 + string.prototype.trimend: 1.0.8 + string.prototype.trimstart: 1.0.8 + typed-array-buffer: 1.0.2 + typed-array-byte-length: 1.0.1 + typed-array-byte-offset: 1.0.2 + typed-array-length: 1.0.6 + unbox-primitive: 1.0.2 + which-typed-array: 1.1.15 + dev: true + + /es-define-property@1.0.0: + resolution: {integrity: sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==} + engines: {node: '>= 0.4'} + dependencies: + get-intrinsic: 1.2.4 + dev: true + + /es-errors@1.3.0: + resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} + engines: {node: '>= 0.4'} + dev: true + + /es-object-atoms@1.0.0: + resolution: {integrity: sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==} + engines: {node: '>= 0.4'} + dependencies: + es-errors: 1.3.0 + dev: true + + /es-set-tostringtag@2.0.3: + resolution: {integrity: sha512-3T8uNMC3OQTHkFUsFq8r/BwAXLHvU/9O9mE0fBc/MY5iq/8H7ncvO947LmYA6ldWw9Uh8Yhf25zu6n7nML5QWQ==} + engines: {node: '>= 0.4'} + dependencies: + get-intrinsic: 1.2.4 + has-tostringtag: 1.0.2 + hasown: 2.0.2 + dev: true + + /es-shim-unscopables@1.0.2: + resolution: {integrity: sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw==} + dependencies: + hasown: 2.0.2 + dev: true + + /es-to-primitive@1.2.1: + resolution: {integrity: sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==} + engines: {node: '>= 0.4'} + dependencies: + is-callable: 1.2.7 + is-date-object: 1.0.5 + is-symbol: 1.0.4 + dev: true + + /escalade@3.1.2: + resolution: {integrity: sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA==} + engines: {node: '>=6'} + dev: true + + /escape-string-regexp@1.0.5: + resolution: {integrity: sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==} + engines: {node: '>=0.8.0'} + + /escape-string-regexp@4.0.0: + resolution: {integrity: sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==} + engines: {node: '>=10'} + + /eslint-config-oclif-typescript@3.1.7(eslint@8.57.0)(typescript@5.4.5): + resolution: {integrity: sha512-5q6Q1NjQt6WrAANGO9Go3uuxZTzf7ywmecRNW7e+bTnlkTk0/ClPd6SogH+qkwOkFJaMHmBp45ZmzvwGzy/Txg==} + engines: {node: '>=18.0.0'} + dependencies: + '@typescript-eslint/eslint-plugin': 6.21.0(@typescript-eslint/parser@6.21.0)(eslint@8.57.0)(typescript@5.4.5) + '@typescript-eslint/parser': 6.21.0(eslint@8.57.0)(typescript@5.4.5) + eslint-config-xo-space: 0.35.0(eslint@8.57.0) + eslint-import-resolver-typescript: 3.6.1(@typescript-eslint/parser@6.21.0)(eslint-plugin-import@2.29.1)(eslint@8.57.0) + eslint-plugin-import: 2.29.1(@typescript-eslint/parser@6.21.0)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0) + eslint-plugin-mocha: 10.4.3(eslint@8.57.0) + eslint-plugin-n: 15.7.0(eslint@8.57.0) + eslint-plugin-perfectionist: 2.10.0(eslint@8.57.0)(typescript@5.4.5) + transitivePeerDependencies: + - astro-eslint-parser + - eslint + - eslint-import-resolver-node + - eslint-import-resolver-webpack + - supports-color + - svelte + - svelte-eslint-parser + - typescript + - vue-eslint-parser + dev: true + + /eslint-config-oclif@5.2.0(eslint@8.57.0): + resolution: {integrity: sha512-fd2rFmm1x5YvTHNklSigbKj8ymo/uAU/PKBic/Yc+9yCRHgOAQos01mBLYVw9oeoyVLx+d79YVidkqgPoyx6RQ==} + engines: {node: '>=18.0.0'} + dependencies: + eslint-config-xo-space: 0.35.0(eslint@8.57.0) + eslint-plugin-mocha: 10.4.3(eslint@8.57.0) + eslint-plugin-n: 15.7.0(eslint@8.57.0) + eslint-plugin-unicorn: 48.0.1(eslint@8.57.0) + transitivePeerDependencies: + - eslint + dev: true + + /eslint-config-prettier@9.1.0(eslint@8.57.0): + resolution: {integrity: sha512-NSWl5BFQWEPi1j4TjVNItzYV7dZXZ+wP6I6ZhrBGpChQhZRUaElihE9uRRkcbRnNb76UMKDF3r+WTmNcGPKsqw==} + hasBin: true + peerDependencies: + eslint: '>=7.0.0' + dependencies: + eslint: 8.57.0 + dev: true + + /eslint-config-xo-space@0.35.0(eslint@8.57.0): + resolution: {integrity: sha512-+79iVcoLi3PvGcjqYDpSPzbLfqYpNcMlhsCBRsnmDoHAn4npJG6YxmHpelQKpXM7v/EeZTUKb4e1xotWlei8KA==} + engines: {node: '>=12'} + peerDependencies: + eslint: '>=8.56.0' + dependencies: + eslint: 8.57.0 + eslint-config-xo: 0.44.0(eslint@8.57.0) + dev: true + + /eslint-config-xo@0.44.0(eslint@8.57.0): + resolution: {integrity: sha512-YG4gdaor0mJJi8UBeRJqDPO42MedTWYMaUyucF5bhm2pi/HS98JIxfFQmTLuyj6hGpQlAazNfyVnn7JuDn+Sew==} + engines: {node: '>=18'} + peerDependencies: + eslint: '>=8.56.0' + dependencies: + confusing-browser-globals: 1.0.11 + eslint: 8.57.0 + dev: true + + /eslint-import-resolver-node@0.3.9: + resolution: {integrity: sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==} + dependencies: + debug: 3.2.7 + is-core-module: 2.13.1 + resolve: 1.22.8 + transitivePeerDependencies: + - supports-color + dev: true + + /eslint-import-resolver-typescript@3.6.1(@typescript-eslint/parser@6.21.0)(eslint-plugin-import@2.29.1)(eslint@8.57.0): + resolution: {integrity: sha512-xgdptdoi5W3niYeuQxKmzVDTATvLYqhpwmykwsh7f6HIOStGWEIL9iqZgQDF9u9OEzrRwR8no5q2VT+bjAujTg==} + engines: {node: ^14.18.0 || >=16.0.0} + peerDependencies: + eslint: '*' + eslint-plugin-import: '*' + dependencies: + debug: 4.3.5(supports-color@9.4.0) + enhanced-resolve: 5.16.1 + eslint: 8.57.0 + eslint-module-utils: 2.8.1(@typescript-eslint/parser@6.21.0)(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0) + eslint-plugin-import: 2.29.1(@typescript-eslint/parser@6.21.0)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0) + fast-glob: 3.3.2 + get-tsconfig: 4.7.5 + is-core-module: 2.13.1 + is-glob: 4.0.3 + transitivePeerDependencies: + - '@typescript-eslint/parser' + - eslint-import-resolver-node + - eslint-import-resolver-webpack + - supports-color + dev: true + + /eslint-module-utils@2.8.1(@typescript-eslint/parser@6.21.0)(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0): + resolution: {integrity: sha512-rXDXR3h7cs7dy9RNpUlQf80nX31XWJEyGq1tRMo+6GsO5VmTe4UTwtmonAD4ZkAsrfMVDA2wlGJ3790Ys+D49Q==} + engines: {node: '>=4'} + peerDependencies: + '@typescript-eslint/parser': '*' + eslint: '*' + eslint-import-resolver-node: '*' + eslint-import-resolver-typescript: '*' + eslint-import-resolver-webpack: '*' + peerDependenciesMeta: + '@typescript-eslint/parser': + optional: true + eslint: + optional: true + eslint-import-resolver-node: + optional: true + eslint-import-resolver-typescript: + optional: true + eslint-import-resolver-webpack: + optional: true + dependencies: + '@typescript-eslint/parser': 6.21.0(eslint@8.57.0)(typescript@5.4.5) + debug: 3.2.7 + eslint: 8.57.0 + eslint-import-resolver-node: 0.3.9 + eslint-import-resolver-typescript: 3.6.1(@typescript-eslint/parser@6.21.0)(eslint-plugin-import@2.29.1)(eslint@8.57.0) + transitivePeerDependencies: + - supports-color + dev: true + + /eslint-plugin-es@4.1.0(eslint@8.57.0): + resolution: {integrity: sha512-GILhQTnjYE2WorX5Jyi5i4dz5ALWxBIdQECVQavL6s7cI76IZTDWleTHkxz/QT3kvcs2QlGHvKLYsSlPOlPXnQ==} + engines: {node: '>=8.10.0'} + peerDependencies: + eslint: '>=4.19.1' + dependencies: + eslint: 8.57.0 + eslint-utils: 2.1.0 + regexpp: 3.2.0 + dev: true + + /eslint-plugin-import@2.29.1(@typescript-eslint/parser@6.21.0)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0): + resolution: {integrity: sha512-BbPC0cuExzhiMo4Ff1BTVwHpjjv28C5R+btTOGaCRC7UEz801up0JadwkeSk5Ued6TG34uaczuVuH6qyy5YUxw==} + engines: {node: '>=4'} + peerDependencies: + '@typescript-eslint/parser': '*' + eslint: ^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 + peerDependenciesMeta: + '@typescript-eslint/parser': + optional: true + dependencies: + '@typescript-eslint/parser': 6.21.0(eslint@8.57.0)(typescript@5.4.5) + array-includes: 3.1.8 + array.prototype.findlastindex: 1.2.5 + array.prototype.flat: 1.3.2 + array.prototype.flatmap: 1.3.2 + debug: 3.2.7 + doctrine: 2.1.0 + eslint: 8.57.0 + eslint-import-resolver-node: 0.3.9 + eslint-module-utils: 2.8.1(@typescript-eslint/parser@6.21.0)(eslint-import-resolver-node@0.3.9)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0) + hasown: 2.0.2 + is-core-module: 2.13.1 + is-glob: 4.0.3 + minimatch: 3.1.2 + object.fromentries: 2.0.8 + object.groupby: 1.0.3 + object.values: 1.2.0 + semver: 6.3.1 + tsconfig-paths: 3.15.0 + transitivePeerDependencies: + - eslint-import-resolver-typescript + - eslint-import-resolver-webpack + - supports-color + dev: true + + /eslint-plugin-mocha@10.4.3(eslint@8.57.0): + resolution: {integrity: sha512-emc4TVjq5Ht0/upR+psftuz6IBG5q279p+1dSRDeHf+NS9aaerBi3lXKo1SEzwC29hFIW21gO89CEWSvRsi8IQ==} + engines: {node: '>=14.0.0'} + peerDependencies: + eslint: '>=7.0.0' + dependencies: + eslint: 8.57.0 + eslint-utils: 3.0.0(eslint@8.57.0) + globals: 13.24.0 + rambda: 7.5.0 + dev: true + + /eslint-plugin-n@15.7.0(eslint@8.57.0): + resolution: {integrity: sha512-jDex9s7D/Qial8AGVIHq4W7NswpUD5DPDL2RH8Lzd9EloWUuvUkHfv4FRLMipH5q2UtyurorBkPeNi1wVWNh3Q==} + engines: {node: '>=12.22.0'} + peerDependencies: + eslint: '>=7.0.0' + dependencies: + builtins: 5.1.0 + eslint: 8.57.0 + eslint-plugin-es: 4.1.0(eslint@8.57.0) + eslint-utils: 3.0.0(eslint@8.57.0) + ignore: 5.3.1 + is-core-module: 2.13.1 + minimatch: 3.1.2 + resolve: 1.22.8 + semver: 7.6.2 + dev: true + + /eslint-plugin-perfectionist@2.10.0(eslint@8.57.0)(typescript@5.4.5): + resolution: {integrity: sha512-P+tdrkHeMWBc55+DZsoDOAftV1WCsEoHaKm6JC7zajFus/syfT4vUPBFb3atGFSuyaVnGQGHlcKpP9X3Q0gH/w==} + peerDependencies: + astro-eslint-parser: ^0.16.0 + eslint: '>=8.0.0' + svelte: '>=3.0.0' + svelte-eslint-parser: ^0.33.0 + vue-eslint-parser: '>=9.0.0' + peerDependenciesMeta: + astro-eslint-parser: + optional: true + svelte: + optional: true + svelte-eslint-parser: + optional: true + vue-eslint-parser: + optional: true + dependencies: + '@typescript-eslint/utils': 7.12.0(eslint@8.57.0)(typescript@5.4.5) + eslint: 8.57.0 + minimatch: 9.0.4 + natural-compare-lite: 1.4.0 + transitivePeerDependencies: + - supports-color + - typescript + dev: true + + /eslint-plugin-unicorn@48.0.1(eslint@8.57.0): + resolution: {integrity: sha512-FW+4r20myG/DqFcCSzoumaddKBicIPeFnTrifon2mWIzlfyvzwyqZjqVP7m4Cqr/ZYisS2aiLghkUWaPg6vtCw==} + engines: {node: '>=16'} + peerDependencies: + eslint: '>=8.44.0' + dependencies: + '@babel/helper-validator-identifier': 7.24.6 + '@eslint-community/eslint-utils': 4.4.0(eslint@8.57.0) + ci-info: 3.9.0 + clean-regexp: 1.0.0 + eslint: 8.57.0 + esquery: 1.5.0 + indent-string: 4.0.0 + is-builtin-module: 3.2.1 + jsesc: 3.0.2 + lodash: 4.17.21 + pluralize: 8.0.0 + read-pkg-up: 7.0.1 + regexp-tree: 0.1.27 + regjsparser: 0.10.0 + semver: 7.6.2 + strip-indent: 3.0.0 + dev: true + + /eslint-scope@7.2.2: + resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dependencies: + esrecurse: 4.3.0 + estraverse: 5.3.0 + dev: true + + /eslint-utils@2.1.0: + resolution: {integrity: sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==} + engines: {node: '>=6'} + dependencies: + eslint-visitor-keys: 1.3.0 + dev: true + + /eslint-utils@3.0.0(eslint@8.57.0): + resolution: {integrity: sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==} + engines: {node: ^10.0.0 || ^12.0.0 || >= 14.0.0} + peerDependencies: + eslint: '>=5' + dependencies: + eslint: 8.57.0 + eslint-visitor-keys: 2.1.0 + dev: true + + /eslint-visitor-keys@1.3.0: + resolution: {integrity: sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==} + engines: {node: '>=4'} + dev: true + + /eslint-visitor-keys@2.1.0: + resolution: {integrity: sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==} + engines: {node: '>=10'} + dev: true + + /eslint-visitor-keys@3.4.3: + resolution: {integrity: sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dev: true + + /eslint@8.57.0: + resolution: {integrity: sha512-dZ6+mexnaTIbSBZWgou51U6OmzIhYM2VcNdtiTtI7qPNZm35Akpr0f6vtw3w1Kmn5PYo+tZVfh13WrhpS6oLqQ==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + hasBin: true + dependencies: + '@eslint-community/eslint-utils': 4.4.0(eslint@8.57.0) + '@eslint-community/regexpp': 4.10.1 + '@eslint/eslintrc': 2.1.4 + '@eslint/js': 8.57.0 + '@humanwhocodes/config-array': 0.11.14 + '@humanwhocodes/module-importer': 1.0.1 + '@nodelib/fs.walk': 1.2.8 + '@ungap/structured-clone': 1.2.0 + ajv: 6.12.6 + chalk: 4.1.2 + cross-spawn: 7.0.3 + debug: 4.3.5(supports-color@9.4.0) + doctrine: 3.0.0 + escape-string-regexp: 4.0.0 + eslint-scope: 7.2.2 + eslint-visitor-keys: 3.4.3 + espree: 9.6.1 + esquery: 1.5.0 + esutils: 2.0.3 + fast-deep-equal: 3.1.3 + file-entry-cache: 6.0.1 + find-up: 5.0.0 + glob-parent: 6.0.2 + globals: 13.24.0 + graphemer: 1.4.0 + ignore: 5.3.1 + imurmurhash: 0.1.4 + is-glob: 4.0.3 + is-path-inside: 3.0.3 + js-yaml: 4.1.0 + json-stable-stringify-without-jsonify: 1.0.1 + levn: 0.4.1 + lodash.merge: 4.6.2 + minimatch: 3.1.2 + natural-compare: 1.4.0 + optionator: 0.9.4 + strip-ansi: 6.0.1 + text-table: 0.2.0 + transitivePeerDependencies: + - supports-color + dev: true + + /espree@9.6.1: + resolution: {integrity: sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + dependencies: + acorn: 8.11.3 + acorn-jsx: 5.3.2(acorn@8.11.3) + eslint-visitor-keys: 3.4.3 + dev: true + + /esprima@4.0.1: + resolution: {integrity: sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==} + engines: {node: '>=4'} + hasBin: true + + /esquery@1.5.0: + resolution: {integrity: sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==} + engines: {node: '>=0.10'} + dependencies: + estraverse: 5.3.0 + dev: true + + /esrecurse@4.3.0: + resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==} + engines: {node: '>=4.0'} + dependencies: + estraverse: 5.3.0 + dev: true + + /estraverse@5.3.0: + resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==} + engines: {node: '>=4.0'} + dev: true + + /esutils@2.0.3: + resolution: {integrity: sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==} + engines: {node: '>=0.10.0'} + dev: true + + /fast-deep-equal@3.1.3: + resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} + dev: true + + /fast-glob@3.3.2: + resolution: {integrity: sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==} + engines: {node: '>=8.6.0'} + dependencies: + '@nodelib/fs.stat': 2.0.5 + '@nodelib/fs.walk': 1.2.8 + glob-parent: 5.1.2 + merge2: 1.4.1 + micromatch: 4.0.7 + + /fast-json-stable-stringify@2.1.0: + resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==} + dev: true + + /fast-levenshtein@2.0.6: + resolution: {integrity: sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==} + dev: true + + /fast-levenshtein@3.0.0: + resolution: {integrity: sha512-hKKNajm46uNmTlhHSyZkmToAc56uZJwYq7yrciZjqOxnlfQwERDQJmHPUp7m1m9wx8vgOe8IaCKZ5Kv2k1DdCQ==} + dependencies: + fastest-levenshtein: 1.0.16 + dev: true + + /fast-xml-parser@4.2.5: + resolution: {integrity: sha512-B9/wizE4WngqQftFPmdaMYlXoJlJOYxGQOanC77fq9k8+Z0v5dDSVh+3glErdIROP//s/jgb7ZuxKfB8nVyo0g==} + hasBin: true + dependencies: + strnum: 1.0.5 + dev: true + + /fastest-levenshtein@1.0.16: + resolution: {integrity: sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg==} + engines: {node: '>= 4.9.1'} + dev: true + + /fastq@1.17.1: + resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} + dependencies: + reusify: 1.0.4 + + /file-entry-cache@6.0.1: + resolution: {integrity: sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==} + engines: {node: ^10.12.0 || >=12.0.0} + dependencies: + flat-cache: 3.2.0 + dev: true + + /filelist@1.0.4: + resolution: {integrity: sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==} + dependencies: + minimatch: 5.1.6 + + /fill-range@7.1.1: + resolution: {integrity: sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==} + engines: {node: '>=8'} + dependencies: + to-regex-range: 5.0.1 + + /find-up@4.1.0: + resolution: {integrity: sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==} + engines: {node: '>=8'} + dependencies: + locate-path: 5.0.0 + path-exists: 4.0.0 + dev: true + + /find-up@5.0.0: + resolution: {integrity: sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==} + engines: {node: '>=10'} + dependencies: + locate-path: 6.0.0 + path-exists: 4.0.0 + dev: true + + /find-yarn-workspace-root@2.0.0: + resolution: {integrity: sha512-1IMnbjt4KzsQfnhnzNd8wUEgXZ44IzZaZmnLYx7D5FZlaHt2gW20Cri8Q+E/t5tIj4+epTBub+2Zxu/vNILzqQ==} + dependencies: + micromatch: 4.0.7 + dev: true + + /flat-cache@3.2.0: + resolution: {integrity: sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==} + engines: {node: ^10.12.0 || >=12.0.0} + dependencies: + flatted: 3.3.1 + keyv: 4.5.4 + rimraf: 3.0.2 + dev: true + + /flat@5.0.2: + resolution: {integrity: sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==} + hasBin: true + dev: true + + /flatted@3.3.1: + resolution: {integrity: sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==} + dev: true + + /follow-redirects@1.15.6: + resolution: {integrity: sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==} + engines: {node: '>=4.0'} + peerDependencies: + debug: '*' + peerDependenciesMeta: + debug: + optional: true + dev: false + + /for-each@0.3.3: + resolution: {integrity: sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==} + dependencies: + is-callable: 1.2.7 + dev: true + + /form-data-encoder@2.1.4: + resolution: {integrity: sha512-yDYSgNMraqvnxiEXO4hi88+YZxaHC6QKzb5N84iRCTDeRO7ZALpir/lVmf/uXUhnwUr2O4HU8s/n6x+yNjQkHw==} + engines: {node: '>= 14.17'} + dev: true + + /form-data@4.0.0: + resolution: {integrity: sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==} + engines: {node: '>= 6'} + dependencies: + asynckit: 0.4.0 + combined-stream: 1.0.8 + mime-types: 2.1.35 + dev: false + + /fs-extra@8.1.0: + resolution: {integrity: sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==} + engines: {node: '>=6 <7 || >=8'} + dependencies: + graceful-fs: 4.2.11 + jsonfile: 4.0.0 + universalify: 0.1.2 + dev: true + + /fs.realpath@1.0.0: + resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} + dev: true + + /fsevents@2.3.3: + resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} + engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} + os: [darwin] + requiresBuild: true + dev: true + optional: true + + /function-bind@1.1.2: + resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} + dev: true + + /function.prototype.name@1.1.6: + resolution: {integrity: sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + functions-have-names: 1.2.3 + dev: true + + /functions-have-names@1.2.3: + resolution: {integrity: sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==} + dev: true + + /get-caller-file@2.0.5: + resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} + engines: {node: 6.* || 8.* || >= 10.*} + dev: true + + /get-func-name@2.0.2: + resolution: {integrity: sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==} + dev: true + + /get-intrinsic@1.2.4: + resolution: {integrity: sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==} + engines: {node: '>= 0.4'} + dependencies: + es-errors: 1.3.0 + function-bind: 1.1.2 + has-proto: 1.0.3 + has-symbols: 1.0.3 + hasown: 2.0.2 + dev: true + + /get-package-type@0.1.0: + resolution: {integrity: sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==} + engines: {node: '>=8.0.0'} + + /get-stdin@9.0.0: + resolution: {integrity: sha512-dVKBjfWisLAicarI2Sf+JuBE/DghV4UzNAVe9yhEJuzeREd3JhOTE9cUaJTeSa77fsbQUK3pcOpJfM59+VKZaA==} + engines: {node: '>=12'} + dev: true + + /get-stream@6.0.1: + resolution: {integrity: sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==} + engines: {node: '>=10'} + dev: true + + /get-symbol-description@1.0.2: + resolution: {integrity: sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + es-errors: 1.3.0 + get-intrinsic: 1.2.4 + dev: true + + /get-tsconfig@4.7.5: + resolution: {integrity: sha512-ZCuZCnlqNzjb4QprAzXKdpp/gh6KTxSJuw3IBsPnV/7fV4NxC9ckB+vPTt8w7fJA0TaSD7c55BR47JD6MEDyDw==} + dependencies: + resolve-pkg-maps: 1.0.0 + dev: true + + /git-hooks-list@3.1.0: + resolution: {integrity: sha512-LF8VeHeR7v+wAbXqfgRlTSX/1BJR9Q1vEMR8JAz1cEg6GX07+zyj3sAdDvYjj/xnlIfVuGgj4qBei1K3hKH+PA==} + dev: true + + /github-slugger@2.0.0: + resolution: {integrity: sha512-IaOQ9puYtjrkq7Y0Ygl9KDZnrf/aiUJYUpVf89y8kyaxbRG7Y1SrX/jaumrv81vc61+kiMempujsM3Yw7w5qcw==} + dev: true + + /glob-parent@5.1.2: + resolution: {integrity: sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==} + engines: {node: '>= 6'} + dependencies: + is-glob: 4.0.3 + + /glob-parent@6.0.2: + resolution: {integrity: sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==} + engines: {node: '>=10.13.0'} + dependencies: + is-glob: 4.0.3 + dev: true + + /glob@7.2.3: + resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==} + deprecated: Glob versions prior to v9 are no longer supported + dependencies: + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 3.1.2 + once: 1.4.0 + path-is-absolute: 1.0.1 + dev: true + + /glob@8.1.0: + resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} + engines: {node: '>=12'} + deprecated: Glob versions prior to v9 are no longer supported + dependencies: + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 5.0.1 + once: 1.4.0 + dev: true + + /globals@13.24.0: + resolution: {integrity: sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==} + engines: {node: '>=8'} + dependencies: + type-fest: 0.20.2 + dev: true + + /globalthis@1.0.4: + resolution: {integrity: sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==} + engines: {node: '>= 0.4'} + dependencies: + define-properties: 1.2.1 + gopd: 1.0.1 + dev: true + + /globby@11.1.0: + resolution: {integrity: sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==} + engines: {node: '>=10'} + dependencies: + array-union: 2.1.0 + dir-glob: 3.0.1 + fast-glob: 3.3.2 + ignore: 5.3.1 + merge2: 1.4.1 + slash: 3.0.0 + + /globby@13.2.2: + resolution: {integrity: sha512-Y1zNGV+pzQdh7H39l9zgB4PJqjRNqydvdYCDG4HFXM4XuvSaQQlEc91IU1yALL8gUTDomgBAfz3XJdmUS+oo0w==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dependencies: + dir-glob: 3.0.1 + fast-glob: 3.3.2 + ignore: 5.3.1 + merge2: 1.4.1 + slash: 4.0.0 + dev: true + + /gopd@1.0.1: + resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} + dependencies: + get-intrinsic: 1.2.4 + dev: true + + /got@13.0.0: + resolution: {integrity: sha512-XfBk1CxOOScDcMr9O1yKkNaQyy865NbYs+F7dr4H0LZMVgCj2Le59k6PqbNHoL5ToeaEQUYh6c6yMfVcc6SJxA==} + engines: {node: '>=16'} + dependencies: + '@sindresorhus/is': 5.6.0 + '@szmarczak/http-timer': 5.0.1 + cacheable-lookup: 7.0.0 + cacheable-request: 10.2.14 + decompress-response: 6.0.0 + form-data-encoder: 2.1.4 + get-stream: 6.0.1 + http2-wrapper: 2.2.1 + lowercase-keys: 3.0.0 + p-cancelable: 3.0.0 + responselike: 3.0.0 + dev: true + + /graceful-fs@4.2.11: + resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==} + dev: true + + /graphemer@1.4.0: + resolution: {integrity: sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==} + dev: true + + /has-bigints@1.0.2: + resolution: {integrity: sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==} + dev: true + + /has-flag@3.0.0: + resolution: {integrity: sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==} + engines: {node: '>=4'} + + /has-flag@4.0.0: + resolution: {integrity: sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==} + engines: {node: '>=8'} + + /has-property-descriptors@1.0.2: + resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} + dependencies: + es-define-property: 1.0.0 + dev: true + + /has-proto@1.0.3: + resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} + engines: {node: '>= 0.4'} + dev: true + + /has-symbols@1.0.3: + resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} + engines: {node: '>= 0.4'} + dev: true + + /has-tostringtag@1.0.2: + resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} + engines: {node: '>= 0.4'} + dependencies: + has-symbols: 1.0.3 + dev: true + + /hasown@2.0.2: + resolution: {integrity: sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==} + engines: {node: '>= 0.4'} + dependencies: + function-bind: 1.1.2 + dev: true + + /he@1.2.0: + resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==} + hasBin: true + dev: true + + /header-case@2.0.4: + resolution: {integrity: sha512-H/vuk5TEEVZwrR0lp2zed9OCo1uAILMlx0JEMgC26rzyJJ3N1v6XkwHHXJQdR2doSjcGPM6OKPYoJgf0plJ11Q==} + dependencies: + capital-case: 1.0.4 + tslib: 2.6.2 + dev: true + + /hosted-git-info@2.8.9: + resolution: {integrity: sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==} + dev: true + + /hosted-git-info@7.0.2: + resolution: {integrity: sha512-puUZAUKT5m8Zzvs72XWy3HtvVbTWljRE66cP60bxJzAqf2DgICo7lYTY2IHUmLnNpjYvw5bvmoHvPc0QO2a62w==} + engines: {node: ^16.14.0 || >=18.0.0} + dependencies: + lru-cache: 10.2.2 + + /http-cache-semantics@4.1.1: + resolution: {integrity: sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==} + dev: true + + /http-call@5.3.0: + resolution: {integrity: sha512-ahwimsC23ICE4kPl9xTBjKB4inbRaeLyZeRunC/1Jy/Z6X8tv22MEAjK+KBOMSVLaqXPTTmd8638waVIKLGx2w==} + engines: {node: '>=8.0.0'} + dependencies: + content-type: 1.0.5 + debug: 4.3.5(supports-color@9.4.0) + is-retry-allowed: 1.2.0 + is-stream: 2.0.1 + parse-json: 4.0.0 + tunnel-agent: 0.6.0 + transitivePeerDependencies: + - supports-color + dev: true + + /http2-wrapper@2.2.1: + resolution: {integrity: sha512-V5nVw1PAOgfI3Lmeaj2Exmeg7fenjhRUgz1lPSezy1CuhPYbgQtbQj4jZfEAEMlaL+vupsvhjqCyjzob0yxsmQ==} + engines: {node: '>=10.19.0'} + dependencies: + quick-lru: 5.1.1 + resolve-alpn: 1.2.1 + dev: true + + /hyperlinker@1.0.0: + resolution: {integrity: sha512-Ty8UblRWFEcfSuIaajM34LdPXIhbs1ajEX/BBPv24J+enSVaEVY63xQ6lTO9VRYS5LAoghIG0IDJ+p+IPzKUQQ==} + engines: {node: '>=4'} + + /ignore@5.3.1: + resolution: {integrity: sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==} + engines: {node: '>= 4'} + + /import-fresh@3.3.0: + resolution: {integrity: sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==} + engines: {node: '>=6'} + dependencies: + parent-module: 1.0.1 + resolve-from: 4.0.0 + + /imurmurhash@0.1.4: + resolution: {integrity: sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==} + engines: {node: '>=0.8.19'} + dev: true + + /indent-string@4.0.0: + resolution: {integrity: sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==} + engines: {node: '>=8'} + + /inflight@1.0.6: + resolution: {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. + dependencies: + once: 1.4.0 + wrappy: 1.0.2 + dev: true + + /inherits@2.0.4: + resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} + dev: true + + /internal-slot@1.0.7: + resolution: {integrity: sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g==} + engines: {node: '>= 0.4'} + dependencies: + es-errors: 1.3.0 + hasown: 2.0.2 + side-channel: 1.0.6 + dev: true + + /interpret@1.4.0: + resolution: {integrity: sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==} + engines: {node: '>= 0.10'} + dev: true + + /is-array-buffer@3.0.4: + resolution: {integrity: sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + get-intrinsic: 1.2.4 + dev: true + + /is-arrayish@0.2.1: + resolution: {integrity: sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==} + + /is-arrayish@0.3.2: + resolution: {integrity: sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==} + + /is-bigint@1.0.4: + resolution: {integrity: sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==} + dependencies: + has-bigints: 1.0.2 + dev: true + + /is-binary-path@2.1.0: + resolution: {integrity: sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==} + engines: {node: '>=8'} + dependencies: + binary-extensions: 2.3.0 + dev: true + + /is-boolean-object@1.1.2: + resolution: {integrity: sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + has-tostringtag: 1.0.2 + dev: true + + /is-builtin-module@3.2.1: + resolution: {integrity: sha512-BSLE3HnV2syZ0FK0iMA/yUGplUeMmNz4AW5fnTunbCIqZi4vG3WjJT9FHMy5D69xmAYBHXQhJdALdpwVxV501A==} + engines: {node: '>=6'} + dependencies: + builtin-modules: 3.3.0 + dev: true + + /is-callable@1.2.7: + resolution: {integrity: sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==} + engines: {node: '>= 0.4'} + dev: true + + /is-core-module@2.13.1: + resolution: {integrity: sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw==} + dependencies: + hasown: 2.0.2 + dev: true + + /is-data-view@1.0.1: + resolution: {integrity: sha512-AHkaJrsUVW6wq6JS8y3JnM/GJF/9cf+k20+iDzlSaJrinEo5+7vRiteOSwBhHRiAyQATN1AmY4hwzxJKPmYf+w==} + engines: {node: '>= 0.4'} + dependencies: + is-typed-array: 1.1.13 + dev: true + + /is-date-object@1.0.5: + resolution: {integrity: sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==} + engines: {node: '>= 0.4'} + dependencies: + has-tostringtag: 1.0.2 + dev: true + + /is-docker@2.2.1: + resolution: {integrity: sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==} + engines: {node: '>=8'} + hasBin: true + + /is-extglob@2.1.1: + resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} + engines: {node: '>=0.10.0'} + + /is-fullwidth-code-point@3.0.0: + resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} + engines: {node: '>=8'} + + /is-glob@4.0.3: + resolution: {integrity: sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==} + engines: {node: '>=0.10.0'} + dependencies: + is-extglob: 2.1.1 + + /is-negative-zero@2.0.3: + resolution: {integrity: sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==} + engines: {node: '>= 0.4'} + dev: true + + /is-number-object@1.0.7: + resolution: {integrity: sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==} + engines: {node: '>= 0.4'} + dependencies: + has-tostringtag: 1.0.2 + dev: true + + /is-number@7.0.0: + resolution: {integrity: sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==} + engines: {node: '>=0.12.0'} + + /is-path-inside@3.0.3: + resolution: {integrity: sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==} + engines: {node: '>=8'} + dev: true + + /is-plain-obj@2.1.0: + resolution: {integrity: sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==} + engines: {node: '>=8'} + dev: true + + /is-plain-obj@4.1.0: + resolution: {integrity: sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg==} + engines: {node: '>=12'} + dev: true + + /is-regex@1.1.4: + resolution: {integrity: sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + has-tostringtag: 1.0.2 + dev: true + + /is-retry-allowed@1.2.0: + resolution: {integrity: sha512-RUbUeKwvm3XG2VYamhJL1xFktgjvPzL0Hq8C+6yrWIswDy3BIXGqCxhxkc30N9jqK311gVU137K8Ei55/zVJRg==} + engines: {node: '>=0.10.0'} + dev: true + + /is-shared-array-buffer@1.0.3: + resolution: {integrity: sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + dev: true + + /is-stream@2.0.1: + resolution: {integrity: sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==} + engines: {node: '>=8'} + dev: true + + /is-string@1.0.7: + resolution: {integrity: sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==} + engines: {node: '>= 0.4'} + dependencies: + has-tostringtag: 1.0.2 + dev: true + + /is-symbol@1.0.4: + resolution: {integrity: sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==} + engines: {node: '>= 0.4'} + dependencies: + has-symbols: 1.0.3 + dev: true + + /is-typed-array@1.1.13: + resolution: {integrity: sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==} + engines: {node: '>= 0.4'} + dependencies: + which-typed-array: 1.1.15 + dev: true + + /is-unicode-supported@0.1.0: + resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} + engines: {node: '>=10'} + dev: true + + /is-weakref@1.0.2: + resolution: {integrity: sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==} + dependencies: + call-bind: 1.0.7 + dev: true + + /is-wsl@2.2.0: + resolution: {integrity: sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==} + engines: {node: '>=8'} + dependencies: + is-docker: 2.2.1 + + /isarray@2.0.5: + resolution: {integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==} + dev: true + + /isexe@2.0.0: + resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} + + /isexe@3.1.1: + resolution: {integrity: sha512-LpB/54B+/2J5hqQ7imZHfdU31OlgQqx7ZicVlkm9kzg9/w8GKLEcFfJl/t7DCEDueOyBAD6zCCwTO6Fzs0NoEQ==} + engines: {node: '>=16'} + dev: false + + /jake@10.9.1: + resolution: {integrity: sha512-61btcOHNnLnsOdtLgA5efqQWjnSi/vow5HbI7HMdKKWqvrKR1bLK3BPlJn9gcSaP2ewuamUSMB5XEy76KUIS2w==} + engines: {node: '>=10'} + hasBin: true + dependencies: + async: 3.2.5 + chalk: 4.1.2 + filelist: 1.0.4 + minimatch: 3.1.2 + + /js-tokens@4.0.0: + resolution: {integrity: sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==} + + /js-yaml@3.14.1: + resolution: {integrity: sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==} + hasBin: true + dependencies: + argparse: 1.0.10 + esprima: 4.0.1 + + /js-yaml@4.1.0: + resolution: {integrity: sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==} + hasBin: true + dependencies: + argparse: 2.0.1 + + /jsesc@0.5.0: + resolution: {integrity: sha512-uZz5UnB7u4T9LvwmFqXii7pZSouaRPorGs5who1Ip7VO0wxanFvBL7GkM6dTHlgX+jhBApRetaWpnDabOeTcnA==} + hasBin: true + dev: true + + /jsesc@3.0.2: + resolution: {integrity: sha512-xKqzzWXDttJuOcawBt4KnKHHIf5oQ/Cxax+0PWFG+DFDgHNAdi+TXECADI+RYiFUMmx8792xsMbbgXj4CwnP4g==} + engines: {node: '>=6'} + hasBin: true + dev: true + + /json-buffer@3.0.1: + resolution: {integrity: sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==} + dev: true + + /json-parse-better-errors@1.0.2: + resolution: {integrity: sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==} + dev: true + + /json-parse-even-better-errors@2.3.1: + resolution: {integrity: sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==} + + /json-schema-traverse@0.4.1: + resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} + dev: true + + /json-stable-stringify-without-jsonify@1.0.1: + resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==} + dev: true + + /json5@1.0.2: + resolution: {integrity: sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==} + hasBin: true + dependencies: + minimist: 1.2.8 + dev: true + + /jsonfile@4.0.0: + resolution: {integrity: sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==} + optionalDependencies: + graceful-fs: 4.2.11 + dev: true + + /keyv@4.5.4: + resolution: {integrity: sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==} + dependencies: + json-buffer: 3.0.1 + dev: true + + /levn@0.4.1: + resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} + engines: {node: '>= 0.8.0'} + dependencies: + prelude-ls: 1.2.1 + type-check: 0.4.0 + dev: true + + /lines-and-columns@1.2.4: + resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} + + /locate-path@5.0.0: + resolution: {integrity: sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==} + engines: {node: '>=8'} + dependencies: + p-locate: 4.1.0 + dev: true + + /locate-path@6.0.0: + resolution: {integrity: sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==} + engines: {node: '>=10'} + dependencies: + p-locate: 5.0.0 + dev: true + + /lodash.merge@4.6.2: + resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==} + dev: true + + /lodash@4.17.21: + resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} + dev: true + + /log-symbols@4.1.0: + resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} + engines: {node: '>=10'} + dependencies: + chalk: 4.1.2 + is-unicode-supported: 0.1.0 + dev: true + + /loupe@2.3.7: + resolution: {integrity: sha512-zSMINGVYkdpYSOBmLi0D1Uo7JU9nVdQKrHxC8eYlV+9YKK9WePqAlL7lSlorG/U2Fw1w0hTBmaa/jrQ3UbPHtA==} + dependencies: + get-func-name: 2.0.2 + dev: true + + /lower-case@2.0.2: + resolution: {integrity: sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==} + dependencies: + tslib: 2.6.2 + dev: true + + /lowercase-keys@3.0.0: + resolution: {integrity: sha512-ozCC6gdQ+glXOQsveKD0YsDy8DSQFjDTz4zyzEHNV5+JP5D62LmfDZ6o1cycFx9ouG940M5dE8C8CTewdj2YWQ==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dev: true + + /lru-cache@10.2.2: + resolution: {integrity: sha512-9hp3Vp2/hFQUiIwKo8XCeFVnrg8Pk3TYNPIR7tJADKi5YfcF7vEaK7avFHTlSy3kOKYaJQaalfEo6YuXdceBOQ==} + engines: {node: 14 || >=16.14} + + /make-error@1.3.6: + resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==} + dev: true + + /merge2@1.4.1: + resolution: {integrity: sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==} + engines: {node: '>= 8'} + + /micromatch@4.0.7: + resolution: {integrity: sha512-LPP/3KorzCwBxfeUuZmaR6bG2kdeHSbe0P2tY3FLRU4vYrjYz5hI4QZwV0njUx3jeuKe67YukQ1LSPZBKDqO/Q==} + engines: {node: '>=8.6'} + dependencies: + braces: 3.0.3 + picomatch: 2.3.1 + + /mime-db@1.52.0: + resolution: {integrity: sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==} + engines: {node: '>= 0.6'} + dev: false + + /mime-types@2.1.35: + resolution: {integrity: sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==} + engines: {node: '>= 0.6'} + dependencies: + mime-db: 1.52.0 + dev: false + + /mimic-response@3.1.0: + resolution: {integrity: sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==} + engines: {node: '>=10'} + dev: true + + /mimic-response@4.0.0: + resolution: {integrity: sha512-e5ISH9xMYU0DzrT+jl8q2ze9D6eWBto+I8CNpe+VI+K2J/F/k3PdkdTdz4wvGVH4NTpo+NRYTVIuMQEMMcsLqg==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dev: true + + /min-indent@1.0.1: + resolution: {integrity: sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==} + engines: {node: '>=4'} + dev: true + + /minimatch@3.1.2: + resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} + dependencies: + brace-expansion: 1.1.11 + + /minimatch@5.0.1: + resolution: {integrity: sha512-nLDxIFRyhDblz3qMuq+SoRZED4+miJ/G+tdDrjkkkRnjAsBexeGpgjLEQ0blJy7rHhR2b93rhQY4SvyWu9v03g==} + engines: {node: '>=10'} + dependencies: + brace-expansion: 2.0.1 + dev: true + + /minimatch@5.1.6: + resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} + engines: {node: '>=10'} + dependencies: + brace-expansion: 2.0.1 + + /minimatch@9.0.3: + resolution: {integrity: sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==} + engines: {node: '>=16 || 14 >=14.17'} + dependencies: + brace-expansion: 2.0.1 + dev: true + + /minimatch@9.0.4: + resolution: {integrity: sha512-KqWh+VchfxcMNRAJjj2tnsSJdNbHsVgnkBhTNrW7AjVo6OvLtxw8zfT9oLw1JSohlFzJ8jCoTgaoXvJ+kHt6fw==} + engines: {node: '>=16 || 14 >=14.17'} + dependencies: + brace-expansion: 2.0.1 + + /minimist@1.2.8: + resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} + dev: true + + /mocha@10.4.0: + resolution: {integrity: sha512-eqhGB8JKapEYcC4ytX/xrzKforgEc3j1pGlAXVy3eRwrtAy5/nIfT1SvgGzfN0XZZxeLq0aQWkOUAmqIJiv+bA==} + engines: {node: '>= 14.0.0'} + hasBin: true + dependencies: + ansi-colors: 4.1.1 + browser-stdout: 1.3.1 + chokidar: 3.5.3 + debug: 4.3.4(supports-color@8.1.1) + diff: 5.0.0 + escape-string-regexp: 4.0.0 + find-up: 5.0.0 + glob: 8.1.0 + he: 1.2.0 + js-yaml: 4.1.0 + log-symbols: 4.1.0 + minimatch: 5.0.1 + ms: 2.1.3 + serialize-javascript: 6.0.0 + strip-json-comments: 3.1.1 + supports-color: 8.1.1 + workerpool: 6.2.1 + yargs: 16.2.0 + yargs-parser: 20.2.4 + yargs-unparser: 2.0.0 + dev: true + + /ms@2.1.2: + resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} + + /ms@2.1.3: + resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} + dev: true + + /mute-stream@1.0.0: + resolution: {integrity: sha512-avsJQhyd+680gKXyG/sQc0nXaC6rBkPOfyHYcFb9+hdkqQkR9bdnkJ0AMZhke0oesPqIO+mFFJ+IdBc7mst4IA==} + engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} + dev: true + + /natural-compare-lite@1.4.0: + resolution: {integrity: sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==} + dev: true + + /natural-compare@1.4.0: + resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} + dev: true + + /natural-orderby@2.0.3: + resolution: {integrity: sha512-p7KTHxU0CUrcOXe62Zfrb5Z13nLvPhSWR/so3kFulUQU0sgUll2Z0LwpsLN351eOOD+hRGu/F1g+6xDfPeD++Q==} + + /no-case@3.0.4: + resolution: {integrity: sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==} + dependencies: + lower-case: 2.0.2 + tslib: 2.6.2 + dev: true + + /normalize-package-data@2.5.0: + resolution: {integrity: sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==} + dependencies: + hosted-git-info: 2.8.9 + resolve: 1.22.8 + semver: 5.7.2 + validate-npm-package-license: 3.0.4 + dev: true + + /normalize-package-data@6.0.1: + resolution: {integrity: sha512-6rvCfeRW+OEZagAB4lMLSNuTNYZWLVtKccK79VSTf//yTY5VOCgcpH80O+bZK8Neps7pUnd5G+QlMg1yV/2iZQ==} + engines: {node: ^16.14.0 || >=18.0.0} + dependencies: + hosted-git-info: 7.0.2 + is-core-module: 2.13.1 + semver: 7.6.2 + validate-npm-package-license: 3.0.4 + dev: true + + /normalize-path@3.0.0: + resolution: {integrity: sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==} + engines: {node: '>=0.10.0'} + dev: true + + /normalize-url@8.0.1: + resolution: {integrity: sha512-IO9QvjUMWxPQQhs60oOu10CRkWCiZzSUkzbXGGV9pviYl1fXYcvkzQ5jV9z8Y6un8ARoVRl4EtC6v6jNqbaJ/w==} + engines: {node: '>=14.16'} + dev: true + + /npm-package-arg@11.0.2: + resolution: {integrity: sha512-IGN0IAwmhDJwy13Wc8k+4PEbTPhpJnMtfR53ZbOyjkvmEcLS4nCwp6mvMWjS5sUjeiW3mpx6cHmuhKEu9XmcQw==} + engines: {node: ^16.14.0 || >=18.0.0} + dependencies: + hosted-git-info: 7.0.2 + proc-log: 4.2.0 + semver: 7.6.2 + validate-npm-package-name: 5.0.1 + dev: false + + /npm-run-path@5.3.0: + resolution: {integrity: sha512-ppwTtiJZq0O/ai0z7yfudtBpWIoxM8yE6nHi1X47eFR2EWORqfbu6CnPlNsjeN683eT0qG6H/Pyf9fCcvjnnnQ==} + engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} + dependencies: + path-key: 4.0.0 + dev: false + + /npm@10.8.1: + resolution: {integrity: sha512-Dp1C6SvSMYQI7YHq/y2l94uvI+59Eqbu1EpuKQHQ8p16txXRuRit5gH3Lnaagk2aXDIjg/Iru9pd05bnneKgdw==} + engines: {node: ^18.17.0 || >=20.5.0} + hasBin: true + dev: false + bundledDependencies: + - '@isaacs/string-locale-compare' + - '@npmcli/arborist' + - '@npmcli/config' + - '@npmcli/fs' + - '@npmcli/map-workspaces' + - '@npmcli/package-json' + - '@npmcli/promise-spawn' + - '@npmcli/redact' + - '@npmcli/run-script' + - '@sigstore/tuf' + - abbrev + - archy + - cacache + - chalk + - ci-info + - cli-columns + - fastest-levenshtein + - fs-minipass + - glob + - graceful-fs + - hosted-git-info + - ini + - init-package-json + - is-cidr + - json-parse-even-better-errors + - libnpmaccess + - libnpmdiff + - libnpmexec + - libnpmfund + - libnpmhook + - libnpmorg + - libnpmpack + - libnpmpublish + - libnpmsearch + - libnpmteam + - libnpmversion + - make-fetch-happen + - minimatch + - minipass + - minipass-pipeline + - ms + - node-gyp + - nopt + - normalize-package-data + - npm-audit-report + - npm-install-checks + - npm-package-arg + - npm-pick-manifest + - npm-profile + - npm-registry-fetch + - npm-user-validate + - p-map + - pacote + - parse-conflict-json + - proc-log + - qrcode-terminal + - read + - semver + - spdx-expression-parse + - ssri + - supports-color + - tar + - text-table + - tiny-relative-date + - treeverse + - validate-npm-package-name + - which + - write-file-atomic + + /object-inspect@1.13.1: + resolution: {integrity: sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==} + dev: true + + /object-keys@1.1.1: + resolution: {integrity: sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==} + engines: {node: '>= 0.4'} + dev: true + + /object-treeify@1.1.33: + resolution: {integrity: sha512-EFVjAYfzWqWsBMRHPMAXLCDIJnpMhdWAqR7xG6M6a2cs6PMFpl/+Z20w9zDW4vkxOFfddegBKq9Rehd0bxWE7A==} + engines: {node: '>= 10'} + + /object-treeify@4.0.1: + resolution: {integrity: sha512-Y6tg5rHfsefSkfKujv2SwHulInROy/rCL5F4w0QOWxut8AnxYxf0YmNhTh95Zfyxpsudo66uqkux0ACFnyMSgQ==} + engines: {node: '>= 16'} + dev: false + + /object.assign@4.1.5: + resolution: {integrity: sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + has-symbols: 1.0.3 + object-keys: 1.1.1 + dev: true + + /object.fromentries@2.0.8: + resolution: {integrity: sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + es-object-atoms: 1.0.0 + dev: true + + /object.groupby@1.0.3: + resolution: {integrity: sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + dev: true + + /object.values@1.2.0: + resolution: {integrity: sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-object-atoms: 1.0.0 + dev: true + + /oclif@4.12.3(typescript@5.4.5): + resolution: {integrity: sha512-o4j5/n0LbI/OOasTiKSO4R2H+8NiG2Cn3nwVbp9GoSPfCZW7Tuup+xXNlh2/lURAYVD69Vx5XUs1PIY9QLgS5Q==} + engines: {node: '>=18.0.0'} + hasBin: true + dependencies: + '@aws-sdk/client-cloudfront': 3.588.0 + '@aws-sdk/client-s3': 3.588.0 + '@inquirer/confirm': 3.1.9 + '@inquirer/input': 2.1.9 + '@inquirer/select': 2.3.5 + '@oclif/core': 4.0.0-beta.15(typescript@5.4.5) + '@oclif/plugin-help': 6.1.0(typescript@5.4.5) + '@oclif/plugin-not-found': 3.2.1(typescript@5.4.5) + '@oclif/plugin-warn-if-update-available': 3.1.3(typescript@5.4.5) + async-retry: 1.3.3 + chalk: 4.1.2 + change-case: 4.1.2 + debug: 4.3.5(supports-color@9.4.0) + ejs: 3.1.10 + find-yarn-workspace-root: 2.0.0 + fs-extra: 8.1.0 + github-slugger: 2.0.0 + got: 13.0.0 + lodash: 4.17.21 + normalize-package-data: 6.0.1 + semver: 7.6.2 + sort-package-json: 2.10.0 + tiny-jsonc: 1.0.1 + validate-npm-package-name: 5.0.1 + transitivePeerDependencies: + - aws-crt + - supports-color + - typescript + dev: true + + /once@1.4.0: + resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==} + dependencies: + wrappy: 1.0.2 + dev: true + + /optionator@0.9.4: + resolution: {integrity: sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==} + engines: {node: '>= 0.8.0'} + dependencies: + deep-is: 0.1.4 + fast-levenshtein: 2.0.6 + levn: 0.4.1 + prelude-ls: 1.2.1 + type-check: 0.4.0 + word-wrap: 1.2.5 + dev: true + + /p-cancelable@3.0.0: + resolution: {integrity: sha512-mlVgR3PGuzlo0MmTdk4cXqXWlwQDLnONTAg6sm62XkMJEiRxN3GL3SffkYvqwonbkJBcrI7Uvv5Zh9yjvn2iUw==} + engines: {node: '>=12.20'} + dev: true + + /p-limit@2.3.0: + resolution: {integrity: sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==} + engines: {node: '>=6'} + dependencies: + p-try: 2.2.0 + dev: true + + /p-limit@3.1.0: + resolution: {integrity: sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==} + engines: {node: '>=10'} + dependencies: + yocto-queue: 0.1.0 + dev: true + + /p-locate@4.1.0: + resolution: {integrity: sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==} + engines: {node: '>=8'} + dependencies: + p-limit: 2.3.0 + dev: true + + /p-locate@5.0.0: + resolution: {integrity: sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==} + engines: {node: '>=10'} + dependencies: + p-limit: 3.1.0 + dev: true + + /p-try@2.2.0: + resolution: {integrity: sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==} + engines: {node: '>=6'} + dev: true + + /param-case@3.0.4: + resolution: {integrity: sha512-RXlj7zCYokReqWpOPH9oYivUzLYZ5vAPIfEmCTNViosC78F8F0H9y7T7gG2M39ymgutxF5gcFEsyZQSph9Bp3A==} + dependencies: + dot-case: 3.0.4 + tslib: 2.6.2 + dev: true + + /parent-module@1.0.1: + resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} + engines: {node: '>=6'} + dependencies: + callsites: 3.1.0 + + /parse-json@4.0.0: + resolution: {integrity: sha512-aOIos8bujGN93/8Ox/jPLh7RwVnPEysynVFE+fQZyg6jKELEHwzgKdLRFHUgXJL6kylijVSBC4BvN9OmsB48Rw==} + engines: {node: '>=4'} + dependencies: + error-ex: 1.3.2 + json-parse-better-errors: 1.0.2 + dev: true + + /parse-json@5.2.0: + resolution: {integrity: sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==} + engines: {node: '>=8'} + dependencies: + '@babel/code-frame': 7.24.6 + error-ex: 1.3.2 + json-parse-even-better-errors: 2.3.1 + lines-and-columns: 1.2.4 + + /pascal-case@3.1.2: + resolution: {integrity: sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g==} + dependencies: + no-case: 3.0.4 + tslib: 2.6.2 + dev: true + + /password-prompt@1.1.3: + resolution: {integrity: sha512-HkrjG2aJlvF0t2BMH0e2LB/EHf3Lcq3fNMzy4GYHcQblAvOl+QQji1Lx7WRBMqpVK8p+KR7bCg7oqAMXtdgqyw==} + dependencies: + ansi-escapes: 4.3.2 + cross-spawn: 7.0.3 + + /path-case@3.0.4: + resolution: {integrity: sha512-qO4qCFjXqVTrcbPt/hQfhTQ+VhFsqNKOPtytgNKkKxSoEp3XPUQ8ObFuePylOIok5gjn69ry8XiULxCwot3Wfg==} + dependencies: + dot-case: 3.0.4 + tslib: 2.6.2 + dev: true + + /path-exists@4.0.0: + resolution: {integrity: sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==} + engines: {node: '>=8'} + dev: true + + /path-is-absolute@1.0.1: + resolution: {integrity: sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==} + engines: {node: '>=0.10.0'} + dev: true + + /path-key@3.1.1: + resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} + engines: {node: '>=8'} + + /path-key@4.0.0: + resolution: {integrity: sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ==} + engines: {node: '>=12'} + dev: false + + /path-parse@1.0.7: + resolution: {integrity: sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==} + dev: true + + /path-type@4.0.0: + resolution: {integrity: sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==} + engines: {node: '>=8'} + + /pathval@1.1.1: + resolution: {integrity: sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==} + dev: true + + /picocolors@1.0.1: + resolution: {integrity: sha512-anP1Z8qwhkbmu7MFP5iTt+wQKXgwzf7zTyGlcdzabySa9vd0Xt392U0rVmz9poOaBj0uHJKyyo9/upk0HrEQew==} + + /picomatch@2.3.1: + resolution: {integrity: sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==} + engines: {node: '>=8.6'} + + /pluralize@8.0.0: + resolution: {integrity: sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==} + engines: {node: '>=4'} + dev: true + + /possible-typed-array-names@1.0.0: + resolution: {integrity: sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==} + engines: {node: '>= 0.4'} + dev: true + + /prelude-ls@1.2.1: + resolution: {integrity: sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==} + engines: {node: '>= 0.8.0'} + dev: true + + /proc-log@4.2.0: + resolution: {integrity: sha512-g8+OnU/L2v+wyiVK+D5fA34J7EH8jZ8DDlvwhRCMxmMj7UCBvxiO1mGeN+36JXIKF4zevU4kRBd8lVgG9vLelA==} + engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} + dev: false + + /proxy-from-env@1.1.0: + resolution: {integrity: sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==} + dev: false + + /punycode@2.3.1: + resolution: {integrity: sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==} + engines: {node: '>=6'} + dev: true + + /queue-microtask@1.2.3: + resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} + + /quick-lru@5.1.1: + resolution: {integrity: sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA==} + engines: {node: '>=10'} + dev: true + + /rambda@7.5.0: + resolution: {integrity: sha512-y/M9weqWAH4iopRd7EHDEQQvpFPHj1AA3oHozE9tfITHUtTR7Z9PSlIRRG2l1GuW7sefC1cXFfIcF+cgnShdBA==} + dev: true + + /randombytes@2.1.0: + resolution: {integrity: sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==} + dependencies: + safe-buffer: 5.2.1 + dev: true + + /read-pkg-up@7.0.1: + resolution: {integrity: sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==} + engines: {node: '>=8'} + dependencies: + find-up: 4.1.0 + read-pkg: 5.2.0 + type-fest: 0.8.1 + dev: true + + /read-pkg@5.2.0: + resolution: {integrity: sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==} + engines: {node: '>=8'} + dependencies: + '@types/normalize-package-data': 2.4.4 + normalize-package-data: 2.5.0 + parse-json: 5.2.0 + type-fest: 0.6.0 + dev: true + + /readdirp@3.6.0: + resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} + engines: {node: '>=8.10.0'} + dependencies: + picomatch: 2.3.1 + dev: true + + /rechoir@0.6.2: + resolution: {integrity: sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==} + engines: {node: '>= 0.10'} + dependencies: + resolve: 1.22.8 + dev: true + + /redeyed@2.1.1: + resolution: {integrity: sha512-FNpGGo1DycYAdnrKFxCMmKYgo/mILAqtRYbkdQD8Ep/Hk2PQ5+aEAEx+IU713RTDmuBaH0c8P5ZozurNu5ObRQ==} + dependencies: + esprima: 4.0.1 + + /regexp-tree@0.1.27: + resolution: {integrity: sha512-iETxpjK6YoRWJG5o6hXLwvjYAoW+FEZn9os0PD/b6AP6xQwsa/Y7lCVgIixBbUPMfhu+i2LtdeAqVTgGlQarfA==} + hasBin: true + dev: true + + /regexp.prototype.flags@1.5.2: + resolution: {integrity: sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-errors: 1.3.0 + set-function-name: 2.0.2 + dev: true + + /regexpp@3.2.0: + resolution: {integrity: sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==} + engines: {node: '>=8'} + dev: true + + /regjsparser@0.10.0: + resolution: {integrity: sha512-qx+xQGZVsy55CH0a1hiVwHmqjLryfh7wQyF5HO07XJ9f7dQMY/gPQHhlyDkIzJKC+x2fUCpCcUODUUUFrm7SHA==} + hasBin: true + dependencies: + jsesc: 0.5.0 + dev: true + + /require-directory@2.1.1: + resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} + engines: {node: '>=0.10.0'} + dev: true + + /resolve-alpn@1.2.1: + resolution: {integrity: sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g==} + dev: true + + /resolve-from@4.0.0: + resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} + engines: {node: '>=4'} + + /resolve-pkg-maps@1.0.0: + resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} + dev: true + + /resolve@1.22.8: + resolution: {integrity: sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==} + hasBin: true + dependencies: + is-core-module: 2.13.1 + path-parse: 1.0.7 + supports-preserve-symlinks-flag: 1.0.0 + dev: true + + /responselike@3.0.0: + resolution: {integrity: sha512-40yHxbNcl2+rzXvZuVkrYohathsSJlMTXKryG5y8uciHv1+xDLHQpgjG64JUO9nrEq2jGLH6IZ8BcZyw3wrweg==} + engines: {node: '>=14.16'} + dependencies: + lowercase-keys: 3.0.0 + dev: true + + /retry@0.13.1: + resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} + engines: {node: '>= 4'} + dev: true + + /reusify@1.0.4: + resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} + engines: {iojs: '>=1.0.0', node: '>=0.10.0'} + + /rimraf@3.0.2: + resolution: {integrity: sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==} + deprecated: Rimraf versions prior to v4 are no longer supported + hasBin: true + dependencies: + glob: 7.2.3 + dev: true + + /run-parallel@1.2.0: + resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} + dependencies: + queue-microtask: 1.2.3 + + /safe-array-concat@1.1.2: + resolution: {integrity: sha512-vj6RsCsWBCf19jIeHEfkRMw8DPiBb+DMXklQ/1SGDHOMlHdPUkZXFQ2YdplS23zESTijAcurb1aSgJA3AgMu1Q==} + engines: {node: '>=0.4'} + dependencies: + call-bind: 1.0.7 + get-intrinsic: 1.2.4 + has-symbols: 1.0.3 + isarray: 2.0.5 + dev: true + + /safe-buffer@5.2.1: + resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} + dev: true + + /safe-regex-test@1.0.3: + resolution: {integrity: sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + es-errors: 1.3.0 + is-regex: 1.1.4 + dev: true + + /semver@5.7.2: + resolution: {integrity: sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==} + hasBin: true + dev: true + + /semver@6.3.1: + resolution: {integrity: sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==} + hasBin: true + dev: true + + /semver@7.6.2: + resolution: {integrity: sha512-FNAIBWCx9qcRhoHcgcJ0gvU7SN1lYU2ZXuSfl04bSC5OpvDHFyJCjdNHomPXxjQlCBU67YW64PzY7/VIEH7F2w==} + engines: {node: '>=10'} + hasBin: true + + /sentence-case@3.0.4: + resolution: {integrity: sha512-8LS0JInaQMCRoQ7YUytAo/xUu5W2XnQxV2HI/6uM6U7CITS1RqPElr30V6uIqyMKM9lJGRVFy5/4CuzcixNYSg==} + dependencies: + no-case: 3.0.4 + tslib: 2.6.2 + upper-case-first: 2.0.2 + dev: true + + /serialize-javascript@6.0.0: + resolution: {integrity: sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==} + dependencies: + randombytes: 2.1.0 + dev: true + + /set-function-length@1.2.2: + resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} + engines: {node: '>= 0.4'} + dependencies: + define-data-property: 1.1.4 + es-errors: 1.3.0 + function-bind: 1.1.2 + get-intrinsic: 1.2.4 + gopd: 1.0.1 + has-property-descriptors: 1.0.2 + dev: true + + /set-function-name@2.0.2: + resolution: {integrity: sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==} + engines: {node: '>= 0.4'} + dependencies: + define-data-property: 1.1.4 + es-errors: 1.3.0 + functions-have-names: 1.2.3 + has-property-descriptors: 1.0.2 + dev: true + + /shebang-command@2.0.0: + resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} + engines: {node: '>=8'} + dependencies: + shebang-regex: 3.0.0 + + /shebang-regex@3.0.0: + resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} + engines: {node: '>=8'} + + /shelljs@0.8.5: + resolution: {integrity: sha512-TiwcRcrkhHvbrZbnRcFYMLl30Dfov3HKqzp5tO5b4pt6G/SezKcYhmDg15zXVBswHmctSAQKznqNW2LO5tTDow==} + engines: {node: '>=4'} + hasBin: true + dependencies: + glob: 7.2.3 + interpret: 1.4.0 + rechoir: 0.6.2 + dev: true + + /shx@0.3.4: + resolution: {integrity: sha512-N6A9MLVqjxZYcVn8hLmtneQWIJtp8IKzMP4eMnx+nqkvXoqinUPCbUFLp2UcWTEIUONhlk0ewxr/jaVGlc+J+g==} + engines: {node: '>=6'} + hasBin: true + dependencies: + minimist: 1.2.8 + shelljs: 0.8.5 + dev: true + + /side-channel@1.0.6: + resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + es-errors: 1.3.0 + get-intrinsic: 1.2.4 + object-inspect: 1.13.1 + dev: true + + /signal-exit@4.1.0: + resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} + engines: {node: '>=14'} + dev: true + + /simple-swizzle@0.2.2: + resolution: {integrity: sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==} + dependencies: + is-arrayish: 0.3.2 + + /slash@3.0.0: + resolution: {integrity: sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==} + engines: {node: '>=8'} + + /slash@4.0.0: + resolution: {integrity: sha512-3dOsAHXXUkQTpOYcoAxLIorMTp4gIQr5IW3iVb7A7lFIp0VHhnynm9izx6TssdrIcVIESAlVjtnO2K8bg+Coew==} + engines: {node: '>=12'} + dev: true + + /slice-ansi@4.0.0: + resolution: {integrity: sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==} + engines: {node: '>=10'} + dependencies: + ansi-styles: 4.3.0 + astral-regex: 2.0.0 + is-fullwidth-code-point: 3.0.0 + + /snake-case@3.0.4: + resolution: {integrity: sha512-LAOh4z89bGQvl9pFfNF8V146i7o7/CqFPbqzYgP+yYzDIDeS9HaNFtXABamRW+AQzEVODcvE79ljJ+8a9YSdMg==} + dependencies: + dot-case: 3.0.4 + tslib: 2.6.2 + dev: true + + /sort-object-keys@1.1.3: + resolution: {integrity: sha512-855pvK+VkU7PaKYPc+Jjnmt4EzejQHyhhF33q31qG8x7maDzkeFhAAThdCYay11CISO+qAMwjOBP+fPZe0IPyg==} + dev: true + + /sort-package-json@2.10.0: + resolution: {integrity: sha512-MYecfvObMwJjjJskhxYfuOADkXp1ZMMnCFC8yhp+9HDsk7HhR336hd7eiBs96lTXfiqmUNI+WQCeCMRBhl251g==} + hasBin: true + dependencies: + detect-indent: 7.0.1 + detect-newline: 4.0.1 + get-stdin: 9.0.0 + git-hooks-list: 3.1.0 + globby: 13.2.2 + is-plain-obj: 4.1.0 + semver: 7.6.2 + sort-object-keys: 1.1.3 + dev: true + + /spdx-correct@3.2.0: + resolution: {integrity: sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==} + dependencies: + spdx-expression-parse: 3.0.1 + spdx-license-ids: 3.0.18 + dev: true + + /spdx-exceptions@2.5.0: + resolution: {integrity: sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w==} + dev: true + + /spdx-expression-parse@3.0.1: + resolution: {integrity: sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==} + dependencies: + spdx-exceptions: 2.5.0 + spdx-license-ids: 3.0.18 + dev: true + + /spdx-license-ids@3.0.18: + resolution: {integrity: sha512-xxRs31BqRYHwiMzudOrpSiHtZ8i/GeionCBDSilhYRj+9gIcI8wCZTlXZKu9vZIVqViP3dcp9qE5G6AlIaD+TQ==} + dev: true + + /sprintf-js@1.0.3: + resolution: {integrity: sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==} + + /string-width@4.2.3: + resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} + engines: {node: '>=8'} + dependencies: + emoji-regex: 8.0.0 + is-fullwidth-code-point: 3.0.0 + strip-ansi: 6.0.1 + + /string.prototype.trim@1.2.9: + resolution: {integrity: sha512-klHuCNxiMZ8MlsOihJhJEBJAiMVqU3Z2nEXWfWnIqjN0gEFS9J9+IxKozWWtQGcgoa1WUZzLjKPTr4ZHNFTFxw==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-abstract: 1.23.3 + es-object-atoms: 1.0.0 + dev: true + + /string.prototype.trimend@1.0.8: + resolution: {integrity: sha512-p73uL5VCHCO2BZZ6krwwQE3kCzM7NKmis8S//xEC6fQonchbum4eP6kR4DLEjQFO3Wnj3Fuo8NM0kOSjVdHjZQ==} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-object-atoms: 1.0.0 + dev: true + + /string.prototype.trimstart@1.0.8: + resolution: {integrity: sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + define-properties: 1.2.1 + es-object-atoms: 1.0.0 + dev: true + + /strip-ansi@6.0.1: + resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} + engines: {node: '>=8'} + dependencies: + ansi-regex: 5.0.1 + + /strip-bom@3.0.0: + resolution: {integrity: sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==} + engines: {node: '>=4'} + dev: true + + /strip-indent@3.0.0: + resolution: {integrity: sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ==} + engines: {node: '>=8'} + dependencies: + min-indent: 1.0.1 + dev: true + + /strip-json-comments@3.1.1: + resolution: {integrity: sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==} + engines: {node: '>=8'} + dev: true + + /strnum@1.0.5: + resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} + dev: true + + /supports-color@5.5.0: + resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} + engines: {node: '>=4'} + dependencies: + has-flag: 3.0.0 + + /supports-color@7.2.0: + resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} + engines: {node: '>=8'} + dependencies: + has-flag: 4.0.0 + + /supports-color@8.1.1: + resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} + engines: {node: '>=10'} + dependencies: + has-flag: 4.0.0 + + /supports-color@9.4.0: + resolution: {integrity: sha512-VL+lNrEoIXww1coLPOmiEmK/0sGigko5COxI09KzHc2VJXJsQ37UaQ+8quuxjDeA7+KnLGTWRyOXSLLR2Wb4jw==} + engines: {node: '>=12'} + + /supports-hyperlinks@2.3.0: + resolution: {integrity: sha512-RpsAZlpWcDwOPQA22aCH4J0t7L8JmAvsCxfOSEwm7cQs3LshN36QaTkwd70DnBOXDWGssw2eUoc8CaRWT0XunA==} + engines: {node: '>=8'} + dependencies: + has-flag: 4.0.0 + supports-color: 7.2.0 + + /supports-preserve-symlinks-flag@1.0.0: + resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} + engines: {node: '>= 0.4'} + dev: true + + /tapable@2.2.1: + resolution: {integrity: sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==} + engines: {node: '>=6'} + dev: true + + /text-table@0.2.0: + resolution: {integrity: sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==} + dev: true + + /tiny-jsonc@1.0.1: + resolution: {integrity: sha512-ik6BCxzva9DoiEfDX/li0L2cWKPPENYvixUprFdl3YPi4bZZUhDnNI9YUkacrv+uIG90dnxR5mNqaoD6UhD6Bw==} + dev: true + + /to-regex-range@5.0.1: + resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} + engines: {node: '>=8.0'} + dependencies: + is-number: 7.0.0 + + /ts-api-utils@1.3.0(typescript@5.4.5): + resolution: {integrity: sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ==} + engines: {node: '>=16'} + peerDependencies: + typescript: '>=4.2.0' + dependencies: + typescript: 5.4.5 + dev: true + + /ts-node@10.9.2(@types/node@18.19.34)(typescript@5.4.5): + resolution: {integrity: sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==} + hasBin: true + peerDependencies: + '@swc/core': '>=1.2.50' + '@swc/wasm': '>=1.2.50' + '@types/node': '*' + typescript: '>=2.7' + peerDependenciesMeta: + '@swc/core': + optional: true + '@swc/wasm': + optional: true + dependencies: + '@cspotcode/source-map-support': 0.8.1 + '@tsconfig/node10': 1.0.11 + '@tsconfig/node12': 1.0.11 + '@tsconfig/node14': 1.0.3 + '@tsconfig/node16': 1.0.4 + '@types/node': 18.19.34 + acorn: 8.11.3 + acorn-walk: 8.3.2 + arg: 4.1.3 + create-require: 1.1.1 + diff: 4.0.2 + make-error: 1.3.6 + typescript: 5.4.5 + v8-compile-cache-lib: 3.0.1 + yn: 3.1.1 + dev: true + + /tsconfig-paths@3.15.0: + resolution: {integrity: sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==} + dependencies: + '@types/json5': 0.0.29 + json5: 1.0.2 + minimist: 1.2.8 + strip-bom: 3.0.0 + dev: true + + /tslib@1.14.1: + resolution: {integrity: sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==} + dev: true + + /tslib@2.6.2: + resolution: {integrity: sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==} + dev: true + + /tunnel-agent@0.6.0: + resolution: {integrity: sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==} + dependencies: + safe-buffer: 5.2.1 + dev: true + + /type-check@0.4.0: + resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==} + engines: {node: '>= 0.8.0'} + dependencies: + prelude-ls: 1.2.1 + dev: true + + /type-detect@4.0.8: + resolution: {integrity: sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==} + engines: {node: '>=4'} + dev: true + + /type-fest@0.20.2: + resolution: {integrity: sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==} + engines: {node: '>=10'} + dev: true + + /type-fest@0.21.3: + resolution: {integrity: sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==} + engines: {node: '>=10'} + + /type-fest@0.6.0: + resolution: {integrity: sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==} + engines: {node: '>=8'} + dev: true + + /type-fest@0.8.1: + resolution: {integrity: sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==} + engines: {node: '>=8'} + dev: true + + /typed-array-buffer@1.0.2: + resolution: {integrity: sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + es-errors: 1.3.0 + is-typed-array: 1.1.13 + dev: true + + /typed-array-byte-length@1.0.1: + resolution: {integrity: sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + for-each: 0.3.3 + gopd: 1.0.1 + has-proto: 1.0.3 + is-typed-array: 1.1.13 + dev: true + + /typed-array-byte-offset@1.0.2: + resolution: {integrity: sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==} + engines: {node: '>= 0.4'} + dependencies: + available-typed-arrays: 1.0.7 + call-bind: 1.0.7 + for-each: 0.3.3 + gopd: 1.0.1 + has-proto: 1.0.3 + is-typed-array: 1.1.13 + dev: true + + /typed-array-length@1.0.6: + resolution: {integrity: sha512-/OxDN6OtAk5KBpGb28T+HZc2M+ADtvRxXrKKbUwtsLgdoxgX13hyy7ek6bFRl5+aBs2yZzB0c4CnQfAtVypW/g==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + for-each: 0.3.3 + gopd: 1.0.1 + has-proto: 1.0.3 + is-typed-array: 1.1.13 + possible-typed-array-names: 1.0.0 + dev: true + + /typescript@5.4.5: + resolution: {integrity: sha512-vcI4UpRgg81oIRUFwR0WSIHKt11nJ7SAVlYNIu+QpqeyXP+gpQJy/Z4+F0aGxSE4MqwjyXvW/TzgkLAx2AGHwQ==} + engines: {node: '>=14.17'} + hasBin: true + + /unbox-primitive@1.0.2: + resolution: {integrity: sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==} + dependencies: + call-bind: 1.0.7 + has-bigints: 1.0.2 + has-symbols: 1.0.3 + which-boxed-primitive: 1.0.2 + dev: true + + /undici-types@5.26.5: + resolution: {integrity: sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==} + + /universalify@0.1.2: + resolution: {integrity: sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==} + engines: {node: '>= 4.0.0'} + dev: true + + /upper-case-first@2.0.2: + resolution: {integrity: sha512-514ppYHBaKwfJRK/pNC6c/OxfGa0obSnAl106u97Ed0I625Nin96KAjttZF6ZL3e1XLtphxnqrOi9iWgm+u+bg==} + dependencies: + tslib: 2.6.2 + dev: true + + /upper-case@2.0.2: + resolution: {integrity: sha512-KgdgDGJt2TpuwBUIjgG6lzw2GWFRCW9Qkfkiv0DxqHHLYJHmtmdUIKcZd8rHgFSjopVTlw6ggzCm1b8MFQwikg==} + dependencies: + tslib: 2.6.2 + dev: true + + /uri-js@4.4.1: + resolution: {integrity: sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==} + dependencies: + punycode: 2.3.1 + dev: true + + /uuid@9.0.1: + resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} + hasBin: true + dev: true + + /v8-compile-cache-lib@3.0.1: + resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==} + dev: true + + /validate-npm-package-license@3.0.4: + resolution: {integrity: sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==} + dependencies: + spdx-correct: 3.2.0 + spdx-expression-parse: 3.0.1 + dev: true + + /validate-npm-package-name@5.0.1: + resolution: {integrity: sha512-OljLrQ9SQdOUqTaQxqL5dEfZWrXExyyWsozYlAWFawPVNuD83igl7uJD2RTkNMbniIYgt8l81eCJGIdQF7avLQ==} + engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} + + /which-boxed-primitive@1.0.2: + resolution: {integrity: sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==} + dependencies: + is-bigint: 1.0.4 + is-boolean-object: 1.1.2 + is-number-object: 1.0.7 + is-string: 1.0.7 + is-symbol: 1.0.4 + dev: true + + /which-typed-array@1.1.15: + resolution: {integrity: sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==} + engines: {node: '>= 0.4'} + dependencies: + available-typed-arrays: 1.0.7 + call-bind: 1.0.7 + for-each: 0.3.3 + gopd: 1.0.1 + has-tostringtag: 1.0.2 + dev: true + + /which@2.0.2: + resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} + engines: {node: '>= 8'} + hasBin: true + dependencies: + isexe: 2.0.0 + + /which@4.0.0: + resolution: {integrity: sha512-GlaYyEb07DPxYCKhKzplCWBJtvxZcZMrL+4UkrTSJHHPyZU4mYYTv3qaOe77H7EODLSSopAUFAc6W8U4yqvscg==} + engines: {node: ^16.13.0 || >=18.0.0} + hasBin: true + dependencies: + isexe: 3.1.1 + dev: false + + /widest-line@3.1.0: + resolution: {integrity: sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==} + engines: {node: '>=8'} + dependencies: + string-width: 4.2.3 + + /word-wrap@1.2.5: + resolution: {integrity: sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==} + engines: {node: '>=0.10.0'} + dev: true + + /wordwrap@1.0.0: + resolution: {integrity: sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==} + + /workerpool@6.2.1: + resolution: {integrity: sha512-ILEIE97kDZvF9Wb9f6h5aXK4swSlKGUcOEGiIYb2OOu/IrDU9iwj0fD//SsA6E5ibwJxpEvhullJY4Sl4GcpAw==} + dev: true + + /wrap-ansi@6.2.0: + resolution: {integrity: sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==} + engines: {node: '>=8'} + dependencies: + ansi-styles: 4.3.0 + string-width: 4.2.3 + strip-ansi: 6.0.1 + dev: true + + /wrap-ansi@7.0.0: + resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} + engines: {node: '>=10'} + dependencies: + ansi-styles: 4.3.0 + string-width: 4.2.3 + strip-ansi: 6.0.1 + + /wrappy@1.0.2: + resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==} + dev: true + + /y18n@5.0.8: + resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} + engines: {node: '>=10'} + dev: true + + /yargs-parser@20.2.4: + resolution: {integrity: sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==} + engines: {node: '>=10'} + dev: true + + /yargs-unparser@2.0.0: + resolution: {integrity: sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==} + engines: {node: '>=10'} + dependencies: + camelcase: 6.3.0 + decamelize: 4.0.0 + flat: 5.0.2 + is-plain-obj: 2.1.0 + dev: true + + /yargs@16.2.0: + resolution: {integrity: sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==} + engines: {node: '>=10'} + dependencies: + cliui: 7.0.4 + escalade: 3.1.2 + get-caller-file: 2.0.5 + require-directory: 2.1.1 + string-width: 4.2.3 + y18n: 5.0.8 + yargs-parser: 20.2.4 + dev: true + + /yarn@1.22.22: + resolution: {integrity: sha512-prL3kGtyG7o9Z9Sv8IPfBNrWTDmXB4Qbes8A9rEzt6wkJV8mUvoirjU0Mp3GGAU06Y0XQyA3/2/RQFVuK7MTfg==} + engines: {node: '>=4.0.0'} + hasBin: true + requiresBuild: true + dev: false + + /yn@3.1.1: + resolution: {integrity: sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==} + engines: {node: '>=6'} + dev: true + + /yocto-queue@0.1.0: + resolution: {integrity: sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==} + engines: {node: '>=10'} + dev: true diff --git a/readme.md b/readme.md index ecfe9df..1bca5ef 100644 --- a/readme.md +++ b/readme.md @@ -1 +1,396 @@ -# cli +dokploy +================= + +A CLI to manage dokploy server remotely + + +[![oclif](https://img.shields.io/badge/cli-oclif-brightgreen.svg)](https://oclif.io) +[![Version](https://img.shields.io/npm/v/dokploy.svg)](https://npmjs.org/package/dokploy) +[![Downloads/week](https://img.shields.io/npm/dw/dokploy.svg)](https://npmjs.org/package/dokploy) + + + +* [Usage](#usage) +* [Commands](#commands) + +# Usage + +```sh-session +$ npm install -g dokploy +$ dokploy COMMAND +running command... +$ dokploy (--version) +dokploy/0.0.0 darwin-arm64 node-v18.18.0 +$ dokploy --help [COMMAND] +USAGE + $ dokploy COMMAND +... +``` + +# Commands + +* [`dokploy hello PERSON`](#dokploy-hello-person) +* [`dokploy hello world`](#dokploy-hello-world) +* [`dokploy help [COMMAND]`](#dokploy-help-command) +* [`dokploy plugins`](#dokploy-plugins) +* [`dokploy plugins add PLUGIN`](#dokploy-plugins-add-plugin) +* [`dokploy plugins:inspect PLUGIN...`](#dokploy-pluginsinspect-plugin) +* [`dokploy plugins install PLUGIN`](#dokploy-plugins-install-plugin) +* [`dokploy plugins link PATH`](#dokploy-plugins-link-path) +* [`dokploy plugins remove [PLUGIN]`](#dokploy-plugins-remove-plugin) +* [`dokploy plugins reset`](#dokploy-plugins-reset) +* [`dokploy plugins uninstall [PLUGIN]`](#dokploy-plugins-uninstall-plugin) +* [`dokploy plugins unlink [PLUGIN]`](#dokploy-plugins-unlink-plugin) +* [`dokploy plugins update`](#dokploy-plugins-update) + +## `dokploy hello PERSON` + +Say hello + +``` +USAGE + $ dokploy hello PERSON -f + +ARGUMENTS + PERSON Person to say hello to + +FLAGS + -f, --from= (required) Who is saying hello + +DESCRIPTION + Say hello + +EXAMPLES + $ dokploy hello friend --from oclif + hello friend from oclif! (./src/commands/hello/index.ts) +``` + +_See code: [src/commands/hello/index.ts](https://github.com/Dokploy/cli/blob/v0.0.0/src/commands/hello/index.ts)_ + +## `dokploy hello world` + +Say hello world + +``` +USAGE + $ dokploy hello world + +DESCRIPTION + Say hello world + +EXAMPLES + $ dokploy hello world + hello world! (./src/commands/hello/world.ts) +``` + +_See code: [src/commands/hello/world.ts](https://github.com/Dokploy/cli/blob/v0.0.0/src/commands/hello/world.ts)_ + +## `dokploy help [COMMAND]` + +Display help for dokploy. + +``` +USAGE + $ dokploy help [COMMAND...] [-n] + +ARGUMENTS + COMMAND... Command to show help for. + +FLAGS + -n, --nested-commands Include all nested commands in the output. + +DESCRIPTION + Display help for dokploy. +``` + +_See code: [@oclif/plugin-help](https://github.com/oclif/plugin-help/blob/v6.1.0/src/commands/help.ts)_ + +## `dokploy plugins` + +List installed plugins. + +``` +USAGE + $ dokploy plugins [--json] [--core] + +FLAGS + --core Show core plugins. + +GLOBAL FLAGS + --json Format output as json. + +DESCRIPTION + List installed plugins. + +EXAMPLES + $ dokploy plugins +``` + +_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.2.1/src/commands/plugins/index.ts)_ + +## `dokploy plugins add PLUGIN` + +Installs a plugin into dokploy. + +``` +USAGE + $ dokploy plugins add PLUGIN... [--json] [-f] [-h] [-s | -v] + +ARGUMENTS + PLUGIN... Plugin to install. + +FLAGS + -f, --force Force npm to fetch remote resources even if a local copy exists on disk. + -h, --help Show CLI help. + -s, --silent Silences npm output. + -v, --verbose Show verbose npm output. + +GLOBAL FLAGS + --json Format output as json. + +DESCRIPTION + Installs a plugin into dokploy. + + Uses bundled npm executable to install plugins into /Users/mauricio/.local/share/dokploy + + Installation of a user-installed plugin will override a core plugin. + + Use the DOKPLOY_NPM_LOG_LEVEL environment variable to set the npm loglevel. + Use the DOKPLOY_NPM_REGISTRY environment variable to set the npm registry. + +ALIASES + $ dokploy plugins add + +EXAMPLES + Install a plugin from npm registry. + + $ dokploy plugins add myplugin + + Install a plugin from a github url. + + $ dokploy plugins add https://github.com/someuser/someplugin + + Install a plugin from a github slug. + + $ dokploy plugins add someuser/someplugin +``` + +## `dokploy plugins:inspect PLUGIN...` + +Displays installation properties of a plugin. + +``` +USAGE + $ dokploy plugins inspect PLUGIN... + +ARGUMENTS + PLUGIN... [default: .] Plugin to inspect. + +FLAGS + -h, --help Show CLI help. + -v, --verbose + +GLOBAL FLAGS + --json Format output as json. + +DESCRIPTION + Displays installation properties of a plugin. + +EXAMPLES + $ dokploy plugins inspect myplugin +``` + +_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.2.1/src/commands/plugins/inspect.ts)_ + +## `dokploy plugins install PLUGIN` + +Installs a plugin into dokploy. + +``` +USAGE + $ dokploy plugins install PLUGIN... [--json] [-f] [-h] [-s | -v] + +ARGUMENTS + PLUGIN... Plugin to install. + +FLAGS + -f, --force Force npm to fetch remote resources even if a local copy exists on disk. + -h, --help Show CLI help. + -s, --silent Silences npm output. + -v, --verbose Show verbose npm output. + +GLOBAL FLAGS + --json Format output as json. + +DESCRIPTION + Installs a plugin into dokploy. + + Uses bundled npm executable to install plugins into /Users/mauricio/.local/share/dokploy + + Installation of a user-installed plugin will override a core plugin. + + Use the DOKPLOY_NPM_LOG_LEVEL environment variable to set the npm loglevel. + Use the DOKPLOY_NPM_REGISTRY environment variable to set the npm registry. + +ALIASES + $ dokploy plugins add + +EXAMPLES + Install a plugin from npm registry. + + $ dokploy plugins install myplugin + + Install a plugin from a github url. + + $ dokploy plugins install https://github.com/someuser/someplugin + + Install a plugin from a github slug. + + $ dokploy plugins install someuser/someplugin +``` + +_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.2.1/src/commands/plugins/install.ts)_ + +## `dokploy plugins link PATH` + +Links a plugin into the CLI for development. + +``` +USAGE + $ dokploy plugins link PATH [-h] [--install] [-v] + +ARGUMENTS + PATH [default: .] path to plugin + +FLAGS + -h, --help Show CLI help. + -v, --verbose + --[no-]install Install dependencies after linking the plugin. + +DESCRIPTION + Links a plugin into the CLI for development. + Installation of a linked plugin will override a user-installed or core plugin. + + e.g. If you have a user-installed or core plugin that has a 'hello' command, installing a linked plugin with a 'hello' + command will override the user-installed or core plugin implementation. This is useful for development work. + + +EXAMPLES + $ dokploy plugins link myplugin +``` + +_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.2.1/src/commands/plugins/link.ts)_ + +## `dokploy plugins remove [PLUGIN]` + +Removes a plugin from the CLI. + +``` +USAGE + $ dokploy plugins remove [PLUGIN...] [-h] [-v] + +ARGUMENTS + PLUGIN... plugin to uninstall + +FLAGS + -h, --help Show CLI help. + -v, --verbose + +DESCRIPTION + Removes a plugin from the CLI. + +ALIASES + $ dokploy plugins unlink + $ dokploy plugins remove + +EXAMPLES + $ dokploy plugins remove myplugin +``` + +## `dokploy plugins reset` + +Remove all user-installed and linked plugins. + +``` +USAGE + $ dokploy plugins reset [--hard] [--reinstall] + +FLAGS + --hard Delete node_modules and package manager related files in addition to uninstalling plugins. + --reinstall Reinstall all plugins after uninstalling. +``` + +_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.2.1/src/commands/plugins/reset.ts)_ + +## `dokploy plugins uninstall [PLUGIN]` + +Removes a plugin from the CLI. + +``` +USAGE + $ dokploy plugins uninstall [PLUGIN...] [-h] [-v] + +ARGUMENTS + PLUGIN... plugin to uninstall + +FLAGS + -h, --help Show CLI help. + -v, --verbose + +DESCRIPTION + Removes a plugin from the CLI. + +ALIASES + $ dokploy plugins unlink + $ dokploy plugins remove + +EXAMPLES + $ dokploy plugins uninstall myplugin +``` + +_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.2.1/src/commands/plugins/uninstall.ts)_ + +## `dokploy plugins unlink [PLUGIN]` + +Removes a plugin from the CLI. + +``` +USAGE + $ dokploy plugins unlink [PLUGIN...] [-h] [-v] + +ARGUMENTS + PLUGIN... plugin to uninstall + +FLAGS + -h, --help Show CLI help. + -v, --verbose + +DESCRIPTION + Removes a plugin from the CLI. + +ALIASES + $ dokploy plugins unlink + $ dokploy plugins remove + +EXAMPLES + $ dokploy plugins unlink myplugin +``` + +## `dokploy plugins update` + +Update installed plugins. + +``` +USAGE + $ dokploy plugins update [-h] [-v] + +FLAGS + -h, --help Show CLI help. + -v, --verbose + +DESCRIPTION + Update installed plugins. +``` + +_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.2.1/src/commands/plugins/update.ts)_ + diff --git a/src/commands/authenticate.ts b/src/commands/authenticate.ts new file mode 100644 index 0000000..07da036 --- /dev/null +++ b/src/commands/authenticate.ts @@ -0,0 +1,29 @@ +import { Command, Flags } from "@oclif/core"; +import * as fs from "node:fs"; +import * as path from "node:path"; + +const configPath = path.join(__dirname, "..", "..", "config.json"); + +export default class Authenticate extends Command { + static description = "Authenticate the user by saving server URL and token"; + + static flags = { + token: Flags.string({ + char: "t", + description: "Authentication token", + required: true, + }), + url: Flags.string({ char: "u", description: "Server URL", required: true }), + }; + + async run() { + const { flags } = await this.parse(Authenticate); + const config = { + token: flags.token, + url: flags.url, + }; + + fs.writeFileSync(configPath, JSON.stringify(config, null, 2)); + this.log("Authentication details saved successfully."); + } +} diff --git a/src/commands/check-server.ts b/src/commands/check-server.ts new file mode 100644 index 0000000..7cb55de --- /dev/null +++ b/src/commands/check-server.ts @@ -0,0 +1,30 @@ +import {Args, Command, Flags} from '@oclif/core' + +export default class CheckServer extends Command { + static override args = { + file: Args.string({description: 'file to read'}), + } + + static override description = 'describe the command here' + + static override examples = [ + '<%= config.bin %> <%= command.id %>', + ] + + static override flags = { + // flag with no value (-f, --force) + force: Flags.boolean({char: 'f'}), + // flag with a value (-n, --name=VALUE) + name: Flags.string({char: 'n', description: 'name to print'}), + } + + public async run(): Promise { + const {args, flags} = await this.parse(CheckServer) + + const name = flags.name ?? 'world' + this.log(`hello ${name} from /Users/mauricio/Documents/Github/Personal/cli/src/commands/check-server.ts`) + if (args.file && flags.force) { + this.log(`you input --force and --file: ${args.file}`) + } + } +} diff --git a/src/commands/hello/greet.ts b/src/commands/hello/greet.ts new file mode 100644 index 0000000..3b77354 --- /dev/null +++ b/src/commands/hello/greet.ts @@ -0,0 +1,29 @@ +import { Command, Flags } from "@oclif/core"; + +export default class Greet extends Command { + static args = { + // Puedes agregar argumentos adicionales si lo necesitas + }; + + static description = "Greet the user with a message"; + + static examples = [ + `<%= config.bin %> <%= command.id %> --name=Mau +hello Mau! +`, + ]; + + static flags = { + name: Flags.string({ + char: "n", + description: "name to greet", + required: false, + }), + }; + + async run(): Promise { + const { flags } = await this.parse(Greet); + const name = flags.name ?? "user"; + this.log(`hello ${name}!`); + } +} diff --git a/src/commands/hello/index.ts b/src/commands/hello/index.ts new file mode 100644 index 0000000..f7b639c --- /dev/null +++ b/src/commands/hello/index.ts @@ -0,0 +1,34 @@ +import { Args, Command, Flags } from "@oclif/core"; + +export default class Hello extends Command { + static args = { + person: Args.string({ + description: "Person to say hello to", + required: true, + }), + }; + + static description = "Say hello"; + + static examples = [ + `<%= config.bin %> <%= command.id %> friend --from oclif +hello friend from oclif! (./src/commands/hello/index.ts) +`, + ]; + + static flags = { + from: Flags.string({ + char: "f", + description: "Who is saying hello", + required: true, + }), + }; + + async run(): Promise { + const { args, flags } = await this.parse(Hello); + + this.log( + `hello ${args.person} from ${flags.from}! (./src/commands/hello/index.ts)`, + ); + } +} diff --git a/src/commands/hello/world.ts b/src/commands/hello/world.ts new file mode 100644 index 0000000..8111043 --- /dev/null +++ b/src/commands/hello/world.ts @@ -0,0 +1,19 @@ +import {Command} from '@oclif/core' + +export default class World extends Command { + static args = {} + + static description = 'Say hello world' + + static examples = [ + `<%= config.bin %> <%= command.id %> +hello world! (./src/commands/hello/world.ts) +`, + ] + + static flags = {} + + async run(): Promise { + this.log('hello world! (./src/commands/hello/world.ts)') + } +} diff --git a/src/index.ts b/src/index.ts new file mode 100644 index 0000000..e32b0b2 --- /dev/null +++ b/src/index.ts @@ -0,0 +1 @@ +export {run} from '@oclif/core' diff --git a/test/commands/check-server.test.ts b/test/commands/check-server.test.ts new file mode 100644 index 0000000..8fbc599 --- /dev/null +++ b/test/commands/check-server.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('check-server', () => { + it('runs check-server cmd', async () => { + const {stdout} = await runCommand('check-server') + expect(stdout).to.contain('hello world') + }) + + it('runs check-server --name oclif', async () => { + const {stdout} = await runCommand('check-server --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/hello/index.test.ts b/test/commands/hello/index.test.ts new file mode 100644 index 0000000..dad0ac3 --- /dev/null +++ b/test/commands/hello/index.test.ts @@ -0,0 +1,9 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('hello', () => { + it('runs hello', async () => { + const {stdout} = await runCommand('hello friend --from oclif') + expect(stdout).to.contain('hello friend from oclif!') + }) +}) diff --git a/test/commands/hello/world.test.ts b/test/commands/hello/world.test.ts new file mode 100644 index 0000000..0f5e90f --- /dev/null +++ b/test/commands/hello/world.test.ts @@ -0,0 +1,9 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('hello world', () => { + it('runs hello world cmd', async () => { + const {stdout} = await runCommand('hello world') + expect(stdout).to.contain('hello world!') + }) +}) diff --git a/test/tsconfig.json b/test/tsconfig.json new file mode 100644 index 0000000..95898fc --- /dev/null +++ b/test/tsconfig.json @@ -0,0 +1,9 @@ +{ + "extends": "../tsconfig", + "compilerOptions": { + "noEmit": true + }, + "references": [ + {"path": ".."} + ] +} diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..e144b78 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,15 @@ +{ + "compilerOptions": { + "declaration": true, + "module": "Node16", + "outDir": "dist", + "rootDir": "src", + "strict": true, + "target": "es2022", + "moduleResolution": "node16" + }, + "include": ["./src/**/*"], + "ts-node": { + "esm": true + } +} From 6dae0587d81b0fe5d78c348eead5997a53a7cfb6 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Tue, 4 Jun 2024 01:10:32 -0600 Subject: [PATCH 02/13] feat: add --- config.json | 4 + package.json | 6 +- pnpm-lock.yaml | 255 ++++++++++++++++++++++++++--- src/commands/authenticate.ts | 84 +++++++++- test/commands/authenticate.test.ts | 14 ++ tsconfig.json | 2 +- 6 files changed, 334 insertions(+), 31 deletions(-) create mode 100644 config.json create mode 100644 test/commands/authenticate.test.ts diff --git a/config.json b/config.json new file mode 100644 index 0000000..cc30a3c --- /dev/null +++ b/config.json @@ -0,0 +1,4 @@ +{ + "token": "hola", + "url": "http://localhost:3000" +} \ No newline at end of file diff --git a/package.json b/package.json index 099d238..510a583 100644 --- a/package.json +++ b/package.json @@ -11,12 +11,16 @@ "@oclif/core": "^3", "@oclif/plugin-help": "^6", "@oclif/plugin-plugins": "^5", - "axios": "^1.7.2" + "axios": "^1.7.2", + "chalk": "^5.3.0", + "inquirer": "^9.2.23", + "superjson": "^2.2.1" }, "devDependencies": { "@oclif/prettier-config": "^0.2.1", "@oclif/test": "^4", "@types/chai": "^4", + "@types/inquirer": "9.0.7", "@types/mocha": "^10", "@types/node": "^18", "chai": "^4", diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 6453d7b..0db1304 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -17,6 +17,15 @@ dependencies: axios: specifier: ^1.7.2 version: 1.7.2 + chalk: + specifier: ^5.3.0 + version: 5.3.0 + inquirer: + specifier: ^9.2.23 + version: 9.2.23 + superjson: + specifier: ^2.2.1 + version: 2.2.1 devDependencies: '@oclif/prettier-config': @@ -28,6 +37,9 @@ devDependencies: '@types/chai': specifier: ^4 version: 4.3.16 + '@types/inquirer': + specifier: 9.0.7 + version: 9.0.7 '@types/mocha': specifier: ^10 version: 10.0.6 @@ -865,7 +877,6 @@ packages: /@inquirer/figures@1.0.3: resolution: {integrity: sha512-ErXXzENMH5pJt5/ssXV0DfWUZqly8nGzf0UcBV9xTnP+KyffE2mqyxIMBrZ8ijQck2nU0TQm40EQB53YreyWHw==} engines: {node: '>=18'} - dev: true /@inquirer/input@2.1.9: resolution: {integrity: sha512-1xTCHmIe48x9CG1+8glAHrVVdH+QfYhzgBUbgyoVpp5NovnXgRcjSn/SNulepxf9Ol8HDq3gzw3ZCAUr+h1Eyg==} @@ -907,6 +918,13 @@ packages: '@jridgewell/sourcemap-codec': 1.4.15 dev: true + /@ljharb/through@2.3.13: + resolution: {integrity: sha512-/gKJun8NNiWGZJkGzI/Ragc53cOdcLNdzjLaIa+GEjguQs0ulsurx8WN0jijdK9yPqDvziX995sMRLyLt1uZMQ==} + engines: {node: '>= 0.4'} + dependencies: + call-bind: 1.0.7 + dev: false + /@nodelib/fs.scandir@2.1.5: resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} engines: {node: '>= 8'} @@ -1574,6 +1592,13 @@ packages: resolution: {integrity: sha512-1m0bIFVc7eJWyve9S0RnuRgcQqF/Xd5QsUZAZeQFr1Q3/p9JWoQQEqmVy+DPTNpGXwhgIetAoYF8JSc33q29QA==} dev: true + /@types/inquirer@9.0.7: + resolution: {integrity: sha512-Q0zyBupO6NxGRZut/JdmqYKOnN95Eg5V8Csg3PGKkP+FnvsUZx1jAyK7fztIszxxMuoBA6E3KXWvdZVXIpx60g==} + dependencies: + '@types/through': 0.0.33 + rxjs: 7.8.1 + dev: true + /@types/json-schema@7.0.15: resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==} dev: true @@ -1611,6 +1636,12 @@ packages: resolution: {integrity: sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==} dev: true + /@types/through@0.0.33: + resolution: {integrity: sha512-HsJ+z3QuETzP3cswwtzt2vEIiHBk/dCcHGhbmG5X3ecnwFD/lPrMpliGXxSCg03L9AhrdwA4Oz/qfspkDW+xGQ==} + dependencies: + '@types/node': 18.19.34 + dev: true + /@types/wrap-ansi@3.0.0: resolution: {integrity: sha512-ltIpx+kM7g/MLRZfkbL7EsCEjfzCcScLpkg37eXEtx5kmrAKBkTJwd1GIAjDSL8wTpM6Hzn5YO4pSb91BEwu1g==} dev: true @@ -2003,11 +2034,23 @@ packages: /balanced-match@1.0.2: resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} + /base64-js@1.5.1: + resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==} + dev: false + /binary-extensions@2.3.0: resolution: {integrity: sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==} engines: {node: '>=8'} dev: true + /bl@4.1.0: + resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==} + dependencies: + buffer: 5.7.1 + inherits: 2.0.4 + readable-stream: 3.6.2 + dev: false + /bowser@2.11.0: resolution: {integrity: sha512-AlcaJBi/pqqJBIQ8U9Mcpc9i8Aqxn88Skv5d+xBX006BY5u8N3mGLHa5Lgppa7L/HfwgwLgZ6NYs+Ag6uUmJRA==} dev: true @@ -2033,6 +2076,13 @@ packages: resolution: {integrity: sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==} dev: true + /buffer@5.7.1: + resolution: {integrity: sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==} + dependencies: + base64-js: 1.5.1 + ieee754: 1.2.1 + dev: false + /builtin-modules@3.3.0: resolution: {integrity: sha512-zhaCDicdLuWN5UbN5IMnFqNMhNfo919sH85y2/ea+5Yg9TsTkeZxpL+JLbp6cgYFS4sRLp3YV4S6yDuqVWHYOw==} engines: {node: '>=6'} @@ -2071,7 +2121,6 @@ packages: function-bind: 1.1.2 get-intrinsic: 1.2.4 set-function-length: 1.2.2 - dev: true /callsites@3.1.0: resolution: {integrity: sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==} @@ -2132,6 +2181,11 @@ packages: ansi-styles: 4.3.0 supports-color: 7.2.0 + /chalk@5.3.0: + resolution: {integrity: sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w==} + engines: {node: ^12.17.0 || ^14.13 || >=16.0.0} + dev: false + /change-case@4.1.2: resolution: {integrity: sha512-bSxY2ws9OtviILG1EiY5K7NNxkqg/JnRnFxLtKQ96JaviiIxi7djMrSd0ECT9AC+lttClmYwKw53BWpOMblo7A==} dependencies: @@ -2149,6 +2203,10 @@ packages: tslib: 2.6.2 dev: true + /chardet@0.7.0: + resolution: {integrity: sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==} + dev: false + /check-error@1.0.3: resolution: {integrity: sha512-iKEoDYaRmd1mxM90a2OEfWhjsjPpYPuQ+lMYsoxB126+t8fw7ySEO48nmDg5COTjxDI65/Y2OWpeEHk3ZOe8zg==} dependencies: @@ -2188,6 +2246,13 @@ packages: dependencies: escape-string-regexp: 4.0.0 + /cli-cursor@3.1.0: + resolution: {integrity: sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==} + engines: {node: '>=8'} + dependencies: + restore-cursor: 3.1.0 + dev: false + /cli-progress@3.12.0: resolution: {integrity: sha512-tRkV3HJ1ASwm19THiiLIXLO7Im7wlTuKnvkYaTkyoAPefqjNg7W7DHKUlGRxy9vxDvbyCYQkQozvptuMkGCg8A==} engines: {node: '>=4'} @@ -2201,7 +2266,6 @@ packages: /cli-width@4.1.0: resolution: {integrity: sha512-ouuZd4/dm2Sw5Gmqy6bGyNNNe1qt9RpmxveLSO7KcgsTnU7RXfsw+/bukWGo1abgBiMAic068rclZsO4IWmmxQ==} engines: {node: '>= 12'} - dev: true /cliui@7.0.4: resolution: {integrity: sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==} @@ -2211,6 +2275,11 @@ packages: wrap-ansi: 7.0.0 dev: true + /clone@1.0.4: + resolution: {integrity: sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==} + engines: {node: '>=0.8'} + dev: false + /color-convert@1.9.3: resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==} dependencies: @@ -2268,6 +2337,13 @@ packages: engines: {node: '>= 0.6'} dev: true + /copy-anything@3.0.5: + resolution: {integrity: sha512-yCEafptTtb4bk7GLEQoM8KVJpxAfdBJYaXyzQEgQQQgYrZiDp8SJmGKlYza6CYjEDNstAdNdKA3UuoULlEbS6w==} + engines: {node: '>=12.13'} + dependencies: + is-what: 4.1.16 + dev: false + /cosmiconfig@9.0.0(typescript@5.4.5): resolution: {integrity: sha512-itvL5h8RETACmOTFc4UfIyB2RfEHi71Ax6E/PivVxq9NseKbOWpeyHEOIbmAw1rs8Ak0VursQNww7lf7YtUwzg==} engines: {node: '>=14'} @@ -2393,6 +2469,12 @@ packages: resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} dev: true + /defaults@1.0.4: + resolution: {integrity: sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==} + dependencies: + clone: 1.0.4 + dev: false + /defer-to-connect@2.0.1: resolution: {integrity: sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg==} engines: {node: '>=10'} @@ -2405,7 +2487,6 @@ packages: es-define-property: 1.0.0 es-errors: 1.3.0 gopd: 1.0.1 - dev: true /define-properties@1.2.1: resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==} @@ -2552,12 +2633,10 @@ packages: engines: {node: '>= 0.4'} dependencies: get-intrinsic: 1.2.4 - dev: true /es-errors@1.3.0: resolution: {integrity: sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==} engines: {node: '>= 0.4'} - dev: true /es-object-atoms@1.0.0: resolution: {integrity: sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==} @@ -2982,6 +3061,15 @@ packages: engines: {node: '>=0.10.0'} dev: true + /external-editor@3.1.0: + resolution: {integrity: sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==} + engines: {node: '>=4'} + dependencies: + chardet: 0.7.0 + iconv-lite: 0.4.24 + tmp: 0.0.33 + dev: false + /fast-deep-equal@3.1.3: resolution: {integrity: sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==} dev: true @@ -3138,7 +3226,6 @@ packages: /function-bind@1.1.2: resolution: {integrity: sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==} - dev: true /function.prototype.name@1.1.6: resolution: {integrity: sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==} @@ -3172,7 +3259,6 @@ packages: has-proto: 1.0.3 has-symbols: 1.0.3 hasown: 2.0.2 - dev: true /get-package-type@0.1.0: resolution: {integrity: sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==} @@ -3289,7 +3375,6 @@ packages: resolution: {integrity: sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==} dependencies: get-intrinsic: 1.2.4 - dev: true /got@13.0.0: resolution: {integrity: sha512-XfBk1CxOOScDcMr9O1yKkNaQyy865NbYs+F7dr4H0LZMVgCj2Le59k6PqbNHoL5ToeaEQUYh6c6yMfVcc6SJxA==} @@ -3332,17 +3417,14 @@ packages: resolution: {integrity: sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==} dependencies: es-define-property: 1.0.0 - dev: true /has-proto@1.0.3: resolution: {integrity: sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==} engines: {node: '>= 0.4'} - dev: true /has-symbols@1.0.3: resolution: {integrity: sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==} engines: {node: '>= 0.4'} - dev: true /has-tostringtag@1.0.2: resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} @@ -3356,7 +3438,6 @@ packages: engines: {node: '>= 0.4'} dependencies: function-bind: 1.1.2 - dev: true /he@1.2.0: resolution: {integrity: sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==} @@ -3410,6 +3491,17 @@ packages: resolution: {integrity: sha512-Ty8UblRWFEcfSuIaajM34LdPXIhbs1ajEX/BBPv24J+enSVaEVY63xQ6lTO9VRYS5LAoghIG0IDJ+p+IPzKUQQ==} engines: {node: '>=4'} + /iconv-lite@0.4.24: + resolution: {integrity: sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==} + engines: {node: '>=0.10.0'} + dependencies: + safer-buffer: 2.1.2 + dev: false + + /ieee754@1.2.1: + resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==} + dev: false + /ignore@5.3.1: resolution: {integrity: sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==} engines: {node: '>= 4'} @@ -3440,7 +3532,27 @@ packages: /inherits@2.0.4: resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} - dev: true + + /inquirer@9.2.23: + resolution: {integrity: sha512-kod5s+FBPIDM2xiy9fu+6wdU/SkK5le5GS9lh4FEBjBHqiMgD9lLFbCbuqFNAjNL2ZOy9Wd9F694IOzN9pZHBA==} + engines: {node: '>=18'} + dependencies: + '@inquirer/figures': 1.0.3 + '@ljharb/through': 2.3.13 + ansi-escapes: 4.3.2 + chalk: 5.3.0 + cli-cursor: 3.1.0 + cli-width: 4.1.0 + external-editor: 3.1.0 + lodash: 4.17.21 + mute-stream: 1.0.0 + ora: 5.4.1 + run-async: 3.0.0 + rxjs: 7.8.1 + string-width: 4.2.3 + strip-ansi: 6.0.1 + wrap-ansi: 6.2.0 + dev: false /internal-slot@1.0.7: resolution: {integrity: sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g==} @@ -3542,6 +3654,11 @@ packages: dependencies: is-extglob: 2.1.1 + /is-interactive@1.0.0: + resolution: {integrity: sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==} + engines: {node: '>=8'} + dev: false + /is-negative-zero@2.0.3: resolution: {integrity: sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==} engines: {node: '>= 0.4'} @@ -3622,7 +3739,6 @@ packages: /is-unicode-supported@0.1.0: resolution: {integrity: sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==} engines: {node: '>=10'} - dev: true /is-weakref@1.0.2: resolution: {integrity: sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==} @@ -3630,6 +3746,11 @@ packages: call-bind: 1.0.7 dev: true + /is-what@4.1.16: + resolution: {integrity: sha512-ZhMwEosbFJkA0YhFnNDgTM4ZxDRsS6HqTo7qsZM08fehyRYIYa0yHu5R6mgo1n/8MgaPBXiPimPD77baVFYg+A==} + engines: {node: '>=12.13'} + dev: false + /is-wsl@2.2.0: resolution: {integrity: sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==} engines: {node: '>=8'} @@ -3754,7 +3875,6 @@ packages: /lodash@4.17.21: resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} - dev: true /log-symbols@4.1.0: resolution: {integrity: sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==} @@ -3762,7 +3882,6 @@ packages: dependencies: chalk: 4.1.2 is-unicode-supported: 0.1.0 - dev: true /loupe@2.3.7: resolution: {integrity: sha512-zSMINGVYkdpYSOBmLi0D1Uo7JU9nVdQKrHxC8eYlV+9YKK9WePqAlL7lSlorG/U2Fw1w0hTBmaa/jrQ3UbPHtA==} @@ -3812,6 +3931,11 @@ packages: mime-db: 1.52.0 dev: false + /mimic-fn@2.1.0: + resolution: {integrity: sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==} + engines: {node: '>=6'} + dev: false + /mimic-response@3.1.0: resolution: {integrity: sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==} engines: {node: '>=10'} @@ -3899,7 +4023,6 @@ packages: /mute-stream@1.0.0: resolution: {integrity: sha512-avsJQhyd+680gKXyG/sQc0nXaC6rBkPOfyHYcFb9+hdkqQkR9bdnkJ0AMZhke0oesPqIO+mFFJ+IdBc7mst4IA==} engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} - dev: true /natural-compare-lite@1.4.0: resolution: {integrity: sha512-Tj+HTDSJJKaZnfiuw+iaF9skdPpTo2GtEly5JHnWV/hfv2Qj/9RKsGISQtLh2ox3l5EAGw487hnBee0sIJ6v2g==} @@ -4137,6 +4260,13 @@ packages: wrappy: 1.0.2 dev: true + /onetime@5.1.2: + resolution: {integrity: sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==} + engines: {node: '>=6'} + dependencies: + mimic-fn: 2.1.0 + dev: false + /optionator@0.9.4: resolution: {integrity: sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==} engines: {node: '>= 0.8.0'} @@ -4149,6 +4279,26 @@ packages: word-wrap: 1.2.5 dev: true + /ora@5.4.1: + resolution: {integrity: sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==} + engines: {node: '>=10'} + dependencies: + bl: 4.1.0 + chalk: 4.1.2 + cli-cursor: 3.1.0 + cli-spinners: 2.9.2 + is-interactive: 1.0.0 + is-unicode-supported: 0.1.0 + log-symbols: 4.1.0 + strip-ansi: 6.0.1 + wcwidth: 1.0.1 + dev: false + + /os-tmpdir@1.0.2: + resolution: {integrity: sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==} + engines: {node: '>=0.10.0'} + dev: false + /p-cancelable@3.0.0: resolution: {integrity: sha512-mlVgR3PGuzlo0MmTdk4cXqXWlwQDLnONTAg6sm62XkMJEiRxN3GL3SffkYvqwonbkJBcrI7Uvv5Zh9yjvn2iUw==} engines: {node: '>=12.20'} @@ -4341,6 +4491,15 @@ packages: type-fest: 0.6.0 dev: true + /readable-stream@3.6.2: + resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} + engines: {node: '>= 6'} + dependencies: + inherits: 2.0.4 + string_decoder: 1.3.0 + util-deprecate: 1.0.2 + dev: false + /readdirp@3.6.0: resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} engines: {node: '>=8.10.0'} @@ -4420,6 +4579,14 @@ packages: lowercase-keys: 3.0.0 dev: true + /restore-cursor@3.1.0: + resolution: {integrity: sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==} + engines: {node: '>=8'} + dependencies: + onetime: 5.1.2 + signal-exit: 3.0.7 + dev: false + /retry@0.13.1: resolution: {integrity: sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==} engines: {node: '>= 4'} @@ -4437,11 +4604,21 @@ packages: glob: 7.2.3 dev: true + /run-async@3.0.0: + resolution: {integrity: sha512-540WwVDOMxA6dN6We19EcT9sc3hkXPw5mzRNGM3FkdN/vtE9NFvj5lFAPNwUDmJjXidm3v7TC1cTE7t17Ulm1Q==} + engines: {node: '>=0.12.0'} + dev: false + /run-parallel@1.2.0: resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} dependencies: queue-microtask: 1.2.3 + /rxjs@7.8.1: + resolution: {integrity: sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==} + dependencies: + tslib: 2.6.2 + /safe-array-concat@1.1.2: resolution: {integrity: sha512-vj6RsCsWBCf19jIeHEfkRMw8DPiBb+DMXklQ/1SGDHOMlHdPUkZXFQ2YdplS23zESTijAcurb1aSgJA3AgMu1Q==} engines: {node: '>=0.4'} @@ -4454,7 +4631,6 @@ packages: /safe-buffer@5.2.1: resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} - dev: true /safe-regex-test@1.0.3: resolution: {integrity: sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==} @@ -4465,6 +4641,10 @@ packages: is-regex: 1.1.4 dev: true + /safer-buffer@2.1.2: + resolution: {integrity: sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==} + dev: false + /semver@5.7.2: resolution: {integrity: sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==} hasBin: true @@ -4504,7 +4684,6 @@ packages: get-intrinsic: 1.2.4 gopd: 1.0.1 has-property-descriptors: 1.0.2 - dev: true /set-function-name@2.0.2: resolution: {integrity: sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==} @@ -4555,6 +4734,10 @@ packages: object-inspect: 1.13.1 dev: true + /signal-exit@3.0.7: + resolution: {integrity: sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==} + dev: false + /signal-exit@4.1.0: resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} engines: {node: '>=14'} @@ -4667,6 +4850,12 @@ packages: es-object-atoms: 1.0.0 dev: true + /string_decoder@1.3.0: + resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==} + dependencies: + safe-buffer: 5.2.1 + dev: false + /strip-ansi@6.0.1: resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} engines: {node: '>=8'} @@ -4694,6 +4883,13 @@ packages: resolution: {integrity: sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==} dev: true + /superjson@2.2.1: + resolution: {integrity: sha512-8iGv75BYOa0xRJHK5vRLEjE2H/i4lulTjzpUXic3Eg8akftYjkmQDa8JARQ42rlczXyFR3IeRoeFCc7RxHsYZA==} + engines: {node: '>=16'} + dependencies: + copy-anything: 3.0.5 + dev: false + /supports-color@5.5.0: resolution: {integrity: sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==} engines: {node: '>=4'} @@ -4741,6 +4937,13 @@ packages: resolution: {integrity: sha512-ik6BCxzva9DoiEfDX/li0L2cWKPPENYvixUprFdl3YPi4bZZUhDnNI9YUkacrv+uIG90dnxR5mNqaoD6UhD6Bw==} dev: true + /tmp@0.0.33: + resolution: {integrity: sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==} + engines: {node: '>=0.6.0'} + dependencies: + os-tmpdir: 1.0.2 + dev: false + /to-regex-range@5.0.1: resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} engines: {node: '>=8.0'} @@ -4802,7 +5005,6 @@ packages: /tslib@2.6.2: resolution: {integrity: sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==} - dev: true /tunnel-agent@0.6.0: resolution: {integrity: sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==} @@ -4925,6 +5127,10 @@ packages: punycode: 2.3.1 dev: true + /util-deprecate@1.0.2: + resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} + dev: false + /uuid@9.0.1: resolution: {integrity: sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==} hasBin: true @@ -4945,6 +5151,12 @@ packages: resolution: {integrity: sha512-OljLrQ9SQdOUqTaQxqL5dEfZWrXExyyWsozYlAWFawPVNuD83igl7uJD2RTkNMbniIYgt8l81eCJGIdQF7avLQ==} engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} + /wcwidth@1.0.1: + resolution: {integrity: sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==} + dependencies: + defaults: 1.0.4 + dev: false + /which-boxed-primitive@1.0.2: resolution: {integrity: sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==} dependencies: @@ -5006,7 +5218,6 @@ packages: ansi-styles: 4.3.0 string-width: 4.2.3 strip-ansi: 6.0.1 - dev: true /wrap-ansi@7.0.0: resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} diff --git a/src/commands/authenticate.ts b/src/commands/authenticate.ts index 07da036..221a8fa 100644 --- a/src/commands/authenticate.ts +++ b/src/commands/authenticate.ts @@ -1,29 +1,99 @@ import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; import * as fs from "node:fs"; import * as path from "node:path"; - +import { fileURLToPath } from "node:url"; +const __filename = fileURLToPath(import.meta.url); +const __dirname = path.dirname(__filename); const configPath = path.join(__dirname, "..", "..", "config.json"); export default class Authenticate extends Command { static description = "Authenticate the user by saving server URL and token"; + static examples = [ + "$ <%= config.bin %> <%= command.id %> --url=https://panel.dokploy.com --token=aslgjasndjanskj123%!@#", + "$ <%= config.bin %> <%= command.id %> -u https://panel.dokploy.com -t aslgjasndjanskj123%!@#", + ]; + static flags = { token: Flags.string({ char: "t", description: "Authentication token", - required: true, }), - url: Flags.string({ char: "u", description: "Server URL", required: true }), + url: Flags.string({ + char: "u", + description: "Server URL", + }), }; async run() { + console.log( + chalk.blue.bold("\n Welcome to Dokploy CLI Authentication \n"), + ); + const { flags } = await this.parse(Authenticate); + + let answers: any; + if (!flags.token || !flags.url) { + answers = await inquirer.prompt([ + { + message: chalk.green( + "Enter your server URL (e.g., https://panel.dokploy.com): ", + ), + name: "url", + type: "input", + validate: (input) => (input ? true : "Server URL is required"), + }, + { + message: chalk.green( + "Enter your authentication token (e.g., aslgjasndjanskj123%!@#): ", + ), + name: "token", + type: "input", + validate: (input) => + input ? true : "Authentication token is required", + }, + ]); + } + + const url = flags.url || answers.url; + const token = flags.token || answers.token; + const config = { - token: flags.token, - url: flags.url, + token, + url, }; - fs.writeFileSync(configPath, JSON.stringify(config, null, 2)); - this.log("Authentication details saved successfully."); + try { + console.log(`\n${chalk.blue("Validating server...")}`); + + const response = await axios.post( + `${url}/api/trpc/auth.verifyToken`, + { + json: { + token, + }, + }, + { + headers: { + "Content-Type": "application/json", + }, + }, + ); + + if (!response.data.result.data.json) { + this.error(chalk.red("Invalid token")); + } + + fs.writeFileSync(configPath, JSON.stringify(config, null, 2)); + this.log(chalk.green("Authentication details saved successfully.")); + } catch (error) { + this.error( + // @ts-expect-error - Type + chalk.red(`Failed to save authentication details: ${error.message}`), + ); + } } } diff --git a/test/commands/authenticate.test.ts b/test/commands/authenticate.test.ts new file mode 100644 index 0000000..a3e158b --- /dev/null +++ b/test/commands/authenticate.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('authenticate', () => { + it('runs authenticate cmd', async () => { + const {stdout} = await runCommand('authenticate') + expect(stdout).to.contain('hello world') + }) + + it('runs authenticate --name oclif', async () => { + const {stdout} = await runCommand('authenticate --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/tsconfig.json b/tsconfig.json index e144b78..387201a 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -6,7 +6,7 @@ "rootDir": "src", "strict": true, "target": "es2022", - "moduleResolution": "node16" + "moduleResolution": "node16", }, "include": ["./src/**/*"], "ts-node": { From 9f033de1b212f9d4bad32725fa8c17d59d95d99c Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Tue, 4 Jun 2024 21:47:36 -0600 Subject: [PATCH 03/13] feat: add verify and authenticate command --- .eslintrc.json | 2 +- src/commands/authenticate.ts | 71 ++++++++++++++++++++-------------- src/commands/verify.ts | 75 ++++++++++++++++++++++++++++++++++++ test/commands/verify.test.ts | 14 +++++++ 4 files changed, 132 insertions(+), 30 deletions(-) create mode 100644 src/commands/verify.ts create mode 100644 test/commands/verify.test.ts diff --git a/.eslintrc.json b/.eslintrc.json index 1dfcfc4..e698178 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -1,3 +1,3 @@ { - "extends": ["oclif", "oclif-typescript", "prettier"] + "extends": ["oclif", "prettier"] } diff --git a/src/commands/authenticate.ts b/src/commands/authenticate.ts index 221a8fa..05ca5e3 100644 --- a/src/commands/authenticate.ts +++ b/src/commands/authenticate.ts @@ -1,7 +1,7 @@ import { Command, Flags } from "@oclif/core"; import axios from "axios"; import chalk from "chalk"; -import inquirer from "inquirer"; +import inquirer, { type Answers, type QuestionCollection } from "inquirer"; import * as fs from "node:fs"; import * as path from "node:path"; import { fileURLToPath } from "node:url"; @@ -13,8 +13,8 @@ export default class Authenticate extends Command { static description = "Authenticate the user by saving server URL and token"; static examples = [ - "$ <%= config.bin %> <%= command.id %> --url=https://panel.dokploy.com --token=aslgjasndjanskj123%!@#", - "$ <%= config.bin %> <%= command.id %> -u https://panel.dokploy.com -t aslgjasndjanskj123%!@#", + "$ <%= config.bin %> <%= command.id %> --url=https://panel.dokploy.com --token=MRTHGZDGMRZWM43EMZSHGZTTMRTHGZDGONSGMZDTMY", + "$ <%= config.bin %> <%= command.id %> -u https://panel.dokploy.com -t MRTHGZDGMRZWM43EMZSHGZTTMRTHGZDGONSGMZDTMY", ]; static flags = { @@ -35,36 +35,50 @@ export default class Authenticate extends Command { const { flags } = await this.parse(Authenticate); - let answers: any; - if (!flags.token || !flags.url) { - answers = await inquirer.prompt([ - { - message: chalk.green( - "Enter your server URL (e.g., https://panel.dokploy.com): ", - ), - name: "url", - type: "input", - validate: (input) => (input ? true : "Server URL is required"), - }, - { - message: chalk.green( - "Enter your authentication token (e.g., aslgjasndjanskj123%!@#): ", - ), - name: "token", - type: "input", - validate: (input) => - input ? true : "Authentication token is required", - }, - ]); + let answers: Answers = {}; + + const questions: QuestionCollection[] = []; + + let config: { token?: string; url?: string } = {}; + if (fs.existsSync(configPath)) { + const configFileContent = fs.readFileSync(configPath, "utf8"); + config = JSON.parse(configFileContent); + } + + if (!flags.url) { + questions.push({ + default: config.url, + message: chalk.green( + "Enter your server URL (e.g., https://panel.dokploy.com): ", + ), + name: "url", + type: "input", + validate: (input) => (input ? true : "Server URL is required"), + }); + } + + if (!flags.token) { + questions.push({ + default: config.token, + message: chalk.green( + "Enter your authentication token (e.g., MRTHGZDGMRZWM43EMZSHGZTTMRTHGZDGONSGMZDTMY=): ", + ), + name: "token", + type: "input", + validate: (input) => + input ? true : "Authentication token is required", + }); + } + + if (questions.length > 0) { + answers = await inquirer.prompt(questions); } const url = flags.url || answers.url; const token = flags.token || answers.token; - const config = { - token, - url, - }; + config.token = token; + config.url = url; try { console.log(`\n${chalk.blue("Validating server...")}`); @@ -86,7 +100,6 @@ export default class Authenticate extends Command { if (!response.data.result.data.json) { this.error(chalk.red("Invalid token")); } - fs.writeFileSync(configPath, JSON.stringify(config, null, 2)); this.log(chalk.green("Authentication details saved successfully.")); } catch (error) { diff --git a/src/commands/verify.ts b/src/commands/verify.ts new file mode 100644 index 0000000..2da837f --- /dev/null +++ b/src/commands/verify.ts @@ -0,0 +1,75 @@ +import { Command } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import * as fs from "node:fs"; +import * as path from "node:path"; +import { fileURLToPath } from "node:url"; + +const __filename = fileURLToPath(import.meta.url); +const __dirname = path.dirname(__filename); +const configPath = path.join(__dirname, "..", "..", "config.json"); + +export default class Verify extends Command { + static description = "Verify if the saved authentication token is valid"; + + static examples = ["$ <%= config.bin %> <%= command.id %>"]; + + async run() { + console.log(chalk.blue.bold("\nVerifying Authentication Token")); + + if (!fs.existsSync(configPath)) { + this.error( + chalk.red( + "No configuration file found. Please authenticate first using `authenticate` command.", + ), + ); + } + + const configFileContent = fs.readFileSync(configPath, "utf8"); + const config = JSON.parse(configFileContent); + const { token, url } = config; + + if (!url || !token) { + this.error( + chalk.red( + "Incomplete authentication details. Please authenticate again using `authenticate` command.", + ), + ); + } + + try { + console.log(`\n${chalk.blue("Validating token...")}`); + + const response = await axios.post( + `${url}/api/trpc/auth.verifyToken`, + { + json: { + token, + }, + }, + { + headers: { + "Content-Type": "application/json", + }, + }, + ); + + if (response.data.result.data.json) { + this.log(chalk.green("Token is valid.")); + } else { + this.error( + chalk.red( + "Invalid token. Please authenticate again using `authenticate` command.", + ), + ); + } + } catch (error) { + this.error( + chalk.red( + // @ts-ignore + `Failed to verify token: ${error.message}. Please authenticate again using 'authenticate' command.`, + ), + ); + } + } +} diff --git a/test/commands/verify.test.ts b/test/commands/verify.test.ts new file mode 100644 index 0000000..7311300 --- /dev/null +++ b/test/commands/verify.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('verify', () => { + it('runs verify cmd', async () => { + const {stdout} = await runCommand('verify') + expect(stdout).to.contain('hello world') + }) + + it('runs verify --name oclif', async () => { + const {stdout} = await runCommand('verify --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) From bec0dfa57f388fe3caafe0835a6a5704e952c606 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Wed, 5 Jun 2024 01:43:19 -0600 Subject: [PATCH 04/13] feat: add list and create project --- .eslintrc.json | 2 +- config.json | 2 +- package.json | 1 + pnpm-lock.yaml | 19 ++++++ src/commands/authenticate.ts | 12 +--- src/commands/project/create.ts | 94 ++++++++++++++++++++++++++++ src/commands/project/list.ts | 60 ++++++++++++++++++ src/commands/verify.ts | 7 +-- src/utils/http.ts | 4 ++ src/utils/utils.ts | 36 +++++++++++ test/commands/project/create.test.ts | 14 +++++ test/commands/project/list.test.ts | 14 +++++ 12 files changed, 249 insertions(+), 16 deletions(-) create mode 100644 src/commands/project/create.ts create mode 100644 src/commands/project/list.ts create mode 100644 src/utils/http.ts create mode 100644 src/utils/utils.ts create mode 100644 test/commands/project/create.test.ts create mode 100644 test/commands/project/list.test.ts diff --git a/.eslintrc.json b/.eslintrc.json index e698178..1dfcfc4 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -1,3 +1,3 @@ { - "extends": ["oclif", "prettier"] + "extends": ["oclif", "oclif-typescript", "prettier"] } diff --git a/config.json b/config.json index cc30a3c..e5834e2 100644 --- a/config.json +++ b/config.json @@ -1,4 +1,4 @@ { - "token": "hola", + "token": "5qdderpq2ejrjlu90hrnbjohvgc8j1u1k7i00um4", "url": "http://localhost:3000" } \ No newline at end of file diff --git a/package.json b/package.json index 510a583..b49cab2 100644 --- a/package.json +++ b/package.json @@ -13,6 +13,7 @@ "@oclif/plugin-plugins": "^5", "axios": "^1.7.2", "chalk": "^5.3.0", + "cli-table3": "^0.6.5", "inquirer": "^9.2.23", "superjson": "^2.2.1" }, diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 0db1304..1e509c1 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -20,6 +20,9 @@ dependencies: chalk: specifier: ^5.3.0 version: 5.3.0 + cli-table3: + specifier: ^0.6.5 + version: 0.6.5 inquirer: specifier: ^9.2.23 version: 9.2.23 @@ -783,6 +786,13 @@ packages: js-tokens: 4.0.0 picocolors: 1.0.1 + /@colors/colors@1.5.0: + resolution: {integrity: sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==} + engines: {node: '>=0.1.90'} + requiresBuild: true + dev: false + optional: true + /@cspotcode/source-map-support@0.8.1: resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==} engines: {node: '>=12'} @@ -2263,6 +2273,15 @@ packages: resolution: {integrity: sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==} engines: {node: '>=6'} + /cli-table3@0.6.5: + resolution: {integrity: sha512-+W/5efTR7y5HRD7gACw9yQjqMVvEMLBHmboM/kPWam+H+Hmyrgjh6YncVKK122YZkXrLudzTuAukUw9FnMf7IQ==} + engines: {node: 10.* || >= 12.*} + dependencies: + string-width: 4.2.3 + optionalDependencies: + '@colors/colors': 1.5.0 + dev: false + /cli-width@4.1.0: resolution: {integrity: sha512-ouuZd4/dm2Sw5Gmqy6bGyNNNe1qt9RpmxveLSO7KcgsTnU7RXfsw+/bukWGo1abgBiMAic068rclZsO4IWmmxQ==} engines: {node: '>= 12'} diff --git a/src/commands/authenticate.ts b/src/commands/authenticate.ts index 05ca5e3..9e58baa 100644 --- a/src/commands/authenticate.ts +++ b/src/commands/authenticate.ts @@ -83,23 +83,17 @@ export default class Authenticate extends Command { try { console.log(`\n${chalk.blue("Validating server...")}`); - const response = await axios.post( + await axios.post( `${url}/api/trpc/auth.verifyToken`, - { - json: { - token, - }, - }, + {}, { headers: { + Authorization: `Bearer ${token}`, "Content-Type": "application/json", }, }, ); - if (!response.data.result.data.json) { - this.error(chalk.red("Invalid token")); - } fs.writeFileSync(configPath, JSON.stringify(config, null, 2)); this.log(chalk.green("Authentication details saved successfully.")); } catch (error) { diff --git a/src/commands/project/create.ts b/src/commands/project/create.ts new file mode 100644 index 0000000..2b3988e --- /dev/null +++ b/src/commands/project/create.ts @@ -0,0 +1,94 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer, { type Answers, type QuestionCollection } from "inquirer"; + +import { readAuthConfig } from "../../utils/utils.js"; + +export default class ProjectCreate extends Command { + static override description = + "Create a new project with an optional description."; + + static override examples = [ + "$ <%= config.bin %> <%= command.id %> -n MyProject -d 'This is my project description'", + "$ <%= config.bin %> <%= command.id %> -n MyProject", + "$ <%= config.bin %> <%= command.id %>", + ]; + + static override flags = { + description: Flags.string({ + char: "d", + description: "Description of the project", + required: false, + }), + name: Flags.string({ + char: "n", + description: "Name of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Create a New Project \n")); + + const { flags } = await this.parse(ProjectCreate); + + let answers: Answers = {}; + + const questions: QuestionCollection[] = []; + + if (!flags.name) { + questions.push({ + message: chalk.green("Enter the project name:"), + name: "name", + type: "input", + validate: (input) => (input ? true : "Project name is required"), + }); + } + + if (!flags.description) { + questions.push({ + default: "", + message: chalk.green("Enter the project description (optional):"), + name: "description", + type: "input", + }); + } + + if (questions.length > 0) { + answers = await inquirer.prompt(questions); + } + + const name = flags.name || answers.name; + const description = flags.description || answers.description; + + try { + const response = await axios.post( + `${auth.url}/api/trpc/project.createCLI`, + { + json: { + description, + name, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error`")); + } + + this.log(chalk.green(`Project '${name}' created successfully.`)); + } catch (error) { + // @ts-expect-error hola + this.error(chalk.red(`Failed to create project: ${error.message}`)); + } + } +} diff --git a/src/commands/project/list.ts b/src/commands/project/list.ts new file mode 100644 index 0000000..fcf355e --- /dev/null +++ b/src/commands/project/list.ts @@ -0,0 +1,60 @@ +import { Command } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import Table from "cli-table3"; + +import { readAuthConfig } from "../../utils/utils.js"; + +export default class ProjectList extends Command { + static description = "List all projects."; + + static examples = ["$ <%= config.bin %> project list"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + } else { + this.log(chalk.green("Projects:")); + const table = new Table({ + colWidths: [10, 30, 50], + head: [ + chalk.cyan("Index"), + chalk.cyan("Name"), + chalk.cyan("Description"), + ], + }); + const index = 1; + for (const project of projects) { + table.push([ + chalk.white(index + 1), + chalk.white(project.name), + chalk.gray(project.description || "No description"), + ]); + } + + this.log(table.toString()); + } + } catch { + // @ts-expect-error error is not defined + this.error(chalk.red(`Failed to list projects: ${error.message}`)); + } + } +} diff --git a/src/commands/verify.ts b/src/commands/verify.ts index 2da837f..d77da35 100644 --- a/src/commands/verify.ts +++ b/src/commands/verify.ts @@ -42,13 +42,10 @@ export default class Verify extends Command { const response = await axios.post( `${url}/api/trpc/auth.verifyToken`, - { - json: { - token, - }, - }, + {}, { headers: { + Authorization: `Bearer ${token}`, "Content-Type": "application/json", }, }, diff --git a/src/utils/http.ts b/src/utils/http.ts new file mode 100644 index 0000000..b27d636 --- /dev/null +++ b/src/utils/http.ts @@ -0,0 +1,4 @@ +export const headers = { + "Content-Type": "application/json", + "User-Agent": "Dokploy CLI", +}; diff --git a/src/utils/utils.ts b/src/utils/utils.ts new file mode 100644 index 0000000..419678f --- /dev/null +++ b/src/utils/utils.ts @@ -0,0 +1,36 @@ +import type { Command } from "@oclif/core"; + +import chalk from "chalk"; +import * as fs from "node:fs"; +import * as path from "node:path"; +import { fileURLToPath } from "node:url"; + +const __filename = fileURLToPath(import.meta.url); +const __dirname = path.dirname(__filename); +const configPath = path.join(__dirname, "..", "..", "config.json"); + +export const readAuthConfig = async ( + command: Command, +): Promise<{ token: string; url: string }> => { + if (!fs.existsSync(configPath)) { + command.error( + chalk.red( + "No configuration file found. Please authenticate first using the 'authenticate' command.", + ), + ); + } + + const configFileContent = fs.readFileSync(configPath, "utf8"); + const config = JSON.parse(configFileContent); + const { token, url } = config; + + if (!url || !token) { + command.error( + chalk.red( + "Incomplete authentication details. Please authenticate again using the 'authenticate' command.", + ), + ); + } + + return { token, url }; +}; diff --git a/test/commands/project/create.test.ts b/test/commands/project/create.test.ts new file mode 100644 index 0000000..cd9e123 --- /dev/null +++ b/test/commands/project/create.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('project:create', () => { + it('runs project:create cmd', async () => { + const {stdout} = await runCommand('project:create') + expect(stdout).to.contain('hello world') + }) + + it('runs project:create --name oclif', async () => { + const {stdout} = await runCommand('project:create --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/project/list.test.ts b/test/commands/project/list.test.ts new file mode 100644 index 0000000..c21edbb --- /dev/null +++ b/test/commands/project/list.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('project:list', () => { + it('runs project:list cmd', async () => { + const {stdout} = await runCommand('project:list') + expect(stdout).to.contain('hello world') + }) + + it('runs project:list --name oclif', async () => { + const {stdout} = await runCommand('project:list --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) From f1616b39094ba288d91a82d1356332c69513d4b3 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Wed, 5 Jun 2024 02:15:24 -0600 Subject: [PATCH 05/13] feat: add info list project --- src/commands/project/info.ts | 197 +++++++++++++++++++++++++++++ test/commands/project/info.test.ts | 14 ++ 2 files changed, 211 insertions(+) create mode 100644 src/commands/project/info.ts create mode 100644 test/commands/project/info.test.ts diff --git a/src/commands/project/info.ts b/src/commands/project/info.ts new file mode 100644 index 0000000..44e3bdb --- /dev/null +++ b/src/commands/project/info.ts @@ -0,0 +1,197 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../utils/utils.js"; + +export default class ProjectInfo extends Command { + static description = + "Get detailed information about a project, including the number of applications and databases."; + + static examples = [ + "$ <%= config.bin %> project info", + "$ <%= config.bin %> project info -p ", + ]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(ProjectInfo); + + if (flags.projectId) { + // Si se proporciona un projectId, mostrar directamente la información del proyecto + await this.showProjectInfo(auth, flags.projectId); + } else { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to view details:", + name: "selectedProject", + type: "list", + }, + ]); + + const selectedProjectId = answers.selectedProject; + + await this.showProjectInfo(auth, selectedProjectId); + } catch (error) { + // @ts-expect-error hola + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + } + + private async showProjectInfo( + auth: { token: string; url: string }, + projectId: string, + ) { + console.log( + chalk.blue.bold(`\n Information for Project ID: ${projectId} \n`), + ); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.one`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + params: { + input: JSON.stringify({ + json: { projectId }, + }), + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching project information")); + } + + const projectInfo = response.data.result.data.json; + + this.log(chalk.green(`Project Name: ${projectInfo.name}`)); + this.log( + chalk.green( + `Description: ${projectInfo?.description || "No description"}`, + ), + ); + this.log( + chalk.green( + `Number of Applications: ${projectInfo.applications.length}`, + ), + ); + this.log( + chalk.green( + `Number of Compose Services: ${projectInfo.compose.length}`, + ), + ); + this.log( + chalk.green( + `Number of MariaDB Databases: ${projectInfo.mariadb.length}`, + ), + ); + this.log( + chalk.green(`Number of MongoDB Databases: ${projectInfo.mongo.length}`), + ); + this.log( + chalk.green(`Number of MySQL Databases: ${projectInfo.mysql.length}`), + ); + this.log( + chalk.green( + `Number of PostgreSQL Databases: ${projectInfo.postgres.length}`, + ), + ); + this.log( + chalk.green(`Number of Redis Databases: ${projectInfo.redis.length}`), + ); + + if (projectInfo.applications.length > 0) { + this.log(chalk.blue("\nApplications:")); + projectInfo.applications.forEach((app: any, index: number) => { + this.log(` ${index + 1}. ${app.name}`); + }); + } + + if (projectInfo.compose.length > 0) { + this.log(chalk.blue("\nCompose Services:")); + projectInfo.compose.forEach((service: any, index: number) => { + this.log(` ${index + 1}. ${service.name}`); + }); + } + + if (projectInfo.mariadb.length > 0) { + this.log(chalk.blue("\nMariaDB Databases:")); + projectInfo.mariadb.forEach((db: any, index: number) => { + this.log(` ${index + 1}. ${db.name}`); + }); + } + + if (projectInfo.mongo.length > 0) { + this.log(chalk.blue("\nMongoDB Databases:")); + projectInfo.mongo.forEach((db: any, index: number) => { + this.log(` ${index + 1}. ${db.name}`); + }); + } + + if (projectInfo.mysql.length > 0) { + this.log(chalk.blue("\nMySQL Databases:")); + projectInfo.mysql.forEach((db: any, index: number) => { + this.log(` ${index + 1}. ${db.name}`); + }); + } + + if (projectInfo.postgres.length > 0) { + this.log(chalk.blue("\nPostgreSQL Databases:")); + projectInfo.postgres.forEach((db: any, index: number) => { + this.log(` ${index + 1}. ${db.name}`); + }); + } + + if (projectInfo.redis.length > 0) { + this.log(chalk.blue("\nRedis Databases:")); + projectInfo.redis.forEach((db: any, index: number) => { + this.log(` ${index + 1}. ${db.name}`); + }); + } + } catch (error) { + this.error( + // @ts-expect-error hola + chalk.red(`Failed to fetch project information: ${error.message}`), + ); + } + } +} diff --git a/test/commands/project/info.test.ts b/test/commands/project/info.test.ts new file mode 100644 index 0000000..5ad12ad --- /dev/null +++ b/test/commands/project/info.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('project:info', () => { + it('runs project:info cmd', async () => { + const {stdout} = await runCommand('project:info') + expect(stdout).to.contain('hello world') + }) + + it('runs project:info --name oclif', async () => { + const {stdout} = await runCommand('project:info --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) From 1b97e2c6f51384148bdc5862e64747863ee2766b Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Wed, 5 Jun 2024 02:33:09 -0600 Subject: [PATCH 06/13] feat: add delete app --- src/commands/app/create.ts | 121 ++++++++++++++++++++++++ src/commands/app/delete.ts | 155 +++++++++++++++++++++++++++++++ test/commands/app/create.test.ts | 14 +++ test/commands/app/delete.test.ts | 14 +++ 4 files changed, 304 insertions(+) create mode 100644 src/commands/app/create.ts create mode 100644 src/commands/app/delete.ts create mode 100644 test/commands/app/create.test.ts create mode 100644 test/commands/app/delete.test.ts diff --git a/src/commands/app/create.ts b/src/commands/app/create.ts new file mode 100644 index 0000000..66fe787 --- /dev/null +++ b/src/commands/app/create.ts @@ -0,0 +1,121 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../utils/utils.js"; + +export default class AppCreate extends Command { + static description = "Create a new application within a project."; + + static examples = ["$ <%= config.bin %> app create"]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(AppCreate); + + let { projectId } = flags; + + if (!projectId) { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to create the application in:", + name: "selectedProject", + type: "list", + }, + ]); + + projectId = answers.selectedProject; + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + + // Solicitar detalles de la nueva aplicación + const appDetails = await inquirer.prompt([ + { + message: "Enter the application name:", + name: "appName", + type: "input", + validate: (input) => (input ? true : "Application name is required"), + }, + { + message: "Enter the application description (optional):", + name: "appDescription", + type: "input", + }, + ]); + + const { appDescription, appName } = appDetails; + + // Crear la aplicación en el proyecto seleccionado + try { + const response = await axios.post( + `${auth.url}/api/trpc/application.create`, + { + json: { + description: appDescription, + name: appName, + projectId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error creating application")); + } + + this.log( + chalk.green( + `Application '${appName}' created successfully in project ID '${projectId}'.`, + ), + ); + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to create application: ${error.message}`)); + } + } +} diff --git a/src/commands/app/delete.ts b/src/commands/app/delete.ts new file mode 100644 index 0000000..6a12f98 --- /dev/null +++ b/src/commands/app/delete.ts @@ -0,0 +1,155 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../utils/utils.js"; + +export default class AppDelete extends Command { + static description = "Delete an application from a project."; + + static examples = [ + "$ <%= config.bin %> app delete", + "$ <%= config.bin %> app delete -p ", + ]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(AppDelete); + + let { projectId } = flags; + + if (!projectId) { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the application from:", + name: "selectedProject", + type: "list", + }, + ]); + + projectId = answers.selectedProject; + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + + // Obtener la lista de aplicaciones del proyecto seleccionado + try { + const response = await axios.get(`${auth.url}/api/trpc/project.one`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + params: { + input: JSON.stringify({ + json: { projectId }, + }), + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching applications")); + } + + const apps = response.data.result.data.json; + + if (apps.applications.length === 0) { + this.log(chalk.yellow("No applications found in this project.")); + return; + } + + // Permitir al usuario seleccionar una aplicación + const appAnswers = await inquirer.prompt([ + { + choices: apps.applications.map((app: any) => ({ + name: app.name, + value: app.applicationId, + })), + message: "Select the application to delete:", + name: "selectedApp", + type: "list", + }, + ]); + + const applicationId = appAnswers.selectedApp; + + // Confirmar eliminación + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to delete this application?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.log(chalk.yellow("Application deletion cancelled.")); + return; + } + + // Eliminar la aplicación seleccionada + const deleteResponse = await axios.post( + `${auth.url}/api/trpc/application.delete`, + { + json: { + applicationId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!deleteResponse.data.result.data.json) { + this.error(chalk.red("Error deleting application")); + } + + this.log(chalk.green("Application deleted successfully.")); + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to delete application: ${error.message}`)); + } + } +} diff --git a/test/commands/app/create.test.ts b/test/commands/app/create.test.ts new file mode 100644 index 0000000..aaa7dbb --- /dev/null +++ b/test/commands/app/create.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('app:create', () => { + it('runs app:create cmd', async () => { + const {stdout} = await runCommand('app:create') + expect(stdout).to.contain('hello world') + }) + + it('runs app:create --name oclif', async () => { + const {stdout} = await runCommand('app:create --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/app/delete.test.ts b/test/commands/app/delete.test.ts new file mode 100644 index 0000000..50dc59b --- /dev/null +++ b/test/commands/app/delete.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('app:delete', () => { + it('runs app:delete cmd', async () => { + const {stdout} = await runCommand('app:delete') + expect(stdout).to.contain('hello world') + }) + + it('runs app:delete --name oclif', async () => { + const {stdout} = await runCommand('app:delete --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) From 99451d416ccb08239ff10a4263d6e191b3279563 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Wed, 12 Jun 2024 23:01:46 -0600 Subject: [PATCH 07/13] feat: add database create --- package.json | 1 + pnpm-lock.yaml | 8 + src/commands/app/create.ts | 74 +++++---- src/commands/database/create.ts | 121 +++++++++++++++ src/commands/database/mariadb/create.ts | 144 ++++++++++++++++++ src/commands/database/mongo/create.ts | 138 +++++++++++++++++ src/commands/database/mysql/create.ts | 143 +++++++++++++++++ src/commands/database/postgres/create.ts | 140 +++++++++++++++++ src/commands/database/redis/create.ts | 125 +++++++++++++++ src/utils/slug.ts | 14 ++ src/utils/slugify.ts | 3 + test/commands/database/create.test.ts | 14 ++ test/commands/database/mariadb/create.test.ts | 14 ++ test/commands/database/mongo/create.test.ts | 14 ++ test/commands/database/mysql/create.test.ts | 14 ++ .../commands/database/postgres/create.test.ts | 14 ++ test/commands/database/redis/create.test.ts | 14 ++ 17 files changed, 965 insertions(+), 30 deletions(-) create mode 100644 src/commands/database/create.ts create mode 100644 src/commands/database/mariadb/create.ts create mode 100644 src/commands/database/mongo/create.ts create mode 100644 src/commands/database/mysql/create.ts create mode 100644 src/commands/database/postgres/create.ts create mode 100644 src/commands/database/redis/create.ts create mode 100644 src/utils/slug.ts create mode 100644 src/utils/slugify.ts create mode 100644 test/commands/database/create.test.ts create mode 100644 test/commands/database/mariadb/create.test.ts create mode 100644 test/commands/database/mongo/create.test.ts create mode 100644 test/commands/database/mysql/create.test.ts create mode 100644 test/commands/database/postgres/create.test.ts create mode 100644 test/commands/database/redis/create.test.ts diff --git a/package.json b/package.json index b49cab2..6451eed 100644 --- a/package.json +++ b/package.json @@ -15,6 +15,7 @@ "chalk": "^5.3.0", "cli-table3": "^0.6.5", "inquirer": "^9.2.23", + "slugify": "^1.6.6", "superjson": "^2.2.1" }, "devDependencies": { diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 1e509c1..b48de68 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -26,6 +26,9 @@ dependencies: inquirer: specifier: ^9.2.23 version: 9.2.23 + slugify: + specifier: ^1.6.6 + version: 1.6.6 superjson: specifier: ^2.2.1 version: 2.2.1 @@ -4784,6 +4787,11 @@ packages: astral-regex: 2.0.0 is-fullwidth-code-point: 3.0.0 + /slugify@1.6.6: + resolution: {integrity: sha512-h+z7HKHYXj6wJU+AnS/+IH8Uh9fdcX1Lrhg1/VMdf9PwoBQXFcXiAdsy2tSK0P6gKwJLXp02r90ahUCqHk9rrw==} + engines: {node: '>=8.0.0'} + dev: false + /snake-case@3.0.4: resolution: {integrity: sha512-LAOh4z89bGQvl9pFfNF8V146i7o7/CqFPbqzYgP+yYzDIDeS9HaNFtXABamRW+AQzEVODcvE79ljJ+8a9YSdMg==} dependencies: diff --git a/src/commands/app/create.ts b/src/commands/app/create.ts index 66fe787..339d859 100644 --- a/src/commands/app/create.ts +++ b/src/commands/app/create.ts @@ -55,7 +55,7 @@ export default class AppCreate extends Command { name: project.name, value: project.projectId, })), - message: "Select a project to create the application in:", + message: "Select a project to create the database in:", name: "selectedProject", type: "list", }, @@ -68,16 +68,32 @@ export default class AppCreate extends Command { } } + const databases = ["postgres", "mysql", "redis", "mariadb", "mongo"]; + + const databaseSelect = await inquirer.prompt([ + { + choices: databases.map((database: any) => ({ + name: database, + value: database, + })), + message: "Select a database to create the application in:", + name: "selectedDatabase", + type: "list", + }, + ]); + + const urlSelected = `${auth.url}/api/trpc/${databaseSelect.selectedDatabase}.create`; + // Solicitar detalles de la nueva aplicación const appDetails = await inquirer.prompt([ { - message: "Enter the application name:", + message: "Enter the database name:", name: "appName", type: "input", - validate: (input) => (input ? true : "Application name is required"), + validate: (input) => (input ? true : "Database name is required"), }, { - message: "Enter the application description (optional):", + message: "Enter the database description (optional):", name: "appDescription", type: "input", }, @@ -87,32 +103,30 @@ export default class AppCreate extends Command { // Crear la aplicación en el proyecto seleccionado try { - const response = await axios.post( - `${auth.url}/api/trpc/application.create`, - { - json: { - description: appDescription, - name: appName, - projectId, - }, - }, - { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }, - ); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error creating application")); - } - - this.log( - chalk.green( - `Application '${appName}' created successfully in project ID '${projectId}'.`, - ), - ); + // const response = await axios.post( + // `${auth.url}/api/trpc/application.create`, + // { + // json: { + // description: appDescription, + // name: appName, + // projectId, + // }, + // }, + // { + // headers: { + // Authorization: `Bearer ${auth.token}`, + // "Content-Type": "application/json", + // }, + // }, + // ); + // if (!response.data.result.data.json) { + // this.error(chalk.red("Error creating application")); + // } + // this.log( + // chalk.green( + // `Application '${appName}' created successfully in project ID '${projectId}'.`, + // ), + // ); } catch (error) { // @ts-expect-error TODO: Fix this this.error(chalk.red(`Failed to create application: ${error.message}`)); diff --git a/src/commands/database/create.ts b/src/commands/database/create.ts new file mode 100644 index 0000000..b9559d4 --- /dev/null +++ b/src/commands/database/create.ts @@ -0,0 +1,121 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../utils/utils.js"; + +export default class DatabaseCreate extends Command { + static description = "Create a new database within a project."; + + static examples = ["$ <%= config.bin %> app create"]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(DatabaseCreate); + + let { projectId } = flags; + + if (!projectId) { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to create the application in:", + name: "selectedProject", + type: "list", + }, + ]); + + projectId = answers.selectedProject; + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + + // Solicitar detalles de la nueva aplicación + const appDetails = await inquirer.prompt([ + { + message: "Enter the database name:", + name: "appName", + type: "input", + validate: (input) => (input ? true : "Application name is required"), + }, + { + message: "Enter the application description (optional):", + name: "appDescription", + type: "input", + }, + ]); + + const { appDescription, appName } = appDetails; + + // Crear la aplicación en el proyecto seleccionado + try { + const response = await axios.post( + `${auth.url}/api/trpc/database.create`, + { + json: { + description: appDescription, + name: appName, + projectId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error creating application")); + } + + this.log( + chalk.green( + `Application '${appName}' created successfully in project ID '${projectId}'.`, + ), + ); + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to create application: ${error.message}`)); + } + } +} diff --git a/src/commands/database/mariadb/create.ts b/src/commands/database/mariadb/create.ts new file mode 100644 index 0000000..38b3f2a --- /dev/null +++ b/src/commands/database/mariadb/create.ts @@ -0,0 +1,144 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { slugify } from "../../../utils/slug.js"; +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabaseMariadbCreate extends Command { + static description = "Create a new database within a project."; + + static examples = ["$ <%= config.bin %> mariadb create"]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(DatabaseMariadbCreate); + + const { projectId } = flags; + if (!projectId) { + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the database in:", + name: "project", + type: "list", + }, + ]); + + const appDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Database name:", + name: "databaseName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database Root Password (optional):", + name: "databaseRootPassword", + type: "input", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "input", + }, + { + default: "mariadb:4", + message: "Docker Image (default: mariadb:4):", + name: "dockerImage", + type: "input", + }, + { + default: "mariadb", + message: "Database User: (default: mariadb):", + name: "databaseUser", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${appDetails.name}`, + message: "Enter the App name: (optional):", + name: "appName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + ]); + + const responseDatabase = await axios.post( + `${auth.url}/api/trpc/mariadb.create`, + { + json: { + ...appDetails, + appName: appName.appName, + projectId: project.projectId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!responseDatabase.data.result.data.json) { + this.error(chalk.red("Error creating database")); + } + + this.log( + chalk.green(`Database '${appDetails.name}' created successfully.`), + ); + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + } +} diff --git a/src/commands/database/mongo/create.ts b/src/commands/database/mongo/create.ts new file mode 100644 index 0000000..df1c007 --- /dev/null +++ b/src/commands/database/mongo/create.ts @@ -0,0 +1,138 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { slugify } from "../../../utils/slug.js"; +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabaseMongoCreate extends Command { + static description = "Create a new database within a project."; + + static examples = ["$ <%= config.bin %> mongo create"]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + const { flags } = await this.parse(DatabaseMongoCreate); + + const { projectId } = flags; + if (!projectId) { + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the database in:", + name: "project", + type: "list", + }, + ]); + + const appDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Database name:", + name: "databaseName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "input", + }, + { + default: "mongo:6", + message: "Docker Image (default: mongo:6):", + name: "dockerImage", + type: "input", + }, + { + default: "mongo", + message: "Database User: (default: mongo):", + name: "databaseUser", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${appDetails.name}`, + message: "Enter the App name: (optional):", + name: "appName", + type: "input", + validate: (input) => (input ? true : "App name is required"), + }, + ]); + + const responseDatabase = await axios.post( + `${auth.url}/api/trpc/mongo.create`, + { + json: { + ...appDetails, + appName: appName.appName, + projectId: project.projectId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!responseDatabase.data.result.data.json) { + this.error(chalk.red("Error creating database")); + } + + this.log( + chalk.green(`Database '${appDetails.name}' created successfully.`), + ); + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + } +} diff --git a/src/commands/database/mysql/create.ts b/src/commands/database/mysql/create.ts new file mode 100644 index 0000000..eb286b7 --- /dev/null +++ b/src/commands/database/mysql/create.ts @@ -0,0 +1,143 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { slugify } from "../../../utils/slug.js"; +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabaseMysqlCreate extends Command { + static description = "Create a new database within a project."; + + static examples = ["$ <%= config.bin %> mysql create"]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + const { flags } = await this.parse(DatabaseMysqlCreate); + + const { projectId } = flags; + if (!projectId) { + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the database in:", + name: "project", + type: "list", + }, + ]); + + const appDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Database name:", + name: "databaseName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database Root Password (optional):", + name: "databaseRootPassword", + type: "input", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "input", + }, + { + default: "mysql:8", + message: "Docker Image (default: mysql:8):", + name: "dockerImage", + type: "input", + }, + { + default: "mysql", + message: "Database User: (default: mysql):", + name: "databaseUser", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${appDetails.name}`, + message: "Enter the App name: (optional):", + name: "appName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + ]); + + const responseDatabase = await axios.post( + `${auth.url}/api/trpc/mysql.create`, + { + json: { + ...appDetails, + appName: appName.appName, + projectId: project.projectId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!responseDatabase.data.result.data.json) { + this.error(chalk.red("Error creating database")); + } + + this.log( + chalk.green(`Database '${appDetails.name}' created successfully.`), + ); + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + } +} diff --git a/src/commands/database/postgres/create.ts b/src/commands/database/postgres/create.ts new file mode 100644 index 0000000..94451db --- /dev/null +++ b/src/commands/database/postgres/create.ts @@ -0,0 +1,140 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { slugify } from "../../../utils/slug.js"; +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabasePostgresCreate extends Command { + static description = "Create a new database within a project."; + + static examples = ["$ <%= config.bin %> postgres create"]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(DatabasePostgresCreate); + + const { projectId } = flags; + if (!projectId) { + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the database in:", + name: "project", + type: "list", + }, + ]); + + const appDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Database name:", + name: "databaseName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "input", + }, + { + default: "postgres:15", + message: "Docker Image (default: postgres:15):", + name: "dockerImage", + type: "input", + }, + + { + default: "postgres", + message: "Database User: (default: postgres):", + name: "databaseUser", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${appDetails.name}`, + message: "Enter the App name: (optional):", + name: "appName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + ]); + + const responseDatabase = await axios.post( + `${auth.url}/api/trpc/postgres.create`, + { + json: { + ...appDetails, + appName: appName.appName, + projectId: project.projectId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!responseDatabase.data.result.data.json) { + this.error(chalk.red("Error creating database")); + } + + this.log( + chalk.green(`Database '${appDetails.name}' created successfully.`), + ); + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + } +} diff --git a/src/commands/database/redis/create.ts b/src/commands/database/redis/create.ts new file mode 100644 index 0000000..a68f00a --- /dev/null +++ b/src/commands/database/redis/create.ts @@ -0,0 +1,125 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { slugify } from "../../../utils/slug.js"; +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabaseRedisCreate extends Command { + static description = "Create a new database within a project."; + + static examples = ["$ <%= config.bin %> redis create"]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + const { flags } = await this.parse(DatabaseRedisCreate); + const { projectId } = flags; + if (!projectId) { + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the database in:", + name: "project", + type: "list", + }, + ]); + + const appDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "input", + }, + { + default: "redis:7", + message: "Docker Image (default: redis:7):", + name: "dockerImage", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${appDetails.name}`, + message: "Enter the App name: (optional):", + name: "appName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + ]); + + const responseDatabase = await axios.post( + `${auth.url}/api/trpc/redis.create`, + { + json: { + ...appDetails, + appName: appName.appName, + projectId: project.projectId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!responseDatabase.data.result.data.json) { + this.error(chalk.red("Error creating database")); + } + + this.log( + chalk.green(`Database '${appDetails.name}' created successfully.`), + ); + } catch (error) { + // @ts-expect-error TODO: Fix this + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + } +} diff --git a/src/utils/slug.ts b/src/utils/slug.ts new file mode 100644 index 0000000..1c79a8c --- /dev/null +++ b/src/utils/slug.ts @@ -0,0 +1,14 @@ +import slug from "./slugify.js"; + +export const slugify = (text: string | undefined) => { + if (!text) { + return ""; + } + + const cleanedText = text.trim().replaceAll(/[^\d\sA-Za-z]/g, ""); + return slug(cleanedText, { + lower: true, + strict: true, + trim: true, + }); +}; diff --git a/src/utils/slugify.ts b/src/utils/slugify.ts new file mode 100644 index 0000000..c68e030 --- /dev/null +++ b/src/utils/slugify.ts @@ -0,0 +1,3 @@ +import slugify from "slugify"; + +export default slugify as unknown as typeof slugify.default; diff --git a/test/commands/database/create.test.ts b/test/commands/database/create.test.ts new file mode 100644 index 0000000..0e9c882 --- /dev/null +++ b/test/commands/database/create.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:create', () => { + it('runs database:create cmd', async () => { + const {stdout} = await runCommand('database:create') + expect(stdout).to.contain('hello world') + }) + + it('runs database:create --name oclif', async () => { + const {stdout} = await runCommand('database:create --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/mariadb/create.test.ts b/test/commands/database/mariadb/create.test.ts new file mode 100644 index 0000000..5891adf --- /dev/null +++ b/test/commands/database/mariadb/create.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:mariadb:create', () => { + it('runs database:mariadb:create cmd', async () => { + const {stdout} = await runCommand('database:mariadb:create') + expect(stdout).to.contain('hello world') + }) + + it('runs database:mariadb:create --name oclif', async () => { + const {stdout} = await runCommand('database:mariadb:create --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/mongo/create.test.ts b/test/commands/database/mongo/create.test.ts new file mode 100644 index 0000000..b04cc22 --- /dev/null +++ b/test/commands/database/mongo/create.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:mongo:create', () => { + it('runs database:mongo:create cmd', async () => { + const {stdout} = await runCommand('database:mongo:create') + expect(stdout).to.contain('hello world') + }) + + it('runs database:mongo:create --name oclif', async () => { + const {stdout} = await runCommand('database:mongo:create --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/mysql/create.test.ts b/test/commands/database/mysql/create.test.ts new file mode 100644 index 0000000..0812e46 --- /dev/null +++ b/test/commands/database/mysql/create.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:mysql:create', () => { + it('runs database:mysql:create cmd', async () => { + const {stdout} = await runCommand('database:mysql:create') + expect(stdout).to.contain('hello world') + }) + + it('runs database:mysql:create --name oclif', async () => { + const {stdout} = await runCommand('database:mysql:create --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/postgres/create.test.ts b/test/commands/database/postgres/create.test.ts new file mode 100644 index 0000000..a032ffd --- /dev/null +++ b/test/commands/database/postgres/create.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:postgres:create', () => { + it('runs database:postgres:create cmd', async () => { + const {stdout} = await runCommand('database:postgres:create') + expect(stdout).to.contain('hello world') + }) + + it('runs database:postgres:create --name oclif', async () => { + const {stdout} = await runCommand('database:postgres:create --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/redis/create.test.ts b/test/commands/database/redis/create.test.ts new file mode 100644 index 0000000..9e910e6 --- /dev/null +++ b/test/commands/database/redis/create.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:redis:create', () => { + it('runs database:redis:create cmd', async () => { + const {stdout} = await runCommand('database:redis:create') + expect(stdout).to.contain('hello world') + }) + + it('runs database:redis:create --name oclif', async () => { + const {stdout} = await runCommand('database:redis:create --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) From fbf6b7f5ccee6132007d8d5e92f77222f7972e05 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Wed, 12 Jun 2024 23:15:00 -0600 Subject: [PATCH 08/13] feat: add delete --- src/commands/database/mariadb/delete.ts | 153 ++++++++++++++++++ src/commands/database/mongo/delete.ts | 153 ++++++++++++++++++ src/commands/database/mysql/delete.ts | 153 ++++++++++++++++++ src/commands/database/postgres/delete.ts | 153 ++++++++++++++++++ src/commands/database/redis/delete.ts | 153 ++++++++++++++++++ test/commands/database/mariadb/delete.test.ts | 14 ++ test/commands/database/mongo/delete.test.ts | 14 ++ test/commands/database/mysql/delete.test.ts | 14 ++ .../commands/database/postgres/delete.test.ts | 14 ++ test/commands/database/redis/delete.test.ts | 14 ++ 10 files changed, 835 insertions(+) create mode 100644 src/commands/database/mariadb/delete.ts create mode 100644 src/commands/database/mongo/delete.ts create mode 100644 src/commands/database/mysql/delete.ts create mode 100644 src/commands/database/postgres/delete.ts create mode 100644 src/commands/database/redis/delete.ts create mode 100644 test/commands/database/mariadb/delete.test.ts create mode 100644 test/commands/database/mongo/delete.test.ts create mode 100644 test/commands/database/mysql/delete.test.ts create mode 100644 test/commands/database/postgres/delete.test.ts create mode 100644 test/commands/database/redis/delete.test.ts diff --git a/src/commands/database/mariadb/delete.ts b/src/commands/database/mariadb/delete.ts new file mode 100644 index 0000000..c79ad81 --- /dev/null +++ b/src/commands/database/mariadb/delete.ts @@ -0,0 +1,153 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabaseMariadbDelete extends Command { + static description = "Delete an application from a project."; + + static examples = [ + "$ <%= config.bin %> mariadb delete", + "$ <%= config.bin %> mariadb delete -p ", + ]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(DatabaseMariadbDelete); + let { projectId } = flags; + + if (!projectId) { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the mariadb database from:", + name: "selectedProject", + type: "list", + }, + ]); + + projectId = answers.selectedProject; + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.one`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + params: { + input: JSON.stringify({ + json: { projectId }, + }), + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching applications")); + } + + const apps = response.data.result.data.json; + + if (apps.mariadb.length === 0) { + this.log(chalk.yellow("No applications found in this project.")); + return; + } + + // Permitir al usuario seleccionar una aplicación + const appAnswers = await inquirer.prompt([ + { + choices: apps.mariadb.map((app: any) => ({ + name: app.name, + value: app.mariadbId, + })), + message: "Select the mariadb database to delete:", + name: "selectedApp", + type: "list", + }, + ]); + + const mariadbId = appAnswers.selectedApp; + + // Confirmar eliminación + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to delete this mysql database?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.log(chalk.yellow("Application deletion cancelled.")); + return; + } + + // Eliminar la aplicación seleccionada + const deleteResponse = await axios.post( + `${auth.url}/api/trpc/mariadb.remove`, + { + json: { + mariadbId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!deleteResponse.data.result.data.json) { + this.error(chalk.red("Error deleting application")); + } + + this.log(chalk.green("Application deleted successfully.")); + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to delete application: ${error.message}`)); + } + } +} diff --git a/src/commands/database/mongo/delete.ts b/src/commands/database/mongo/delete.ts new file mode 100644 index 0000000..40c829f --- /dev/null +++ b/src/commands/database/mongo/delete.ts @@ -0,0 +1,153 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabaseMongoDelete extends Command { + static description = "Delete an application from a project."; + + static examples = [ + "$ <%= config.bin %> mongo delete", + "$ <%= config.bin %> mongo delete -p ", + ]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(DatabaseMongoDelete); + let { projectId } = flags; + + if (!projectId) { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the mongo database from:", + name: "selectedProject", + type: "list", + }, + ]); + + projectId = answers.selectedProject; + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.one`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + params: { + input: JSON.stringify({ + json: { projectId }, + }), + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching applications")); + } + + const apps = response.data.result.data.json; + + if (apps.mongo.length === 0) { + this.log(chalk.yellow("No applications found in this project.")); + return; + } + + // Permitir al usuario seleccionar una aplicación + const appAnswers = await inquirer.prompt([ + { + choices: apps.mongo.map((app: any) => ({ + name: app.name, + value: app.mongoId, + })), + message: "Select the mongo database to delete:", + name: "selectedApp", + type: "list", + }, + ]); + + const mongoId = appAnswers.selectedApp; + + // Confirmar eliminación + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to delete this mongo database?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.log(chalk.yellow("Application deletion cancelled.")); + return; + } + + // Eliminar la aplicación seleccionada + const deleteResponse = await axios.post( + `${auth.url}/api/trpc/mongo.remove`, + { + json: { + mongoId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!deleteResponse.data.result.data.json) { + this.error(chalk.red("Error deleting application")); + } + + this.log(chalk.green("Application deleted successfully.")); + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to delete application: ${error.message}`)); + } + } +} diff --git a/src/commands/database/mysql/delete.ts b/src/commands/database/mysql/delete.ts new file mode 100644 index 0000000..bb49888 --- /dev/null +++ b/src/commands/database/mysql/delete.ts @@ -0,0 +1,153 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabaseMysqlDelete extends Command { + static description = "Delete an application from a project."; + + static examples = [ + "$ <%= config.bin %> mysql delete", + "$ <%= config.bin %> mysql delete -p ", + ]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(DatabaseMysqlDelete); + let { projectId } = flags; + + if (!projectId) { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the mysql database from:", + name: "selectedProject", + type: "list", + }, + ]); + + projectId = answers.selectedProject; + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.one`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + params: { + input: JSON.stringify({ + json: { projectId }, + }), + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching applications")); + } + + const apps = response.data.result.data.json; + + if (apps.mysql.length === 0) { + this.log(chalk.yellow("No applications found in this project.")); + return; + } + + // Permitir al usuario seleccionar una aplicación + const appAnswers = await inquirer.prompt([ + { + choices: apps.mysql.map((app: any) => ({ + name: app.name, + value: app.mysqlId, + })), + message: "Select the mysql database to delete:", + name: "selectedApp", + type: "list", + }, + ]); + + const mysqlId = appAnswers.selectedApp; + + // Confirmar eliminación + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to delete this mysql database?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.log(chalk.yellow("Application deletion cancelled.")); + return; + } + + // Eliminar la aplicación seleccionada + const deleteResponse = await axios.post( + `${auth.url}/api/trpc/mysql.remove`, + { + json: { + mysqlId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!deleteResponse.data.result.data.json) { + this.error(chalk.red("Error deleting application")); + } + + this.log(chalk.green("Application deleted successfully.")); + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to delete application: ${error.message}`)); + } + } +} diff --git a/src/commands/database/postgres/delete.ts b/src/commands/database/postgres/delete.ts new file mode 100644 index 0000000..47aaf0e --- /dev/null +++ b/src/commands/database/postgres/delete.ts @@ -0,0 +1,153 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabasePostgresDelete extends Command { + static description = "Delete an application from a project."; + + static examples = [ + "$ <%= config.bin %> postgres delete", + "$ <%= config.bin %> postgres delete -p ", + ]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(DatabasePostgresDelete); + let { projectId } = flags; + + if (!projectId) { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the postgres database from:", + name: "selectedProject", + type: "list", + }, + ]); + + projectId = answers.selectedProject; + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.one`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + params: { + input: JSON.stringify({ + json: { projectId }, + }), + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching applications")); + } + + const apps = response.data.result.data.json; + + if (apps.postgres.length === 0) { + this.log(chalk.yellow("No applications found in this project.")); + return; + } + + // Permitir al usuario seleccionar una aplicación + const appAnswers = await inquirer.prompt([ + { + choices: apps.postgres.map((app: any) => ({ + name: app.name, + value: app.postgresId, + })), + message: "Select the postgres database to delete:", + name: "selectedApp", + type: "list", + }, + ]); + + const postgresId = appAnswers.selectedApp; + + // Confirmar eliminación + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to delete this postgres database?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.log(chalk.yellow("Application deletion cancelled.")); + return; + } + + // Eliminar la aplicación seleccionada + const deleteResponse = await axios.post( + `${auth.url}/api/trpc/postgres.remove`, + { + json: { + postgresId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!deleteResponse.data.result.data.json) { + this.error(chalk.red("Error deleting application")); + } + + this.log(chalk.green("Application deleted successfully.")); + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to delete application: ${error.message}`)); + } + } +} diff --git a/src/commands/database/redis/delete.ts b/src/commands/database/redis/delete.ts new file mode 100644 index 0000000..258927f --- /dev/null +++ b/src/commands/database/redis/delete.ts @@ -0,0 +1,153 @@ +import { Command, Flags } from "@oclif/core"; +import axios from "axios"; +import chalk from "chalk"; +import inquirer from "inquirer"; + +import { readAuthConfig } from "../../../utils/utils.js"; + +export default class DatabaseRedisDelete extends Command { + static description = "Delete an application from a project."; + + static examples = [ + "$ <%= config.bin %> redis delete", + "$ <%= config.bin %> redis delete -p ", + ]; + + static flags = { + projectId: Flags.string({ + char: "p", + description: "ID of the project", + required: false, + }), + }; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + const { flags } = await this.parse(DatabaseRedisDelete); + let { projectId } = flags; + + if (!projectId) { + // Obtener la lista de proyectos y permitir la selección + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; + } + + // Permitir al usuario seleccionar un proyecto + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the mariadb database from:", + name: "selectedProject", + type: "list", + }, + ]); + + projectId = answers.selectedProject; + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } + } + + try { + const response = await axios.get(`${auth.url}/api/trpc/project.one`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + params: { + input: JSON.stringify({ + json: { projectId }, + }), + }, + }); + + if (!response.data.result.data.json) { + this.error(chalk.red("Error fetching applications")); + } + + const apps = response.data.result.data.json; + + if (apps.redis.length === 0) { + this.log(chalk.yellow("No applications found in this project.")); + return; + } + + // Permitir al usuario seleccionar una aplicación + const appAnswers = await inquirer.prompt([ + { + choices: apps.redis.map((app: any) => ({ + name: app.name, + value: app.redisId, + })), + message: "Select the redis database to delete:", + name: "selectedApp", + type: "list", + }, + ]); + + const redisId = appAnswers.selectedApp; + + // Confirmar eliminación + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to delete this mysql database?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.log(chalk.yellow("Application deletion cancelled.")); + return; + } + + // Eliminar la aplicación seleccionada + const deleteResponse = await axios.post( + `${auth.url}/api/trpc/redis.remove`, + { + json: { + redisId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (!deleteResponse.data.result.data.json) { + this.error(chalk.red("Error deleting application")); + } + + this.log(chalk.green("Application deleted successfully.")); + } catch (error) { + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + this.error(chalk.red(`Failed to delete application: ${error.message}`)); + } + } +} diff --git a/test/commands/database/mariadb/delete.test.ts b/test/commands/database/mariadb/delete.test.ts new file mode 100644 index 0000000..1351ffe --- /dev/null +++ b/test/commands/database/mariadb/delete.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:mariadb:delete', () => { + it('runs database:mariadb:delete cmd', async () => { + const {stdout} = await runCommand('database:mariadb:delete') + expect(stdout).to.contain('hello world') + }) + + it('runs database:mariadb:delete --name oclif', async () => { + const {stdout} = await runCommand('database:mariadb:delete --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/mongo/delete.test.ts b/test/commands/database/mongo/delete.test.ts new file mode 100644 index 0000000..486a81b --- /dev/null +++ b/test/commands/database/mongo/delete.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:mongo:delete', () => { + it('runs database:mongo:delete cmd', async () => { + const {stdout} = await runCommand('database:mongo:delete') + expect(stdout).to.contain('hello world') + }) + + it('runs database:mongo:delete --name oclif', async () => { + const {stdout} = await runCommand('database:mongo:delete --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/mysql/delete.test.ts b/test/commands/database/mysql/delete.test.ts new file mode 100644 index 0000000..9588a8f --- /dev/null +++ b/test/commands/database/mysql/delete.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:mysql:delete', () => { + it('runs database:mysql:delete cmd', async () => { + const {stdout} = await runCommand('database:mysql:delete') + expect(stdout).to.contain('hello world') + }) + + it('runs database:mysql:delete --name oclif', async () => { + const {stdout} = await runCommand('database:mysql:delete --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/postgres/delete.test.ts b/test/commands/database/postgres/delete.test.ts new file mode 100644 index 0000000..8b5817e --- /dev/null +++ b/test/commands/database/postgres/delete.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:postgres:delete', () => { + it('runs database:postgres:delete cmd', async () => { + const {stdout} = await runCommand('database:postgres:delete') + expect(stdout).to.contain('hello world') + }) + + it('runs database:postgres:delete --name oclif', async () => { + const {stdout} = await runCommand('database:postgres:delete --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) diff --git a/test/commands/database/redis/delete.test.ts b/test/commands/database/redis/delete.test.ts new file mode 100644 index 0000000..52be139 --- /dev/null +++ b/test/commands/database/redis/delete.test.ts @@ -0,0 +1,14 @@ +import {runCommand} from '@oclif/test' +import {expect} from 'chai' + +describe('database:redis:delete', () => { + it('runs database:redis:delete cmd', async () => { + const {stdout} = await runCommand('database:redis:delete') + expect(stdout).to.contain('hello world') + }) + + it('runs database:redis:delete --name oclif', async () => { + const {stdout} = await runCommand('database:redis:delete --name oclif') + expect(stdout).to.contain('hello oclif') + }) +}) From d610c967d979f3312125df7c68096e290c552c84 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Sat, 22 Jun 2024 00:52:17 -0600 Subject: [PATCH 09/13] refactor: add commands --- .eslintrc.json | 2 +- config.json | 2 +- src/commands/app/create.ts | 158 +++++++----------- src/commands/app/delete.ts | 89 +++------- src/commands/database/create.ts | 121 -------------- src/commands/database/mariadb/create.ts | 203 +++++++++++------------ src/commands/database/mariadb/delete.ts | 118 +++++-------- src/commands/database/mongo/create.ts | 169 +++++++++---------- src/commands/database/mongo/delete.ts | 102 ++++-------- src/commands/database/mysql/create.ts | 176 ++++++++++---------- src/commands/database/mysql/delete.ts | 81 +++------ src/commands/database/postgres/create.ts | 168 +++++++++---------- src/commands/database/postgres/delete.ts | 96 ++++------- src/commands/database/redis/create.ts | 136 ++++++++------- src/commands/database/redis/delete.ts | 99 ++++------- src/commands/project/info.ts | 56 ++----- src/commands/project/list.ts | 19 +-- src/utils/shared.ts | 81 +++++++++ src/utils/utils.ts | 9 +- 19 files changed, 755 insertions(+), 1130 deletions(-) delete mode 100644 src/commands/database/create.ts create mode 100644 src/utils/shared.ts diff --git a/.eslintrc.json b/.eslintrc.json index 1dfcfc4..b315bce 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -1,3 +1,3 @@ { - "extends": ["oclif", "oclif-typescript", "prettier"] + "extends": [ "prettier"] } diff --git a/config.json b/config.json index e5834e2..14d290a 100644 --- a/config.json +++ b/config.json @@ -1,4 +1,4 @@ { - "token": "5qdderpq2ejrjlu90hrnbjohvgc8j1u1k7i00um4", + "token": "icsy0appti460sbh5be1sevami702rc8a57l2e8h", "url": "http://localhost:3000" } \ No newline at end of file diff --git a/src/commands/app/create.ts b/src/commands/app/create.ts index 339d859..a9be784 100644 --- a/src/commands/app/create.ts +++ b/src/commands/app/create.ts @@ -3,8 +3,14 @@ import axios from "axios"; import chalk from "chalk"; import inquirer from "inquirer"; +import { type Project, getProjects } from "../../utils/shared.js"; +import { slugify } from "../../utils/slug.js"; import { readAuthConfig } from "../../utils/utils.js"; +export interface Answers { + project: Project; +} + export default class AppCreate extends Command { static description = "Create a new application within a project."; @@ -26,110 +32,72 @@ export default class AppCreate extends Command { let { projectId } = flags; if (!projectId) { - // Obtener la lista de proyectos y permitir la selección console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the application in:", + name: "project", + type: "list", + }, + ]); + + projectId = project.projectId; + + const appDetails = await inquirer.prompt([ + { + message: "Enter the application name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Application name is required"), + }, + { + message: "Enter the application description (optional):", + name: "appDescription", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${appDetails.name}`, + message: "Enter the App name: (optional):", + name: "appName", + type: "input", + validate: (input) => (input ? true : "App name is required"), + }, + ]); + + const response = await axios.post( + `${auth.url}/api/trpc/application.create`, + { + json: { + ...appDetails, + appName: appName.appName, + projectId: project.projectId, + }, + }, + { headers: { Authorization: `Bearer ${auth.token}`, "Content-Type": "application/json", }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - // Permitir al usuario seleccionar un proyecto - const answers = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project.projectId, - })), - message: "Select a project to create the database in:", - name: "selectedProject", - type: "list", - }, - ]); + }, + ); - projectId = answers.selectedProject; - } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + if (response.status !== 200) { + this.error(chalk.red("Error creating application")); } - } - const databases = ["postgres", "mysql", "redis", "mariadb", "mongo"]; - - const databaseSelect = await inquirer.prompt([ - { - choices: databases.map((database: any) => ({ - name: database, - value: database, - })), - message: "Select a database to create the application in:", - name: "selectedDatabase", - type: "list", - }, - ]); - - const urlSelected = `${auth.url}/api/trpc/${databaseSelect.selectedDatabase}.create`; - - // Solicitar detalles de la nueva aplicación - const appDetails = await inquirer.prompt([ - { - message: "Enter the database name:", - name: "appName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Enter the database description (optional):", - name: "appDescription", - type: "input", - }, - ]); - - const { appDescription, appName } = appDetails; - - // Crear la aplicación en el proyecto seleccionado - try { - // const response = await axios.post( - // `${auth.url}/api/trpc/application.create`, - // { - // json: { - // description: appDescription, - // name: appName, - // projectId, - // }, - // }, - // { - // headers: { - // Authorization: `Bearer ${auth.token}`, - // "Content-Type": "application/json", - // }, - // }, - // ); - // if (!response.data.result.data.json) { - // this.error(chalk.red("Error creating application")); - // } - // this.log( - // chalk.green( - // `Application '${appName}' created successfully in project ID '${projectId}'.`, - // ), - // ); - } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to create application: ${error.message}`)); + this.log( + chalk.green(`Application '${appDetails.name}' created successfully.`), + ); } } } diff --git a/src/commands/app/delete.ts b/src/commands/app/delete.ts index 6a12f98..b8ca78b 100644 --- a/src/commands/app/delete.ts +++ b/src/commands/app/delete.ts @@ -3,7 +3,9 @@ import axios from "axios"; import chalk from "chalk"; import inquirer from "inquirer"; +import { getProject, getProjects } from "../../utils/shared.js"; import { readAuthConfig } from "../../utils/utils.js"; +import type { Answers } from "./create.js"; export default class AppDelete extends Command { static description = "Delete an application from a project."; @@ -29,77 +31,33 @@ export default class AppDelete extends Command { let { projectId } = flags; if (!projectId) { - // Obtener la lista de proyectos y permitir la selección console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - // Permitir al usuario seleccionar un proyecto - const answers = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project.projectId, - })), - message: "Select a project to delete the application from:", - name: "selectedProject", - type: "list", - }, - ]); - - projectId = answers.selectedProject; - } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); - } - } + const projects = await getProjects(auth, this); - // Obtener la lista de aplicaciones del proyecto seleccionado - try { - const response = await axios.get(`${auth.url}/api/trpc/project.one`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - params: { - input: JSON.stringify({ - json: { projectId }, - }), + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the application in:", + name: "project", + type: "list", }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching applications")); - } + ]); - const apps = response.data.result.data.json; + projectId = project.projectId; + const projectSelected = await getProject(projectId, auth, this); - if (apps.applications.length === 0) { - this.log(chalk.yellow("No applications found in this project.")); - return; + if (projectSelected.applications.length === 0) { + this.error(chalk.yellow("No applications found in this project.")); } - // Permitir al usuario seleccionar una aplicación const appAnswers = await inquirer.prompt([ { - choices: apps.applications.map((app: any) => ({ + // @ts-ignore + choices: projectSelected.applications.map((app) => ({ name: app.name, value: app.applicationId, })), @@ -111,7 +69,7 @@ export default class AppDelete extends Command { const applicationId = appAnswers.selectedApp; - // Confirmar eliminación + // // Confirmar eliminación const confirmAnswers = await inquirer.prompt([ { default: false, @@ -122,11 +80,9 @@ export default class AppDelete extends Command { ]); if (!confirmAnswers.confirmDelete) { - this.log(chalk.yellow("Application deletion cancelled.")); - return; + this.error(chalk.yellow("Application deletion cancelled.")); } - // Eliminar la aplicación seleccionada const deleteResponse = await axios.post( `${auth.url}/api/trpc/application.delete`, { @@ -147,9 +103,6 @@ export default class AppDelete extends Command { } this.log(chalk.green("Application deleted successfully.")); - } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to delete application: ${error.message}`)); } } } diff --git a/src/commands/database/create.ts b/src/commands/database/create.ts deleted file mode 100644 index b9559d4..0000000 --- a/src/commands/database/create.ts +++ /dev/null @@ -1,121 +0,0 @@ -import { Command, Flags } from "@oclif/core"; -import axios from "axios"; -import chalk from "chalk"; -import inquirer from "inquirer"; - -import { readAuthConfig } from "../../utils/utils.js"; - -export default class DatabaseCreate extends Command { - static description = "Create a new database within a project."; - - static examples = ["$ <%= config.bin %> app create"]; - - static flags = { - projectId: Flags.string({ - char: "p", - description: "ID of the project", - required: false, - }), - }; - - public async run(): Promise { - const auth = await readAuthConfig(this); - - const { flags } = await this.parse(DatabaseCreate); - - let { projectId } = flags; - - if (!projectId) { - // Obtener la lista de proyectos y permitir la selección - console.log(chalk.blue.bold("\n Listing all Projects \n")); - - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - // Permitir al usuario seleccionar un proyecto - const answers = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project.projectId, - })), - message: "Select a project to create the application in:", - name: "selectedProject", - type: "list", - }, - ]); - - projectId = answers.selectedProject; - } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); - } - } - - // Solicitar detalles de la nueva aplicación - const appDetails = await inquirer.prompt([ - { - message: "Enter the database name:", - name: "appName", - type: "input", - validate: (input) => (input ? true : "Application name is required"), - }, - { - message: "Enter the application description (optional):", - name: "appDescription", - type: "input", - }, - ]); - - const { appDescription, appName } = appDetails; - - // Crear la aplicación en el proyecto seleccionado - try { - const response = await axios.post( - `${auth.url}/api/trpc/database.create`, - { - json: { - description: appDescription, - name: appName, - projectId, - }, - }, - { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }, - ); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error creating application")); - } - - this.log( - chalk.green( - `Application '${appName}' created successfully in project ID '${projectId}'.`, - ), - ); - } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to create application: ${error.message}`)); - } - } -} diff --git a/src/commands/database/mariadb/create.ts b/src/commands/database/mariadb/create.ts index 38b3f2a..676405c 100644 --- a/src/commands/database/mariadb/create.ts +++ b/src/commands/database/mariadb/create.ts @@ -2,12 +2,13 @@ import { Command, Flags } from "@oclif/core"; import axios from "axios"; import chalk from "chalk"; import inquirer from "inquirer"; - -import { slugify } from "../../../utils/slug.js"; import { readAuthConfig } from "../../../utils/utils.js"; +import { getProjects } from "../../../utils/shared.js"; +import { slugify } from "../../../utils/slug.js"; +import type { Answers } from "../../app/create.js"; export default class DatabaseMariadbCreate extends Command { - static description = "Create a new database within a project."; + static description = "Create a new MariaDB database within a project."; static examples = ["$ <%= config.bin %> mariadb create"]; @@ -24,121 +25,105 @@ export default class DatabaseMariadbCreate extends Command { const { flags } = await this.parse(DatabaseMariadbCreate); - const { projectId } = flags; + let { projectId } = flags; + if (!projectId) { console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the MariaDB database in:", + name: "project", + type: "list", + }, + ]); + + projectId = project.projectId; + + const dbDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Database name:", + name: "databaseName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database Root Password (optional):", + name: "databaseRootPassword", + type: "password", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "password", + }, + { + default: "mariadb:4", + message: "Docker Image (default: mariadb:4):", + name: "dockerImage", + type: "input", + }, + { + default: "mariadb", + message: "Database User: (default: mariadb):", + name: "databaseUser", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${dbDetails.name}`, + message: "Enter the App name:", + name: "appName", + type: "input", + validate: (input) => (input ? true : "App name is required"), + }, + ]); + + const response = await axios.post( + `${auth.url}/api/trpc/mariadb.create`, + { + json: { + ...dbDetails, + appName: appName.appName, + projectId: project.projectId, + }, + }, + { headers: { Authorization: `Bearer ${auth.token}`, "Content-Type": "application/json", }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; + }, + ); - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - const { project } = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project, - })), - message: "Select a project to create the database in:", - name: "project", - type: "list", - }, - ]); - - const appDetails = await inquirer.prompt([ - { - message: "Enter the name:", - name: "name", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Database name:", - name: "databaseName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Enter the database description (optional):", - name: "description", - type: "input", - }, - { - message: "Database Root Password (optional):", - name: "databaseRootPassword", - type: "input", - }, - { - message: "Database password (optional):", - name: "databasePassword", - type: "input", - }, - { - default: "mariadb:4", - message: "Docker Image (default: mariadb:4):", - name: "dockerImage", - type: "input", - }, - { - default: "mariadb", - message: "Database User: (default: mariadb):", - name: "databaseUser", - type: "input", - }, - ]); - - const appName = await inquirer.prompt([ - { - default: `${slugify(project.name)}-${appDetails.name}`, - message: "Enter the App name: (optional):", - name: "appName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - ]); - - const responseDatabase = await axios.post( - `${auth.url}/api/trpc/mariadb.create`, - { - json: { - ...appDetails, - appName: appName.appName, - projectId: project.projectId, - }, - }, - { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }, - ); - - if (!responseDatabase.data.result.data.json) { - this.error(chalk.red("Error creating database")); - } - - this.log( - chalk.green(`Database '${appDetails.name}' created successfully.`), - ); - } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + if (!response.data.result.data.json) { + this.error(chalk.red("Error creating MariaDB database")); } + + this.log( + chalk.green( + `MariaDB database '${dbDetails.name}' created successfully.`, + ), + ); } } } diff --git a/src/commands/database/mariadb/delete.ts b/src/commands/database/mariadb/delete.ts index c79ad81..bc8dd4c 100644 --- a/src/commands/database/mariadb/delete.ts +++ b/src/commands/database/mariadb/delete.ts @@ -3,16 +3,15 @@ import axios from "axios"; import chalk from "chalk"; import inquirer from "inquirer"; +import { getProject, getProjects } from "../../../utils/shared.js"; import { readAuthConfig } from "../../../utils/utils.js"; export default class DatabaseMariadbDelete extends Command { - static description = "Delete an application from a project."; - + static description = "Delete a MariaDB database from a project."; static examples = [ "$ <%= config.bin %> mariadb delete", "$ <%= config.bin %> mariadb delete -p ", ]; - static flags = { projectId: Flags.string({ char: "p", @@ -23,108 +22,68 @@ export default class DatabaseMariadbDelete extends Command { public async run(): Promise { const auth = await readAuthConfig(this); - const { flags } = await this.parse(DatabaseMariadbDelete); let { projectId } = flags; if (!projectId) { - // Obtener la lista de proyectos y permitir la selección console.log(chalk.blue.bold("\n Listing all Projects \n")); + const projects = await getProjects(auth, this); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - // Permitir al usuario seleccionar un proyecto - const answers = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project.projectId, - })), - message: "Select a project to delete the mariadb database from:", - name: "selectedProject", - type: "list", - }, - ]); - - projectId = answers.selectedProject; - } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; } - } - try { - const response = await axios.get(`${auth.url}/api/trpc/project.one`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - params: { - input: JSON.stringify({ - json: { projectId }, - }), + const answers = await inquirer.prompt([ + { + type: "list", + name: "selectedProject", + message: "Select a project to delete the MariaDB database from:", + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching applications")); - } + ]); + projectId = answers.selectedProject; + } - const apps = response.data.result.data.json; + try { + const project = await getProject(projectId, auth, this); - if (apps.mariadb.length === 0) { - this.log(chalk.yellow("No applications found in this project.")); + if (!project.mariadb || project.mariadb.length === 0) { + this.log(chalk.yellow("No MariaDB databases found in this project.")); return; } - // Permitir al usuario seleccionar una aplicación const appAnswers = await inquirer.prompt([ { - choices: apps.mariadb.map((app: any) => ({ - name: app.name, - value: app.mariadbId, - })), - message: "Select the mariadb database to delete:", - name: "selectedApp", type: "list", + name: "selectedDb", + message: "Select the MariaDB database to delete:", + choices: project.mariadb.map((db: any) => ({ + name: db.name, + value: db.mariadbId, + })), }, ]); - const mariadbId = appAnswers.selectedApp; + const mariadbId = appAnswers.selectedDb; - // Confirmar eliminación const confirmAnswers = await inquirer.prompt([ { - default: false, - message: "Are you sure you want to delete this mysql database?", - name: "confirmDelete", type: "confirm", + name: "confirmDelete", + message: "Are you sure you want to delete this MariaDB database?", + default: false, }, ]); if (!confirmAnswers.confirmDelete) { - this.log(chalk.yellow("Application deletion cancelled.")); + this.log(chalk.yellow("Database deletion cancelled.")); return; } - // Eliminar la aplicación seleccionada const deleteResponse = await axios.post( `${auth.url}/api/trpc/mariadb.remove`, { @@ -141,13 +100,14 @@ export default class DatabaseMariadbDelete extends Command { ); if (!deleteResponse.data.result.data.json) { - this.error(chalk.red("Error deleting application")); + this.error(chalk.red("Error deleting mariadb database")); } - - this.log(chalk.green("Application deleted successfully.")); + this.log(chalk.green("MariaDB database deleted successfully.")); } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to delete application: ${error.message}`)); + this.error( + // @ts-ignore + chalk.red(`Failed to delete MariaDB database: ${error.message}`), + ); } } } diff --git a/src/commands/database/mongo/create.ts b/src/commands/database/mongo/create.ts index df1c007..ecb486a 100644 --- a/src/commands/database/mongo/create.ts +++ b/src/commands/database/mongo/create.ts @@ -2,12 +2,13 @@ import { Command, Flags } from "@oclif/core"; import axios from "axios"; import chalk from "chalk"; import inquirer from "inquirer"; - -import { slugify } from "../../../utils/slug.js"; import { readAuthConfig } from "../../../utils/utils.js"; +import { getProjects } from "../../../utils/shared.js"; +import { slugify } from "../../../utils/slug.js"; +import type { Answers } from "../../app/create.js"; export default class DatabaseMongoCreate extends Command { - static description = "Create a new database within a project."; + static description = "Create a new MongoDB database within a project."; static examples = ["$ <%= config.bin %> mongo create"]; @@ -21,95 +22,83 @@ export default class DatabaseMongoCreate extends Command { public async run(): Promise { const auth = await readAuthConfig(this); + const { flags } = await this.parse(DatabaseMongoCreate); - const { projectId } = flags; + let { projectId } = flags; + if (!projectId) { console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the MongoDB database in:", + name: "project", + type: "list", + }, + ]); + + projectId = project.projectId; + + const dbDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Database name:", + name: "databaseName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "password", + }, + { + default: "mongo:6", + message: "Docker Image (default: mongo:6):", + name: "dockerImage", + type: "input", + }, + { + default: "mongo", + message: "Database User: (default: mongo):", + name: "databaseUser", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${dbDetails.name}`, + message: "Enter the App name:", + name: "appName", + type: "input", + validate: (input) => (input ? true : "App name is required"), + }, + ]); - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - const { project } = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project, - })), - message: "Select a project to create the database in:", - name: "project", - type: "list", - }, - ]); - - const appDetails = await inquirer.prompt([ - { - message: "Enter the name:", - name: "name", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Database name:", - name: "databaseName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Enter the database description (optional):", - name: "description", - type: "input", - }, - { - message: "Database password (optional):", - name: "databasePassword", - type: "input", - }, - { - default: "mongo:6", - message: "Docker Image (default: mongo:6):", - name: "dockerImage", - type: "input", - }, - { - default: "mongo", - message: "Database User: (default: mongo):", - name: "databaseUser", - type: "input", - }, - ]); - - const appName = await inquirer.prompt([ - { - default: `${slugify(project.name)}-${appDetails.name}`, - message: "Enter the App name: (optional):", - name: "appName", - type: "input", - validate: (input) => (input ? true : "App name is required"), - }, - ]); - - const responseDatabase = await axios.post( + try { + const response = await axios.post( `${auth.url}/api/trpc/mongo.create`, { json: { - ...appDetails, + ...dbDetails, appName: appName.appName, projectId: project.projectId, }, @@ -122,16 +111,20 @@ export default class DatabaseMongoCreate extends Command { }, ); - if (!responseDatabase.data.result.data.json) { - this.error(chalk.red("Error creating database")); + if (!response.data.result.data.json) { + this.error(chalk.red("Error creating MongoDB database")); } this.log( - chalk.green(`Database '${appDetails.name}' created successfully.`), + chalk.green( + `MongoDB database '${dbDetails.name}' created successfully.`, + ), ); } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + this.error( + // @ts-ignore + chalk.red(`Failed to create MongoDB database: ${error.message}`), + ); } } } diff --git a/src/commands/database/mongo/delete.ts b/src/commands/database/mongo/delete.ts index 40c829f..6bdabbc 100644 --- a/src/commands/database/mongo/delete.ts +++ b/src/commands/database/mongo/delete.ts @@ -4,9 +4,10 @@ import chalk from "chalk"; import inquirer from "inquirer"; import { readAuthConfig } from "../../../utils/utils.js"; +import { getProject, getProjects } from "../../../utils/shared.js"; export default class DatabaseMongoDelete extends Command { - static description = "Delete an application from a project."; + static description = "Delete a MongoDB database from a project."; static examples = [ "$ <%= config.bin %> mongo delete", @@ -28,80 +29,45 @@ export default class DatabaseMongoDelete extends Command { let { projectId } = flags; if (!projectId) { - // Obtener la lista de proyectos y permitir la selección console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - // Permitir al usuario seleccionar un proyecto - const answers = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project.projectId, - })), - message: "Select a project to delete the mongo database from:", - name: "selectedProject", - type: "list", - }, - ]); + const projects = await getProjects(auth, this); - projectId = answers.selectedProject; - } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; } - } - try { - const response = await axios.get(`${auth.url}/api/trpc/project.one`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - params: { - input: JSON.stringify({ - json: { projectId }, - }), + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the MongoDB database from:", + name: "selectedProject", + type: "list", }, - }); + ]); - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching applications")); - } + projectId = answers.selectedProject; + } - const apps = response.data.result.data.json; + try { + const project = await getProject(projectId, auth, this); - if (apps.mongo.length === 0) { - this.log(chalk.yellow("No applications found in this project.")); + if (!project.mongo || project.mongo.length === 0) { + this.log(chalk.yellow("No MongoDB databases found in this project.")); return; } - // Permitir al usuario seleccionar una aplicación const appAnswers = await inquirer.prompt([ { - choices: apps.mongo.map((app: any) => ({ - name: app.name, - value: app.mongoId, + choices: project.mongo.map((db: any) => ({ + name: db.name, + value: db.mongoId, })), - message: "Select the mongo database to delete:", + message: "Select the MongoDB database to delete:", name: "selectedApp", type: "list", }, @@ -109,22 +75,20 @@ export default class DatabaseMongoDelete extends Command { const mongoId = appAnswers.selectedApp; - // Confirmar eliminación const confirmAnswers = await inquirer.prompt([ { default: false, - message: "Are you sure you want to delete this mongo database?", + message: "Are you sure you want to delete this MongoDB database?", name: "confirmDelete", type: "confirm", }, ]); if (!confirmAnswers.confirmDelete) { - this.log(chalk.yellow("Application deletion cancelled.")); + this.log(chalk.yellow("Database deletion cancelled.")); return; } - // Eliminar la aplicación seleccionada const deleteResponse = await axios.post( `${auth.url}/api/trpc/mongo.remove`, { @@ -141,13 +105,15 @@ export default class DatabaseMongoDelete extends Command { ); if (!deleteResponse.data.result.data.json) { - this.error(chalk.red("Error deleting application")); + this.error(chalk.red("Error deleting MongoDB database")); } - this.log(chalk.green("Application deleted successfully.")); + this.log(chalk.green("MongoDB database deleted successfully.")); } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to delete application: ${error.message}`)); + this.error( + // @ts-ignore + chalk.red(`Failed to delete MongoDB database: ${error.message}`), + ); } } } diff --git a/src/commands/database/mysql/create.ts b/src/commands/database/mysql/create.ts index eb286b7..8f654c5 100644 --- a/src/commands/database/mysql/create.ts +++ b/src/commands/database/mysql/create.ts @@ -5,9 +5,11 @@ import inquirer from "inquirer"; import { slugify } from "../../../utils/slug.js"; import { readAuthConfig } from "../../../utils/utils.js"; +import { getProjects } from "../../../utils/shared.js"; +import type { Answers } from "../../app/create.js"; export default class DatabaseMysqlCreate extends Command { - static description = "Create a new database within a project."; + static description = "Create a new MySQL database within a project."; static examples = ["$ <%= config.bin %> mysql create"]; @@ -21,100 +23,88 @@ export default class DatabaseMysqlCreate extends Command { public async run(): Promise { const auth = await readAuthConfig(this); + const { flags } = await this.parse(DatabaseMysqlCreate); - const { projectId } = flags; + let { projectId } = flags; + if (!projectId) { console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - const { project } = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project, - })), - message: "Select a project to create the database in:", - name: "project", - type: "list", - }, - ]); - - const appDetails = await inquirer.prompt([ - { - message: "Enter the name:", - name: "name", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Database name:", - name: "databaseName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Enter the database description (optional):", - name: "description", - type: "input", - }, - { - message: "Database Root Password (optional):", - name: "databaseRootPassword", - type: "input", - }, - { - message: "Database password (optional):", - name: "databasePassword", - type: "input", - }, - { - default: "mysql:8", - message: "Docker Image (default: mysql:8):", - name: "dockerImage", - type: "input", - }, - { - default: "mysql", - message: "Database User: (default: mysql):", - name: "databaseUser", - type: "input", - }, - ]); - - const appName = await inquirer.prompt([ - { - default: `${slugify(project.name)}-${appDetails.name}`, - message: "Enter the App name: (optional):", - name: "appName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - ]); + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the MySQL database in:", + name: "project", + type: "list", + }, + ]); + + projectId = project.projectId; + + const dbDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Database name:", + name: "databaseName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database Root Password (optional):", + name: "databaseRootPassword", + type: "password", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "password", + }, + { + default: "mysql:8", + message: "Docker Image (default: mysql:8):", + name: "dockerImage", + type: "input", + }, + { + default: "mysql", + message: "Database User: (default: mysql):", + name: "databaseUser", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${dbDetails.name}`, + message: "Enter the App name:", + name: "appName", + type: "input", + validate: (input) => (input ? true : "App name is required"), + }, + ]); - const responseDatabase = await axios.post( + try { + const response = await axios.post( `${auth.url}/api/trpc/mysql.create`, { json: { - ...appDetails, + ...dbDetails, appName: appName.appName, projectId: project.projectId, }, @@ -127,16 +117,20 @@ export default class DatabaseMysqlCreate extends Command { }, ); - if (!responseDatabase.data.result.data.json) { - this.error(chalk.red("Error creating database")); + if (!response.data.result.data.json) { + this.error(chalk.red("Error creating MySQL database")); } this.log( - chalk.green(`Database '${appDetails.name}' created successfully.`), + chalk.green( + `MySQL database '${dbDetails.name}' created successfully.`, + ), ); } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + this.error( + // @ts-ignore + chalk.red(`Failed to create MySQL database: ${error.message}`), + ); } } } diff --git a/src/commands/database/mysql/delete.ts b/src/commands/database/mysql/delete.ts index bb49888..8ea0c01 100644 --- a/src/commands/database/mysql/delete.ts +++ b/src/commands/database/mysql/delete.ts @@ -4,9 +4,10 @@ import chalk from "chalk"; import inquirer from "inquirer"; import { readAuthConfig } from "../../../utils/utils.js"; +import { getProject, getProjects } from "../../../utils/shared.js"; export default class DatabaseMysqlDelete extends Command { - static description = "Delete an application from a project."; + static description = "Delete a MySQL database from a project."; static examples = [ "$ <%= config.bin %> mysql delete", @@ -28,80 +29,46 @@ export default class DatabaseMysqlDelete extends Command { let { projectId } = flags; if (!projectId) { - // Obtener la lista de proyectos y permitir la selección console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - // Permitir al usuario seleccionar un proyecto - const answers = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project.projectId, - })), - message: "Select a project to delete the mysql database from:", - name: "selectedProject", - type: "list", - }, - ]); + const projects = await getProjects(auth, this); - projectId = answers.selectedProject; - } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; } - } - try { - const response = await axios.get(`${auth.url}/api/trpc/project.one`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - params: { - input: JSON.stringify({ - json: { projectId }, - }), + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the MySQL database from:", + name: "selectedProject", + type: "list", }, - }); + ]); - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching applications")); - } + projectId = answers.selectedProject; + } - const apps = response.data.result.data.json; + try { + const project = await getProject(projectId, auth, this); - if (apps.mysql.length === 0) { - this.log(chalk.yellow("No applications found in this project.")); + if (!project.mysql || project.mysql.length === 0) { + this.log(chalk.yellow("No MySQL databases found in this project.")); return; } // Permitir al usuario seleccionar una aplicación const appAnswers = await inquirer.prompt([ { - choices: apps.mysql.map((app: any) => ({ + choices: project.mysql.map((app: any) => ({ name: app.name, value: app.mysqlId, })), - message: "Select the mysql database to delete:", + message: "Select the MySQL database to delete:", name: "selectedApp", type: "list", }, diff --git a/src/commands/database/postgres/create.ts b/src/commands/database/postgres/create.ts index 94451db..6b8514f 100644 --- a/src/commands/database/postgres/create.ts +++ b/src/commands/database/postgres/create.ts @@ -2,12 +2,12 @@ import { Command, Flags } from "@oclif/core"; import axios from "axios"; import chalk from "chalk"; import inquirer from "inquirer"; - import { slugify } from "../../../utils/slug.js"; import { readAuthConfig } from "../../../utils/utils.js"; - +import { getProjects } from "../../../utils/shared.js"; +import type { Answers } from "../../app/create.js"; export default class DatabasePostgresCreate extends Command { - static description = "Create a new database within a project."; + static description = "Create a new PostgreSQL database within a project."; static examples = ["$ <%= config.bin %> postgres create"]; @@ -24,94 +24,80 @@ export default class DatabasePostgresCreate extends Command { const { flags } = await this.parse(DatabasePostgresCreate); - const { projectId } = flags; + let { projectId } = flags; + if (!projectId) { console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - const { project } = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project, - })), - message: "Select a project to create the database in:", - name: "project", - type: "list", - }, - ]); - - const appDetails = await inquirer.prompt([ - { - message: "Enter the name:", - name: "name", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Database name:", - name: "databaseName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Enter the database description (optional):", - name: "description", - type: "input", - }, - { - message: "Database password (optional):", - name: "databasePassword", - type: "input", - }, - { - default: "postgres:15", - message: "Docker Image (default: postgres:15):", - name: "dockerImage", - type: "input", - }, - - { - default: "postgres", - message: "Database User: (default: postgres):", - name: "databaseUser", - type: "input", - }, - ]); - - const appName = await inquirer.prompt([ - { - default: `${slugify(project.name)}-${appDetails.name}`, - message: "Enter the App name: (optional):", - name: "appName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - ]); + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the PostgreSQL database in:", + name: "project", + type: "list", + }, + ]); + + projectId = project.projectId; + + const dbDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Database name:", + name: "databaseName", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "password", + }, + { + default: "postgres:15", + message: "Docker Image (default: postgres:15):", + name: "dockerImage", + type: "input", + }, + { + default: "postgres", + message: "Database User: (default: postgres):", + name: "databaseUser", + type: "input", + }, + ]); + + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${dbDetails.name}`, + message: "Enter the App name:", + name: "appName", + type: "input", + validate: (input) => (input ? true : "App name is required"), + }, + ]); - const responseDatabase = await axios.post( + try { + const response = await axios.post( `${auth.url}/api/trpc/postgres.create`, { json: { - ...appDetails, + ...dbDetails, appName: appName.appName, projectId: project.projectId, }, @@ -124,16 +110,20 @@ export default class DatabasePostgresCreate extends Command { }, ); - if (!responseDatabase.data.result.data.json) { - this.error(chalk.red("Error creating database")); + if (!response.data.result.data.json) { + this.error(chalk.red("Error creating PostgreSQL database")); } this.log( - chalk.green(`Database '${appDetails.name}' created successfully.`), + chalk.green( + `PostgreSQL database '${dbDetails.name}' created successfully.`, + ), ); } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + this.error( + // @ts-ignore + chalk.red(`Failed to create PostgreSQL database: ${error.message}`), + ); } } } diff --git a/src/commands/database/postgres/delete.ts b/src/commands/database/postgres/delete.ts index 47aaf0e..e91298a 100644 --- a/src/commands/database/postgres/delete.ts +++ b/src/commands/database/postgres/delete.ts @@ -4,9 +4,10 @@ import chalk from "chalk"; import inquirer from "inquirer"; import { readAuthConfig } from "../../../utils/utils.js"; +import { getProject, getProjects } from "../../../utils/shared.js"; export default class DatabasePostgresDelete extends Command { - static description = "Delete an application from a project."; + static description = "Delete a PostgreSQL database from a project."; static examples = [ "$ <%= config.bin %> postgres delete", @@ -28,80 +29,47 @@ export default class DatabasePostgresDelete extends Command { let { projectId } = flags; if (!projectId) { - // Obtener la lista de proyectos y permitir la selección console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - // Permitir al usuario seleccionar un proyecto - const answers = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project.projectId, - })), - message: "Select a project to delete the postgres database from:", - name: "selectedProject", - type: "list", - }, - ]); + const projects = await getProjects(auth, this); - projectId = answers.selectedProject; - } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; } - } - try { - const response = await axios.get(`${auth.url}/api/trpc/project.one`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - params: { - input: JSON.stringify({ - json: { projectId }, - }), + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the PostgreSQL database from:", + name: "selectedProject", + type: "list", }, - }); + ]); - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching applications")); - } + projectId = answers.selectedProject; + } - const apps = response.data.result.data.json; + try { + const project = await getProject(projectId, auth, this); - if (apps.postgres.length === 0) { - this.log(chalk.yellow("No applications found in this project.")); + if (!project.postgres || project.postgres.length === 0) { + this.log( + chalk.yellow("No PostgreSQL databases found in this project."), + ); return; } - // Permitir al usuario seleccionar una aplicación const appAnswers = await inquirer.prompt([ { - choices: apps.postgres.map((app: any) => ({ - name: app.name, - value: app.postgresId, + choices: project.postgres.map((db: any) => ({ + name: db.name, + value: db.postgresId, })), - message: "Select the postgres database to delete:", + message: "Select the PostgreSQL database to delete:", name: "selectedApp", type: "list", }, @@ -109,7 +77,6 @@ export default class DatabasePostgresDelete extends Command { const postgresId = appAnswers.selectedApp; - // Confirmar eliminación const confirmAnswers = await inquirer.prompt([ { default: false, @@ -120,11 +87,10 @@ export default class DatabasePostgresDelete extends Command { ]); if (!confirmAnswers.confirmDelete) { - this.log(chalk.yellow("Application deletion cancelled.")); + this.log(chalk.yellow("Database deletion cancelled.")); return; } - // Eliminar la aplicación seleccionada const deleteResponse = await axios.post( `${auth.url}/api/trpc/postgres.remove`, { @@ -141,10 +107,10 @@ export default class DatabasePostgresDelete extends Command { ); if (!deleteResponse.data.result.data.json) { - this.error(chalk.red("Error deleting application")); + this.error(chalk.red("Error deleting PostgreSQL database")); } - this.log(chalk.green("Application deleted successfully.")); + this.log(chalk.green("PostgreSQL database deleted successfully.")); } catch (error) { // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. this.error(chalk.red(`Failed to delete application: ${error.message}`)); diff --git a/src/commands/database/redis/create.ts b/src/commands/database/redis/create.ts index a68f00a..6a419e5 100644 --- a/src/commands/database/redis/create.ts +++ b/src/commands/database/redis/create.ts @@ -2,12 +2,13 @@ import { Command, Flags } from "@oclif/core"; import axios from "axios"; import chalk from "chalk"; import inquirer from "inquirer"; - import { slugify } from "../../../utils/slug.js"; import { readAuthConfig } from "../../../utils/utils.js"; +import { getProjects } from "../../../utils/shared.js"; +import type { Answers } from "../../app/create.js"; export default class DatabaseRedisCreate extends Command { - static description = "Create a new database within a project."; + static description = "Create a new Redis database within a project."; static examples = ["$ <%= config.bin %> redis create"]; @@ -21,82 +22,71 @@ export default class DatabaseRedisCreate extends Command { public async run(): Promise { const auth = await readAuthConfig(this); + const { flags } = await this.parse(DatabaseRedisCreate); - const { projectId } = flags; - if (!projectId) { - console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); + let { projectId } = flags; - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } + if (!projectId) { + console.log(chalk.blue.bold("\n Listing all Projects \n")); - const projects = response.data.result.data.json; + const projects = await getProjects(auth, this); - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to create the Redis database in:", + name: "project", + type: "list", + }, + ]); - const { project } = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project, - })), - message: "Select a project to create the database in:", - name: "project", - type: "list", - }, - ]); + projectId = project.projectId; - const appDetails = await inquirer.prompt([ - { - message: "Enter the name:", - name: "name", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - { - message: "Enter the database description (optional):", - name: "description", - type: "input", - }, - { - message: "Database password (optional):", - name: "databasePassword", - type: "input", - }, - { - default: "redis:7", - message: "Docker Image (default: redis:7):", - name: "dockerImage", - type: "input", - }, - ]); + const dbDetails = await inquirer.prompt([ + { + message: "Enter the name:", + name: "name", + type: "input", + validate: (input) => (input ? true : "Database name is required"), + }, + { + message: "Enter the database description (optional):", + name: "description", + type: "input", + }, + { + message: "Database password (optional):", + name: "databasePassword", + type: "password", + }, + { + default: "redis:7", + message: "Docker Image (default: redis:7):", + name: "dockerImage", + type: "input", + }, + ]); - const appName = await inquirer.prompt([ - { - default: `${slugify(project.name)}-${appDetails.name}`, - message: "Enter the App name: (optional):", - name: "appName", - type: "input", - validate: (input) => (input ? true : "Database name is required"), - }, - ]); + const appName = await inquirer.prompt([ + { + default: `${slugify(project.name)}-${dbDetails.name}`, + message: "Enter the App name:", + name: "appName", + type: "input", + validate: (input) => (input ? true : "App name is required"), + }, + ]); - const responseDatabase = await axios.post( + try { + const response = await axios.post( `${auth.url}/api/trpc/redis.create`, { json: { - ...appDetails, + ...dbDetails, appName: appName.appName, projectId: project.projectId, }, @@ -109,16 +99,20 @@ export default class DatabaseRedisCreate extends Command { }, ); - if (!responseDatabase.data.result.data.json) { - this.error(chalk.red("Error creating database")); + if (!response.data.result.data.json) { + this.error(chalk.red("Error creating Redis database")); } this.log( - chalk.green(`Database '${appDetails.name}' created successfully.`), + chalk.green( + `Redis database '${dbDetails.name}' created successfully.`, + ), ); } catch (error) { - // @ts-expect-error TODO: Fix this - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + this.error( + // @ts-ignore + chalk.red(`Failed to create Redis database: ${error.message}`), + ); } } } diff --git a/src/commands/database/redis/delete.ts b/src/commands/database/redis/delete.ts index 258927f..e47c5f1 100644 --- a/src/commands/database/redis/delete.ts +++ b/src/commands/database/redis/delete.ts @@ -4,9 +4,10 @@ import chalk from "chalk"; import inquirer from "inquirer"; import { readAuthConfig } from "../../../utils/utils.js"; +import { getProject, getProjects } from "../../../utils/shared.js"; export default class DatabaseRedisDelete extends Command { - static description = "Delete an application from a project."; + static description = "Delete an redis database from a project."; static examples = [ "$ <%= config.bin %> redis delete", @@ -28,78 +29,44 @@ export default class DatabaseRedisDelete extends Command { let { projectId } = flags; if (!projectId) { - // Obtener la lista de proyectos y permitir la selección console.log(chalk.blue.bold("\n Listing all Projects \n")); - try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; - - if (projects.length === 0) { - this.log(chalk.yellow("No projects found.")); - return; - } - - // Permitir al usuario seleccionar un proyecto - const answers = await inquirer.prompt([ - { - choices: projects.map((project: any) => ({ - name: project.name, - value: project.projectId, - })), - message: "Select a project to delete the mariadb database from:", - name: "selectedProject", - type: "list", - }, - ]); + const projects = await getProjects(auth, this); - projectId = answers.selectedProject; - } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + if (projects.length === 0) { + this.log(chalk.yellow("No projects found.")); + return; } - } - try { - const response = await axios.get(`${auth.url}/api/trpc/project.one`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - params: { - input: JSON.stringify({ - json: { projectId }, - }), + const answers = await inquirer.prompt([ + { + choices: projects.map((project: any) => ({ + name: project.name, + value: project.projectId, + })), + message: "Select a project to delete the redis database from:", + name: "selectedProject", + type: "list", }, - }); + ]); - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching applications")); - } + projectId = answers.selectedProject; + } - const apps = response.data.result.data.json; + try { + const project = await getProject(projectId, auth, this); - if (apps.redis.length === 0) { - this.log(chalk.yellow("No applications found in this project.")); + if (!project.redis || project.redis.length === 0) { + this.log(chalk.yellow("No redis databases found in this project.")); return; } // Permitir al usuario seleccionar una aplicación const appAnswers = await inquirer.prompt([ { - choices: apps.redis.map((app: any) => ({ - name: app.name, - value: app.redisId, + choices: project.redis.map((db: any) => ({ + name: db.name, + value: db.redisId, })), message: "Select the redis database to delete:", name: "selectedApp", @@ -109,22 +76,20 @@ export default class DatabaseRedisDelete extends Command { const redisId = appAnswers.selectedApp; - // Confirmar eliminación const confirmAnswers = await inquirer.prompt([ { default: false, - message: "Are you sure you want to delete this mysql database?", + message: "Are you sure you want to delete this redis database?", name: "confirmDelete", type: "confirm", }, ]); if (!confirmAnswers.confirmDelete) { - this.log(chalk.yellow("Application deletion cancelled.")); + this.log(chalk.yellow("Database deletion cancelled.")); return; } - // Eliminar la aplicación seleccionada const deleteResponse = await axios.post( `${auth.url}/api/trpc/redis.remove`, { @@ -141,13 +106,15 @@ export default class DatabaseRedisDelete extends Command { ); if (!deleteResponse.data.result.data.json) { - this.error(chalk.red("Error deleting application")); + this.error(chalk.red("Error deleting redis database")); } - this.log(chalk.green("Application deleted successfully.")); + this.log(chalk.green("Redis database deleted successfully.")); } catch (error) { - // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. - this.error(chalk.red(`Failed to delete application: ${error.message}`)); + this.error( + // @ts-expect-error - TS2339: Property 'data' does not exist on type 'AxiosError'. + chalk.red(`Failed to delete redis database: ${error.message}`), + ); } } } diff --git a/src/commands/project/info.ts b/src/commands/project/info.ts index 44e3bdb..ded21ce 100644 --- a/src/commands/project/info.ts +++ b/src/commands/project/info.ts @@ -1,9 +1,9 @@ import { Command, Flags } from "@oclif/core"; -import axios from "axios"; import chalk from "chalk"; import inquirer from "inquirer"; import { readAuthConfig } from "../../utils/utils.js"; +import { getProject, getProjects } from "../../utils/shared.js"; export default class ProjectInfo extends Command { static description = @@ -28,35 +28,21 @@ export default class ProjectInfo extends Command { const { flags } = await this.parse(ProjectInfo); if (flags.projectId) { - // Si se proporciona un projectId, mostrar directamente la información del proyecto await this.showProjectInfo(auth, flags.projectId); } else { - // Obtener la lista de proyectos y permitir la selección console.log(chalk.blue.bold("\n Listing all Projects \n")); try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; + const projects = await getProjects(auth, this); if (projects.length === 0) { this.log(chalk.yellow("No projects found.")); return; } - // Permitir al usuario seleccionar un proyecto const answers = await inquirer.prompt([ { - choices: projects.map((project: any) => ({ + choices: projects.map((project) => ({ name: project.name, value: project.projectId, })), @@ -68,7 +54,7 @@ export default class ProjectInfo extends Command { const selectedProjectId = answers.selectedProject; - await this.showProjectInfo(auth, selectedProjectId); + await await this.showProjectInfo(auth, selectedProjectId); } catch (error) { // @ts-expect-error hola this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); @@ -85,23 +71,7 @@ export default class ProjectInfo extends Command { ); try { - const response = await axios.get(`${auth.url}/api/trpc/project.one`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - params: { - input: JSON.stringify({ - json: { projectId }, - }), - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching project information")); - } - - const projectInfo = response.data.result.data.json; + const projectInfo = await getProject(projectId, auth, this); this.log(chalk.green(`Project Name: ${projectInfo.name}`)); this.log( @@ -141,55 +111,55 @@ export default class ProjectInfo extends Command { if (projectInfo.applications.length > 0) { this.log(chalk.blue("\nApplications:")); - projectInfo.applications.forEach((app: any, index: number) => { + projectInfo.applications.forEach((app, index: number) => { this.log(` ${index + 1}. ${app.name}`); }); } if (projectInfo.compose.length > 0) { this.log(chalk.blue("\nCompose Services:")); - projectInfo.compose.forEach((service: any, index: number) => { + projectInfo.compose.forEach((service, index: number) => { this.log(` ${index + 1}. ${service.name}`); }); } if (projectInfo.mariadb.length > 0) { this.log(chalk.blue("\nMariaDB Databases:")); - projectInfo.mariadb.forEach((db: any, index: number) => { + projectInfo.mariadb.forEach((db, index: number) => { this.log(` ${index + 1}. ${db.name}`); }); } if (projectInfo.mongo.length > 0) { this.log(chalk.blue("\nMongoDB Databases:")); - projectInfo.mongo.forEach((db: any, index: number) => { + projectInfo.mongo.forEach((db, index: number) => { this.log(` ${index + 1}. ${db.name}`); }); } if (projectInfo.mysql.length > 0) { this.log(chalk.blue("\nMySQL Databases:")); - projectInfo.mysql.forEach((db: any, index: number) => { + projectInfo.mysql.forEach((db, index: number) => { this.log(` ${index + 1}. ${db.name}`); }); } if (projectInfo.postgres.length > 0) { this.log(chalk.blue("\nPostgreSQL Databases:")); - projectInfo.postgres.forEach((db: any, index: number) => { + projectInfo.postgres.forEach((db, index: number) => { this.log(` ${index + 1}. ${db.name}`); }); } if (projectInfo.redis.length > 0) { this.log(chalk.blue("\nRedis Databases:")); - projectInfo.redis.forEach((db: any, index: number) => { + projectInfo.redis.forEach((db, index: number) => { this.log(` ${index + 1}. ${db.name}`); }); } } catch (error) { this.error( - // @ts-expect-error hola + // @ts-expect-error chalk.red(`Failed to fetch project information: ${error.message}`), ); } diff --git a/src/commands/project/list.ts b/src/commands/project/list.ts index fcf355e..91242b6 100644 --- a/src/commands/project/list.ts +++ b/src/commands/project/list.ts @@ -1,9 +1,9 @@ import { Command } from "@oclif/core"; -import axios from "axios"; import chalk from "chalk"; import Table from "cli-table3"; import { readAuthConfig } from "../../utils/utils.js"; +import { getProjects } from "../../utils/shared.js"; export default class ProjectList extends Command { static description = "List all projects."; @@ -16,18 +16,7 @@ export default class ProjectList extends Command { console.log(chalk.blue.bold("\n Listing all Projects \n")); try { - const response = await axios.get(`${auth.url}/api/trpc/project.all`, { - headers: { - Authorization: `Bearer ${auth.token}`, - "Content-Type": "application/json", - }, - }); - - if (!response.data.result.data.json) { - this.error(chalk.red("Error fetching projects")); - } - - const projects = response.data.result.data.json; + const projects = await getProjects(auth, this); if (projects.length === 0) { this.log(chalk.yellow("No projects found.")); @@ -52,9 +41,9 @@ export default class ProjectList extends Command { this.log(table.toString()); } - } catch { + } catch (error) { // @ts-expect-error error is not defined - this.error(chalk.red(`Failed to list projects: ${error.message}`)); + this.error(chalk.red(`Failed to list projects: ${error?.message}`)); } } } diff --git a/src/utils/shared.ts b/src/utils/shared.ts new file mode 100644 index 0000000..c1ff412 --- /dev/null +++ b/src/utils/shared.ts @@ -0,0 +1,81 @@ +import type { Command } from "@oclif/core"; + +import axios from "axios"; +import chalk from "chalk"; + +import type { AuthConfig } from "./utils.js"; + +export type Project = { + adminId: string; + name: string; + projectId?: string | undefined; + description?: string | undefined; +}; + +export const getProjects = async ( + auth: AuthConfig, + command: Command, +): Promise => { + try { + const response = await axios.get(`${auth.url}/api/trpc/project.all`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }); + + if (!response.data.result.data.json) { + command.error(chalk.red("Error fetching projects")); + } + + const projects = response.data.result.data.json; + + if (projects.length === 0) { + command.log(chalk.yellow("No projects found.")); + return []; + } + + return projects; + } catch { + // @ts-expect-error TODO: Fix this + command.error(chalk.red(`Failed to fetch project list: ${error.message}`)); + } +}; + +export const getProject = async ( + projectId: string | undefined, + auth: AuthConfig, + command: Command, +) => { + try { + if (!projectId) { + command.error(chalk.red("Project ID is required")); + } + const response = await axios.get(`${auth.url}/api/trpc/project.one`, { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + params: { + input: JSON.stringify({ + json: { projectId }, + }), + }, + }); + + if (!response.data.result.data.json) { + command.error(chalk.red("Error fetching project")); + } + + const project = response.data.result.data.json; + + if (!project) { + command.error(chalk.red("Error fetching project")); + } + + return project; + } catch { + // @ts-expect-error TODO: Fix this + command.error(chalk.red(`Failed to fetch project: ${error.message}`)); + } +}; diff --git a/src/utils/utils.ts b/src/utils/utils.ts index 419678f..c5db541 100644 --- a/src/utils/utils.ts +++ b/src/utils/utils.ts @@ -9,9 +9,12 @@ const __filename = fileURLToPath(import.meta.url); const __dirname = path.dirname(__filename); const configPath = path.join(__dirname, "..", "..", "config.json"); -export const readAuthConfig = async ( - command: Command, -): Promise<{ token: string; url: string }> => { +export type AuthConfig = { + token: string; + url: string; +}; + +export const readAuthConfig = async (command: Command): Promise => { if (!fs.existsSync(configPath)) { command.error( chalk.red( From 209eadce993ed98fd2fd71525e8db04ca9cd1659 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Sat, 22 Jun 2024 15:53:49 -0600 Subject: [PATCH 10/13] clean files --- src/commands/hello/greet.ts | 29 ----------------------------- src/commands/hello/index.ts | 34 ---------------------------------- src/commands/hello/world.ts | 19 ------------------- src/commands/project/info.ts | 2 +- 4 files changed, 1 insertion(+), 83 deletions(-) delete mode 100644 src/commands/hello/greet.ts delete mode 100644 src/commands/hello/index.ts delete mode 100644 src/commands/hello/world.ts diff --git a/src/commands/hello/greet.ts b/src/commands/hello/greet.ts deleted file mode 100644 index 3b77354..0000000 --- a/src/commands/hello/greet.ts +++ /dev/null @@ -1,29 +0,0 @@ -import { Command, Flags } from "@oclif/core"; - -export default class Greet extends Command { - static args = { - // Puedes agregar argumentos adicionales si lo necesitas - }; - - static description = "Greet the user with a message"; - - static examples = [ - `<%= config.bin %> <%= command.id %> --name=Mau -hello Mau! -`, - ]; - - static flags = { - name: Flags.string({ - char: "n", - description: "name to greet", - required: false, - }), - }; - - async run(): Promise { - const { flags } = await this.parse(Greet); - const name = flags.name ?? "user"; - this.log(`hello ${name}!`); - } -} diff --git a/src/commands/hello/index.ts b/src/commands/hello/index.ts deleted file mode 100644 index f7b639c..0000000 --- a/src/commands/hello/index.ts +++ /dev/null @@ -1,34 +0,0 @@ -import { Args, Command, Flags } from "@oclif/core"; - -export default class Hello extends Command { - static args = { - person: Args.string({ - description: "Person to say hello to", - required: true, - }), - }; - - static description = "Say hello"; - - static examples = [ - `<%= config.bin %> <%= command.id %> friend --from oclif -hello friend from oclif! (./src/commands/hello/index.ts) -`, - ]; - - static flags = { - from: Flags.string({ - char: "f", - description: "Who is saying hello", - required: true, - }), - }; - - async run(): Promise { - const { args, flags } = await this.parse(Hello); - - this.log( - `hello ${args.person} from ${flags.from}! (./src/commands/hello/index.ts)`, - ); - } -} diff --git a/src/commands/hello/world.ts b/src/commands/hello/world.ts deleted file mode 100644 index 8111043..0000000 --- a/src/commands/hello/world.ts +++ /dev/null @@ -1,19 +0,0 @@ -import {Command} from '@oclif/core' - -export default class World extends Command { - static args = {} - - static description = 'Say hello world' - - static examples = [ - `<%= config.bin %> <%= command.id %> -hello world! (./src/commands/hello/world.ts) -`, - ] - - static flags = {} - - async run(): Promise { - this.log('hello world! (./src/commands/hello/world.ts)') - } -} diff --git a/src/commands/project/info.ts b/src/commands/project/info.ts index ded21ce..a64d4b1 100644 --- a/src/commands/project/info.ts +++ b/src/commands/project/info.ts @@ -54,7 +54,7 @@ export default class ProjectInfo extends Command { const selectedProjectId = answers.selectedProject; - await await this.showProjectInfo(auth, selectedProjectId); + await this.showProjectInfo(auth, selectedProjectId); } catch (error) { // @ts-expect-error hola this.error(chalk.red(`Failed to fetch project list: ${error.message}`)); From e48c58c8834d11fbefe5f190e32b3588e830eaf0 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Sat, 22 Jun 2024 16:22:30 -0600 Subject: [PATCH 11/13] feat: add deploy command --- src/commands/app/deploy.ts | 89 ++++++++++++++++++++++++ src/commands/app/stop.ts | 89 ++++++++++++++++++++++++ src/commands/database/mariadb/create.ts | 4 +- src/commands/database/mariadb/deploy.ts | 89 ++++++++++++++++++++++++ src/commands/database/mongo/deploy.ts | 89 ++++++++++++++++++++++++ src/commands/database/mysql/deploy.ts | 89 ++++++++++++++++++++++++ src/commands/database/postgres/deploy.ts | 89 ++++++++++++++++++++++++ src/commands/database/redis/deploy.ts | 89 ++++++++++++++++++++++++ 8 files changed, 625 insertions(+), 2 deletions(-) create mode 100644 src/commands/app/deploy.ts create mode 100644 src/commands/app/stop.ts create mode 100644 src/commands/database/mariadb/deploy.ts create mode 100644 src/commands/database/mongo/deploy.ts create mode 100644 src/commands/database/mysql/deploy.ts create mode 100644 src/commands/database/postgres/deploy.ts create mode 100644 src/commands/database/redis/deploy.ts diff --git a/src/commands/app/deploy.ts b/src/commands/app/deploy.ts new file mode 100644 index 0000000..4ecf97a --- /dev/null +++ b/src/commands/app/deploy.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import { readAuthConfig } from "../../utils/utils.js"; +import chalk from "chalk"; +import { getProject, getProjects } from "../../utils/shared.js"; +import inquirer from "inquirer"; +import type { Answers } from "./create.js"; +import axios from "axios"; + +export default class AppDeploy extends Command { + static description = "Deploy an application to a project."; + + static examples = ["$ <%= config.bin %> app deploy"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to deploy the application in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.applications.length === 0) { + this.error(chalk.yellow("No applications found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.applications.map((app) => ({ + name: app.name, + value: app.applicationId, + })), + message: "Select the application to deploy:", + name: "selectedApp", + type: "list", + }, + ]); + + const applicationId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to deploy this application?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("Application deployment cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/application.deploy`, + { + json: { + applicationId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error deploying application")); + } + this.log(chalk.green("Application deploy successful.")); + } +} diff --git a/src/commands/app/stop.ts b/src/commands/app/stop.ts new file mode 100644 index 0000000..b04f690 --- /dev/null +++ b/src/commands/app/stop.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import { readAuthConfig } from "../../utils/utils.js"; +import chalk from "chalk"; +import inquirer from "inquirer"; +import { getProject, getProjects } from "../../utils/shared.js"; +import type { Answers } from "./create.js"; +import axios from "axios"; + +export default class AppStop extends Command { + static description = "Stop an application from a project."; + + static examples = ["$ <%= config.bin %> app stop"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to stop the application in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.applications.length === 0) { + this.error(chalk.yellow("No applications found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.applications.map((app) => ({ + name: app.name, + value: app.applicationId, + })), + message: "Select the application to stop:", + name: "selectedApp", + type: "list", + }, + ]); + + const applicationId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to stop this application?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("Application stop cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/application.stop`, + { + json: { + applicationId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error stopping application")); + } + this.log(chalk.green("Application stop successful.")); + } +} diff --git a/src/commands/database/mariadb/create.ts b/src/commands/database/mariadb/create.ts index 676405c..71a32df 100644 --- a/src/commands/database/mariadb/create.ts +++ b/src/commands/database/mariadb/create.ts @@ -75,8 +75,8 @@ export default class DatabaseMariadbCreate extends Command { type: "password", }, { - default: "mariadb:4", - message: "Docker Image (default: mariadb:4):", + default: "mariadb:11", + message: "Docker Image (default: mariadb:11):", name: "dockerImage", type: "input", }, diff --git a/src/commands/database/mariadb/deploy.ts b/src/commands/database/mariadb/deploy.ts new file mode 100644 index 0000000..0874a60 --- /dev/null +++ b/src/commands/database/mariadb/deploy.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import { readAuthConfig } from "../../../utils/utils.js"; +import chalk from "chalk"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import inquirer from "inquirer"; +import type { Answers } from "../../app/create.js"; +import axios from "axios"; + +export default class DatabaseMariadbDeploy extends Command { + static description = "Deploy an mariadb to a project."; + + static examples = ["$ <%= config.bin %> app deploy"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to deploy the mariadb in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.mariadb.length === 0) { + this.error(chalk.yellow("No mariadb found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.mariadb.map((app) => ({ + name: app.name, + value: app.mariadbId, + })), + message: "Select the mariadb to deploy:", + name: "selectedApp", + type: "list", + }, + ]); + + const mariadbId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to deploy this mariadb?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("mariadb deployment cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/mariadb.deploy`, + { + json: { + mariadbId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error deploying mariadb")); + } + this.log(chalk.green("Mariadb deploy successful.")); + } +} diff --git a/src/commands/database/mongo/deploy.ts b/src/commands/database/mongo/deploy.ts new file mode 100644 index 0000000..cbee40e --- /dev/null +++ b/src/commands/database/mongo/deploy.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import { readAuthConfig } from "../../../utils/utils.js"; +import chalk from "chalk"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import inquirer from "inquirer"; +import type { Answers } from "../../app/create.js"; +import axios from "axios"; + +export default class DatabaseMongoDeploy extends Command { + static description = "Deploy an mongo to a project."; + + static examples = ["$ <%= config.bin %> app deploy"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to deploy the mongo in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.mongo.length === 0) { + this.error(chalk.yellow("No mongo found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.mongo.map((app) => ({ + name: app.name, + value: app.mongoId, + })), + message: "Select the mongo to deploy:", + name: "selectedApp", + type: "list", + }, + ]); + + const mongoId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to deploy this mongo?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("mongo deployment cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/mongo.deploy`, + { + json: { + mongoId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error deploying mongo")); + } + this.log(chalk.green("Mongo deploy successful.")); + } +} diff --git a/src/commands/database/mysql/deploy.ts b/src/commands/database/mysql/deploy.ts new file mode 100644 index 0000000..b1cb76b --- /dev/null +++ b/src/commands/database/mysql/deploy.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import { readAuthConfig } from "../../../utils/utils.js"; +import chalk from "chalk"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import inquirer from "inquirer"; +import type { Answers } from "../../app/create.js"; +import axios from "axios"; + +export default class DatabaseMysqlDeploy extends Command { + static description = "Deploy an mysql to a project."; + + static examples = ["$ <%= config.bin %> app deploy"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to deploy the mysql in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.mysql.length === 0) { + this.error(chalk.yellow("No mysql found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.mysql.map((app) => ({ + name: app.name, + value: app.mysqlId, + })), + message: "Select the mysql to deploy:", + name: "selectedApp", + type: "list", + }, + ]); + + const mysqlId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to deploy this mysql?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("mysql deployment cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/mysql.deploy`, + { + json: { + mysqlId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error deploying mysql")); + } + this.log(chalk.green("Mysql deployed successful.")); + } +} diff --git a/src/commands/database/postgres/deploy.ts b/src/commands/database/postgres/deploy.ts new file mode 100644 index 0000000..36675c1 --- /dev/null +++ b/src/commands/database/postgres/deploy.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import { readAuthConfig } from "../../../utils/utils.js"; +import chalk from "chalk"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import inquirer from "inquirer"; +import type { Answers } from "../../app/create.js"; +import axios from "axios"; + +export default class DatabasePostgresDeploy extends Command { + static description = "Deploy an postgres to a project."; + + static examples = ["$ <%= config.bin %> app deploy"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to deploy the postgres in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.postgres.length === 0) { + this.error(chalk.yellow("No postgres found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.postgres.map((app) => ({ + name: app.name, + value: app.postgresId, + })), + message: "Select the postgres to deploy:", + name: "selectedApp", + type: "list", + }, + ]); + + const postgresId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to deploy this postgres?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("postgres deployment cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/postgres.deploy`, + { + json: { + postgresId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error deploying postgres")); + } + this.log(chalk.green("Postgres deployed successful.")); + } +} diff --git a/src/commands/database/redis/deploy.ts b/src/commands/database/redis/deploy.ts new file mode 100644 index 0000000..fdda563 --- /dev/null +++ b/src/commands/database/redis/deploy.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import { readAuthConfig } from "../../../utils/utils.js"; +import chalk from "chalk"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import inquirer from "inquirer"; +import type { Answers } from "../../app/create.js"; +import axios from "axios"; + +export default class DatabaseRedisDeploy extends Command { + static description = "Deploy an redis to a project."; + + static examples = ["$ <%= config.bin %> app deploy"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to deploy the redis in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.redis.length === 0) { + this.error(chalk.yellow("No redis found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.redis.map((app) => ({ + name: app.name, + value: app.redisId, + })), + message: "Select the redis to deploy:", + name: "selectedApp", + type: "list", + }, + ]); + + const redisId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to deploy this redis?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("redis deployment cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/redis.deploy`, + { + json: { + redisId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error deploying redis")); + } + this.log(chalk.green("Redis deployed successful.")); + } +} From 3b12f55ae40ea2e13226b3af20b5876ac59b55a9 Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Sat, 22 Jun 2024 16:46:05 -0600 Subject: [PATCH 12/13] feat: add stop command --- src/commands/database/mariadb/stop.ts | 89 ++++++++++++++++++++++++++ src/commands/database/mongo/stop.ts | 89 ++++++++++++++++++++++++++ src/commands/database/mysql/stop.ts | 89 ++++++++++++++++++++++++++ src/commands/database/postgres/stop.ts | 89 ++++++++++++++++++++++++++ src/commands/database/redis/stop.ts | 89 ++++++++++++++++++++++++++ 5 files changed, 445 insertions(+) create mode 100644 src/commands/database/mariadb/stop.ts create mode 100644 src/commands/database/mongo/stop.ts create mode 100644 src/commands/database/mysql/stop.ts create mode 100644 src/commands/database/postgres/stop.ts create mode 100644 src/commands/database/redis/stop.ts diff --git a/src/commands/database/mariadb/stop.ts b/src/commands/database/mariadb/stop.ts new file mode 100644 index 0000000..a39bba6 --- /dev/null +++ b/src/commands/database/mariadb/stop.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import chalk from "chalk"; +import inquirer from "inquirer"; +import axios from "axios"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import { readAuthConfig } from "../../../utils/utils.js"; +import type { Answers } from "../../app/create.js"; + +export default class DatabaseMariadbStop extends Command { + static description = "Stop an mariadb from a project."; + + static examples = ["$ <%= config.bin %> mariadb stop"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to stop the mariadb in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.mariadb.length === 0) { + this.error(chalk.yellow("No mariadb found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.mariadb.map((app) => ({ + name: app.name, + value: app.mariadbId, + })), + message: "Select the mariadb to stop:", + name: "selectedApp", + type: "list", + }, + ]); + + const mariadbId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to stop this mariadb?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("Mariadb stop cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/mariadb.stop`, + { + json: { + mariadbId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error stopping mariadb")); + } + this.log(chalk.green("Mariadb stop successful.")); + } +} diff --git a/src/commands/database/mongo/stop.ts b/src/commands/database/mongo/stop.ts new file mode 100644 index 0000000..96b3975 --- /dev/null +++ b/src/commands/database/mongo/stop.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import chalk from "chalk"; +import inquirer from "inquirer"; +import axios from "axios"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import { readAuthConfig } from "../../../utils/utils.js"; +import type { Answers } from "../../app/create.js"; + +export default class DatabaseMongoStop extends Command { + static description = "Stop an mongo from a project."; + + static examples = ["$ <%= config.bin %> mongo stop"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to stop the mongo in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.mongo.length === 0) { + this.error(chalk.yellow("No mongo found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.mongo.map((app) => ({ + name: app.name, + value: app.mongoId, + })), + message: "Select the mongo to stop:", + name: "selectedApp", + type: "list", + }, + ]); + + const mongoId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to stop this mongo?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("mongo stop cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/mongo.stop`, + { + json: { + mongoId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error stopping mongo")); + } + this.log(chalk.green("Mongo stop successful.")); + } +} diff --git a/src/commands/database/mysql/stop.ts b/src/commands/database/mysql/stop.ts new file mode 100644 index 0000000..e13c51a --- /dev/null +++ b/src/commands/database/mysql/stop.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import chalk from "chalk"; +import inquirer from "inquirer"; +import axios from "axios"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import { readAuthConfig } from "../../../utils/utils.js"; +import type { Answers } from "../../app/create.js"; + +export default class DatabaseMysqlStop extends Command { + static description = "Stop an mysql from a project."; + + static examples = ["$ <%= config.bin %> mysql stop"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to stop the mysql in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.mysql.length === 0) { + this.error(chalk.yellow("No mysql found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.mysql.map((app) => ({ + name: app.name, + value: app.mysqlId, + })), + message: "Select the mysql to stop:", + name: "selectedApp", + type: "list", + }, + ]); + + const mysqlId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to stop this mysql?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("mysql stop cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/mysql.stop`, + { + json: { + mysqlId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error stopping mysql")); + } + this.log(chalk.green("Mysql stop successful.")); + } +} diff --git a/src/commands/database/postgres/stop.ts b/src/commands/database/postgres/stop.ts new file mode 100644 index 0000000..1fcfc29 --- /dev/null +++ b/src/commands/database/postgres/stop.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import chalk from "chalk"; +import inquirer from "inquirer"; +import axios from "axios"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import { readAuthConfig } from "../../../utils/utils.js"; +import type { Answers } from "../../app/create.js"; + +export default class DatabasePostgresStop extends Command { + static description = "Stop an postgres from a project."; + + static examples = ["$ <%= config.bin %> postgres stop"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to stop the postgres in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.postgres.length === 0) { + this.error(chalk.yellow("No postgres found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.postgres.map((app) => ({ + name: app.name, + value: app.postgresId, + })), + message: "Select the postgres to stop:", + name: "selectedApp", + type: "list", + }, + ]); + + const postgresId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to stop this postgres?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("postgres stop cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/postgres.stop`, + { + json: { + postgresId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error stopping postgres")); + } + this.log(chalk.green("Postgres stop successful.")); + } +} diff --git a/src/commands/database/redis/stop.ts b/src/commands/database/redis/stop.ts new file mode 100644 index 0000000..4e0dfb3 --- /dev/null +++ b/src/commands/database/redis/stop.ts @@ -0,0 +1,89 @@ +import { Command } from "@oclif/core"; +import chalk from "chalk"; +import inquirer from "inquirer"; +import axios from "axios"; +import { getProject, getProjects } from "../../../utils/shared.js"; +import { readAuthConfig } from "../../../utils/utils.js"; +import type { Answers } from "../../app/create.js"; + +export default class DatabaseRedisStop extends Command { + static description = "Stop an redis from a project."; + + static examples = ["$ <%= config.bin %> redis stop"]; + + public async run(): Promise { + const auth = await readAuthConfig(this); + + console.log(chalk.blue.bold("\n Listing all Projects \n")); + + const projects = await getProjects(auth, this); + + const { project } = await inquirer.prompt([ + { + choices: projects.map((project) => ({ + name: project.name, + value: project, + })), + message: "Select a project to stop the redis in:", + name: "project", + type: "list", + }, + ]); + + const projectId = project.projectId; + + const projectSelected = await getProject(projectId, auth, this); + + if (projectSelected.redis.length === 0) { + this.error(chalk.yellow("No redis found in this project.")); + } + + const appAnswers = await inquirer.prompt([ + { + // @ts-ignore + choices: projectSelected.redis.map((app) => ({ + name: app.name, + value: app.redisId, + })), + message: "Select the redis to stop:", + name: "selectedApp", + type: "list", + }, + ]); + + const redisId = appAnswers.selectedApp; + + const confirmAnswers = await inquirer.prompt([ + { + default: false, + message: "Are you sure you want to stop this redis?", + name: "confirmDelete", + type: "confirm", + }, + ]); + + if (!confirmAnswers.confirmDelete) { + this.error(chalk.yellow("redis stop cancelled.")); + } + + const response = await axios.post( + `${auth.url}/api/trpc/redis.stop`, + { + json: { + redisId, + }, + }, + { + headers: { + Authorization: `Bearer ${auth.token}`, + "Content-Type": "application/json", + }, + }, + ); + + if (response.status !== 200) { + this.error(chalk.red("Error stopping redis")); + } + this.log(chalk.green("Redis stop successful.")); + } +} From 791f8c96c3c3c9b7518f1a29e0284718bf6f9baf Mon Sep 17 00:00:00 2001 From: Mauricio Siu <47042324+Siumauricio@users.noreply.github.com> Date: Sat, 22 Jun 2024 17:01:47 -0600 Subject: [PATCH 13/13] remove --- src/commands/check-server.ts | 30 ------------------------------ 1 file changed, 30 deletions(-) delete mode 100644 src/commands/check-server.ts diff --git a/src/commands/check-server.ts b/src/commands/check-server.ts deleted file mode 100644 index 7cb55de..0000000 --- a/src/commands/check-server.ts +++ /dev/null @@ -1,30 +0,0 @@ -import {Args, Command, Flags} from '@oclif/core' - -export default class CheckServer extends Command { - static override args = { - file: Args.string({description: 'file to read'}), - } - - static override description = 'describe the command here' - - static override examples = [ - '<%= config.bin %> <%= command.id %>', - ] - - static override flags = { - // flag with no value (-f, --force) - force: Flags.boolean({char: 'f'}), - // flag with a value (-n, --name=VALUE) - name: Flags.string({char: 'n', description: 'name to print'}), - } - - public async run(): Promise { - const {args, flags} = await this.parse(CheckServer) - - const name = flags.name ?? 'world' - this.log(`hello ${name} from /Users/mauricio/Documents/Github/Personal/cli/src/commands/check-server.ts`) - if (args.file && flags.force) { - this.log(`you input --force and --file: ${args.file}`) - } - } -}