diff --git a/.eslintignore b/.eslintignore deleted file mode 100644 index 8894739..0000000 --- a/.eslintignore +++ /dev/null @@ -1,2 +0,0 @@ -./node_modules/**/*.js -./src/node_modules/**/*.js \ No newline at end of file diff --git a/.eslintrc.yml b/.eslintrc.yml deleted file mode 100644 index 61fcb09..0000000 --- a/.eslintrc.yml +++ /dev/null @@ -1,32 +0,0 @@ -env: - # es6: true - es2023: true - node: true - # commonjs: false -extends: - - airbnb-base - # - eslint:recommended - - prettier -parserOptions: - ecmaVersion: 2023 - sourceType: module - requireConfigFile: false - babelOptions: - plugins: - - '@babel/plugin-syntax-import-assertions' -parser: '@babel/eslint-parser' -rules: - prettier/prettier: error - import/no-unresolved: 2 - import/no-commonjs: 2 - import/extensions: [2, 'ignorePackages'] -plugins: - - prettier - - import -globals: - beforeAll: readonly - afterAll: readonly - describe: readonly - expect: readonly - test: readonly - it: readonly diff --git a/.gitattributes b/.gitattributes index 1e41f64..8e5a691 100644 --- a/.gitattributes +++ b/.gitattributes @@ -15,11 +15,11 @@ src/.prettierignore export-ignore src/.prettierrc.yaml export-ignore src/.snyk export-ignore src/Dockerfile export-ignore -src/jest.config.js export-ignore -release-config/* export-ignore -release-config export-ignore -src/scriptlog export-ignore -scriptlog export-ignore +src/jest.config.js export-ignore +release-config/* export-ignore +release-config export-ignore +src/scriptlog export-ignore +scriptlog export-ignore .codeclimate.yml export-ignore .jshintrc export-ignore diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index f05f278..35b2ebb 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -24,11 +24,11 @@ jobs: steps: - name: Show github.ref run: echo "$GITHUB_REF" - + - uses: googleapis/release-please-action@v4 id: release if: | - github.repository_owner == 'ptarmiganlabs' + github.repository_owner == 'ptarmiganlabs' with: token: ${{ secrets.RELEASE_PLEASE_PAT }} # optional. customize path to release-please-config.json @@ -68,7 +68,7 @@ jobs: - macos - sp53 # timeout-minutes: 15 - + if: needs.release-please.outputs.releases_created == 'true' env: DIST_FILE_NAME: ctrl-q @@ -79,32 +79,32 @@ jobs: MACOS_CI_KEYCHAIN_PWD: ${{ secrets.PROD_MACOS_CI_KEYCHAIN_PWD }} PROD_MACOS_NOTARIZATION_APPLE_ID: ${{ secrets.PROD_MACOS_NOTARIZATION_APPLE_ID }} PROD_MACOS_NOTARIZATION_TEAM_ID: ${{ secrets.PROD_MACOS_NOTARIZATION_TEAM_ID }} - PROD_MACOS_NOTARIZATION_PWD: ${{ secrets.PROD_MACOS_NOTARIZATION_PWD }} + PROD_MACOS_NOTARIZATION_PWD: ${{ secrets.PROD_MACOS_NOTARIZATION_PWD }} steps: - name: Release tag and upload url from previous job run: | echo "tag_name : ${{ needs.release-please.outputs.release_tag_name }}" echo "version : ${{ needs.release-please.outputs.release_version }}" echo "upload_url : ${{ needs.release-please.outputs.release_upload_url }}" - + - name: Checkout repository uses: actions/checkout@v4 - + - name: Setup Node.js uses: actions/setup-node@v4 with: node-version: lts/* - + - name: Install tool for creating stand-alone executables run: | npm install pkg --location=global npm install --save-exact esbuild - + - name: Install dependencies run: | - pwd + pwd npm ci --include=prod - + - name: Build binaries run: | pwd @@ -115,42 +115,42 @@ jobs: security delete-keychain build.keychain || true # Turn our base64-encoded certificate back to a regular .p12 file - + echo $MACOS_CERTIFICATE | base64 --decode > certificate.p12 - + # We need to create a new keychain, otherwise using the certificate will prompt # with a UI dialog asking for the certificate password, which we can't # use in a headless CI environment - + security create-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain security list-keychains -d user -s build.keychain security default-keychain -d user -s build.keychain security unlock-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain security import certificate.p12 -k build.keychain -P "$MACOS_CERTIFICATE_PWD" -T /usr/bin/codesign security set-key-partition-list -S apple-tool:,apple:,codesign: -s -k "$MACOS_CI_KEYCHAIN_PWD" build.keychain - + codesign --force -s "$MACOS_CERTIFICATE_NAME" -v "./${DIST_FILE_NAME}" --deep --strict --options=runtime --timestamp --entitlements ./release-config/${DIST_FILE_NAME}.entitlements - - + + # We can't notarize an app bundle directly, but we need to compress it as an archive. # Therefore, we create a zip file containing our app bundle, so that we can send it to the # notarization service - + # Notarize release binary echo "Creating temp notarization archive for release binary" # ditto -c -k --keepParent "./${DIST_FILE_NAME}" "./${DIST_FILE_NAME}.zip" ditto -c -k --keepParent "./${DIST_FILE_NAME}" "./${DIST_FILE_NAME}-${{ needs.release-please.outputs.release_version }}-macos.zip" - + # Here we send the notarization request to the Apple's Notarization service, waiting for the result. # This typically takes a few seconds inside a CI environment, but it might take more depending on the App # characteristics. Visit the Notarization docs for more information and strategies on how to optimize it if # you're curious echo "Notarize release app" xcrun notarytool submit "./${DIST_FILE_NAME}-${{ needs.release-please.outputs.release_version }}-macos.zip" --keychain-profile "notarytool-profile" --wait - + # Delete build keychain security delete-keychain build.keychain - + - name: Upload to existing release uses: ncipollo/release-action@v1 with: @@ -163,15 +163,15 @@ jobs: tag: ${{ needs.release-please.outputs.release_tag_name }} artifacts: ./ctrl-q-${{ needs.release-please.outputs.release_version }}-macos.zip token: ${{ github.token }} - + - name: Tidy up before existing run: | pwd - ls -la + ls -la rm build.cjs rm "./${DIST_FILE_NAME}" rm "./${DIST_FILE_NAME}-${{ needs.release-please.outputs.release_version }}-macos.zip" - + ##################### release-win64: needs: release-please @@ -196,22 +196,22 @@ jobs: Write-Output 'tag_name : ${{ needs.release-please.outputs.release_tag_name }}' Write-Output 'version : ${{ needs.release-please.outputs.release_version }}' Write-Output 'upload_url : ${{ needs.release-please.outputs.release_upload_url }}' - + - name: Checkout repository uses: actions/checkout@v4 - + - name: Setup Node.js uses: actions/setup-node@v4 with: node-version: lts/* - + - name: Install tool for creating stand-alone executables run: | npm install pkg --location=global - + - name: Install dependencies run: | - pwd + pwd npm ci --include=prod - name: Build binaries @@ -299,7 +299,7 @@ jobs: - name: Install dependencies run: | - pwd + pwd npm ci - name: Build binaries diff --git a/.github/workflows/insiders-build.yaml b/.github/workflows/insiders-build.yaml index baeb307..ac9a059 100644 --- a/.github/workflows/insiders-build.yaml +++ b/.github/workflows/insiders-build.yaml @@ -12,8 +12,21 @@ jobs: include: - os: win-code-sign build: | - ./node_modules/.bin/esbuild src/ctrl-q.js --bundle --external:axios --external:xdg-open --external:enigma.js --outfile=build.cjs --format=cjs --platform=node --target=node18 --minify --inject:./src/lib/util/import-meta-url.js --define:import.meta.url=import_meta_url - pkg --output "./${env:DIST_FILE_NAME}.exe" -t node18-win-x64 ./build.cjs --config package.json --compress GZip + ./node_modules/.bin/esbuild src/ctrl-q.js --bundle --outfile=build.cjs --format=cjs --platform=node --target=node23 --inject:./src/lib/util/import-meta-url.js --define:import.meta.url=import_meta_url + node --experimental-sea-config sea-config.json + node -e "require('fs').copyFileSync(process.execPath, 'ctrl-q.exe')" + + # Remove the signature from the executable + $processOptions1 = @{ + FilePath = "C:\Program Files (x86)/Windows Kits/10/bin/10.0.22621.0/x64/signtool.exe" + Wait = $true + ArgumentList = "remove", "/s", "./${env:DIST_FILE_NAME}.exe" + WorkingDirectory = "." + NoNewWindow = $true + } + Start-Process @processOptions1 + + npx postject ctrl-q.exe NODE_SEA_BLOB sea-prep.blob --sentinel-fuse NODE_SEA_FUSE_fce680ab2cc467b6e072b8b5df1996b2 dir @@ -46,48 +59,51 @@ jobs: } Compress-Archive @compress - + artifact_insider: ctrl-q--win-x64--${{ github.sha }}.zip - os: mac-build1 build: | - ./node_modules/.bin/esbuild src/ctrl-q.js --bundle --external:axios --external:xdg-open --external:enigma.js --outfile=build.cjs --format=cjs --platform=node --target=node18 --minify --inject:./src/lib/util/import-meta-url.js --define:import.meta.url=import_meta_url - pkg --output "./${DIST_FILE_NAME}" -t node18-macos-x64 ./build.cjs --config package.json --compress GZip + # ------------------- + ./node_modules/.bin/esbuild src/ctrl-q.js --bundle --outfile=build.cjs --format=cjs --platform=node --target=node23 --inject:./src/lib/util/import-meta-url.js --define:import.meta.url=import_meta_url + node --experimental-sea-config sea-config.json + cp $(command -v node) ${DIST_FILE_NAME} + codesign --remove-signature ${DIST_FILE_NAME} + npx postject ctrl-q NODE_SEA_BLOB sea-prep.blob --sentinel-fuse NODE_SEA_FUSE_fce680ab2cc467b6e072b8b5df1996b2 --macho-segment-name NODE_SEA - chmod +x "${DIST_FILE_NAME}" security delete-keychain build.keychain || true pwd ls -la + # ------------------- # Turn our base64-encoded certificate back to a regular .p12 file - echo $MACOS_CERTIFICATE | base64 --decode > certificate.p12 + # ------------------- # We need to create a new keychain, otherwise using the certificate will prompt # with a UI dialog asking for the certificate password, which we can't # use in a headless CI environment - + security create-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain security list-keychains -d user -s build.keychain security default-keychain -d user -s build.keychain security unlock-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain security import certificate.p12 -k build.keychain -P "$MACOS_CERTIFICATE_PWD" -T /usr/bin/codesign security set-key-partition-list -S apple-tool:,apple:,codesign: -s -k "$MACOS_CI_KEYCHAIN_PWD" build.keychain - - codesign --force -s "$MACOS_CERTIFICATE_NAME" -v "./${DIST_FILE_NAME}" --deep --strict --options=runtime --timestamp --entitlements ./release-config/${DIST_FILE_NAME}.entitlements + codesign --force -s "$MACOS_CERTIFICATE_NAME" -v "./${DIST_FILE_NAME}" --deep --strict --options=runtime --timestamp --entitlements ./release-config/${DIST_FILE_NAME}.entitlements + # ------------------- # Notarize # Store the notarization credentials so that we can prevent a UI password dialog from blocking the CI echo "Create keychain profile" xcrun notarytool store-credentials "notarytool-profile" --apple-id "$PROD_MACOS_NOTARIZATION_APPLE_ID" --team-id "$PROD_MACOS_NOTARIZATION_TEAM_ID" --password "$PROD_MACOS_NOTARIZATION_PWD" + # ------------------- # We can't notarize an app bundle directly, but we need to compress it as an archive. # Therefore, we create a zip file containing our app bundle, so that we can send it to the # notarization service - - # Notarize insider binary echo "Creating temp notarization archive for insider build" ditto -c -k --keepParent "./${DIST_FILE_NAME}" "./${DIST_FILE_NAME}--macos-x64--${{ github.sha }}.zip" @@ -99,14 +115,20 @@ jobs: echo "Notarize insider app" xcrun notarytool submit "./${DIST_FILE_NAME}--macos-x64--${{ github.sha }}.zip" --keychain-profile "notarytool-profile" --wait + # ------------------- + # Clean up # Delete build keychain security delete-keychain build.keychain + rm build.cjs + artifact_insider: ctrl-q--macos-x64--${{ github.sha }}.zip - os: ubuntu-latest build: | - ./node_modules/.bin/esbuild src/ctrl-q.js --bundle --external:axios --external:xdg-open --external:enigma.js --outfile=build.cjs --format=cjs --platform=node --target=node18 --minify --inject:./src/lib/util/import-meta-url.js --define:import.meta.url=import_meta_url - pkg --output "./${DIST_FILE_NAME}" -t node18-linux-x64 ./build.cjs --config package.json --compress GZip + ./node_modules/.bin/esbuild src/ctrl-q.js --bundle --outfile=build.cjs --format=cjs --platform=node --target=node23 --inject:./src/lib/util/import-meta-url.js --define:import.meta.url=import_meta_url + node --experimental-sea-config sea-config.json + cp $(command -v node) ${DIST_FILE_NAME} + npx postject ctrl-q NODE_SEA_BLOB sea-prep.blob --sentinel-fuse NODE_SEA_FUSE_fce680ab2cc467b6e072b8b5df1996b2 chmod +x ${DIST_FILE_NAME} @@ -122,7 +144,7 @@ jobs: - name: Setup Node.js uses: actions/setup-node@v4 with: - node-version: lts/* + node-version: 23.x - name: Install tool for creating stand-alone executables run: | @@ -131,7 +153,7 @@ jobs: - name: Install dependencies run: | - pwd + pwd npm ci --include=prod - name: Run Snyk to check for vulnerabilities @@ -144,7 +166,7 @@ jobs: SNYK_TOKEN: ${{ secrets.SNYK_TOKEN }} with: args: --file=./package.json --sarif-file-output=snyk.sarif - + - name: Upload Snyk result to GitHub Code Scanning if: | github.repository_owner == 'ptarmiganlabs' && @@ -156,7 +178,7 @@ jobs: - name: Create binaries env: - DIST_FILE_NAME: ctrl-q + DIST_FILE_NAME: ctrl-q GITHUB_TOKEN: ${{ secrets.RELEASE_PLEASE_PAT }} MACOS_CERTIFICATE: ${{ secrets.PROD_MACOS_CERTIFICATE_BASE64_CODESIGN }} MACOS_CERTIFICATE_PWD: ${{ secrets.PROD_MACOS_CERTIFICATE_CODESIGN_PWD }} @@ -175,4 +197,3 @@ jobs: with: name: ${{ matrix.artifact_insider }} path: ${{ matrix.artifact_insider }} - \ No newline at end of file diff --git a/.gitignore b/.gitignore index 8423812..279d02a 100644 --- a/.gitignore +++ b/.gitignore @@ -140,3 +140,15 @@ task-chain.csv .vscode/launch.json build.cjs ctrl-q +a.json +a1.csv +build-sea.sh +build +certtest.js +logcertfile +sea-config.json +sea-prep.blob +.vscode/launch.json +.vscode/launch.json +certificate.p12 +.vscode/launch.json diff --git a/.jshintrc b/.jshintrc deleted file mode 100644 index 31a0964..0000000 --- a/.jshintrc +++ /dev/null @@ -1,3 +0,0 @@ -{ - "esversion": 9 -} diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index ab0bdc9..00b6ff3 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,10 +1,10 @@ repos: - repo: https://github.com/gitguardian/ggshield - rev: v1.12.0 + rev: v1.33.0 hooks: - id: ggshield language_version: python3 - stages: [commit] + stages: [pre-commit] # - repo: https://github.com/pre-commit/mirrors-jshint # rev: '' # Use the sha / tag you want to point at @@ -12,7 +12,7 @@ repos: # - id: jshint - repo: https://github.com/pre-commit/mirrors-prettier - rev: 'v2.7.1' # Use the sha / tag you want to point at + rev: 'v4.0.0-alpha.8' # Use the sha / tag you want to point at hooks: - id: prettier exclude: | @@ -25,9 +25,10 @@ repos: )$ - repo: https://github.com/pre-commit/pre-commit-hooks - rev: v4.3.0 # Use the ref you want to point at + rev: v5.0.0 # Use the ref you want to point at hooks: - id: check-case-conflict + - id: trailing-whitespace - id: check-json exclude: | (?x)^( diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..0967ef4 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1 @@ +{} diff --git a/docs/README.md b/docs/README.md index 8b002db..743145f 100644 --- a/docs/README.md +++ b/docs/README.md @@ -5,7 +5,7 @@ ## Process for creating terminal captures -Note: This process only works on macOS and Linux. +Note: This process only works on macOS and Linux. The tools used do not support Windows. The steps on macOS are diff --git a/eslint.config.js b/eslint.config.js new file mode 100644 index 0000000..44fc837 --- /dev/null +++ b/eslint.config.js @@ -0,0 +1,37 @@ +import prettier from 'eslint-plugin-prettier'; +import globals from 'globals'; +import path from 'node:path'; +import { fileURLToPath } from 'node:url'; +import js from '@eslint/js'; +import { FlatCompat } from '@eslint/eslintrc'; + +const __filename = fileURLToPath(import.meta.url); +const __dirname = path.dirname(__filename); +const compat = new FlatCompat({ + baseDirectory: __dirname, + recommendedConfig: js.configs.recommended, + allConfig: js.configs.all, +}); + +// export default [...compat.extends("airbnb-base", "prettier"), { +export default [ + ...compat.extends('prettier'), + { + plugins: { + prettier, + }, + + languageOptions: { + globals: { + ...globals.node, + }, + + ecmaVersion: 12, + sourceType: 'module', + }, + + rules: { + 'prettier/prettier': 'error', + }, + }, +]; diff --git a/package-lock.json b/package-lock.json index 425a1c7..a68c76e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9,41 +9,41 @@ "version": "3.18.1", "license": "MIT", "dependencies": { - "axios": "^1.7.2", + "@qlik/api": "^1.23.0", + "axios": "^1.7.7", "commander": "^12.1.0", "csv-parse": "^5.5.6", - "csv-stringify": "^6.5.0", + "csv-stringify": "^6.5.1", "enigma.js": "^2.14.0", - "esbuild": "^0.21.4", - "form-data": "^4.0.0", + "esbuild": "^0.24.0", + "form-data": "^4.0.1", "fs-extra": "^11.2.0", "handlebars": "^4.7.8", - "luxon": "^3.4.4", + "luxon": "^3.5.0", "node-xlsx": "^0.24.0", "qrs-interact": "^6.3.1", "random-words": "^2.0.1", "retry-axios": "^3.1.3", "table": "^6.8.2", "text-treeview": "^1.0.2", + "undici": "^6.20.1", "upath": "^2.0.1", - "uuid": "^9.0.1", - "winston": "^3.13.0", + "uuid": "^11.0.0", + "winston": "^3.15.0", "winston-daily-rotate-file": "^5.0.0", - "ws": "^8.17.0", + "ws": "^8.18.0", "yesno": "^0.4.0" }, "devDependencies": { - "@babel/eslint-parser": "^7.24.6", - "@babel/plugin-syntax-import-assertions": "^7.24.6", + "@babel/eslint-parser": "^7.25.9", + "@babel/plugin-syntax-import-assertions": "^7.26.0", + "@eslint/js": "^9.13.0", "@jest/globals": "^29.7.0", - "eslint": "^8.57.0", - "eslint-config-airbnb-base": "^15.0.0", "eslint-config-prettier": "^9.1.0", - "eslint-plugin-import": "^2.29.1", - "eslint-plugin-prettier": "^5.1.3", + "eslint-plugin-prettier": "^5.2.1", "jest": "^29.7.0", - "prettier": "^3.3.0", - "snyk": "^1.1291.1" + "prettier": "^3.3.3", + "snyk": "^1.1294.0" } }, "node_modules/@aashutoshrathi/word-wrap": { @@ -51,6 +51,7 @@ "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz", "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==", "dev": true, + "peer": true, "engines": { "node": ">=0.10.0" } @@ -204,10 +205,11 @@ } }, "node_modules/@babel/eslint-parser": { - "version": "7.24.6", - "resolved": "https://registry.npmjs.org/@babel/eslint-parser/-/eslint-parser-7.24.6.tgz", - "integrity": "sha512-Q1BfQX42zXHx732PLW0w4+Y3wJjoZKEMaatFUEAmQ7Z+jCXxinzeqX9bvv2Q8xNPes/H6F0I23oGkcgjaItmLw==", + "version": "7.25.9", + "resolved": "https://registry.npmjs.org/@babel/eslint-parser/-/eslint-parser-7.25.9.tgz", + "integrity": "sha512-5UXfgpK0j0Xr/xIdgdLEhOFxaDZ0bRPWJJchRpqOSur/3rZoPbqqki5mm0p4NE2cs28krBEiSM2MB7//afRSQQ==", "dev": true, + "license": "MIT", "dependencies": { "@nicolo-ribaudo/eslint-scope-5-internals": "5.1.1-v1", "eslint-visitor-keys": "^2.1.0", @@ -342,10 +344,11 @@ } }, "node_modules/@babel/helper-plugin-utils": { - "version": "7.24.6", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.24.6.tgz", - "integrity": "sha512-MZG/JcWfxybKwsA9N9PmtF2lOSFSEMVCpIRrbxccZFLJPrJciJdG/UhSh5W96GEteJI2ARqm5UAHxISwRDLSNg==", + "version": "7.25.9", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.25.9.tgz", + "integrity": "sha512-kSMlyUVdWe25rEsRGviIgOWnoT/nfABVWlqt9N19/dIPWViAOW2s9wznP5tURbs/IDuNk4gPy3YdYRgH3uxhBw==", "dev": true, + "license": "MIT", "engines": { "node": ">=6.9.0" } @@ -549,12 +552,13 @@ } }, "node_modules/@babel/plugin-syntax-import-assertions": { - "version": "7.24.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-assertions/-/plugin-syntax-import-assertions-7.24.6.tgz", - "integrity": "sha512-BE6o2BogJKJImTmGpkmOic4V0hlRRxVtzqxiSPa8TIFxyhi4EFjHm08nq1M4STK4RytuLMgnSz0/wfflvGFNOg==", + "version": "7.26.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-assertions/-/plugin-syntax-import-assertions-7.26.0.tgz", + "integrity": "sha512-QCWT5Hh830hK5EQa7XzuqIkQU9tT/whqbDz7kuaZMHFl1inRRg7JnuAEOQ0Ur0QUl0NufCk1msK2BeY79Aj/eg==", "dev": true, + "license": "MIT", "dependencies": { - "@babel/helper-plugin-utils": "^7.24.6" + "@babel/helper-plugin-utils": "^7.25.9" }, "engines": { "node": ">=6.9.0" @@ -769,9 +773,10 @@ "dev": true }, "node_modules/@colors/colors": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.5.0.tgz", - "integrity": "sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==", + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.6.0.tgz", + "integrity": "sha512-Ir+AOibqzrIsL6ajt3Rz3LskB7OiMVHqltZmspbW/TJuTVuyOMirVqAkjfY6JISiLHgyNqicAC8AyHHGzNd/dA==", + "license": "MIT", "engines": { "node": ">=0.1.90" } @@ -787,348 +792,387 @@ } }, "node_modules/@esbuild/aix-ppc64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.4.tgz", - "integrity": "sha512-Zrm+B33R4LWPLjDEVnEqt2+SLTATlru1q/xYKVn8oVTbiRBGmK2VIMoIYGJDGyftnGaC788IuzGFAlb7IQ0Y8A==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.24.0.tgz", + "integrity": "sha512-WtKdFM7ls47zkKHFVzMz8opM7LkcsIp9amDUBIAWirg70RM71WRSjdILPsY5Uv1D42ZpUfaPILDlfactHgsRkw==", "cpu": [ "ppc64" ], + "license": "MIT", "optional": true, "os": [ "aix" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/android-arm": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.4.tgz", - "integrity": "sha512-E7H/yTd8kGQfY4z9t3nRPk/hrhaCajfA3YSQSBrst8B+3uTcgsi8N+ZWYCaeIDsiVs6m65JPCaQN/DxBRclF3A==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.24.0.tgz", + "integrity": "sha512-arAtTPo76fJ/ICkXWetLCc9EwEHKaeya4vMrReVlEIUCAUncH7M4bhMQ+M9Vf+FFOZJdTNMXNBrWwW+OXWpSew==", "cpu": [ "arm" ], + "license": "MIT", "optional": true, "os": [ "android" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/android-arm64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.4.tgz", - "integrity": "sha512-fYFnz+ObClJ3dNiITySBUx+oNalYUT18/AryMxfovLkYWbutXsct3Wz2ZWAcGGppp+RVVX5FiXeLYGi97umisA==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.24.0.tgz", + "integrity": "sha512-Vsm497xFM7tTIPYK9bNTYJyF/lsP590Qc1WxJdlB6ljCbdZKU9SY8i7+Iin4kyhV/KV5J2rOKsBQbB77Ab7L/w==", "cpu": [ "arm64" ], + "license": "MIT", "optional": true, "os": [ "android" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/android-x64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.4.tgz", - "integrity": "sha512-mDqmlge3hFbEPbCWxp4fM6hqq7aZfLEHZAKGP9viq9wMUBVQx202aDIfc3l+d2cKhUJM741VrCXEzRFhPDKH3Q==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.24.0.tgz", + "integrity": "sha512-t8GrvnFkiIY7pa7mMgJd7p8p8qqYIz1NYiAoKc75Zyv73L3DZW++oYMSHPRarcotTKuSs6m3hTOa5CKHaS02TQ==", "cpu": [ "x64" ], + "license": "MIT", "optional": true, "os": [ "android" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/darwin-arm64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.4.tgz", - "integrity": "sha512-72eaIrDZDSiWqpmCzVaBD58c8ea8cw/U0fq/PPOTqE3c53D0xVMRt2ooIABZ6/wj99Y+h4ksT/+I+srCDLU9TA==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.24.0.tgz", + "integrity": "sha512-CKyDpRbK1hXwv79soeTJNHb5EiG6ct3efd/FTPdzOWdbZZfGhpbcqIpiD0+vwmpu0wTIL97ZRPZu8vUt46nBSw==", "cpu": [ "arm64" ], + "license": "MIT", "optional": true, "os": [ "darwin" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/darwin-x64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.4.tgz", - "integrity": "sha512-uBsuwRMehGmw1JC7Vecu/upOjTsMhgahmDkWhGLWxIgUn2x/Y4tIwUZngsmVb6XyPSTXJYS4YiASKPcm9Zitag==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.24.0.tgz", + "integrity": "sha512-rgtz6flkVkh58od4PwTRqxbKH9cOjaXCMZgWD905JOzjFKW+7EiUObfd/Kav+A6Gyud6WZk9w+xu6QLytdi2OA==", "cpu": [ "x64" ], + "license": "MIT", "optional": true, "os": [ "darwin" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/freebsd-arm64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.4.tgz", - "integrity": "sha512-8JfuSC6YMSAEIZIWNL3GtdUT5NhUA/CMUCpZdDRolUXNAXEE/Vbpe6qlGLpfThtY5NwXq8Hi4nJy4YfPh+TwAg==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.24.0.tgz", + "integrity": "sha512-6Mtdq5nHggwfDNLAHkPlyLBpE5L6hwsuXZX8XNmHno9JuL2+bg2BX5tRkwjyfn6sKbxZTq68suOjgWqCicvPXA==", "cpu": [ "arm64" ], + "license": "MIT", "optional": true, "os": [ "freebsd" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/freebsd-x64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.4.tgz", - "integrity": "sha512-8d9y9eQhxv4ef7JmXny7591P/PYsDFc4+STaxC1GBv0tMyCdyWfXu2jBuqRsyhY8uL2HU8uPyscgE2KxCY9imQ==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.24.0.tgz", + "integrity": "sha512-D3H+xh3/zphoX8ck4S2RxKR6gHlHDXXzOf6f/9dbFt/NRBDIE33+cVa49Kil4WUjxMGW0ZIYBYtaGCa2+OsQwQ==", "cpu": [ "x64" ], + "license": "MIT", "optional": true, "os": [ "freebsd" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-arm": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.4.tgz", - "integrity": "sha512-2rqFFefpYmpMs+FWjkzSgXg5vViocqpq5a1PSRgT0AvSgxoXmGF17qfGAzKedg6wAwyM7UltrKVo9kxaJLMF/g==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.24.0.tgz", + "integrity": "sha512-gJKIi2IjRo5G6Glxb8d3DzYXlxdEj2NlkixPsqePSZMhLudqPhtZ4BUrpIuTjJYXxvF9njql+vRjB2oaC9XpBw==", "cpu": [ "arm" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-arm64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.4.tgz", - "integrity": "sha512-/GLD2orjNU50v9PcxNpYZi+y8dJ7e7/LhQukN3S4jNDXCKkyyiyAz9zDw3siZ7Eh1tRcnCHAo/WcqKMzmi4eMQ==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.24.0.tgz", + "integrity": "sha512-TDijPXTOeE3eaMkRYpcy3LarIg13dS9wWHRdwYRnzlwlA370rNdZqbcp0WTyyV/k2zSxfko52+C7jU5F9Tfj1g==", "cpu": [ "arm64" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-ia32": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.4.tgz", - "integrity": "sha512-pNftBl7m/tFG3t2m/tSjuYeWIffzwAZT9m08+9DPLizxVOsUl8DdFzn9HvJrTQwe3wvJnwTdl92AonY36w/25g==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.24.0.tgz", + "integrity": "sha512-K40ip1LAcA0byL05TbCQ4yJ4swvnbzHscRmUilrmP9Am7//0UjPreh4lpYzvThT2Quw66MhjG//20mrufm40mA==", "cpu": [ "ia32" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-loong64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.4.tgz", - "integrity": "sha512-cSD2gzCK5LuVX+hszzXQzlWya6c7hilO71L9h4KHwqI4qeqZ57bAtkgcC2YioXjsbfAv4lPn3qe3b00Zt+jIfQ==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.24.0.tgz", + "integrity": "sha512-0mswrYP/9ai+CU0BzBfPMZ8RVm3RGAN/lmOMgW4aFUSOQBjA31UP8Mr6DDhWSuMwj7jaWOT0p0WoZ6jeHhrD7g==", "cpu": [ "loong64" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-mips64el": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.4.tgz", - "integrity": "sha512-qtzAd3BJh7UdbiXCrg6npWLYU0YpufsV9XlufKhMhYMJGJCdfX/G6+PNd0+v877X1JG5VmjBLUiFB0o8EUSicA==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.24.0.tgz", + "integrity": "sha512-hIKvXm0/3w/5+RDtCJeXqMZGkI2s4oMUGj3/jM0QzhgIASWrGO5/RlzAzm5nNh/awHE0A19h/CvHQe6FaBNrRA==", "cpu": [ "mips64el" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-ppc64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.4.tgz", - "integrity": "sha512-yB8AYzOTaL0D5+2a4xEy7OVvbcypvDR05MsB/VVPVA7nL4hc5w5Dyd/ddnayStDgJE59fAgNEOdLhBxjfx5+dg==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.24.0.tgz", + "integrity": "sha512-HcZh5BNq0aC52UoocJxaKORfFODWXZxtBaaZNuN3PUX3MoDsChsZqopzi5UupRhPHSEHotoiptqikjN/B77mYQ==", "cpu": [ "ppc64" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-riscv64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.4.tgz", - "integrity": "sha512-Y5AgOuVzPjQdgU59ramLoqSSiXddu7F3F+LI5hYy/d1UHN7K5oLzYBDZe23QmQJ9PIVUXwOdKJ/jZahPdxzm9w==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.24.0.tgz", + "integrity": "sha512-bEh7dMn/h3QxeR2KTy1DUszQjUrIHPZKyO6aN1X4BCnhfYhuQqedHaa5MxSQA/06j3GpiIlFGSsy1c7Gf9padw==", "cpu": [ "riscv64" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-s390x": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.4.tgz", - "integrity": "sha512-Iqc/l/FFwtt8FoTK9riYv9zQNms7B8u+vAI/rxKuN10HgQIXaPzKZc479lZ0x6+vKVQbu55GdpYpeNWzjOhgbA==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.24.0.tgz", + "integrity": "sha512-ZcQ6+qRkw1UcZGPyrCiHHkmBaj9SiCD8Oqd556HldP+QlpUIe2Wgn3ehQGVoPOvZvtHm8HPx+bH20c9pvbkX3g==", "cpu": [ "s390x" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-x64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.4.tgz", - "integrity": "sha512-Td9jv782UMAFsuLZINfUpoF5mZIbAj+jv1YVtE58rFtfvoKRiKSkRGQfHTgKamLVT/fO7203bHa3wU122V/Bdg==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.24.0.tgz", + "integrity": "sha512-vbutsFqQ+foy3wSSbmjBXXIJ6PL3scghJoM8zCL142cGaZKAdCZHyf+Bpu/MmX9zT9Q0zFBVKb36Ma5Fzfa8xA==", "cpu": [ "x64" ], + "license": "MIT", "optional": true, "os": [ "linux" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/netbsd-x64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.4.tgz", - "integrity": "sha512-Awn38oSXxsPMQxaV0Ipb7W/gxZtk5Tx3+W+rAPdZkyEhQ6968r9NvtkjhnhbEgWXYbgV+JEONJ6PcdBS+nlcpA==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.24.0.tgz", + "integrity": "sha512-hjQ0R/ulkO8fCYFsG0FZoH+pWgTTDreqpqY7UnQntnaKv95uP5iW3+dChxnx7C3trQQU40S+OgWhUVwCjVFLvg==", "cpu": [ "x64" ], + "license": "MIT", "optional": true, "os": [ "netbsd" ], "engines": { - "node": ">=12" + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.24.0.tgz", + "integrity": "sha512-MD9uzzkPQbYehwcN583yx3Tu5M8EIoTD+tUgKF982WYL9Pf5rKy9ltgD0eUgs8pvKnmizxjXZyLt0z6DC3rRXg==", + "cpu": [ + "arm64" + ], + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" } }, "node_modules/@esbuild/openbsd-x64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.4.tgz", - "integrity": "sha512-IsUmQeCY0aU374R82fxIPu6vkOybWIMc3hVGZ3ChRwL9hA1TwY+tS0lgFWV5+F1+1ssuvvXt3HFqe8roCip8Hg==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.24.0.tgz", + "integrity": "sha512-4ir0aY1NGUhIC1hdoCzr1+5b43mw99uNwVzhIq1OY3QcEwPDO3B7WNXBzaKY5Nsf1+N11i1eOfFcq+D/gOS15Q==", "cpu": [ "x64" ], + "license": "MIT", "optional": true, "os": [ "openbsd" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/sunos-x64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.4.tgz", - "integrity": "sha512-hsKhgZ4teLUaDA6FG/QIu2q0rI6I36tZVfM4DBZv3BG0mkMIdEnMbhc4xwLvLJSS22uWmaVkFkqWgIS0gPIm+A==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.24.0.tgz", + "integrity": "sha512-jVzdzsbM5xrotH+W5f1s+JtUy1UWgjU0Cf4wMvffTB8m6wP5/kx0KiaLHlbJO+dMgtxKV8RQ/JvtlFcdZ1zCPA==", "cpu": [ "x64" ], + "license": "MIT", "optional": true, "os": [ "sunos" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/win32-arm64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.4.tgz", - "integrity": "sha512-UUfMgMoXPoA/bvGUNfUBFLCh0gt9dxZYIx9W4rfJr7+hKe5jxxHmfOK8YSH4qsHLLN4Ck8JZ+v7Q5fIm1huErg==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.24.0.tgz", + "integrity": "sha512-iKc8GAslzRpBytO2/aN3d2yb2z8XTVfNV0PjGlCxKo5SgWmNXx82I/Q3aG1tFfS+A2igVCY97TJ8tnYwpUWLCA==", "cpu": [ "arm64" ], + "license": "MIT", "optional": true, "os": [ "win32" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/win32-ia32": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.4.tgz", - "integrity": "sha512-yIxbspZb5kGCAHWm8dexALQ9en1IYDfErzjSEq1KzXFniHv019VT3mNtTK7t8qdy4TwT6QYHI9sEZabONHg+aw==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.24.0.tgz", + "integrity": "sha512-vQW36KZolfIudCcTnaTpmLQ24Ha1RjygBo39/aLkM2kmjkWmZGEJ5Gn9l5/7tzXA42QGIoWbICfg6KLLkIw6yw==", "cpu": [ "ia32" ], + "license": "MIT", "optional": true, "os": [ "win32" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/win32-x64": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.4.tgz", - "integrity": "sha512-sywLRD3UK/qRJt0oBwdpYLBibk7KiRfbswmWRDabuncQYSlf8aLEEUor/oP6KRz8KEG+HoiVLBhPRD5JWjS8Sg==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.24.0.tgz", + "integrity": "sha512-7IAFPrjSQIJrGsK6flwg7NFmwBoSTyF3rl7If0hNUFQU4ilTsEPL6GuMuU9BfIWVVGuRnuIidkSMC+c0Otu8IA==", "cpu": [ "x64" ], + "license": "MIT", "optional": true, "os": [ "win32" ], "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@eslint-community/eslint-utils": { @@ -1136,6 +1180,7 @@ "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.3.0.tgz", "integrity": "sha512-v3oplH6FYCULtFuCeqyuTd9D2WKO937Dxdq+GmHOLL72TTRriLxz2VLlNfkZRsvj6PKnOPAtuT6dwrs/pA5DvA==", "dev": true, + "peer": true, "dependencies": { "eslint-visitor-keys": "^3.3.0" }, @@ -1147,24 +1192,55 @@ } }, "node_modules/@eslint-community/regexpp": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.7.0.tgz", - "integrity": "sha512-+HencqxU7CFJnQb7IKtuNBqS6Yx3Tz4kOL8BJXo+JyeiBm5MEX6pO8onXDkjrkCRlfYXS1Axro15ZjVFe9YgsA==", + "version": "4.11.1", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.11.1.tgz", + "integrity": "sha512-m4DVN9ZqskZoLU5GlWZadwDnYo3vAEydiUayB9widCl9ffWx2IvPnp6n3on5rJmziJSw9Bv+Z3ChDVdMwXCY8Q==", "dev": true, + "license": "MIT", + "peer": true, "engines": { "node": "^12.0.0 || ^14.0.0 || >=16.0.0" } }, + "node_modules/@eslint/config-array": { + "version": "0.18.0", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.18.0.tgz", + "integrity": "sha512-fTxvnS1sRMu3+JjXwJG0j/i4RT9u4qJ+lqS/yCGap4lH4zZGzQ7tu+xZqQmcMZq5OBZDL4QRxQzRjkWcGt8IVw==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "@eslint/object-schema": "^2.1.4", + "debug": "^4.3.1", + "minimatch": "^3.1.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/core": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.7.0.tgz", + "integrity": "sha512-xp5Jirz5DyPYlPiKat8jaq0EmYvDXKKpzTbxXMpT9eqlRJkRKIz9AGMdlvYjih+im+QlhWrpvVjl8IPC/lHlUw==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, "node_modules/@eslint/eslintrc": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", - "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.1.0.tgz", + "integrity": "sha512-4Bfj15dVJdoy3RfZmmo86RK1Fwzn6SstsvK9JS+BaVKqC6QQQQyXekNaC+g+LKNgkQ+2VhGAzm6hO40AhMR3zQ==", "dev": true, + "license": "MIT", + "peer": true, "dependencies": { "ajv": "^6.12.4", "debug": "^4.3.2", - "espree": "^9.6.0", - "globals": "^13.19.0", + "espree": "^10.0.1", + "globals": "^14.0.0", "ignore": "^5.2.0", "import-fresh": "^3.2.1", "js-yaml": "^4.1.0", @@ -1172,7 +1248,7 @@ "strip-json-comments": "^3.1.1" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" @@ -1183,6 +1259,8 @@ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", "dev": true, + "license": "MIT", + "peer": true, "dependencies": { "fast-deep-equal": "^3.1.1", "fast-json-stable-stringify": "^2.0.0", @@ -1198,29 +1276,69 @@ "version": "0.4.1", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true + "dev": true, + "license": "MIT", + "peer": true }, "node_modules/@eslint/js": { - "version": "8.57.0", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.0.tgz", - "integrity": "sha512-Ys+3g2TaW7gADOJzPt83SJtCDhMjndcDMFVQ/Tj9iA1BfJzFKD9mAUXT3OenpuPHbI6P/myECxRJrofUsDx/5g==", + "version": "9.13.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.13.0.tgz", + "integrity": "sha512-IFLyoY4d72Z5y/6o/BazFBezupzI/taV8sGumxTAVw3lXG9A6md1Dc34T9s1FoD/an9pJH8RHbAxsaEbBed9lA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/object-schema": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.4.tgz", + "integrity": "sha512-BsWiH1yFGjXXS2yvrf5LyuoSIIbPrGUWob917o+BTKuZ7qJdxX8aJLRxs1fS9n6r7vESrq1OUqb68dANcFXuQQ==", "dev": true, + "license": "Apache-2.0", + "peer": true, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" } }, - "node_modules/@humanwhocodes/config-array": { - "version": "0.11.14", - "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.14.tgz", - "integrity": "sha512-3T8LkOmg45BV5FICb15QQMsyUSWrQ8AygVfC7ZG32zOalnqrilm018ZVCw0eapXux8FtA33q8PSRSstjee3jSg==", + "node_modules/@eslint/plugin-kit": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.1.tgz", + "integrity": "sha512-HFZ4Mp26nbWk9d/BpvP0YNL6W4UoZF0VFcTw/aPPA8RpOxeFQgK+ClABGgAUXs9Y/RGX/l1vOmrqz1MQt9MNuw==", "dev": true, + "license": "Apache-2.0", + "peer": true, "dependencies": { - "@humanwhocodes/object-schema": "^2.0.2", - "debug": "^4.3.1", - "minimatch": "^3.0.5" + "levn": "^0.4.1" }, "engines": { - "node": ">=10.10.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@humanfs/core": { + "version": "0.19.0", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.0.tgz", + "integrity": "sha512-2cbWIHbZVEweE853g8jymffCA+NCMiuqeECeBBLm8dg2oFdjuGJhgN4UAbI+6v0CKbbhvtXA4qV8YR5Ji86nmw==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node": { + "version": "0.16.5", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.5.tgz", + "integrity": "sha512-KSPA4umqSG4LHYRodq31VDwKAvaTF4xmVlzM8Aeh4PlU1JQ3IG0wiA8C25d3RQ9nJyM3mBHyI53K06VVL/oFFg==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "@humanfs/core": "^0.19.0", + "@humanwhocodes/retry": "^0.3.0" + }, + "engines": { + "node": ">=18.18.0" } }, "node_modules/@humanwhocodes/module-importer": { @@ -1228,6 +1346,7 @@ "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", "dev": true, + "peer": true, "engines": { "node": ">=12.22" }, @@ -1236,11 +1355,20 @@ "url": "https://github.com/sponsors/nzakas" } }, - "node_modules/@humanwhocodes/object-schema": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.2.tgz", - "integrity": "sha512-6EwiSjwWYP7pTckG6I5eyFANjPhmPjUX9JRLUSfNPC7FX7zK9gyZAfUEaECL6ALTpGX5AjnBq3C9XmVWPitNpw==", - "dev": true + "node_modules/@humanwhocodes/retry": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.1.tgz", + "integrity": "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } }, "node_modules/@istanbuljs/load-nyc-config": { "version": "1.1.0", @@ -1661,116 +1789,114 @@ "node": ">=4.0" } }, - "node_modules/@nodelib/fs.scandir": { - "version": "2.1.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", - "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "node_modules/@pkgr/core": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.1.1.tgz", + "integrity": "sha512-cq8o4cWH0ibXh9VGi5P20Tu9XF/0fFXl9EUinr9QfTM7a7p0oTA4iJRCQWppXR1Pg8dSM0UCItCkPwsk9qWWYA==", "dev": true, - "dependencies": { - "@nodelib/fs.stat": "2.0.5", - "run-parallel": "^1.1.9" - }, + "license": "MIT", "engines": { - "node": ">= 8" + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/unts" } }, - "node_modules/@nodelib/fs.stat": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", - "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", - "dev": true, + "node_modules/@qlik/api": { + "version": "1.23.0", + "resolved": "https://registry.npmjs.org/@qlik/api/-/api-1.23.0.tgz", + "integrity": "sha512-QrwN7iQP5NYeAevbF9gBRzOnbKuV5MKUNXczyfqN3KnbDFCMRKi0BlwesHD15mdpkpydCd8A8uvPtkHz9OiT0Q==", + "license": "ISC", + "dependencies": { + "enigma.js": "^2.14.0", + "lodash": "^4.17.21", + "nanoid": "^5.0.7", + "ws": "^8.18.0" + }, "engines": { - "node": ">= 8" + "node": ">=18" } }, - "node_modules/@nodelib/fs.walk": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", - "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "node_modules/@sentry-internal/tracing": { + "version": "7.119.2", + "resolved": "https://registry.npmjs.org/@sentry-internal/tracing/-/tracing-7.119.2.tgz", + "integrity": "sha512-V2W+STWrafyGJhQv3ulMFXYDwWHiU6wHQAQBShsHVACiFaDrJ2kPRet38FKv4dMLlLlP2xN+ss2e5zv3tYlTiQ==", "dev": true, + "license": "MIT", "dependencies": { - "@nodelib/fs.scandir": "2.1.5", - "fastq": "^1.6.0" + "@sentry/core": "7.119.2", + "@sentry/types": "7.119.2", + "@sentry/utils": "7.119.2" }, "engines": { - "node": ">= 8" + "node": ">=8" } }, - "node_modules/@pkgr/utils": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/@pkgr/utils/-/utils-2.4.2.tgz", - "integrity": "sha512-POgTXhjrTfbTV63DiFXav4lBHiICLKKwDeaKn9Nphwj7WH6m0hMMCaJkMyRWjgtPFyRKRVoMXXjczsTQRDEhYw==", + "node_modules/@sentry/core": { + "version": "7.119.2", + "resolved": "https://registry.npmjs.org/@sentry/core/-/core-7.119.2.tgz", + "integrity": "sha512-hQr3d2yWq/2lMvoyBPOwXw1IHqTrCjOsU1vYKhAa6w9vGbJZFGhKGGE2KEi/92c3gqGn+gW/PC7cV6waCTDuVA==", "dev": true, + "license": "MIT", "dependencies": { - "cross-spawn": "^7.0.3", - "fast-glob": "^3.3.0", - "is-glob": "^4.0.3", - "open": "^9.1.0", - "picocolors": "^1.0.0", - "tslib": "^2.6.0" + "@sentry/types": "7.119.2", + "@sentry/utils": "7.119.2" }, "engines": { - "node": "^12.20.0 || ^14.18.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/unts" + "node": ">=8" } }, - "node_modules/@pkgr/utils/node_modules/tslib": { - "version": "2.6.2", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", - "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==", - "dev": true - }, - "node_modules/@sentry/core": { - "version": "7.42.0", - "resolved": "https://registry.npmjs.org/@sentry/core/-/core-7.42.0.tgz", - "integrity": "sha512-vNcTyoQz5kUXo5vMGDyc5BJMO0UugPvMfYMQVxqt/BuDNR30LVhY+DL2tW1DFZDvRvyn5At+H7kSTj6GFrANXQ==", + "node_modules/@sentry/integrations": { + "version": "7.119.2", + "resolved": "https://registry.npmjs.org/@sentry/integrations/-/integrations-7.119.2.tgz", + "integrity": "sha512-dCuXKvbUE3gXVVa696SYMjlhSP6CxpMH/gl4Jk26naEB8Xjsn98z/hqEoXLg6Nab73rjR9c/9AdKqBbwVMHyrQ==", "dev": true, + "license": "MIT", "dependencies": { - "@sentry/types": "7.42.0", - "@sentry/utils": "7.42.0", - "tslib": "^1.9.3" + "@sentry/core": "7.119.2", + "@sentry/types": "7.119.2", + "@sentry/utils": "7.119.2", + "localforage": "^1.8.1" }, "engines": { "node": ">=8" } }, "node_modules/@sentry/node": { - "version": "7.42.0", - "resolved": "https://registry.npmjs.org/@sentry/node/-/node-7.42.0.tgz", - "integrity": "sha512-mmpVSDeoM5aEbKOMq3Wt54wAvH53bkivhRh3Ip+R7Uj3aOKkcVJST2XlbghHgoYQXTWz+pl475EVyODWgY9QYg==", + "version": "7.119.2", + "resolved": "https://registry.npmjs.org/@sentry/node/-/node-7.119.2.tgz", + "integrity": "sha512-TPNnqxh+Myooe4jTyRiXrzrM2SH08R4+nrmBls4T7lKp2E5R/3mDSe/YTn5rRcUt1k1hPx1NgO/taG0DoS5cXA==", "dev": true, + "license": "MIT", "dependencies": { - "@sentry/core": "7.42.0", - "@sentry/types": "7.42.0", - "@sentry/utils": "7.42.0", - "cookie": "^0.4.1", - "https-proxy-agent": "^5.0.0", - "lru_map": "^0.3.3", - "tslib": "^1.9.3" + "@sentry-internal/tracing": "7.119.2", + "@sentry/core": "7.119.2", + "@sentry/integrations": "7.119.2", + "@sentry/types": "7.119.2", + "@sentry/utils": "7.119.2" }, "engines": { "node": ">=8" } }, "node_modules/@sentry/types": { - "version": "7.42.0", - "resolved": "https://registry.npmjs.org/@sentry/types/-/types-7.42.0.tgz", - "integrity": "sha512-Ga0xaBIR/peuXQ88hI9a5TNY3GLNoH8jpsgPaAjAtRHkLsTx0y3AR+PrD7pUysza9QjvG+Qux01DRvLgaNKOHA==", + "version": "7.119.2", + "resolved": "https://registry.npmjs.org/@sentry/types/-/types-7.119.2.tgz", + "integrity": "sha512-ydq1tWsdG7QW+yFaTp0gFaowMLNVikIqM70wxWNK+u98QzKnVY/3XTixxNLsUtnAB4Y+isAzFhrc6Vb5GFdFeg==", "dev": true, + "license": "MIT", "engines": { "node": ">=8" } }, "node_modules/@sentry/utils": { - "version": "7.42.0", - "resolved": "https://registry.npmjs.org/@sentry/utils/-/utils-7.42.0.tgz", - "integrity": "sha512-cBiDZVipC+is+IVgsTQLJyZWUZQxlLZ9GarNT+XZOZ5BFh0acFtz88hO6+S7vGmhcx2aCvsdC9yb2Yf+BphK6Q==", + "version": "7.119.2", + "resolved": "https://registry.npmjs.org/@sentry/utils/-/utils-7.119.2.tgz", + "integrity": "sha512-TLdUCvcNgzKP0r9YD7tgCL1PEUp42TObISridsPJ5rhpVGQJvpr+Six0zIkfDUxerLYWZoK8QMm9KgFlPLNQzA==", "dev": true, + "license": "MIT", "dependencies": { - "@sentry/types": "7.42.0", - "tslib": "^1.9.3" + "@sentry/types": "7.119.2" }, "engines": { "node": ">=8" @@ -1841,6 +1967,14 @@ "@babel/types": "^7.20.7" } }, + "node_modules/@types/estree": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.6.tgz", + "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==", + "dev": true, + "license": "MIT", + "peer": true + }, "node_modules/@types/graceful-fs": { "version": "4.1.7", "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.7.tgz", @@ -1874,11 +2008,13 @@ "@types/istanbul-lib-report": "*" } }, - "node_modules/@types/json5": { - "version": "0.0.29", - "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", - "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==", - "dev": true + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true, + "license": "MIT", + "peer": true }, "node_modules/@types/node": { "version": "20.6.5", @@ -1892,6 +2028,12 @@ "integrity": "sha512-Hl219/BT5fLAaz6NDkSuhzasy49dwQS/DSdu4MdggFB8zcXv7vflBI3xp7FEmkmdDkBUI2bPUNeMttp2knYdxw==", "dev": true }, + "node_modules/@types/triple-beam": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/@types/triple-beam/-/triple-beam-1.3.5.tgz", + "integrity": "sha512-6WaYesThRMCl19iryMYP7/x2OVgCtbIVflDGFpWnb9irXI3UjYE4AzmYuiUKY1AJstGijoY+MgUszMgRxIYTYw==", + "license": "MIT" + }, "node_modules/@types/yargs": { "version": "17.0.25", "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.25.tgz", @@ -1907,17 +2049,13 @@ "integrity": "sha512-axdPBuLuEJt0c4yI5OZssC19K2Mq1uKdrfZBzuxLvaztgqUtFYZUNw7lETExPYJR9jdEoIg4mb7RQKRQzOkeGQ==", "dev": true }, - "node_modules/@ungap/structured-clone": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz", - "integrity": "sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ==", - "dev": true - }, "node_modules/acorn": { - "version": "8.11.2", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.2.tgz", - "integrity": "sha512-nc0Axzp/0FILLEVsm4fNwLCwMttvhEI263QtVPQcbpfZZ3ts0hLsZGOpE6czNlid7CJ9MlyH8reXkpsf3YUY4w==", + "version": "8.13.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.13.0.tgz", + "integrity": "sha512-8zSiw54Oxrdym50NlZ9sUusyO1Z1ZchgRLWRaK6c86XJFClyCgFKetdowBg5bKxyp/u+CDBJG4Mpp0m3HLZl9w==", "dev": true, + "license": "MIT", + "peer": true, "bin": { "acorn": "bin/acorn" }, @@ -1930,22 +2068,12 @@ "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", "dev": true, + "license": "MIT", + "peer": true, "peerDependencies": { "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" } }, - "node_modules/agent-base": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", - "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", - "dev": true, - "dependencies": { - "debug": "4" - }, - "engines": { - "node": ">= 6.0.0" - } - }, "node_modules/ajv": { "version": "8.6.1", "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.6.1.tgz", @@ -2027,149 +2155,33 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "dev": true - }, - "node_modules/array-buffer-byte-length": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.0.tgz", - "integrity": "sha512-LPuwb2P+NrQw3XhxGc36+XSvuBPopovXYTR9Ew++Du9Yb/bx5AzBfrIsBoj0EZUifjQU+sHL21sseZ3jerWO/A==", "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "is-array-buffer": "^3.0.1" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } + "license": "Python-2.0", + "peer": true }, - "node_modules/array-includes": { - "version": "3.1.7", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.7.tgz", - "integrity": "sha512-dlcsNBIiWhPkHdOEEKnehA+RNUWDc4UqFtnIXU4uuYDPtA4LDkr7qip2p0VvFAEXNDr0yWZ9PJyIRiGjRLQzwQ==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1", - "get-intrinsic": "^1.2.1", - "is-string": "^1.0.7" - }, + "node_modules/astral-regex": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", + "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">=8" } }, - "node_modules/array.prototype.findlastindex": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.3.tgz", - "integrity": "sha512-LzLoiOMAxvy+Gd3BAq3B7VeIgPdo+Q8hthvKtXybMvRV0jrXfJM/t8mw7nNlpEcVlVUnCnM2KSX4XU5HmpodOA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1", - "es-shim-unscopables": "^1.0.0", - "get-intrinsic": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.flat": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.2.tgz", - "integrity": "sha512-djYB+Zx2vLewY8RWlNCUdHjDXs2XOgm602S9E7P/UpHgfeHL00cRiIF+IN/G/aUJ7kGPb6yO/ErDI5V2s8iycA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1", - "es-shim-unscopables": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/array.prototype.flatmap": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.2.tgz", - "integrity": "sha512-Ewyx0c9PmpcsByhSW4r+9zDU7sGjFc86qf/kKtuSCRdhfbk0SNLLkaT5qvcHnRGgc5NP/ly/y+qkXkqONX54CQ==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1", - "es-shim-unscopables": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/arraybuffer.prototype.slice": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.1.tgz", - "integrity": "sha512-09x0ZWFEjj4WD8PDbykUwo3t9arLn8NIzmmYEJFpYekOAQjpkGSyrQhNoRTcwwcFRu+ycWF78QZ63oWTqSjBcw==", - "dev": true, - "dependencies": { - "array-buffer-byte-length": "^1.0.0", - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "get-intrinsic": "^1.2.1", - "is-array-buffer": "^3.0.2", - "is-shared-array-buffer": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/astral-regex": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", - "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", - "engines": { - "node": ">=8" - } - }, - "node_modules/async": { - "version": "3.2.3", - "resolved": "https://registry.npmjs.org/async/-/async-3.2.3.tgz", - "integrity": "sha512-spZRyzKL5l5BZQrr/6m/SqFdBN0q3OCI0f9rjfBzCMBIP4p75P620rR3gTmaksNOhmzgdxcaxdNfMy6anrbM0g==" + "node_modules/async": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.3.tgz", + "integrity": "sha512-spZRyzKL5l5BZQrr/6m/SqFdBN0q3OCI0f9rjfBzCMBIP4p75P620rR3gTmaksNOhmzgdxcaxdNfMy6anrbM0g==" }, "node_modules/asynckit": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" }, - "node_modules/available-typed-arrays": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz", - "integrity": "sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==", - "dev": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/axios": { - "version": "1.7.2", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.2.tgz", - "integrity": "sha512-2A8QhOMrbomlDuiLeK9XibIBzuHeRcqqNOHp0Cyp5EoJ1IFDh+XZH3A6BkXtv0K4gFGCI0Y4BM7B1wOEi0Rmgw==", + "version": "1.7.7", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.7.tgz", + "integrity": "sha512-S4kL7XrjgBmvdGut0sN3yJxqYzrDOnivkBiN0OFs6hLiUam3UPvswUo0kqGyhqUZGEOytHyumEdXsAkgCOUf3Q==", + "license": "MIT", "dependencies": { "follow-redirects": "^1.15.6", "form-data": "^4.0.0", @@ -2289,15 +2301,6 @@ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", "dev": true }, - "node_modules/big-integer": { - "version": "1.6.52", - "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.52.tgz", - "integrity": "sha512-QxD8cf2eVqJOOz63z6JIN9BzvVs/dlySa5HGSBH5xtR8dPteIRQnBxxKqkNTiT6jbDTF6jAfrd4oMcND9RGbQg==", - "dev": true, - "engines": { - "node": ">=0.6" - } - }, "node_modules/bluebird": { "version": "3.7.2", "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", @@ -2309,18 +2312,6 @@ "integrity": "sha512-d0II/GO9uf9lfUHH2BQsjxzRJZBdsjgsBiW4BvhWk/3qoKwQFjIDVN19PfX8F2D/r9PCMTtLWjYVCFrpeYUzsw==", "dev": true }, - "node_modules/bplist-parser": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/bplist-parser/-/bplist-parser-0.2.0.tgz", - "integrity": "sha512-z0M+byMThzQmD9NILRniCUXYsYpjwnlO8N5uCFaCqIOpqRsJCrQL9NK3JsD67CN5a08nF5oIL2bD6loTdHOuKw==", - "dev": true, - "dependencies": { - "big-integer": "^1.6.44" - }, - "engines": { - "node": ">= 5.10.0" - } - }, "node_modules/brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", @@ -2332,12 +2323,13 @@ } }, "node_modules/braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", "dev": true, + "license": "MIT", "dependencies": { - "fill-range": "^7.0.1" + "fill-range": "^7.1.1" }, "engines": { "node": ">=8" @@ -2390,34 +2382,6 @@ "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", "dev": true }, - "node_modules/bundle-name": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/bundle-name/-/bundle-name-3.0.0.tgz", - "integrity": "sha512-PKA4BeSvBpQKQ8iPOGCSiell+N8P+Tf1DlwqmYhpe2gAhKPHn8EYOxVT+ShuGmhg8lN8XiSlS80yiExKXrURlw==", - "dev": true, - "dependencies": { - "run-applescript": "^5.0.0" - }, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/call-bind": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", - "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", - "dev": true, - "dependencies": { - "function-bind": "^1.1.1", - "get-intrinsic": "^1.0.2" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/callsites": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", @@ -2613,27 +2577,12 @@ "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", "dev": true }, - "node_modules/confusing-browser-globals": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/confusing-browser-globals/-/confusing-browser-globals-1.0.11.tgz", - "integrity": "sha512-JsPKdmh8ZkmnHxDk55FZ1TqVLvEQTvoByJZRN9jzI0UjxK/QgAmsphz7PGtqgPieQZ/CQcHWXCR7ATDNhGe+YA==", - "dev": true - }, "node_modules/convert-source-map": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", "dev": true }, - "node_modules/cookie": { - "version": "0.4.2", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.2.tgz", - "integrity": "sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==", - "dev": true, - "engines": { - "node": ">= 0.6" - } - }, "node_modules/create-jest": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", @@ -2675,9 +2624,10 @@ "integrity": "sha512-uNpm30m/AGSkLxxy7d9yRXpJQFrZzVWLFBkS+6ngPcZkw/5k3L/jjFuj7tVnEpRn+QgmiXr21nDlhCiUK4ij2A==" }, "node_modules/csv-stringify": { - "version": "6.5.0", - "resolved": "https://registry.npmjs.org/csv-stringify/-/csv-stringify-6.5.0.tgz", - "integrity": "sha512-edlXFVKcUx7r8Vx5zQucsuMg4wb/xT6qyz+Sr1vnLrdXqlLD1+UKyWNyZ9zn6mUW1ewmGxrpVwAcChGF0HQ/2Q==" + "version": "6.5.1", + "resolved": "https://registry.npmjs.org/csv-stringify/-/csv-stringify-6.5.1.tgz", + "integrity": "sha512-+9lpZfwpLntpTIEpFbwQyWuW/hmI/eHuJZD1XzeZpfZTqkf1fyvBbBLXTJJMsBuuS11uTShMqPwzx4A6ffXgRQ==", + "license": "MIT" }, "node_modules/debug": { "version": "4.3.4", @@ -2720,7 +2670,8 @@ "version": "0.1.4", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", - "dev": true + "dev": true, + "peer": true }, "node_modules/deepmerge": { "version": "4.3.1", @@ -2731,168 +2682,32 @@ "node": ">=0.10.0" } }, - "node_modules/default-browser": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/default-browser/-/default-browser-4.0.0.tgz", - "integrity": "sha512-wX5pXO1+BrhMkSbROFsyxUm0i/cJEScyNhA4PPxc41ICuv05ZZB/MX28s8aZx6xjmatvebIapF6hLEKEcpneUA==", - "dev": true, - "dependencies": { - "bundle-name": "^3.0.0", - "default-browser-id": "^3.0.0", - "execa": "^7.1.1", - "titleize": "^3.0.0" - }, - "engines": { - "node": ">=14.16" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/default-browser-id": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/default-browser-id/-/default-browser-id-3.0.0.tgz", - "integrity": "sha512-OZ1y3y0SqSICtE8DE4S8YOE9UZOJ8wO16fKWVP5J1Qz42kV9jcnMVFrEE/noXb/ss3Q4pZIH79kxofzyNNtUNA==", - "dev": true, - "dependencies": { - "bplist-parser": "^0.2.0", - "untildify": "^4.0.0" - }, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/default-browser/node_modules/execa": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/execa/-/execa-7.2.0.tgz", - "integrity": "sha512-UduyVP7TLB5IcAQl+OzLyLcS/l32W/GLg+AhHJ+ow40FOk2U3SAllPwR44v4vmdFwIWqpdwxxpQbF1n5ta9seA==", - "dev": true, - "dependencies": { - "cross-spawn": "^7.0.3", - "get-stream": "^6.0.1", - "human-signals": "^4.3.0", - "is-stream": "^3.0.0", - "merge-stream": "^2.0.0", - "npm-run-path": "^5.1.0", - "onetime": "^6.0.0", - "signal-exit": "^3.0.7", - "strip-final-newline": "^3.0.0" - }, - "engines": { - "node": "^14.18.0 || ^16.14.0 || >=18.0.0" - }, - "funding": { - "url": "https://github.com/sindresorhus/execa?sponsor=1" - } - }, - "node_modules/default-browser/node_modules/human-signals": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-4.3.1.tgz", - "integrity": "sha512-nZXjEF2nbo7lIw3mgYjItAfgQXog3OjJogSbKa2CQIIvSGWcKgeJnQlNXip6NglNzYH45nSRiEVimMvYL8DDqQ==", - "dev": true, - "engines": { - "node": ">=14.18.0" - } - }, - "node_modules/default-browser/node_modules/is-stream": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-3.0.0.tgz", - "integrity": "sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA==", - "dev": true, - "engines": { - "node": "^12.20.0 || ^14.13.1 || >=16.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/default-browser/node_modules/mimic-fn": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-4.0.0.tgz", - "integrity": "sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw==", - "dev": true, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/default-browser/node_modules/npm-run-path": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-5.2.0.tgz", - "integrity": "sha512-W4/tgAXFqFA0iL7fk0+uQ3g7wkL8xJmx3XdK0VGb4cHW//eZTtKGvFBBoRKVTpY7n6ze4NL9ly7rgXcHufqXKg==", - "dev": true, - "dependencies": { - "path-key": "^4.0.0" - }, - "engines": { - "node": "^12.20.0 || ^14.13.1 || >=16.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/default-browser/node_modules/onetime": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/onetime/-/onetime-6.0.0.tgz", - "integrity": "sha512-1FlR+gjXK7X+AsAHso35MnyN5KqGwJRi/31ft6x0M194ht7S+rWAvd7PHss9xSKMzE0asv1pyIHaJYq+BbacAQ==", + "node_modules/define-data-property": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", + "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", "dev": true, + "license": "MIT", "dependencies": { - "mimic-fn": "^4.0.0" - }, - "engines": { - "node": ">=12" + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/default-browser/node_modules/path-key": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-4.0.0.tgz", - "integrity": "sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ==", - "dev": true, "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/default-browser/node_modules/strip-final-newline": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-3.0.0.tgz", - "integrity": "sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw==", - "dev": true, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/define-lazy-prop": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/define-lazy-prop/-/define-lazy-prop-3.0.0.tgz", - "integrity": "sha512-N+MeXYoqr3pOgn8xfyRPREN7gHakLYjhsHhWGT3fWAiL4IkAt0iDw14QiiEm2bE30c5XX5q0FtAA3CK5f9/BUg==", - "dev": true, - "engines": { - "node": ">=12" + "node": ">= 0.4" }, "funding": { - "url": "https://github.com/sponsors/sindresorhus" + "url": "https://github.com/sponsors/ljharb" } }, "node_modules/define-properties": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.0.tgz", - "integrity": "sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA==", + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", + "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", "dev": true, + "license": "MIT", "dependencies": { + "define-data-property": "^1.0.1", "has-property-descriptors": "^1.0.0", "object-keys": "^1.1.1" }, @@ -2935,18 +2750,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/doctrine": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", - "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", - "dev": true, - "dependencies": { - "esutils": "^2.0.2" - }, - "engines": { - "node": ">=6.0.0" - } - }, "node_modules/electron-to-chromium": { "version": "1.4.528", "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.528.tgz", @@ -2995,97 +2798,27 @@ "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", "dev": true }, - "node_modules/es-abstract": { - "version": "1.22.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.22.1.tgz", - "integrity": "sha512-ioRRcXMO6OFyRpyzV3kE1IIBd4WG5/kltnzdxSCqoP8CMGs/Li+M1uF5o7lOkZVFjDs+NLesthnF66Pg/0q0Lw==", - "dev": true, - "dependencies": { - "array-buffer-byte-length": "^1.0.0", - "arraybuffer.prototype.slice": "^1.0.1", - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", - "es-set-tostringtag": "^2.0.1", - "es-to-primitive": "^1.2.1", - "function.prototype.name": "^1.1.5", - "get-intrinsic": "^1.2.1", - "get-symbol-description": "^1.0.0", - "globalthis": "^1.0.3", - "gopd": "^1.0.1", - "has": "^1.0.3", - "has-property-descriptors": "^1.0.0", - "has-proto": "^1.0.1", - "has-symbols": "^1.0.3", - "internal-slot": "^1.0.5", - "is-array-buffer": "^3.0.2", - "is-callable": "^1.2.7", - "is-negative-zero": "^2.0.2", - "is-regex": "^1.1.4", - "is-shared-array-buffer": "^1.0.2", - "is-string": "^1.0.7", - "is-typed-array": "^1.1.10", - "is-weakref": "^1.0.2", - "object-inspect": "^1.12.3", - "object-keys": "^1.1.1", - "object.assign": "^4.1.4", - "regexp.prototype.flags": "^1.5.0", - "safe-array-concat": "^1.0.0", - "safe-regex-test": "^1.0.0", - "string.prototype.trim": "^1.2.7", - "string.prototype.trimend": "^1.0.6", - "string.prototype.trimstart": "^1.0.6", - "typed-array-buffer": "^1.0.0", - "typed-array-byte-length": "^1.0.0", - "typed-array-byte-offset": "^1.0.0", - "typed-array-length": "^1.0.4", - "unbox-primitive": "^1.0.2", - "which-typed-array": "^1.1.10" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/es-set-tostringtag": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz", - "integrity": "sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==", + "node_modules/es-define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", + "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==", "dev": true, + "license": "MIT", "dependencies": { - "get-intrinsic": "^1.1.3", - "has": "^1.0.3", - "has-tostringtag": "^1.0.0" + "get-intrinsic": "^1.2.4" }, "engines": { "node": ">= 0.4" } }, - "node_modules/es-shim-unscopables": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.0.2.tgz", - "integrity": "sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw==", - "dev": true, - "dependencies": { - "hasown": "^2.0.0" - } - }, - "node_modules/es-to-primitive": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", - "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", "dev": true, - "dependencies": { - "is-callable": "^1.1.4", - "is-date-object": "^1.0.1", - "is-symbol": "^1.0.2" - }, + "license": "MIT", "engines": { "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" } }, "node_modules/es6-error": { @@ -3095,40 +2828,42 @@ "dev": true }, "node_modules/esbuild": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.4.tgz", - "integrity": "sha512-sFMcNNrj+Q0ZDolrp5pDhH0nRPN9hLIM3fRPwgbLYJeSHHgnXSnbV3xYgSVuOeLWH9c73VwmEverVzupIv5xuA==", + "version": "0.24.0", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.24.0.tgz", + "integrity": "sha512-FuLPevChGDshgSicjisSooU0cemp/sGXR841D5LHMB7mTVOmsEHcAxaH3irL53+8YDIeVNQEySh4DaYU/iuPqQ==", "hasInstallScript": true, + "license": "MIT", "bin": { "esbuild": "bin/esbuild" }, "engines": { - "node": ">=12" + "node": ">=18" }, "optionalDependencies": { - "@esbuild/aix-ppc64": "0.21.4", - "@esbuild/android-arm": "0.21.4", - "@esbuild/android-arm64": "0.21.4", - "@esbuild/android-x64": "0.21.4", - "@esbuild/darwin-arm64": "0.21.4", - "@esbuild/darwin-x64": "0.21.4", - "@esbuild/freebsd-arm64": "0.21.4", - "@esbuild/freebsd-x64": "0.21.4", - "@esbuild/linux-arm": "0.21.4", - "@esbuild/linux-arm64": "0.21.4", - "@esbuild/linux-ia32": "0.21.4", - "@esbuild/linux-loong64": "0.21.4", - "@esbuild/linux-mips64el": "0.21.4", - "@esbuild/linux-ppc64": "0.21.4", - "@esbuild/linux-riscv64": "0.21.4", - "@esbuild/linux-s390x": "0.21.4", - "@esbuild/linux-x64": "0.21.4", - "@esbuild/netbsd-x64": "0.21.4", - "@esbuild/openbsd-x64": "0.21.4", - "@esbuild/sunos-x64": "0.21.4", - "@esbuild/win32-arm64": "0.21.4", - "@esbuild/win32-ia32": "0.21.4", - "@esbuild/win32-x64": "0.21.4" + "@esbuild/aix-ppc64": "0.24.0", + "@esbuild/android-arm": "0.24.0", + "@esbuild/android-arm64": "0.24.0", + "@esbuild/android-x64": "0.24.0", + "@esbuild/darwin-arm64": "0.24.0", + "@esbuild/darwin-x64": "0.24.0", + "@esbuild/freebsd-arm64": "0.24.0", + "@esbuild/freebsd-x64": "0.24.0", + "@esbuild/linux-arm": "0.24.0", + "@esbuild/linux-arm64": "0.24.0", + "@esbuild/linux-ia32": "0.24.0", + "@esbuild/linux-loong64": "0.24.0", + "@esbuild/linux-mips64el": "0.24.0", + "@esbuild/linux-ppc64": "0.24.0", + "@esbuild/linux-riscv64": "0.24.0", + "@esbuild/linux-s390x": "0.24.0", + "@esbuild/linux-x64": "0.24.0", + "@esbuild/netbsd-x64": "0.24.0", + "@esbuild/openbsd-arm64": "0.24.0", + "@esbuild/openbsd-x64": "0.24.0", + "@esbuild/sunos-x64": "0.24.0", + "@esbuild/win32-arm64": "0.24.0", + "@esbuild/win32-ia32": "0.24.0", + "@esbuild/win32-x64": "0.24.0" } }, "node_modules/escalade": { @@ -3153,77 +2888,65 @@ } }, "node_modules/eslint": { - "version": "8.57.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.0.tgz", - "integrity": "sha512-dZ6+mexnaTIbSBZWgou51U6OmzIhYM2VcNdtiTtI7qPNZm35Akpr0f6vtw3w1Kmn5PYo+tZVfh13WrhpS6oLqQ==", + "version": "9.13.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.13.0.tgz", + "integrity": "sha512-EYZK6SX6zjFHST/HRytOdA/zE72Cq/bfw45LSyuwrdvcclb/gqV8RRQxywOBEWO2+WDpva6UZa4CcDeJKzUCFA==", "dev": true, + "license": "MIT", + "peer": true, "dependencies": { "@eslint-community/eslint-utils": "^4.2.0", - "@eslint-community/regexpp": "^4.6.1", - "@eslint/eslintrc": "^2.1.4", - "@eslint/js": "8.57.0", - "@humanwhocodes/config-array": "^0.11.14", + "@eslint-community/regexpp": "^4.11.0", + "@eslint/config-array": "^0.18.0", + "@eslint/core": "^0.7.0", + "@eslint/eslintrc": "^3.1.0", + "@eslint/js": "9.13.0", + "@eslint/plugin-kit": "^0.2.0", + "@humanfs/node": "^0.16.5", "@humanwhocodes/module-importer": "^1.0.1", - "@nodelib/fs.walk": "^1.2.8", - "@ungap/structured-clone": "^1.2.0", + "@humanwhocodes/retry": "^0.3.1", + "@types/estree": "^1.0.6", + "@types/json-schema": "^7.0.15", "ajv": "^6.12.4", "chalk": "^4.0.0", "cross-spawn": "^7.0.2", "debug": "^4.3.2", - "doctrine": "^3.0.0", "escape-string-regexp": "^4.0.0", - "eslint-scope": "^7.2.2", - "eslint-visitor-keys": "^3.4.3", - "espree": "^9.6.1", - "esquery": "^1.4.2", + "eslint-scope": "^8.1.0", + "eslint-visitor-keys": "^4.1.0", + "espree": "^10.2.0", + "esquery": "^1.5.0", "esutils": "^2.0.2", "fast-deep-equal": "^3.1.3", - "file-entry-cache": "^6.0.1", + "file-entry-cache": "^8.0.0", "find-up": "^5.0.0", "glob-parent": "^6.0.2", - "globals": "^13.19.0", - "graphemer": "^1.4.0", "ignore": "^5.2.0", "imurmurhash": "^0.1.4", "is-glob": "^4.0.0", - "is-path-inside": "^3.0.3", - "js-yaml": "^4.1.0", "json-stable-stringify-without-jsonify": "^1.0.1", - "levn": "^0.4.1", "lodash.merge": "^4.6.2", "minimatch": "^3.1.2", "natural-compare": "^1.4.0", "optionator": "^0.9.3", - "strip-ansi": "^6.0.1", "text-table": "^0.2.0" }, "bin": { "eslint": "bin/eslint.js" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/eslint-config-airbnb-base": { - "version": "15.0.0", - "resolved": "https://registry.npmjs.org/eslint-config-airbnb-base/-/eslint-config-airbnb-base-15.0.0.tgz", - "integrity": "sha512-xaX3z4ZZIcFLvh2oUNvcX5oEofXda7giYmuplVxoOg5A7EXJMrUyqRgR+mhDhPK8LZ4PttFOBvCYDbX3sUoUig==", - "dev": true, - "dependencies": { - "confusing-browser-globals": "^1.0.10", - "object.assign": "^4.1.2", - "object.entries": "^1.1.5", - "semver": "^6.3.0" - }, - "engines": { - "node": "^10.12.0 || >=12.0.0" + "url": "https://eslint.org/donate" }, "peerDependencies": { - "eslint": "^7.32.0 || ^8.2.0", - "eslint-plugin-import": "^2.25.2" + "jiti": "*" + }, + "peerDependenciesMeta": { + "jiti": { + "optional": true + } } }, "node_modules/eslint-config-prettier": { @@ -3238,112 +2961,15 @@ "eslint": ">=7.0.0" } }, - "node_modules/eslint-import-resolver-node": { - "version": "0.3.9", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz", - "integrity": "sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==", - "dev": true, - "dependencies": { - "debug": "^3.2.7", - "is-core-module": "^2.13.0", - "resolve": "^1.22.4" - } - }, - "node_modules/eslint-import-resolver-node/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-module-utils": { - "version": "2.8.0", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.8.0.tgz", - "integrity": "sha512-aWajIYfsqCKRDgUfjEXNN/JlrzauMuSEy5sbd7WXbtW3EH6A6MpwEh42c7qD+MqQo9QMJ6fWLAeIJynx0g6OAw==", - "dev": true, - "dependencies": { - "debug": "^3.2.7" - }, - "engines": { - "node": ">=4" - }, - "peerDependenciesMeta": { - "eslint": { - "optional": true - } - } - }, - "node_modules/eslint-module-utils/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-plugin-import": { - "version": "2.29.1", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.29.1.tgz", - "integrity": "sha512-BbPC0cuExzhiMo4Ff1BTVwHpjjv28C5R+btTOGaCRC7UEz801up0JadwkeSk5Ued6TG34uaczuVuH6qyy5YUxw==", - "dev": true, - "dependencies": { - "array-includes": "^3.1.7", - "array.prototype.findlastindex": "^1.2.3", - "array.prototype.flat": "^1.3.2", - "array.prototype.flatmap": "^1.3.2", - "debug": "^3.2.7", - "doctrine": "^2.1.0", - "eslint-import-resolver-node": "^0.3.9", - "eslint-module-utils": "^2.8.0", - "hasown": "^2.0.0", - "is-core-module": "^2.13.1", - "is-glob": "^4.0.3", - "minimatch": "^3.1.2", - "object.fromentries": "^2.0.7", - "object.groupby": "^1.0.1", - "object.values": "^1.1.7", - "semver": "^6.3.1", - "tsconfig-paths": "^3.15.0" - }, - "engines": { - "node": ">=4" - }, - "peerDependencies": { - "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8" - } - }, - "node_modules/eslint-plugin-import/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-plugin-import/node_modules/doctrine": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", - "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", - "dev": true, - "dependencies": { - "esutils": "^2.0.2" - }, - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/eslint-plugin-prettier": { - "version": "5.1.3", - "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.1.3.tgz", - "integrity": "sha512-C9GCVAs4Eq7ZC/XFQHITLiHJxQngdtraXaM+LoUFoFp/lHNl2Zn8f3WQbe9HvTBBQ9YnKFB0/2Ajdqwo5D1EAw==", + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.2.1.tgz", + "integrity": "sha512-gH3iR3g4JfF+yYPaJYkN7jEl9QbweL/YfkoRlNnuIEHEz1vHVlCmWOS+eGGiRuzHQXdJFCOTxRgvju9b8VUmrw==", "dev": true, + "license": "MIT", "dependencies": { "prettier-linter-helpers": "^1.0.0", - "synckit": "^0.8.6" + "synckit": "^0.9.1" }, "engines": { "node": "^14.18.0 || >=16.0.0" @@ -3367,16 +2993,18 @@ } }, "node_modules/eslint-scope": { - "version": "7.2.2", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", - "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.1.0.tgz", + "integrity": "sha512-14dSvlhaVhKKsa9Fx1l8A17s7ah7Ef7wCakJ10LYk6+GYmP9yDti2oq2SEwcyndt6knfcZyhyxwY3i9yL78EQw==", "dev": true, + "license": "BSD-2-Clause", + "peer": true, "dependencies": { "esrecurse": "^4.3.0", "estraverse": "^5.2.0" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" @@ -3387,6 +3015,7 @@ "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", "dev": true, + "peer": true, "engines": { "node": "^12.22.0 || ^14.17.0 || >=16.0.0" }, @@ -3399,6 +3028,7 @@ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", "dev": true, + "peer": true, "dependencies": { "fast-deep-equal": "^3.1.1", "fast-json-stable-stringify": "^2.0.0", @@ -3410,11 +3040,26 @@ "url": "https://github.com/sponsors/epoberezkin" } }, + "node_modules/eslint/node_modules/eslint-visitor-keys": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.1.0.tgz", + "integrity": "sha512-Q7lok0mqMUSf5a/AdAZkA5a/gHcO6snwQClVNNvFKCAVlxXucdU8pKydU5ZVZjBx5xr37vGbFFWtLQYreLzrZg==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, "node_modules/eslint/node_modules/find-up": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", "dev": true, + "peer": true, "dependencies": { "locate-path": "^6.0.0", "path-exists": "^4.0.0" @@ -3430,13 +3075,15 @@ "version": "0.4.1", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true + "dev": true, + "peer": true }, "node_modules/eslint/node_modules/locate-path": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", "dev": true, + "peer": true, "dependencies": { "p-locate": "^5.0.0" }, @@ -3452,6 +3099,7 @@ "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", "dev": true, + "peer": true, "dependencies": { "p-limit": "^3.0.2" }, @@ -3463,17 +3111,33 @@ } }, "node_modules/espree": { - "version": "9.6.1", - "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", - "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "version": "10.2.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.2.0.tgz", + "integrity": "sha512-upbkBJbckcCNBDBDXEbuhjbP68n+scUd3k/U2EkyM9nw+I/jPiL4cLF/Al06CF96wRltFda16sxDFrxsI1v0/g==", "dev": true, + "license": "BSD-2-Clause", + "peer": true, "dependencies": { - "acorn": "^8.9.0", + "acorn": "^8.12.0", "acorn-jsx": "^5.3.2", - "eslint-visitor-keys": "^3.4.1" + "eslint-visitor-keys": "^4.1.0" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree/node_modules/eslint-visitor-keys": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.1.0.tgz", + "integrity": "sha512-Q7lok0mqMUSf5a/AdAZkA5a/gHcO6snwQClVNNvFKCAVlxXucdU8pKydU5ZVZjBx5xr37vGbFFWtLQYreLzrZg==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "url": "https://opencollective.com/eslint" @@ -3497,6 +3161,7 @@ "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", "dev": true, + "peer": true, "dependencies": { "estraverse": "^5.1.0" }, @@ -3530,6 +3195,7 @@ "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", "dev": true, + "peer": true, "engines": { "node": ">=0.10.0" } @@ -3598,34 +3264,6 @@ "integrity": "sha512-xJuoT5+L99XlZ8twedaRf6Ax2TgQVxvgZOYoPKqZufmJib0tL2tegPBOZb1pVNgIhlqDlA0eO0c3wBvQcmzx4w==", "dev": true }, - "node_modules/fast-glob": { - "version": "3.3.2", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz", - "integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==", - "dev": true, - "dependencies": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.2", - "merge2": "^1.3.0", - "micromatch": "^4.0.4" - }, - "engines": { - "node": ">=8.6.0" - } - }, - "node_modules/fast-glob/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, "node_modules/fast-json-stable-stringify": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", @@ -3636,16 +3274,8 @@ "version": "2.0.6", "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", - "dev": true - }, - "node_modules/fastq": { - "version": "1.13.0", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.13.0.tgz", - "integrity": "sha512-YpkpUnK8od0o1hmeSc7UUs/eB/vIPWJYjKck2QKIzAf71Vm1AAQ3EbuZB3g2JIy+pg+ERD0vqI79KyZiB2e2Nw==", "dev": true, - "dependencies": { - "reusify": "^1.0.4" - } + "peer": true }, "node_modules/fb-watchman": { "version": "2.0.2", @@ -3662,15 +3292,17 @@ "integrity": "sha512-OP2IUU6HeYKJi3i0z4A19kHMQoLVs4Hc+DPqqxI2h/DPZHTm/vjsfC6P0b4jCMy14XizLBqvndQ+UilD7707Jw==" }, "node_modules/file-entry-cache": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", - "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", "dev": true, + "license": "MIT", + "peer": true, "dependencies": { - "flat-cache": "^3.0.4" + "flat-cache": "^4.0.0" }, "engines": { - "node": "^10.12.0 || >=12.0.0" + "node": ">=16.0.0" } }, "node_modules/file-stream-rotator": { @@ -3682,10 +3314,11 @@ } }, "node_modules/fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", "dev": true, + "license": "MIT", "dependencies": { "to-regex-range": "^5.0.1" }, @@ -3707,23 +3340,27 @@ } }, "node_modules/flat-cache": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", - "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", + "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", "dev": true, + "license": "MIT", + "peer": true, "dependencies": { - "flatted": "^3.1.0", - "rimraf": "^3.0.2" + "flatted": "^3.2.9", + "keyv": "^4.5.4" }, "engines": { - "node": "^10.12.0 || >=12.0.0" + "node": ">=16" } }, "node_modules/flatted": { - "version": "3.2.5", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.5.tgz", - "integrity": "sha512-WIWGi2L3DyTUvUrwRKgGi9TwxQMUEqPOPQBVi71R96jZXJdFskXEmf54BoZaS1kknGODoIGASGEzBUYdyMCBJg==", - "dev": true + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz", + "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", + "dev": true, + "license": "ISC", + "peer": true }, "node_modules/fn.name": { "version": "1.1.0", @@ -3749,19 +3386,11 @@ } } }, - "node_modules/for-each": { - "version": "0.3.3", - "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", - "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==", - "dev": true, - "dependencies": { - "is-callable": "^1.1.3" - } - }, "node_modules/form-data": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", - "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.1.tgz", + "integrity": "sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==", + "license": "MIT", "dependencies": { "asynckit": "^0.4.0", "combined-stream": "^1.0.8", @@ -3813,33 +3442,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/function.prototype.name": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz", - "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.3", - "es-abstract": "^1.19.0", - "functions-have-names": "^1.2.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/functions-have-names": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", - "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", - "dev": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/gensync": { "version": "1.0.0-beta.2", "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", @@ -3859,15 +3461,20 @@ } }, "node_modules/get-intrinsic": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.1.tgz", - "integrity": "sha512-2DcsyfABl+gVHEfCOaTrWgyt+tb6MSEGmKq+kI5HwLbIYgjgmMcV8KQ41uaKz1xxUcn9tJtgFbQUEVcEbd0FYw==", + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", + "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", "dev": true, + "license": "MIT", "dependencies": { - "function-bind": "^1.1.1", - "has": "^1.0.3", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", "has-proto": "^1.0.1", - "has-symbols": "^1.0.3" + "has-symbols": "^1.0.3", + "hasown": "^2.0.0" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -3894,22 +3501,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/get-symbol-description": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz", - "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/glob": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", @@ -3935,6 +3526,7 @@ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", "dev": true, + "peer": true, "dependencies": { "is-glob": "^4.0.3" }, @@ -3975,15 +3567,14 @@ } }, "node_modules/globals": { - "version": "13.24.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", - "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "version": "14.0.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", + "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", "dev": true, - "dependencies": { - "type-fest": "^0.20.2" - }, + "license": "MIT", + "peer": true, "engines": { - "node": ">=8" + "node": ">=18" }, "funding": { "url": "https://github.com/sponsors/sindresorhus" @@ -4009,6 +3600,7 @@ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz", "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==", "dev": true, + "license": "MIT", "dependencies": { "get-intrinsic": "^1.1.3" }, @@ -4021,12 +3613,6 @@ "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==" }, - "node_modules/graphemer": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", - "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", - "dev": true - }, "node_modules/handlebars": { "version": "4.7.8", "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.8.tgz", @@ -4047,27 +3633,6 @@ "uglify-js": "^3.1.4" } }, - "node_modules/has": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", - "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", - "dev": true, - "dependencies": { - "function-bind": "^1.1.1" - }, - "engines": { - "node": ">= 0.4.0" - } - }, - "node_modules/has-bigints": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", - "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==", - "dev": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", @@ -4078,22 +3643,24 @@ } }, "node_modules/has-property-descriptors": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz", - "integrity": "sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", + "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", "dev": true, + "license": "MIT", "dependencies": { - "get-intrinsic": "^1.1.1" + "es-define-property": "^1.0.0" }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, "node_modules/has-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz", - "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz", + "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==", "dev": true, + "license": "MIT", "engines": { "node": ">= 0.4" }, @@ -4113,26 +3680,12 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/has-tostringtag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz", - "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==", - "dev": true, - "dependencies": { - "has-symbols": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/hasown": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.0.tgz", - "integrity": "sha512-vUptKVTpIJhcczKBbgnS+RtcuYMB8+oNzPK2/Hp3hanz8JmpATdmmgLgSaadVREkDm+e2giHwY3ZRkyjSIDDFA==", + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", "dev": true, + "license": "MIT", "dependencies": { "function-bind": "^1.1.2" }, @@ -4146,19 +3699,6 @@ "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", "dev": true }, - "node_modules/https-proxy-agent": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz", - "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==", - "dev": true, - "dependencies": { - "agent-base": "6", - "debug": "4" - }, - "engines": { - "node": ">= 6" - } - }, "node_modules/human-signals": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", @@ -4169,19 +3709,30 @@ } }, "node_modules/ignore": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.0.tgz", - "integrity": "sha512-g7dmpshy+gD7mh88OC9NwSGTKoc3kyLAZQRU1mt53Aw/vnvfXnbC+F/7F7QoYVKbV+KNvJx8wArewKy1vXMtlg==", + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", "dev": true, + "license": "MIT", + "peer": true, "engines": { "node": ">= 4" } }, + "node_modules/immediate": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", + "integrity": "sha512-XXOFtyqDjNDAQxVfYxuF7g9Il/IbWmmlQg2MYKOH8ExIT1qg6xc4zyS3HaEEATgs1btfzxq15ciUiY7gjSXRGQ==", + "dev": true, + "license": "MIT" + }, "node_modules/import-fresh": { "version": "3.3.0", "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", "dev": true, + "license": "MIT", + "peer": true, "dependencies": { "parent-module": "^1.0.0", "resolve-from": "^4.0.0" @@ -4236,98 +3787,19 @@ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" }, - "node_modules/internal-slot": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.5.tgz", - "integrity": "sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ==", - "dev": true, - "dependencies": { - "get-intrinsic": "^1.2.0", - "has": "^1.0.3", - "side-channel": "^1.0.4" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/is-array-buffer": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.2.tgz", - "integrity": "sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.2.0", - "is-typed-array": "^1.1.10" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-arrayish": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==" }, - "node_modules/is-bigint": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", - "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==", - "dev": true, - "dependencies": { - "has-bigints": "^1.0.1" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-boolean-object": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz", - "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "has-tostringtag": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-callable": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", - "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", - "dev": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-core-module": { - "version": "2.13.1", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.13.1.tgz", - "integrity": "sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw==", - "dev": true, - "dependencies": { - "hasown": "^2.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-date-object": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz", - "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==", + "node_modules/is-core-module": { + "version": "2.15.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.15.1.tgz", + "integrity": "sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==", "dev": true, + "license": "MIT", "dependencies": { - "has-tostringtag": "^1.0.0" + "hasown": "^2.0.2" }, "engines": { "node": ">= 0.4" @@ -4336,26 +3808,12 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/is-docker": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-3.0.0.tgz", - "integrity": "sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ==", - "dev": true, - "bin": { - "is-docker": "cli.js" - }, - "engines": { - "node": "^12.20.0 || ^14.13.1 || >=16.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, "node_modules/is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", "dev": true, + "peer": true, "engines": { "node": ">=0.10.0" } @@ -4382,6 +3840,7 @@ "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", "dev": true, + "peer": true, "dependencies": { "is-extglob": "^2.1.1" }, @@ -4389,97 +3848,16 @@ "node": ">=0.10.0" } }, - "node_modules/is-inside-container": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-inside-container/-/is-inside-container-1.0.0.tgz", - "integrity": "sha512-KIYLCCJghfHZxqjYBE7rEy0OBuTd5xCHS7tHVgvCLkx7StIoaxwNW3hCALgEUjFfeRk+MG/Qxmp/vtETEF3tRA==", - "dev": true, - "dependencies": { - "is-docker": "^3.0.0" - }, - "bin": { - "is-inside-container": "cli.js" - }, - "engines": { - "node": ">=14.16" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/is-negative-zero": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", - "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", - "dev": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", "dev": true, + "license": "MIT", "engines": { "node": ">=0.12.0" } }, - "node_modules/is-number-object": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.7.tgz", - "integrity": "sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==", - "dev": true, - "dependencies": { - "has-tostringtag": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-path-inside": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", - "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", - "dev": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/is-regex": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", - "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "has-tostringtag": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-shared-array-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", - "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-stream": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", @@ -4488,100 +3866,6 @@ "node": ">=8" } }, - "node_modules/is-string": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", - "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", - "dev": true, - "dependencies": { - "has-tostringtag": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-symbol": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", - "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", - "dev": true, - "dependencies": { - "has-symbols": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-typed-array": { - "version": "1.1.10", - "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.10.tgz", - "integrity": "sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A==", - "dev": true, - "dependencies": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "has-tostringtag": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakref": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", - "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-wsl": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", - "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", - "dev": true, - "dependencies": { - "is-docker": "^2.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/is-wsl/node_modules/is-docker": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz", - "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", - "dev": true, - "bin": { - "is-docker": "cli.js" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", - "dev": true - }, "node_modules/isexe": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", @@ -5259,6 +4543,8 @@ "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", "dev": true, + "license": "MIT", + "peer": true, "dependencies": { "argparse": "^2.0.1" }, @@ -5278,6 +4564,14 @@ "node": ">=4" } }, + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true, + "license": "MIT", + "peer": true + }, "node_modules/json-parse-even-better-errors": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", @@ -5293,7 +4587,8 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", - "dev": true + "dev": true, + "peer": true }, "node_modules/json-stringify-safe": { "version": "5.0.1", @@ -5301,18 +4596,6 @@ "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==", "dev": true }, - "node_modules/json5": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", - "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", - "dev": true, - "dependencies": { - "minimist": "^1.2.0" - }, - "bin": { - "json5": "lib/cli.js" - } - }, "node_modules/jsonfile": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", @@ -5324,6 +4607,17 @@ "graceful-fs": "^4.1.6" } }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "json-buffer": "3.0.1" + } + }, "node_modules/kleur": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", @@ -5352,6 +4646,7 @@ "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", "dev": true, + "peer": true, "dependencies": { "prelude-ls": "^1.2.1", "type-check": "~0.4.0" @@ -5360,12 +4655,32 @@ "node": ">= 0.8.0" } }, + "node_modules/lie": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/lie/-/lie-3.1.1.tgz", + "integrity": "sha512-RiNhHysUjhrDQntfYSfY4MU24coXXdEOgw9WGcKHNeEwffDYbF//u87M1EWaMGzuFoSbqW0C9C6lEEhDOAswfw==", + "dev": true, + "license": "MIT", + "dependencies": { + "immediate": "~3.0.5" + } + }, "node_modules/lines-and-columns": { "version": "1.2.4", "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", "dev": true }, + "node_modules/localforage": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/localforage/-/localforage-1.10.0.tgz", + "integrity": "sha512-14/H1aX7hzBBmmh7sGPd+AOMkkIrHM3Z1PAyGgZigA1H1p5O5ANnMyWzvpAETtG68/dC4pC0ncy3+PPGzXZHPg==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "lie": "3.1.1" + } + }, "node_modules/locate-path": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", @@ -5378,11 +4693,18 @@ "node": ">=8" } }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "license": "MIT" + }, "node_modules/lodash.merge": { "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true + "dev": true, + "peer": true }, "node_modules/lodash.truncate": { "version": "4.4.2", @@ -5390,23 +4712,22 @@ "integrity": "sha1-WjUNoLERO4N+z//VgSy+WNbq4ZM=" }, "node_modules/logform": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/logform/-/logform-2.4.0.tgz", - "integrity": "sha512-CPSJw4ftjf517EhXZGGvTHHkYobo7ZCc0kvwUoOYcjfR2UVrI66RHj8MCrfAdEitdmFqbu2BYdYs8FHHZSb6iw==", + "version": "2.6.1", + "resolved": "https://registry.npmjs.org/logform/-/logform-2.6.1.tgz", + "integrity": "sha512-CdaO738xRapbKIMVn2m4F6KTj4j7ooJ8POVnebSgKo3KBz5axNXRAL7ZdRjIV6NOr2Uf4vjtRkxrFETOioCqSA==", + "license": "MIT", "dependencies": { - "@colors/colors": "1.5.0", + "@colors/colors": "1.6.0", + "@types/triple-beam": "^1.3.2", "fecha": "^4.2.0", "ms": "^2.1.1", "safe-stable-stringify": "^2.3.1", "triple-beam": "^1.3.0" + }, + "engines": { + "node": ">= 12.0.0" } }, - "node_modules/lru_map": { - "version": "0.3.3", - "resolved": "https://registry.npmjs.org/lru_map/-/lru_map-0.3.3.tgz", - "integrity": "sha512-Pn9cox5CsMYngeDbmChANltQl+5pi6XmTrraMSzhPmMBbmgcxmqWry0U3PGapCU1yB4/LqCcom7qhHZiF/jGfQ==", - "dev": true - }, "node_modules/lru-cache": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", @@ -5420,9 +4741,10 @@ } }, "node_modules/luxon": { - "version": "3.4.4", - "resolved": "https://registry.npmjs.org/luxon/-/luxon-3.4.4.tgz", - "integrity": "sha512-zobTr7akeGHnv7eBOXcRgMeCP6+uyYsczwmeRCauvpvaAltgNyTbLH/+VaEAPUeWBT+1GuNmz4wC/6jtQzbbVA==", + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/luxon/-/luxon-3.5.0.tgz", + "integrity": "sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==", + "license": "MIT", "engines": { "node": ">=12" } @@ -5484,22 +4806,14 @@ "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", "dev": true }, - "node_modules/merge2": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", - "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "dev": true, - "engines": { - "node": ">= 8" - } - }, "node_modules/micromatch": { - "version": "4.0.5", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", - "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", "dev": true, + "license": "MIT", "dependencies": { - "braces": "^3.0.2", + "braces": "^3.0.3", "picomatch": "^2.3.1" }, "engines": { @@ -5567,6 +4881,24 @@ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" }, + "node_modules/nanoid": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-5.0.8.tgz", + "integrity": "sha512-TcJPw+9RV9dibz1hHUzlLVy8N4X9TnwirAjrU08Juo6BNKggzVfP2ZJ/3ZUSq15Xl5i85i+Z89XBO90pB2PghQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "bin": { + "nanoid": "bin/nanoid.js" + }, + "engines": { + "node": "^18 || >=20" + } + }, "node_modules/natural-compare": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", @@ -5633,15 +4965,6 @@ "node": ">= 6" } }, - "node_modules/object-inspect": { - "version": "1.12.3", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.3.tgz", - "integrity": "sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==", - "dev": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/object-keys": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", @@ -5651,84 +4974,6 @@ "node": ">= 0.4" } }, - "node_modules/object.assign": { - "version": "4.1.4", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", - "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "has-symbols": "^1.0.3", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object.entries": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.5.tgz", - "integrity": "sha512-TyxmjUoZggd4OrrU1W66FMDG6CuqJxsFvymeyXI51+vQLN67zYfZseptRge703kKQdo4uccgAKebXFcRCzk4+g==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.3", - "es-abstract": "^1.19.1" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object.fromentries": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.7.tgz", - "integrity": "sha512-UPbPHML6sL8PI/mOqPwsH4G6iyXcCGzLin8KvEPenOZN5lpCNBZZQ+V62vdjB1mQHrmqGQt5/OJzemUA+KJmEA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object.groupby": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.1.tgz", - "integrity": "sha512-HqaQtqLnp/8Bn4GL16cj+CUYbnpe1bh0TtEaWvybszDG4tgxCJuRpV8VGuvNaI1fAnI4lUJzDG55MXcOH4JZcQ==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1", - "get-intrinsic": "^1.2.1" - } - }, - "node_modules/object.values": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.7.tgz", - "integrity": "sha512-aU6xnDFYT3x17e/f0IiiwlGPTy2jzMySGfUB4fq6z7CV8l85CWHDk5ErhyhpfDHhrOMwGFhSQkhMGHaIotA6Ng==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/once": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", @@ -5748,32 +4993,14 @@ }, "node_modules/onetime": { "version": "5.1.2", - "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", - "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", - "dev": true, - "dependencies": { - "mimic-fn": "^2.1.0" - }, - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/open": { - "version": "9.1.0", - "resolved": "https://registry.npmjs.org/open/-/open-9.1.0.tgz", - "integrity": "sha512-OS+QTnw1/4vrf+9hh1jc1jnYjzSG4ttTBB8UxOwAnInG3Uo4ssetzC1ihqaIHjLJnA5GGlRl6QlZXOTQhRBUvg==", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", "dev": true, "dependencies": { - "default-browser": "^4.0.0", - "define-lazy-prop": "^3.0.0", - "is-inside-container": "^1.0.0", - "is-wsl": "^2.2.0" + "mimic-fn": "^2.1.0" }, "engines": { - "node": ">=14.16" + "node": ">=6" }, "funding": { "url": "https://github.com/sponsors/sindresorhus" @@ -5784,6 +5011,7 @@ "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==", "dev": true, + "peer": true, "dependencies": { "@aashutoshrathi/word-wrap": "^1.2.3", "deep-is": "^0.1.3", @@ -5852,6 +5080,8 @@ "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", "dev": true, + "license": "MIT", + "peer": true, "dependencies": { "callsites": "^3.0.0" }, @@ -5954,15 +5184,17 @@ "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", "dev": true, + "peer": true, "engines": { "node": ">= 0.8.0" } }, "node_modules/prettier": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.3.0.tgz", - "integrity": "sha512-J9odKxERhCQ10OC2yb93583f6UnYutOeiV5i0zEDS7UGTdUt0u+y8erxl3lBKvwo/JHyyoEdXjwp4dke9oyZ/g==", + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.3.3.tgz", + "integrity": "sha512-i2tDNA0O5IrMO757lfrdQZCc2jPNDVntV0m/+4whiDfWaTKfMNgR7Qz0NAeGz/nRqF4m5/6CLzbP4/liHt12Ew==", "dev": true, + "license": "MIT", "bin": { "prettier": "bin/prettier.cjs" }, @@ -6062,26 +5294,6 @@ "extend": "^3.0.2" } }, - "node_modules/queue-microtask": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", - "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ] - }, "node_modules/random-words": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/random-words/-/random-words-2.0.1.tgz", @@ -6109,23 +5321,6 @@ "node": ">= 6" } }, - "node_modules/regexp.prototype.flags": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.0.tgz", - "integrity": "sha512-0SutC3pNudRKgquxGoRGIz946MZVHqbNfPjBdxeOhBrdgDKlRoXmYLQN9xRbrR09ZXWeGAdPuif7egofn6v5LA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "functions-have-names": "^1.2.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", @@ -6186,6 +5381,8 @@ "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", "dev": true, + "license": "MIT", + "peer": true, "engines": { "node": ">=4" } @@ -6210,31 +5407,6 @@ "axios": "*" } }, - "node_modules/reusify": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", - "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", - "dev": true, - "engines": { - "iojs": ">=1.0.0", - "node": ">=0.10.0" - } - }, - "node_modules/rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", - "dev": true, - "dependencies": { - "glob": "^7.1.3" - }, - "bin": { - "rimraf": "bin.js" - }, - "funding": { - "url": "https://github.com/sponsors/isaacs" - } - }, "node_modules/roarr": { "version": "2.15.4", "resolved": "https://registry.npmjs.org/roarr/-/roarr-2.15.4.tgz", @@ -6252,62 +5424,6 @@ "node": ">=8.0" } }, - "node_modules/run-applescript": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/run-applescript/-/run-applescript-5.0.0.tgz", - "integrity": "sha512-XcT5rBksx1QdIhlFOCtgZkB99ZEouFZ1E2Kc2LHqNW13U3/74YGdkQRmThTwxy4QIyookibDKYZOPqX//6BlAg==", - "dev": true, - "dependencies": { - "execa": "^5.0.0" - }, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/run-parallel": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", - "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "dependencies": { - "queue-microtask": "^1.2.2" - } - }, - "node_modules/safe-array-concat": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.0.0.tgz", - "integrity": "sha512-9dVEFruWIsnie89yym+xWTAYASdpw3CJV7Li/6zBewGf9z2i1j31rP6jnY0pHEO4QZh6N0K11bFjWmdR8UGdPQ==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.2.0", - "has-symbols": "^1.0.3", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">=0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/safe-buffer": { "version": "5.2.1", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", @@ -6327,20 +5443,6 @@ } ] }, - "node_modules/safe-regex-test": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.0.tgz", - "integrity": "sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.3", - "is-regex": "^1.1.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/safe-stable-stringify": { "version": "2.3.1", "resolved": "https://registry.npmjs.org/safe-stable-stringify/-/safe-stable-stringify-2.3.1.tgz", @@ -6417,20 +5519,6 @@ "node": ">=8" } }, - "node_modules/side-channel": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", - "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.0", - "get-intrinsic": "^1.0.2", - "object-inspect": "^1.9.0" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/signal-exit": { "version": "3.0.7", "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", @@ -6477,11 +5565,12 @@ } }, "node_modules/snyk": { - "version": "1.1291.1", - "resolved": "https://registry.npmjs.org/snyk/-/snyk-1.1291.1.tgz", - "integrity": "sha512-a+phPmN0HrXzf81fx2qcaAnbZ7rK3WG1OhOHXoBHvRUIlAKIsoAozDSlWPSs0OuQI1hpQL/15O7xUFo3kDwrew==", + "version": "1.1294.0", + "resolved": "https://registry.npmjs.org/snyk/-/snyk-1.1294.0.tgz", + "integrity": "sha512-4RBj3Lfccz5+6L2Kw9bt7icF+ex3antwt9PkSl2oEulI7mgqvc8VUFLnezg8c6PY60IPM9DrSSmNjXBac10I3Q==", "dev": true, "hasInstallScript": true, + "license": "Apache-2.0", "dependencies": { "@sentry/node": "^7.36.0", "global-agent": "^3.0.0" @@ -6580,51 +5669,6 @@ "node": ">=8" } }, - "node_modules/string.prototype.trim": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.7.tgz", - "integrity": "sha512-p6TmeT1T3411M8Cgg9wBTMRtY2q9+PNy9EV1i2lIXUN/btt763oIfxwN3RR8VU6wHX8j/1CFy0L+YuThm6bgOg==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimend": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz", - "integrity": "sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimstart": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz", - "integrity": "sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/strip-ansi": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", @@ -6636,15 +5680,6 @@ "node": ">=8" } }, - "node_modules/strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", - "dev": true, - "engines": { - "node": ">=4" - } - }, "node_modules/strip-final-newline": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", @@ -6691,12 +5726,13 @@ } }, "node_modules/synckit": { - "version": "0.8.6", - "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.8.6.tgz", - "integrity": "sha512-laHF2savN6sMeHCjLRkheIU4wo3Zg9Ln5YOjOo7sZ5dVQW8yF5pPE5SIw1dsPhq3TRp1jisKRCdPhfs/1WMqDA==", + "version": "0.9.2", + "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.9.2.tgz", + "integrity": "sha512-vrozgXDQwYO72vHjUb/HnFbQx1exDjoKzqx23aXEg2a9VIg2TSFZ8FmeZpTjUCFMYw7mpX4BE2SFu8wI7asYsw==", "dev": true, + "license": "MIT", "dependencies": { - "@pkgr/utils": "^2.4.2", + "@pkgr/core": "^0.1.0", "tslib": "^2.6.2" }, "engines": { @@ -6707,10 +5743,11 @@ } }, "node_modules/synckit/node_modules/tslib": { - "version": "2.6.2", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", - "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==", - "dev": true + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.0.tgz", + "integrity": "sha512-jWVzBLplnCmoaTr13V9dYbiQ99wvZRd0vNWaDRg+aVYRcjDF3nDksxFDE/+fkXnKhpnUUkmx5pK/v8mCtLVqZA==", + "dev": true, + "license": "0BSD" }, "node_modules/table": { "version": "6.8.2", @@ -6750,25 +5787,14 @@ "version": "0.2.0", "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", - "dev": true + "dev": true, + "peer": true }, "node_modules/text-treeview": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/text-treeview/-/text-treeview-1.0.2.tgz", "integrity": "sha512-VrajSWiDpufalLVHo136GZe0nI7+I6DXLmVtuCGIkdXJ0pewVGVSpucZa2lIQMbPqxI8G/b7OA+WRiTdcOUUDw==" }, - "node_modules/titleize": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/titleize/-/titleize-3.0.0.tgz", - "integrity": "sha512-KxVu8EYHDPBdUYdKZdKtU2aj2XfEx9AfjXxE/Aj0vT06w2icA09Vus1rh6eSu1y01akYg6BjIK/hxyLJINoMLQ==", - "dev": true, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, "node_modules/tmpl": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", @@ -6789,6 +5815,7 @@ "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", "dev": true, + "license": "MIT", "dependencies": { "is-number": "^7.0.0" }, @@ -6804,29 +5831,12 @@ "node": ">= 14.0.0" } }, - "node_modules/tsconfig-paths": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz", - "integrity": "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==", - "dev": true, - "dependencies": { - "@types/json5": "^0.0.29", - "json5": "^1.0.2", - "minimist": "^1.2.6", - "strip-bom": "^3.0.0" - } - }, - "node_modules/tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", - "dev": true - }, "node_modules/type-check": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", "dev": true, + "peer": true, "dependencies": { "prelude-ls": "^1.2.1" }, @@ -6843,83 +5853,6 @@ "node": ">=4" } }, - "node_modules/type-fest": { - "version": "0.20.2", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", - "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/typed-array-buffer": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.0.tgz", - "integrity": "sha512-Y8KTSIglk9OZEr8zywiIHG/kmQ7KWyjseXs1CbSo8vC42w7hg2HgYTxSWwP0+is7bWDc1H+Fo026CpHFwm8tkw==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.2.1", - "is-typed-array": "^1.1.10" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/typed-array-byte-length": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.0.tgz", - "integrity": "sha512-Or/+kvLxNpeQ9DtSydonMxCx+9ZXOswtwJn17SNLvhptaXYDJvkFFP5zbfU/uLmvnBJlI4yrnXRxpdWH/M5tNA==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "for-each": "^0.3.3", - "has-proto": "^1.0.1", - "is-typed-array": "^1.1.10" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-byte-offset": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.0.tgz", - "integrity": "sha512-RD97prjEt9EL8YgAgpOkf3O4IF9lhJFr9g0htQkm0rchFp/Vx7LW5Q8fSXXub7BXAODyUQohRMyOc3faCPd0hg==", - "dev": true, - "dependencies": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", - "for-each": "^0.3.3", - "has-proto": "^1.0.1", - "is-typed-array": "^1.1.10" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-length": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.4.tgz", - "integrity": "sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "for-each": "^0.3.3", - "is-typed-array": "^1.1.9" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/uglify-js": { "version": "3.17.4", "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.17.4.tgz", @@ -6932,19 +5865,13 @@ "node": ">=0.8.0" } }, - "node_modules/unbox-primitive": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz", - "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "has-bigints": "^1.0.2", - "has-symbols": "^1.0.3", - "which-boxed-primitive": "^1.0.2" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node_modules/undici": { + "version": "6.20.1", + "resolved": "https://registry.npmjs.org/undici/-/undici-6.20.1.tgz", + "integrity": "sha512-AjQF1QsmqfJys+LXfGTNum+qw4S88CojRInG/6t31W/1fk6G59s92bnAvGz5Cmur+kQv2SURXEvvudLmbrE8QA==", + "license": "MIT", + "engines": { + "node": ">=18.17" } }, "node_modules/universalify": { @@ -6955,15 +5882,6 @@ "node": ">= 10.0.0" } }, - "node_modules/untildify": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/untildify/-/untildify-4.0.0.tgz", - "integrity": "sha512-KK8xQ1mkzZeg9inewmFVDNkg3l5LUhoq9kN6iWYB/CC9YMG8HA+c1Q8HwDe6dEX7kErrEVNVBO3fWsVq5iDgtw==", - "dev": true, - "engines": { - "node": ">=8" - } - }, "node_modules/upath": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/upath/-/upath-2.0.1.tgz", @@ -7017,15 +5935,16 @@ "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" }, "node_modules/uuid": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.1.tgz", - "integrity": "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==", + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.0.0.tgz", + "integrity": "sha512-iE8Fa5fgBY4rN5GvNUJ8TSwO1QG7TzdPfhrJczf6XJ6mZUxh/GX433N70fCiJL9h8EKP5ayEIo0Q6EBQGWHFqA==", "funding": [ "https://github.com/sponsors/broofa", "https://github.com/sponsors/ctavan" ], + "license": "MIT", "bin": { - "uuid": "dist/bin/uuid" + "uuid": "dist/esm/bin/uuid" } }, "node_modules/v8-to-istanbul": { @@ -7072,51 +5991,17 @@ "node": ">= 8" } }, - "node_modules/which-boxed-primitive": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz", - "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==", - "dev": true, - "dependencies": { - "is-bigint": "^1.0.1", - "is-boolean-object": "^1.1.0", - "is-number-object": "^1.0.4", - "is-string": "^1.0.5", - "is-symbol": "^1.0.3" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-typed-array": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.11.tgz", - "integrity": "sha512-qe9UWWpkeG5yzZ0tNYxDmd7vo58HDBc39mZ0xWWpolAGADdFOzkfamWLDxkOWcvHQKVmdTyQdLD4NOfjLWTKew==", - "dev": true, - "dependencies": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "has-tostringtag": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/winston": { - "version": "3.13.0", - "resolved": "https://registry.npmjs.org/winston/-/winston-3.13.0.tgz", - "integrity": "sha512-rwidmA1w3SE4j0E5MuIufFhyJPBDG7Nu71RkZor1p2+qHvJSZ9GYDA81AyleQcZbh/+V6HjeBdfnTZJm9rSeQQ==", + "version": "3.15.0", + "resolved": "https://registry.npmjs.org/winston/-/winston-3.15.0.tgz", + "integrity": "sha512-RhruH2Cj0bV0WgNL+lOfoUBI4DVfdUNjVnJGVovWZmrcKtrFTTRzgXYK2O9cymSGjrERCtaAeHwMNnUWXlwZow==", + "license": "MIT", "dependencies": { "@colors/colors": "^1.6.0", "@dabh/diagnostics": "^2.0.2", "async": "^3.2.3", "is-stream": "^2.0.0", - "logform": "^2.4.0", + "logform": "^2.6.0", "one-time": "^1.0.0", "readable-stream": "^3.4.0", "safe-stable-stringify": "^2.3.1", @@ -7158,14 +6043,6 @@ "node": ">= 12.0.0" } }, - "node_modules/winston/node_modules/@colors/colors": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.6.0.tgz", - "integrity": "sha512-Ir+AOibqzrIsL6ajt3Rz3LskB7OiMVHqltZmspbW/TJuTVuyOMirVqAkjfY6JISiLHgyNqicAC8AyHHGzNd/dA==", - "engines": { - "node": ">=0.1.90" - } - }, "node_modules/wordwrap": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", @@ -7208,9 +6085,10 @@ } }, "node_modules/ws": { - "version": "8.17.0", - "resolved": "https://registry.npmjs.org/ws/-/ws-8.17.0.tgz", - "integrity": "sha512-uJq6108EgZMAl20KagGkzCKfMEjxmKvZHG7Tlq0Z6nOky7YF7aq4mOx6xK8TJ/i1LeK4Qus7INktacctDgY8Ow==", + "version": "8.18.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.18.0.tgz", + "integrity": "sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==", + "license": "MIT", "engines": { "node": ">=10.0.0" }, diff --git a/package.json b/package.json index 26e6d16..13d198e 100644 --- a/package.json +++ b/package.json @@ -30,41 +30,41 @@ "license": "MIT", "type": "module", "dependencies": { - "axios": "^1.7.2", + "@qlik/api": "^1.23.0", + "axios": "^1.7.7", "commander": "^12.1.0", "csv-parse": "^5.5.6", - "csv-stringify": "^6.5.0", + "csv-stringify": "^6.5.1", "enigma.js": "^2.14.0", - "esbuild": "^0.21.4", - "form-data": "^4.0.0", + "esbuild": "^0.24.0", + "form-data": "^4.0.1", "fs-extra": "^11.2.0", "handlebars": "^4.7.8", - "luxon": "^3.4.4", + "luxon": "^3.5.0", "node-xlsx": "^0.24.0", "qrs-interact": "^6.3.1", "random-words": "^2.0.1", "retry-axios": "^3.1.3", "table": "^6.8.2", "text-treeview": "^1.0.2", + "undici": "^6.20.1", "upath": "^2.0.1", - "uuid": "^9.0.1", - "winston": "^3.13.0", + "uuid": "^11.0.0", + "winston": "^3.15.0", "winston-daily-rotate-file": "^5.0.0", - "ws": "^8.17.0", + "ws": "^8.18.0", "yesno": "^0.4.0" }, "devDependencies": { - "@babel/eslint-parser": "^7.24.6", - "@babel/plugin-syntax-import-assertions": "^7.24.6", + "@babel/eslint-parser": "^7.25.9", + "@babel/plugin-syntax-import-assertions": "^7.26.0", + "@eslint/js": "^9.13.0", "@jest/globals": "^29.7.0", - "eslint": "^8.57.0", - "eslint-config-airbnb-base": "^15.0.0", "eslint-config-prettier": "^9.1.0", - "eslint-plugin-import": "^2.29.1", - "eslint-plugin-prettier": "^5.1.3", + "eslint-plugin-prettier": "^5.2.1", "jest": "^29.7.0", - "prettier": "^3.3.0", - "snyk": "^1.1291.1" + "prettier": "^3.3.3", + "snyk": "^1.1294.0" }, "pkg": { "assets": [ diff --git a/src/__tests__/activity_custom_property_create_cert.test.js b/src/__tests__/activity_custom_property_create_cert.test.js new file mode 100644 index 0000000..5b642be --- /dev/null +++ b/src/__tests__/activity_custom_property_create_cert.test.js @@ -0,0 +1,50 @@ +import { jest, test, expect, describe } from '@jest/globals'; + +import { createUserActivityBucketsCustomProperty } from '../lib/cmd/qseow/createuseractivitycp.js'; + +const options = { + logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', + authType: process.env.CTRL_Q_AUTH_TYPE || 'cert', + authCertFile: process.env.CTRL_Q_AUTH_CERT_FILE || './cert/client.pem', + authCertKeyFile: process.env.CTRL_Q_AUTH_CERT_KEY_FILE || './cert/client_key.pem', + authRootCertFile: process.env.CTRL_Q_AUTH_ROOT_CERT_FILE || './cert/root.pem', + host: process.env.CTRL_Q_HOST || '', + port: process.env.CTRL_Q_PORT || '4242', + schemaVersion: process.env.CTRL_Q_SCHEMA_VERSION || '12.612.0', + virtualProxy: process.env.CTRL_Q_VIRTUAL_PROXY || '', + secure: process.env.CTRL_Q_SECURE || true, + authUserDir: process.env.CTRL_Q_AUTH_USER_DIR || '', + authUserId: process.env.CTRL_Q_AUTH_USER_ID || '', + + updateUserSleep: process.env.CTRL_Q_UPDATE_USER_SLEEP || 500, + updateBatchSleep: process.env.CTRL_Q_UPDATE_BATCH_SLEEP || 3, + updateBatchSize: process.env.CTRL_Q_UPDATE_BATCH_SIZE || 10, + updateBatchSize: process.env.CTRL_Q_UPDATE_BATCH_SIZE || 10, + activityBuckets: process.env.CTRL_Q_ACTIVITY_BUCKETS || ['1', '7', '14', '30', '90', '180', '365'], + licenseType: process.env.CTRL_Q_LICENSE_TYPE || ['analyzer', 'analyzer-time', 'login', 'professional', 'user'], +}; + +const defaultTestTimeout = process.env.CTRL_Q_TEST_TIMEOUT || 600000; // 10 minute default timeout +console.log(`Jest timeout: ${defaultTestTimeout}`); +jest.setTimeout(defaultTestTimeout); + +test('get tasks (verify parameters) ', async () => { + expect(options.authCertFile).not.toHaveLength(0); + expect(options.authCertKeyFile).not.toHaveLength(0); + expect(options.authRootCertFile).not.toHaveLength(0); + expect(options.host).not.toHaveLength(0); + expect(options.authUserDir).not.toHaveLength(0); + expect(options.authUserId).not.toHaveLength(0); +}); + +// Test suite for creating user activity buckets custom property +describe('create user activity custom property (cert auth)', () => { + test('do not overwrite existing custom property, default buckets, user directory [LAB]', async () => { + options.userDirectory = ['LAB']; + options.customPropertyName = 'Ctrl_Q_User_Activity_Bucket'; + options.force = true; + + const result = await createUserActivityBucketsCustomProperty(options); + expect(result).toBe(true); + }); +}); diff --git a/src/__tests__/app_cert.test.js b/src/__tests__/app_cert.test.js index 88d1725..388efb8 100644 --- a/src/__tests__/app_cert.test.js +++ b/src/__tests__/app_cert.test.js @@ -1,7 +1,7 @@ /* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import { getApps, getAppById } from '../lib/util/app.js'; +import { getApps, getAppById } from '../lib/util/qseow/app.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/app_export_cert.test.js b/src/__tests__/app_export_cert.test.js index 17c3004..8f011b4 100644 --- a/src/__tests__/app_export_cert.test.js +++ b/src/__tests__/app_export_cert.test.js @@ -1,9 +1,9 @@ /* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import fs from 'fs'; -import path from 'path'; -import exportAppToFile from '../lib/cmd/exportapp.js'; +import fs from 'node:fs'; +import path from 'node:path'; +import exportAppToFile from '../lib/cmd/qseow/exportapp.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/app_export_jwt.test.js b/src/__tests__/app_export_jwt.test.js index 3c1ffd3..68f593a 100644 --- a/src/__tests__/app_export_jwt.test.js +++ b/src/__tests__/app_export_jwt.test.js @@ -1,9 +1,9 @@ /* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import fs from 'fs'; -import path from 'path'; -import exportAppToFile from '../lib/cmd/exportapp.js'; +import fs from 'node:fs'; +import path from 'node:path'; +import exportAppToFile from '../lib/cmd/qseow/exportapp.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/app_import_cert.test.js b/src/__tests__/app_import_cert.test.js index d6ee4ab..0671688 100644 --- a/src/__tests__/app_import_cert.test.js +++ b/src/__tests__/app_import_cert.test.js @@ -1,8 +1,8 @@ /* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import importAppFromFile from '../lib/cmd/importapp.js'; -import { appExistById, deleteAppById } from '../lib/util/app.js'; +import importAppFromFile from '../lib/cmd/qseow/importapp.js'; +import { appExistById, deleteAppById } from '../lib/util/qseow/app.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -64,12 +64,10 @@ describe('import apps from QVF files (cert auth)', () => { // console.log(`App ID: ${appId}`); // Check if app exists - // eslint-disable-next-line no-await-in-loop const appExists = await appExistById(appId, options); if (appExists) { // Delete app - // eslint-disable-next-line no-await-in-loop const resultDelete = await deleteAppById(appId, options); expect(resultDelete).toBe(true); } else { diff --git a/src/__tests__/app_import_jwt.test.js b/src/__tests__/app_import_jwt.test.js index 6e16dd5..1ae4730 100644 --- a/src/__tests__/app_import_jwt.test.js +++ b/src/__tests__/app_import_jwt.test.js @@ -1,8 +1,8 @@ /* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import importAppFromFile from '../lib/cmd/importapp.js'; -import { appExistById, deleteAppById } from '../lib/util/app.js'; +import importAppFromFile from '../lib/cmd/qseow/importapp.js'; +import { appExistById, deleteAppById } from '../lib/util/qseow/app.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -64,12 +64,10 @@ describe('import apps from QVF files (cert auth)', () => { // console.log(`App ID: ${appId}`); // Check if app exists - // eslint-disable-next-line no-await-in-loop const appExists = await appExistById(appId, options); if (appExists) { // Delete app - // eslint-disable-next-line no-await-in-loop const resultDelete = await deleteAppById(appId, options); expect(resultDelete).toBe(true); } else { diff --git a/src/__tests__/app_jwt.test.js b/src/__tests__/app_jwt.test.js index 03803c8..c39676c 100644 --- a/src/__tests__/app_jwt.test.js +++ b/src/__tests__/app_jwt.test.js @@ -1,8 +1,8 @@ /* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import { getApps, getAppById, appExistById, deleteAppById } from '../lib/util/app.js'; -import importAppFromFile from '../lib/cmd/importapp.js'; +import { getApps, getAppById, appExistById, deleteAppById } from '../lib/util/qseow/app.js'; +import importAppFromFile from '../lib/cmd/qseow/importapp.js'; import { sleep } from '../globals.js'; const options = { @@ -103,12 +103,10 @@ describe('deleteAppById (JWT auth)', () => { // console.log(`App ID: ${appId}`); // Check if app exists - // eslint-disable-next-line no-await-in-loop const appExists = await appExistById(appId, options); if (appExists) { // Delete app - // eslint-disable-next-line no-await-in-loop const resultDelete = await deleteAppById(appId, options); expect(resultDelete).toBe(true); } else { diff --git a/src/__tests__/bookmark_get_cert.test.js b/src/__tests__/bookmark_get_cert.test.js index 6a9ec50..33cbf38 100644 --- a/src/__tests__/bookmark_get_cert.test.js +++ b/src/__tests__/bookmark_get_cert.test.js @@ -1,7 +1,7 @@ /* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import getBookmark from '../lib/cmd/getbookmark.js'; +import getBookmark from '../lib/cmd/qseow/getbookmark.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -99,8 +99,8 @@ describe('get in-app bookmarks (cert auth)', () => { expect(result.length).toBe(2); // Verify that the bookmarks have the correct IDs - expect(result[0].qInfo.qId).toBe(appIdExistsHasBookmarks1Bookmark1); - expect(result[1].qInfo.qId).toBe(appIdExistsHasBookmarks1Bookmark2); + expect(result[0].qInfo.qId).toBe(appIdExistsHasBookmarks1Bookmark2); + expect(result[1].qInfo.qId).toBe(appIdExistsHasBookmarks1Bookmark1); }); /** @@ -122,8 +122,9 @@ describe('get in-app bookmarks (cert auth)', () => { expect(result.length).toBe(2); // Verify that the bookmarks have the correct IDs - expect(result[0].qInfo.qId).toBe(appIdExistsHasBookmarks1Bookmark1); - expect(result[1].qInfo.qId).toBe(appIdExistsHasBookmarks1Bookmark2); + // Compare against all bookmarks in app, i.e. appIdExistsHasBookmarks1Bookmark1 and appIdExistsHasBookmarks1Bookmark2 + expect(result[0].qInfo.qId).toBe(appIdExistsHasBookmarks1Bookmark2); + expect(result[1].qInfo.qId).toBe(appIdExistsHasBookmarks1Bookmark1); }); /** diff --git a/src/__tests__/bookmark_get_jwt.test.js b/src/__tests__/bookmark_get_jwt.test.js index b6447eb..85b8a5d 100644 --- a/src/__tests__/bookmark_get_jwt.test.js +++ b/src/__tests__/bookmark_get_jwt.test.js @@ -1,7 +1,6 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import getBookmark from '../lib/cmd/getbookmark.js'; +import getBookmark from '../lib/cmd/qseow/getbookmark.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/connection_test_cert.test.js b/src/__tests__/connection_test_cert.test.js index 244c9ae..7c0eb79 100644 --- a/src/__tests__/connection_test_cert.test.js +++ b/src/__tests__/connection_test_cert.test.js @@ -1,7 +1,6 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import testConnection from '../lib/cmd/testconnection.js'; +import testConnection from '../lib/cmd/qseow/testconnection.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/connection_test_jwt.test.js b/src/__tests__/connection_test_jwt.test.js index 28002c6..cc1a1d5 100644 --- a/src/__tests__/connection_test_jwt.test.js +++ b/src/__tests__/connection_test_jwt.test.js @@ -1,7 +1,6 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import testConnection from '../lib/cmd/testconnection.js'; +import testConnection from '../lib/cmd/qseow/testconnection.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -25,10 +24,13 @@ describe('connection test (JWT auth)', () => { options.authType = 'jwt'; options.port = '443'; options.virtualProxy = 'jwt'; + options.secure = false; test('Verify parameters', async () => { + expect(options.authType).toBe('jwt'); expect(options.host).not.toHaveLength(0); expect(options.port).not.toHaveLength(0); + expect(options.port).toBe('443'); expect(options.authJwt).not.toHaveLength(0); }); @@ -38,8 +40,6 @@ describe('connection test (JWT auth)', () => { * Should succeed */ test('do connection test (virtual proxy=jwt)', async () => { - options.virtualProxy = 'jwt'; - const result = await testConnection(options); // Result should be a JSON object diff --git a/src/__tests__/script_get_cert.test.js b/src/__tests__/script_get_cert.test.js index 11ea145..7162292 100644 --- a/src/__tests__/script_get_cert.test.js +++ b/src/__tests__/script_get_cert.test.js @@ -1,7 +1,6 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import getScript from '../lib/cmd/getscript.js'; +import getScript from '../lib/cmd/qseow/getscript.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -43,7 +42,7 @@ describe('get app script (cert auth)', () => { expect(result.appId).toBe('a3e0f5d2-000a-464f-998d-33d333b175d7'); expect(result.appCreatedDate).toBe('2021-06-03T22:04:52.283Z'); - expect(result.appModifiedDate).toBe('2023-05-05T06:17:05.456Z'); + expect(result.appModifiedDate).toBe('2024-03-20T08:02:25.153Z'); expect(result.appScript.length).toBe(1989); }); }); diff --git a/src/__tests__/script_get_jwt.test.js b/src/__tests__/script_get_jwt.test.js index 301472c..eda1ce4 100644 --- a/src/__tests__/script_get_jwt.test.js +++ b/src/__tests__/script_get_jwt.test.js @@ -1,7 +1,6 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import getScript from '../lib/cmd/getscript.js'; +import getScript from '../lib/cmd/qseow/getscript.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -18,6 +17,7 @@ const options = { }; const defaultTestTimeout = process.env.CTRL_Q_TEST_TIMEOUT || 120000; // 2 minute default timeout +console.log(`Jest timeout: ${defaultTestTimeout}`); jest.setTimeout(defaultTestTimeout); // Get app script @@ -41,7 +41,7 @@ describe('get app script (jwt auth)', () => { expect(result.appId).toBe('a3e0f5d2-000a-464f-998d-33d333b175d7'); expect(result.appCreatedDate).toBe('2021-06-03T22:04:52.283Z'); - expect(result.appModifiedDate).toBe('2023-05-05T06:17:05.456Z'); + expect(result.appModifiedDate).toBe('2024-03-20T08:02:25.153Z'); expect(result.appScript.length).toBe(1989); }); }); diff --git a/src/__tests__/task_cert.test.js b/src/__tests__/task_cert.test.js index 39cde42..0a3146f 100644 --- a/src/__tests__/task_cert.test.js +++ b/src/__tests__/task_cert.test.js @@ -1,7 +1,6 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import { taskExistById, getTaskByName, getTaskById } from '../lib/util/task.js'; +import { taskExistById, getTaskByName, getTaskById } from '../lib/util/qseow/task.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -69,9 +68,10 @@ describe('getTaskByName: Get task by name (cert auth)', () => { expect(result).toEqual(false); // Ensure correct substring was written to global console log - expect(global.console.log).toHaveBeenCalledWith( - expect.stringContaining(`More than one task with name ${multipleMatchingTaskNames} found.`) - ); + // TODO: Fix this test + // expect(global.console.log).toHaveBeenCalledWith( + // expect.stringContaining(`More than one task with name ${multipleMatchingTaskNames} found.`) + // ); }); test('no task name provided', async () => { @@ -80,20 +80,20 @@ describe('getTaskByName: Get task by name (cert auth)', () => { }); }); -// // Get task by ID -// describe('getTaskById: Get task by ID (cert auth)', () => { -// test('no matching task', async () => { -// const result = await getTaskById(nonExistingTaskId, options); -// expect(result).toEqual(false); -// }); - -// test('1 matching task', async () => { -// const result = await getTaskById(existingTaskId, options); -// expect(result.id).toEqual(existingTaskId); -// }); - -// test('no task id provided', async () => { -// const result = await getTaskById('', options); -// expect(result).toEqual(false); -// }); -// }); +// Get task by ID +describe('getTaskById: Get task by ID (cert auth)', () => { + test('no matching task', async () => { + const result = await getTaskById(nonExistingTaskId, options); + expect(result).toEqual(false); + }); + + test('1 matching task', async () => { + const result = await getTaskById(existingTaskId, options); + expect(result.id).toEqual(existingTaskId); + }); + + test('no task id provided', async () => { + const result = await getTaskById('', options); + expect(result).toEqual(false); + }); +}); diff --git a/src/__tests__/task_custom_property_set_cert.test.js b/src/__tests__/task_custom_property_set_cert.test.js index bc38bfb..5b77c4c 100644 --- a/src/__tests__/task_custom_property_set_cert.test.js +++ b/src/__tests__/task_custom_property_set_cert.test.js @@ -1,8 +1,7 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import setTaskCustomProperty from '../lib/cmd/settaskcp.js'; -import { getTaskById } from '../lib/util/task.js'; +import setTaskCustomProperty from '../lib/cmd/qseow/settaskcp.js'; +import { getTaskById } from '../lib/util/qseow/task.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -64,11 +63,14 @@ describe('set custom property on reload task (cert auth)', () => { // Get task and verify that CPs have been set const task = await getTaskById(existingTaskId, options); - expect(task.customProperties.length).toBe(2); + + // Find how many values the ctrl_q_unit_test_1 CP has + const cp = task.customProperties.filter((item) => item.definition.name === options.customPropertyName); + expect(cp.length).toBe(2); // Verify that CP has the correct values - expect(task.customProperties[0].value).toBe('Value 1'); - expect(task.customProperties[1].value).toBe('Value 2'); + expect(cp[0].value).toBe('Value 1'); + expect(cp[1].value).toBe('Value 2'); }); /** @@ -91,9 +93,12 @@ describe('set custom property on reload task (cert auth)', () => { // Get task and verify that CPs have been set const task = await getTaskById(existingTaskId, options); - expect(task.customProperties.length).toBe(3); + + // Find how many values the ctrl_q_unit_test_1 CP has + const cp = task.customProperties.filter((item) => item.definition.name === options.customPropertyName); + expect(cp.length).toBe(3); // Verify that CP has the correct values - expect(task.customProperties[2].value).toBe('Value 3'); + expect(cp[2].value).toBe('Value 3'); }); }); diff --git a/src/__tests__/task_custom_property_set_jwt.test.js b/src/__tests__/task_custom_property_set_jwt.test.js index 3ce06af..f99ca3c 100644 --- a/src/__tests__/task_custom_property_set_jwt.test.js +++ b/src/__tests__/task_custom_property_set_jwt.test.js @@ -1,8 +1,7 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import setTaskCustomProperty from '../lib/cmd/settaskcp.js'; -import { getTaskById } from '../lib/util/task.js'; +import setTaskCustomProperty from '../lib/cmd/qseow/settaskcp.js'; +import { getTaskById } from '../lib/util/qseow/task.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -60,11 +59,14 @@ describe('set custom property on reload task (jwt auth)', () => { // Get task and verify that CPs have been set const task = await getTaskById(existingTaskId, options); - expect(task.customProperties.length).toBe(2); + + // Find how many values the ctrl_q_unit_test_1 CP has + const cp = task.customProperties.filter((item) => item.definition.name === options.customPropertyName); + expect(cp.length).toBe(2); // Verify that CP has the correct values - expect(task.customProperties[0].value).toBe('Value 1'); - expect(task.customProperties[1].value).toBe('Value 2'); + expect(cp[0].value).toBe('Value 1'); + expect(cp[1].value).toBe('Value 2'); }); /** @@ -87,9 +89,12 @@ describe('set custom property on reload task (jwt auth)', () => { // Get task and verify that CPs have been set const task = await getTaskById(existingTaskId, options); - expect(task.customProperties.length).toBe(3); + + // Find how many values the ctrl_q_unit_test_1 CP has + const cp = task.customProperties.filter((item) => item.definition.name === options.customPropertyName); + expect(cp.length).toBe(3); // Verify that CP has the correct values - expect(task.customProperties[2].value).toBe('Value 3'); + expect(cp[2].value).toBe('Value 3'); }); }); diff --git a/src/__tests__/task_get_cert.test.js b/src/__tests__/task_get_cert.test.js index bc128a2..7d690b8 100644 --- a/src/__tests__/task_get_cert.test.js +++ b/src/__tests__/task_get_cert.test.js @@ -1,9 +1,8 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import fs from 'fs'; -import path from 'path'; -import getTask from '../lib/cmd/gettask.js'; +import fs from 'node:fs'; +import path from 'node:path'; +import getTask from '../lib/cmd/qseow/gettask.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/task_get_jwt.test.js b/src/__tests__/task_get_jwt.test.js index b6540a2..81e4848 100644 --- a/src/__tests__/task_get_jwt.test.js +++ b/src/__tests__/task_get_jwt.test.js @@ -1,9 +1,8 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import fs from 'fs'; -import path from 'path'; -import getTask from '../lib/cmd/gettask.js'; +import fs from 'node:fs'; +import path from 'node:path'; +import getTask from '../lib/cmd/qseow/gettask.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/task_import_cert.test.js b/src/__tests__/task_import_cert.test.js index b2666e4..29b6d20 100644 --- a/src/__tests__/task_import_cert.test.js +++ b/src/__tests__/task_import_cert.test.js @@ -1,10 +1,8 @@ -/* eslint-disable no-await-in-loop */ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import importTaskFromFile from '../lib/cmd/importtask.js'; -import { getTaskById, deleteExternalProgramTaskById, deleteReloadTaskById } from '../lib/util/task.js'; -import { mapTaskType } from '../lib/util/lookups.js'; +import importTaskFromFile from '../lib/cmd/qseow/importtask.js'; +import { getTaskById, deleteExternalProgramTaskById, deleteReloadTaskById } from '../lib/util/qseow/task.js'; +import { mapTaskType } from '../lib/util/qseow/lookups.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/task_import_jwt.tes.js b/src/__tests__/task_import_jwt.test.js similarity index 96% rename from src/__tests__/task_import_jwt.tes.js rename to src/__tests__/task_import_jwt.test.js index 87d88dc..7d9faad 100644 --- a/src/__tests__/task_import_jwt.tes.js +++ b/src/__tests__/task_import_jwt.test.js @@ -1,10 +1,8 @@ -/* eslint-disable no-await-in-loop */ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import importTaskFromFile from '../lib/cmd/importtask.js'; -import { getTaskById, deleteExternalProgramTaskById, deleteReloadTaskById } from '../lib/util/task.js'; -import { mapTaskType } from '../lib/util/lookups.js'; +import importTaskFromFile from '../lib/cmd/qseow/importtask.js'; +import { getTaskById, deleteExternalProgramTaskById, deleteReloadTaskById } from '../lib/util/qseow/task.js'; +import { mapTaskType } from '../lib/util/qseow/lookups.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', diff --git a/src/__tests__/task_jwt.test.js b/src/__tests__/task_jwt.test.js index 6ebbf33..1d1b4f1 100644 --- a/src/__tests__/task_jwt.test.js +++ b/src/__tests__/task_jwt.test.js @@ -1,7 +1,6 @@ -/* eslint-disable no-console */ import { jest, test, expect, describe } from '@jest/globals'; -import { taskExistById, getTaskByName, getTaskById } from '../lib/util/task.js'; +import { taskExistById, getTaskByName, getTaskById } from '../lib/util/qseow/task.js'; const options = { logLevel: process.env.CTRL_Q_LOG_LEVEL || 'info', @@ -71,9 +70,10 @@ describe('getTaskByName: Get task by name (jwt auth)', () => { expect(result).toEqual(false); // Ensure correct substring was written to global console log - expect(global.console.log).toHaveBeenCalledWith( - expect.stringContaining(`More than one task with name ${multipleMatchingTaskNames} found.`) - ); + // TODO: Fix this test + // expect(global.console.log).toHaveBeenCalledWith( + // expect.stringContaining(`More than one task with name ${multipleMatchingTaskNames} found.`) + // ); }); test('no task name provided', async () => { diff --git a/src/ctrl-q.js b/src/ctrl-q.js index 2d43db2..c85c2bc 100644 --- a/src/ctrl-q.js +++ b/src/ctrl-q.js @@ -1,48 +1,33 @@ import { Command, Option } from 'commander'; import { logger, appVersion, setLoggingLevel, setCliOptions } from './globals.js'; -import { catchLog, logStartupInfo } from './lib/util/log.js'; - -// const { createUserActivityCustomProperty } = require('./lib/createuseractivitycp'); - -import getMasterDimension from './lib/cmd/getdim.js'; - -import deleteMasterDimension from './lib/cmd/deletedim.js'; -import getMasterMeasure from './lib/cmd/getmeasure.js'; -import deleteMasterMeasure from './lib/cmd/deletemeasure.js'; -import getVariable from './lib/cmd/getvariable.js'; -import deleteVariable from './lib/cmd/deletevariable.js'; -import getBookmark from './lib/cmd/getbookmark.js'; -import importMasterItemFromFile from './lib/cmd/import-masteritem-excel.js'; -import scrambleField from './lib/cmd/scramblefield.js'; -import getScript from './lib/cmd/getscript.js'; -import getTask from './lib/cmd/gettask.js'; -import setTaskCustomProperty from './lib/cmd/settaskcp.js'; -import importTaskFromFile from './lib/cmd/importtask.js'; -import importAppFromFile from './lib/cmd/importapp.js'; -import exportAppToFile from './lib/cmd/exportapp.js'; -import testConnection from './lib/cmd/testconnection.js'; -import visTask from './lib/cmd/vistask.js'; -import getSessions from './lib/cmd/getsessions.js'; -import deleteSessions from './lib/cmd/deletesessions.js'; - -import { - sharedParamAssertOptions, - masterItemImportAssertOptions, - masterItemMeasureDeleteAssertOptions, - masterItemDimDeleteAssertOptions, - masterItemGetAssertOptions, - variableGetAssertOptions, - variableDeleteAssertOptions, - getScriptAssertOptions, - getBookmarkAssertOptions, - getTaskAssertOptions, - setTaskCustomPropertyAssertOptions, - taskImportAssertOptions, - appImportAssertOptions, - appExportAssertOptions, - getSessionsAssertOptions, - deleteSessionsAssertOptions, -} from './lib/util/assert-options.js'; +import { logStartupInfo } from './lib/util/log.js'; + +// Import command setup functions +// QSEoW +import { setupQseowUserActivityCustomPropertyCommand } from './lib/cli/qseow-cp-user-activity-bucket.js'; +import { setupQseowMasterItemImportCommand } from './lib/cli/qseow-master-item-import.js'; +import { setupQseowGetMasterMeasureCommand } from './lib/cli/qseow-get-master-measure.js'; +import { setupQseowDeleteMasterMeasureCommand } from './lib/cli/qseow-delete-master-measure.js'; +import { setupQseowGetMasterDimensionCommand } from './lib/cli/qseow-get-master-dimension.js'; +import { setupQseowDeleteMasterDimensionCommand } from './lib/cli/qseow-delete-master-dimension.js'; +import { setpQseowGetVariableCommand } from './lib/cli/qseow-get-variable.js'; +import { setupQseowDeleteVariableCommand } from './lib/cli/qseow-delete-variable.js'; +import { setupQseowScrambleFieldCommand } from './lib/cli/qseow-scramble-field.js'; +import { setupGetScriptCommand } from './lib/cli/qseow-get-script.js'; +import { setupQseowGetBookmarkCommand } from './lib/cli/qseow-get-bookmark.js'; +import { setupGetTaskCommand } from './lib/cli/qseow-get-task.js'; +import { setupQseowSetTaskCustomPropertyCommand } from './lib/cli/qseow-set-task-cp.js'; +import { setupQseowImportTaskFromFileCommand } from './lib/cli/qseow-import-task-from-file.js'; +import { setupQseowImportAppFromFileCommand } from './lib/cli/qseow-import-app-from-file.js'; +import { setupQseowExportAppCommand } from './lib/cli/qseow-export-app-to-file.js'; +import { setupQseowTestConnectionCommand } from './lib/cli/qseow-test-connection.js'; +import { setupQseowShowVersionCommand } from './lib/cli/qseow-show-version.js'; +import { setupQseowVisualiseTaskCommand } from './lib/cli/qseow-visualise-task.js'; +import { setupQseowGetProxySessionsCommand } from './lib/cli/qseow-get-proxy-session.js'; +import { setupQseowDeleteProxySessionsCommand } from './lib/cli/qseow-delete-proxy-session.js'; + +// QS Cloud +import { setupQscloudTestConnectionCommand } from './lib/cli/qscloud-test-connection.js'; const program = new Command(); @@ -65,6 +50,7 @@ program.configureHelp({ // Basic app info program .version(appVersion) + .name('ctrl-q') .description( `Ctrl-Q is a command line utility for interacting with client-managed Qlik Sense Enterprise on Windows servers.\nAmong other things the tool does bulk import of apps and tasks, manipulates master items and scrambles in-app data.\n\nVersion: ${appVersion}` ) @@ -76,953 +62,100 @@ program.configureHelp({ // Set log level & show startup info setLoggingLevel(options.logLevel); - // eslint-disable-next-line no-underscore-dangle logStartupInfo(options, actionCommand._name, actionCommand._description); logger.verbose(`About to call action handler for subcommand: ${actionCommand.name()}`); }); - // Create custom properties for tracking user activity buckets, i.e. how long ago a user was last active (last login) in Sense - // program - // .command('user-activity-cp-create') - // .description( - // 'create custom property and populate it with values ("activity buckets") indicating how long ago users last logged into Sense' - // ) - // .action(async (options) => { - // try { - // let optionsLocal = options; - // await sharedParamAssertOptions(options); - // optionsLocal = userActivityCustomPropertyAssertOptions(options); - // createUserActivityCustomProperty(optionsLocal); - // } catch (err) { - // logger.error(`USER ACTIVITY CP: ${err}`); - // } - // }) - // .addOption( - // new Option('--log-level ', 'log level') - // .choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']) - // .default('info') - // ) - // .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - // .option('--port ', 'Qlik Sense repository API port', '4242') - // .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - // .requiredOption('--secure ', 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', true) - // .option('--auth-user-dir ', 'user directory for user to connect with', 'Internal') - // .option('--auth-user-id ', 'user ID for user to connect with', 'sa_repository') - - // .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - // .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - // .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - // .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - // .option('--jwt ', 'JSON Web Token (JWT) to use for authenticating with Qlik Sense', '') - - // .requiredOption('--user-directory ', 'name of user directory whose users will be updated with activity info') - // .requiredOption('--custom-property-name ', 'name of custom property that will hold user activity buckets') - // .addOption( - // new Option('--force ', 'forcibly overwrite and replace custom property and its values if it already exists') - // .choices(['true', 'false']) - // .default('false') - // ) - // .option('--activity-buckets ', 'custom property values/user activity buckets to be defined. In days.', [ - // '1', - // '7', - // '14', - // '30', - // '90', - // '180', - // '365', - // ]); - - // Import dimensions/measures from definitions in Excel file - program - .command('master-item-import') - .description('create master items based on definitions in a file on disk') - .action(async (options) => { - try { - await sharedParamAssertOptions(options); - masterItemImportAssertOptions(options); - importMasterItemFromFile(options); - } catch (err) { - catchLog('IMPORT EXCEL', err); - } - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--app-id ', 'Qlik Sense app ID') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .addOption(new Option('-t, --file-type ', 'source file type').choices(['excel']).default('excel')) - .requiredOption('--file ', 'file containing master item definitions') - .requiredOption('--sheet ', 'name of Excel sheet where dim/measure flag column is found') - .addOption( - new Option( - '--col-ref-by ', - 'how to refer to columns in the source file. Options are by name or by position (zero based)' - ) - .choices(['name', 'position']) - .default('name') - ) - .requiredOption( - '--col-item-type ', - 'column where dim/measure flag is found. Use "dim-single" in that column to create dimension, "dim-drilldown" for drill-down dimension, "measure" for measure', - 'Master item type' - ) - .requiredOption( - '--col-master-item-name ', - 'column number (zero based) or name to use as master item name', - 'Master item name' - ) - .requiredOption( - '--col-master-item-descr ', - 'column number (zero based) or name to use as master item description', - 'Description' - ) - .requiredOption( - '--col-master-item-label ', - 'column number (zero based) or name to use as master item label', - 'Label' - ) - .requiredOption( - '--col-master-item-expr ', - 'column number (zero based) or name to use as master item expression', - 'Expression' - ) - .requiredOption( - '--col-master-item-tag ', - 'column number (zero based) or name to use as master item tags', - 'Tag' - ) - .requiredOption( - '--col-master-item-color ', - 'column number (zero based) or name to use as color for dimensions/measures', - 'Color' - ) - .requiredOption( - '--col-master-item-per-value-color ', - 'column number (zero based) or name to use as per-value/segment color for dimensions/measures', - 'Per value color' - ) - - .requiredOption('--sleep-between-imports ', 'sleep this many milliseconds between imports. Set to 0 to disable', 1000) - .requiredOption( - '--limit-import-count ', - 'import at most x number of master items from the Excel file. Defaults to 0 = no limit', - 0 - ) - .option('--dry-run', 'do a dry run, i.e. do not create or update anything - just show what would be done'); - - // Get measure command - program - .command('master-item-measure-get') - .description('get info about one or more master measures') - .action(async (options) => { - await sharedParamAssertOptions(options); - masterItemGetAssertOptions(options); - - getMasterMeasure(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--app-id ', 'Qlik Sense app ID') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .addOption( - new Option('--id-type ', 'type of identifier passed in the --master-item option').choices(['id', 'name']).default('name') - ) - .option('--master-item ', 'master measure to retrieve. If not specified all measures will be retrieved') - .addOption(new Option('--output-format ', 'output format').choices(['json', 'table']).default('json')); - - // Delete measure command - program - .command('master-item-measure-delete') - .description('delete master measure(s)') - .action(async (options) => { - await sharedParamAssertOptions(options); - masterItemMeasureDeleteAssertOptions(options); - - deleteMasterMeasure(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--app-id ', 'Qlik Sense app ID') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .addOption(new Option('--id-type ', 'type of identifier passed in the --master-item option').choices(['id', 'name'])) - .option('--master-item ', 'names or IDs of master measures to be deleted. Multiple IDs should be space separated') - .option('--delete-all', 'delete all master measures') - .option('--dry-run', 'do a dry run, i.e. do not delete anything - just show what would be deleted'); - - // Get dimension command - program - .command('master-item-dim-get') - .description('get info about one or more master dimensions') - .action(async (options) => { - await sharedParamAssertOptions(options); - - getMasterDimension(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--app-id ', 'Qlik Sense app ID') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .requiredOption('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .requiredOption('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .requiredOption('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .addOption( - new Option('--id-type ', 'type of identifier passed in the --master-item option').choices(['id', 'name']).default('name') - ) - .option('--master-item ', 'master dimension to retrieve. If not specified all dimensions will be retrieved') - .addOption(new Option('--output-format ', 'output format').choices(['json', 'table']).default('json')); - - // Delete dimension command - program - .command('master-item-dim-delete') - .description('delete master dimension(s)') - .action(async (options) => { - await sharedParamAssertOptions(options); - masterItemDimDeleteAssertOptions(options); - - deleteMasterDimension(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--app-id ', 'Qlik Sense app ID') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .addOption(new Option('--id-type ', 'type of identifier passed in the --master-item option').choices(['id', 'name'])) - .option('--master-item ', 'names or IDs of master dimensions to be deleted. Multiple IDs should be space separated') - .option('--delete-all', 'delete all master dimensions') - .option('--dry-run', 'do a dry run, i.e. do not delete anything - just show what would be deleted'); - - // Get variable command - program - .command('variable-get') - .description('get variable definitions in one or more apps') - .action(async (options) => { - await sharedParamAssertOptions(options); - variableGetAssertOptions(options); - - getVariable(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--engine-port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--qrs-port ', 'Qlik Sense repository service (QRS) port (usually 4747 for cert auth, 443 for jwt auth)', '4242') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .option('--app-id ', 'Qlik Sense app ID(s) to get variables from') - .option('--app-tag ', 'Qlik Sense app tag(s) to get variables') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .addOption( - new Option('--id-type ', 'type of identifier passed in the --variable option').choices(['id', 'name']).default('name') - ) - .option('--variable ', 'variables to retrieve. If not specified all variables will be retrieved') - .addOption(new Option('--output-format ', 'output format').choices(['json', 'table']).default('json')); - - // Delete variable command - program - .command('variable-delete') - .description('delete one or more variables in one or more apps') - .action(async (options) => { - await sharedParamAssertOptions(options); - variableDeleteAssertOptions(options); - - deleteVariable(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--engine-port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--qrs-port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .option('--app-id ', 'Qlik Sense app ID(s) to get variables from') - .option('--app-tag ', 'Qlik Sense app tag(s) to get variables') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .addOption( - new Option('--id-type ', 'type of identifier passed in the --variable option').choices(['id', 'name']).default('name') - ) - .option('--variable ', 'variables to retrieve. If not specified all variables will be retrieved') - .option('--delete-all', 'delete all variables') - .option('--dry-run', 'do a dry run, i.e. do not delete anything - just show what would be deleted'); - - // Scramble field command - program - .command('field-scramble') - .description('scramble one or more fields in an app. A new app with the scrambled data is created.') - .action(async (options) => { - await sharedParamAssertOptions(options); - - scrambleField(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--app-id ', 'Qlik Sense app ID') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .requiredOption('--field-name ', 'name of field(s) to be scrambled') - .requiredOption('--new-app-name ', 'name of new app that will contain scrambled data'); - - // Get script command - program - .command('script-get') - .description('get script from Qlik Sense app') - .action(async (options) => { - await sharedParamAssertOptions(options); - getScriptAssertOptions(options); - - getScript(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--app-id ', 'Qlik Sense app ID') - .addOption(new Option('--open-without-data ', 'open app without data').choices(['true', 'false']).default('true')) - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server'); + // -------------------------------------------------------- + // Create a command for QSEoW-related sub-commands + function createQseowCommands() { + // Create a new command + // const qseow = program.command('qseow'); + const qseow = new Command('qseow'); - // Get bookmark command - program - .command('bookmark-get') - .description('get info about one or more bookmarks') - .action(async (options) => { - await sharedParamAssertOptions(options); - getBookmarkAssertOptions(options); + // Create custom properties for tracking user activity buckets, i.e. how long ago a user was last active (last login) in Sense + setupQseowUserActivityCustomPropertyCommand(qseow); - getBookmark(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--app-id ', 'Qlik Sense app ID') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') + // QSEoW: Import dimensions/measures from definitions in Excel file + setupQseowMasterItemImportCommand(qseow); - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + // QSEoW: Get measure command + setupQseowGetMasterMeasureCommand(qseow); - .addOption( - new Option('--id-type ', 'type of bookmark identifier passed in the --bookmark option') - .choices(['id', 'name']) - .default('name') - ) - .option('--bookmark ', 'bookmark to retrieve. If not specified all bookmarks will be retrieved') - .option('--output-format ', 'output format', 'json'); + // QSEoW: Delete measure command + setupQseowDeleteMasterMeasureCommand(qseow); - // Get tasks command - program - .command('task-get') - .description('get info about one or more tasks') - .action(async (options) => { - const newOptions = options; - // If options.tableDetails is true, it means --table-details was passed as options without any explicit value. - // This is allowed, but should be interpreted as "all" table details. - // Make options.tableDetails an array with all possible table details. - if (options.tableDetails === true) { - newOptions.tableDetails = ['common', 'lastexecution', 'tag', 'customproperty', 'schematrigger', 'compositetrigger']; - } + // QSEoW: Get dimension command + setupQseowGetMasterDimensionCommand(qseow); - await sharedParamAssertOptions(newOptions); - getTaskAssertOptions(newOptions); + // QSEoW: Delete dimension command + setupQseowDeleteMasterDimensionCommand(qseow); - // If --output-format=table and --task-type is not specified, default to ['reload', 'ext-program'] - if (newOptions.outputFormat === 'table' && !newOptions.taskType) { - newOptions.taskType = ['reload', 'ext-program']; - } + // QSEoW: Get variable command + setpQseowGetVariableCommand(qseow); - getTask(newOptions); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') + // QSEoW: Delete variable command + setupQseowDeleteVariableCommand(qseow); - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + // QSEoW: Scramble field command + setupQseowScrambleFieldCommand(qseow); - .addOption(new Option('--task-type ', 'type of tasks to include').choices(['reload', 'ext-program'])) - .option('--task-id ', 'use task IDs to select which tasks to retrieve. Only allowed when --output-format=table') - .option('--task-tag ', 'use tags to select which tasks to retrieve. Only allowed when --output-format=table') + // QSEoW: Get script command + setupGetScriptCommand(qseow); - .addOption(new Option('--output-format ', 'output format').choices(['table', 'tree']).default('tree')) - .addOption(new Option('--output-dest ', 'where to send task info').choices(['screen', 'file']).default('screen')) - .addOption(new Option('--output-file-name ', 'file name to store task info in').default('')) - .addOption(new Option('--output-file-format ', 'file type/format').choices(['excel', 'csv', 'json']).default('excel')) - .option('--output-file-overwrite', 'overwrite output file without asking') + // QSEoW: Get bookmark command + setupQseowGetBookmarkCommand(qseow); - .addOption(new Option('--text-color ', 'use colored text in task views').choices(['yes', 'no']).default('yes')) + // QSEoW: Get tasks command + setupGetTaskCommand(qseow); - .option('--tree-icons', 'display task status icons in tree view') - .addOption( - new Option('--tree-details [detail...]', 'display details for each task in tree view') - .choices(['taskid', 'laststart', 'laststop', 'nextstart', 'appname', 'appstream']) - .default('') - ) + // QSEoW: Set custom property on tasks command + setupQseowSetTaskCustomPropertyCommand(qseow); - .addOption( - new Option( - '--table-details [detail...]', - 'which aspects of tasks should be included in table view. Not choosing any details will show all' - ) - .choices(['common', 'lastexecution', 'tag', 'customproperty', 'schematrigger', 'compositetrigger']) - .default('') - ); + // QSEoW: Import tasks from definitions in Excel/CSV file + setupQseowImportTaskFromFileCommand(qseow); - // Set custom property on tasks command - program - .command('task-custom-property-set') - .description('update a custom property of one or more tasks') - .action(async (options) => { - await sharedParamAssertOptions(options); - setTaskCustomPropertyAssertOptions(options); + // QSEoW: Import apps from definitions in Excel file + setupQseowImportAppFromFileCommand(qseow); - await setTaskCustomProperty(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + // QSEoW: Export apps to QVF files + setupQseowExportAppCommand(qseow); - .addOption( - new Option('--task-type ', 'type of tasks to list').choices(['reload']).default(['reload']) - // .choices(['reload', 'ext-program']) - // .default(['reload', 'ext-program']) - ) - .option('--task-id ', 'use task IDs to select which tasks to retrieve') - .option('--task-tag ', 'use tags to select which tasks to retrieve') - - .requiredOption('--custom-property-name ', 'name of custom property that will be updated') - .requiredOption('--custom-property-value ', 'one or more values name of custom property that will be updated') - .option('--overwrite', 'overwrite existing custom property values without asking') - .addOption( - new Option('--update-mode ', 'append or replace value(s) to existing custom property') - .choices(['append', 'replace']) - .default('append') - ) - .option('--dry-run', 'do a dry run, i.e. do not modify any reload tasks - just show what would be updated'); + // QSEoW: Test connection command + setupQseowTestConnectionCommand(qseow); - // Import tasks from definitions in Excel/CSV file - program - .command('task-import') - .description('create tasks based on definitions in a file on disk, optionally also importing apps from QVF files.') - .action(async (options) => { - try { - await sharedParamAssertOptions(options); - taskImportAssertOptions(options); - importTaskFromFile(options); - } catch (err) { - catchLog('IMPORT TASK 1', err); - } - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') + // QSEoW: Show version command + setupQseowShowVersionCommand(qseow); - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + // QSEoW: Visualise task network + setupQseowVisualiseTaskCommand(qseow); - .addOption(new Option('-t, --file-type ', 'source file type').choices(['excel', 'csv']).default('excel')) - .requiredOption('--file-name ', 'file containing task definitions') - .option('--sheet-name ', 'name of Excel sheet where task info is found') + // QSEoW: Get proxy sessions + setupQseowGetProxySessionsCommand(qseow); - .addOption(new Option('--update-mode ', 'create new or update existing tasks').choices(['create']).default('create')) + // QSEoW: Delete proxy sessions + setupQseowDeleteProxySessionsCommand(qseow); - .requiredOption( - '--limit-import-count ', - 'import at most x number of tasks from the source file. Defaults to 0 = no limit', - 0 - ) - .requiredOption( - '--sleep-app-upload ', - 'Wait this long before continuing after each app has been uploaded to Sense. Defaults to 1000 = 1 second', - 1000 - ) - - .option('--import-app', 'import Sense app QVFs from specified directory') - .option('--import-app-sheet-name ', 'name of Excel sheet where app definitions are found') - - .option('--dry-run', 'do a dry run, i.e. do not create any reload tasks - just show what would be done'); - - // Import apps from definitions in Excel file - program - .command('app-import') - .description('import apps/upload QVF files on disk to Sense based on definitions in Excel file.') - .action(async (options) => { - try { - await sharedParamAssertOptions(options); - appImportAssertOptions(options); - importAppFromFile(options); - } catch (err) { - catchLog('IMPORT APP', err); - } - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .addOption(new Option('-t, --file-type ', 'source file type').choices(['excel']).default('excel')) - .requiredOption('--file-name ', 'file containing app definitions') - .requiredOption('--sheet-name ', 'name of Excel sheet where app info is found') - - .requiredOption('--limit-import-count ', 'import at most x number of apps. Defaults to 0 = no limit', 0) - .requiredOption( - '--sleep-app-upload ', - 'Wait this long before continuing after each app has been uploaded to Sense. Defaults to 1000 = 1 second', - 1000 - ) - - .option('--dry-run', 'do a dry run, i.e. do not import any apps - just show what would be done'); - - // Export apps to QVF files - program - .command('app-export') - .description('export Qlik Sense apps to QVF files on disk.') - .action(async (options) => { - try { - await sharedParamAssertOptions(options); - await appExportAssertOptions(options); - exportAppToFile(options); - } catch (err) { - catchLog('EXPORT APP', err); - } - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - .option('--app-id ', 'use app IDs to select which apps to export') - .option('--app-tag ', 'use app tags to select which apps to export') - .requiredOption('--app-published', 'export all published apps ', false) - - .requiredOption('--output-dir ', 'relative or absolut path in which QVF files should be stored.', 'qvf-export') - .addOption( - new Option('--qvf-name-format ', 'structure of QVF file name format') - .choices(['app-id', 'app-name', 'export-date', 'export-time']) - .default(['app-name']) - ) - .addOption( - new Option('--qvf-name-separator ', 'character used to separate parts of the QVF file name') - .choices(['-', '--', '_', '__']) - .default('_') - ) - .option('--qvf-overwrite', 'overwrite existing QVF files without asking') + return qseow; + } - .requiredOption('--exclude-app-data ', 'exclude or include app data in QVF file', true) - .requiredOption('--limit-export-count ', 'export at most x number of apps. Defaults to 0 = no limit', 0) - .requiredOption( - '--sleep-app-export ', - 'Wait this long before continuing after each app has been exported. Defaults to 1000 = 1 second', - 1000 - ) - - // Export of app metadata - .option('--metadata-file-create', 'create a separate file with information about all exported apps') - .addOption(new Option('--metadata-file-name ', 'file name to store app metadata in').default('app_export.xlsx')) - .addOption(new Option('--metadata-file-format ', 'file type/format').choices(['excel']).default('excel')) - .option('--metadata-file-overwrite', 'overwrite app metadata file without asking') - - .option('--dry-run', 'do a dry run, i.e. do not export any apps - just show what would be done'); - - // Test connection command - program - .command('connection-test') - .description('test connection to Qlik Sense server.') - .action(async (options) => { - try { - await sharedParamAssertOptions(options); - testConnection(options); - } catch (err) { - catchLog('CONNECTION TEST', err); - } - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense proxy service port', '4242') - .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server'); - - // Show version command - program - .command('version') - .description('show version info') - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - // eslint-disable-next-line no-unused-vars - .action(async (options) => { - logger.verbose(`Version: ${appVersion}`); - }); - - // Visualise task network - program - .command('task-vis') - .description('visualise task network') - .action(async (options) => { - await sharedParamAssertOptions(options); - - await visTask(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - .requiredOption('--host ', 'Qlik Sense server IP/FQDN') - .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') - // .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') - - // Options for visualisation host - .option('--vis-host ', 'host for visualisation server', 'localhost') - .option('--vis-port ', 'port for visualisation server', '3000'); - - // Get proxy sessions - program - .command('session-get') - .description('get info about proxy sessions on one or more virtual proxies') - .action(async (options) => { - await sharedParamAssertOptions(options); - await getSessionsAssertOptions(options); - - getSessions(options, null); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - - .requiredOption('--host ', 'Qlik Sense host (IP/FQDN) where Qlik Repository Service (QRS) is running') - .option('--qrs-port ', 'Qlik Sense repository service (QRS) port (usually 4242)', '4242') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix to access QRS via', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - - .option('--session-virtual-proxy ', 'one or more Qlik Sense virtual proxies to get sessions for') - .option( - '--host-proxy ', - 'Qlik Sense hosts/proxies (IP/FQDN) to get sessions from. Must match the host names of the Sense nodes' - ) - .option('--qps-port ', 'Qlik Sense proxy service (QPS) port (usually 4243)', '4243') - - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') - - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') - - .option('--output-format ', 'output format', 'json') - - .addOption( - new Option('-s, --sort-by ', 'column to sort output table by') - .choices(['prefix', 'proxyhost', 'proxyname', 'userdir', 'userid', 'username']) - .default('prefix') - ); - - // Delete proxy sessions - program - .command('session-delete') - .description('delete proxy session(s) on a specific virtual proxy and proxy service') - .action(async (options) => { - await sharedParamAssertOptions(options); - await deleteSessionsAssertOptions(options); - - deleteSessions(options); - }) - .addOption( - new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') - ) - - .requiredOption('--host ', 'Qlik Sense host (IP/FQDN) where Qlik Repository Service (QRS) is running') - .option('--qrs-port ', 'Qlik Sense repository service (QRS) port (usually 4242)', '4242') - .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix to access QRS via', '') - .requiredOption( - '--secure ', - 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', - true - ) - - .option('--session-id ', 'session IDs to delete') - .requiredOption('--session-virtual-proxy ', 'Qlik Sense virtual proxy (prefix) to delete proxy session(s) on', '') - .requiredOption( - '--host-proxy ', - 'Qlik Sense proxy (IP/FQDN) where sessions should be deleted. Must match the host name of a Sense node' - ) - .option('--qps-port ', 'Qlik Sense proxy service (QPS) port (usually 4243)', '4243') + // -------------------------------------------------------- + // Create a command for QS Cloud-related sub-commands + function createQsCloudCommands() { + // Create a new command + const qsCloud = new Command('qscloud'); - .requiredOption('--auth-user-dir ', 'user directory for user to connect with') - .requiredOption('--auth-user-id ', 'user ID for user to connect with') + // QSEoW: Test connection command + setupQscloudTestConnectionCommand(qsCloud); - .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert']).default('cert')) - .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') - .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') - .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem'); + return qsCloud; + } - // .option('--dry-run', 'do a dry run, i.e. do not delete any sessions - just show what would be deleted') + // Add all command definitions + program.addCommand(createQseowCommands()); + program.addCommand(createQsCloudCommands()); - // Parse command line params + // Parse command definitions await program.parseAsync(process.argv); })(); diff --git a/src/globals.js b/src/globals.js index 072c975..30499a8 100644 --- a/src/globals.js +++ b/src/globals.js @@ -1,15 +1,17 @@ import winston from 'winston'; import upath from 'upath'; -import { fileURLToPath } from 'url'; -import { readFileSync, promises as Fs } from 'fs'; +import { fileURLToPath } from 'node:url'; +import { readFileSync, promises as Fs } from 'node:fs'; import 'winston-daily-rotate-file'; +import sea from 'node:sea'; -// Get app version from package.json file // Get app version from package.json file const filenamePackage = `./package.json`; let a; let b; let c; +export let appVersion; + // Are we running as a packaged app? if (process.pkg) { // Get path to JS file @@ -20,6 +22,15 @@ if (process.pkg) { // Add path to package.json file c = upath.join(b, filenamePackage); + + const { version } = JSON.parse(readFileSync(c)); + appVersion = version; +} else if (sea.isSea()) { + // Get contents of package.json file + packageJson = sea.getAsset('package.json', 'utf8'); + const version = JSON.parse(packageJson).version; + + appVersion = version; } else { // Get path to JS file a = fileURLToPath(import.meta.url); @@ -29,10 +40,10 @@ if (process.pkg) { // Add path to package.json file c = upath.join(b, '..', filenamePackage); -} -const { version } = JSON.parse(readFileSync(c)); -export const appVersion = version; + const { version } = JSON.parse(readFileSync(c)); + appVersion = version; +} // Set up logger with timestamps and colors, and optional logging to disk file const logTransports = []; @@ -122,7 +133,7 @@ export const mergeDirFilePath = (pathElements) => { export const generateXrfKey = () => { let xrfString = ''; - // eslint-disable-next-line no-plusplus + for (let i = 0; i < 16; i++) { if (Math.floor(Math.random() * 2) === 0) { xrfString += Math.floor(Math.random() * 10).toString(); @@ -156,7 +167,6 @@ export function isNumeric(str) { } export function sleep(ms) { - // eslint-disable-next-line no-promise-executor-return return new Promise((resolve) => setTimeout(resolve, ms)); } diff --git a/src/lib/app/class_allapps.js b/src/lib/app/class_allapps.js index 99c2091..4d0e837 100644 --- a/src/lib/app/class_allapps.js +++ b/src/lib/app/class_allapps.js @@ -1,22 +1,23 @@ import * as rax from 'retry-axios'; import axios from 'axios'; -import path from 'path'; +import path from 'node:path'; import FormData from 'form-data'; -import fs from 'fs/promises'; -import fs2 from 'fs'; +import fs from 'node:fs/promises'; +import fs2 from 'node:fs'; import { v4 as uuidv4, validate } from 'uuid'; import yesno from 'yesno'; -import { logger, execPath, mergeDirFilePath, verifyFileExists, sleep, isPkg } from '../../globals.js'; -import setupQRSConnection from '../util/qrs.js'; -import { getAppColumnPosFromHeaderRow } from '../util/lookups.js'; + +import { logger, execPath, mergeDirFilePath, verifyFileExists, sleep } from '../../globals.js'; +import { setupQrsConnection } from '../util/qseow/qrs.js'; +import { getAppColumnPosFromHeaderRow } from '../util/qseow/lookups.js'; import QlikSenseApp from './class_app.js'; -import { getTagIdByName } from '../util/tag.js'; -import { getAppById, deleteAppById } from '../util/app.js'; -import { getCustomPropertyDefinitionByName, doesCustomPropertyValueExist } from '../util/customproperties.js'; +import { getTagIdByName } from '../util/qseow/tag.js'; +import { getAppById, deleteAppById } from '../util/qseow/app.js'; +import { getCustomPropertyDefinitionByName, doesCustomPropertyValueExist } from '../util/qseow/customproperties.js'; import { catchLog } from '../util/log.js'; +import { getCertFilePaths } from '../util/qseow/cert.js'; class QlikSenseApps { - // eslint-disable-next-line no-useless-constructor constructor() { // } @@ -26,12 +27,14 @@ class QlikSenseApps { this.appList = []; this.options = options; - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? if (options.authType === 'cert') { - // Make sure certificates exist - this.fileCert = path.resolve(execPath, options.authCertFile); - this.fileCertKey = path.resolve(execPath, options.authCertKeyFile); - this.fileCertCA = path.resolve(execPath, options.authRootCertFile); + // Get certificate paths + const { fileCert, fileCertKey, fileCertCA } = getCertFilePaths(options); + + this.fileCert = fileCert; + this.fileCertKey = fileCertKey; + this.fileCertCA = fileCertCA; } // Map that will connect app counter from Excel file with ID an app gets after import to QSEoW @@ -118,11 +121,11 @@ class QlikSenseApps { } } - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfig; if (this.options.authType === 'cert') { if (filter === '') { - axiosConfig = await setupQRSConnection(this.options, { + axiosConfig = await setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -130,7 +133,7 @@ class QlikSenseApps { path: '/qrs/app/full', }); } else { - axiosConfig = await setupQRSConnection(this.options, { + axiosConfig = await setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -141,12 +144,12 @@ class QlikSenseApps { } } else if (this.options.authType === 'jwt') { if (filter === '') { - axiosConfig = await setupQRSConnection(this.options, { + axiosConfig = await setupQrsConnection(this.options, { method: 'get', path: '/qrs/app/full', }); } else { - axiosConfig = await setupQRSConnection(this.options, { + axiosConfig = await setupQrsConnection(this.options, { method: 'get', path: '/qrs/app/full', queryParameters: [{ name: 'filter', value: filter }], @@ -162,7 +165,6 @@ class QlikSenseApps { this.clear(); for (let i = 0; i < apps.length; i += 1) { - // eslint-disable-next-line no-await-in-loop await this.addApp(apps[i], apps[i].id); } @@ -175,7 +177,6 @@ class QlikSenseApps { } async importAppsFromFiles(appsFromFile, tagsExisting, cpExisting) { - // eslint-disable-next-line no-async-promise-executor return new Promise(async (resolve, reject) => { logger.debug('PARSE APPS FROM EXCEL FILE: Starting get apps from data in file'); @@ -263,7 +264,6 @@ class QlikSenseApps { }"` ); - // eslint-disable-next-line no-await-in-loop const qvfFileExists = await verifyFileExists(currentApp.fullQvfPath); if (!qvfFileExists) { logger.error( @@ -281,9 +281,7 @@ class QlikSenseApps { .filter((item) => item.trim().length !== 0) .map((item) => item.trim()); - // eslint-disable-next-line no-restricted-syntax for (const item of tmpTags) { - // eslint-disable-next-line no-await-in-loop const tagId = await getTagIdByName(item, tagsExisting); if (tagId === false) { // Failed getting tag id, given name. Most likely becuase the tag doesn't exist @@ -308,7 +306,6 @@ class QlikSenseApps { .filter((item) => item.trim().length !== 0) .map((cp) => cp.trim()); - // eslint-disable-next-line no-restricted-syntax for (const item of tmpCustomProperties) { const tmpCustomProperty = item .split('=') @@ -316,8 +313,7 @@ class QlikSenseApps { .map((cp) => cp.trim()); if (tmpCustomProperty?.length === 2) { - // eslint-disable-next-line no-await-in-loop - const customProperty = await getCustomPropertyDefinitionByName('App', tmpCustomProperty[0], cpExisting); + const customProperty = getCustomPropertyDefinitionByName('App', tmpCustomProperty[0], cpExisting); if (customProperty === false) { // Failed getting custom property id, most likely because the custom property does not exist. logger.error( @@ -329,8 +325,7 @@ class QlikSenseApps { } // Verify custom property value is valid - // eslint-disable-next-line no-await-in-loop - const cpValueExists = await doesCustomPropertyValueExist( + const cpValueExists = doesCustomPropertyValueExist( 'App', tmpCustomProperty[0], tmpCustomProperty[1], @@ -357,7 +352,6 @@ class QlikSenseApps { // Import app to QSEoW if (this.options.dryRun === false || this.options.dryRun === undefined) { // 1. Upload the app specified in the Excel file. - // eslint-disable-next-line no-await-in-loop const uploadedAppId = await this.uploadAppToQseow(currentApp); // false returned if the app could not be uploaded to Sense @@ -370,21 +364,18 @@ class QlikSenseApps { currentApp.createdAppId = uploadedAppId; // Update tags, custom properties and owner of uploaded app - // eslint-disable-next-line no-await-in-loop const result = await this.updateUploadedApp(currentApp, uploadedAppId); // Should the app be published to a stream? if (currentApp?.appPublishToStream?.length > 0) { // Yes, publish to stream after app upload - // eslint-disable-next-line no-await-in-loop const { streamId, streamName } = await this.getStreamInfo(currentApp); let tmpAppId; // Do we know which stream to publish to? Publish if so! if (streamId) { if (currentApp.appPublishToStreamOption === 'publish-replace') { - // eslint-disable-next-line no-await-in-loop const result2 = await this.streamAppPublishReplace( currentApp.appCounter, uploadedAppId, @@ -422,10 +413,8 @@ class QlikSenseApps { this.appCounterIdMap.set(tmpAppId, uploadedAppId); } - // eslint-disable-next-line no-await-in-loop await this.addApp(currentApp, tmpAppId); } else if (currentApp.appPublishToStreamOption === 'publish-another') { - // eslint-disable-next-line no-await-in-loop const result2 = await this.streamAppPublishAnother( currentApp.appCounter, uploadedAppId, @@ -461,10 +450,8 @@ class QlikSenseApps { this.appCounterIdMap.set(tmpAppId, uploadedAppId); } - // eslint-disable-next-line no-await-in-loop await this.addApp(currentApp, tmpAppId); } else if (currentApp.appPublishToStreamOption === 'delete-publish') { - // eslint-disable-next-line no-await-in-loop const result2 = await this.streamAppDeletePublish( currentApp.appCounter, uploadedAppId, @@ -503,7 +490,6 @@ class QlikSenseApps { this.appCounterIdMap.set(tmpAppId, uploadedAppId); } - // eslint-disable-next-line no-await-in-loop await this.addApp(currentApp, tmpAppId); } else { logger.error( @@ -528,7 +514,6 @@ class QlikSenseApps { const tmpAppId = `newapp-${currentApp.appCounter}`; this.appCounterIdMap.set(tmpAppId, uploadedAppId); - // eslint-disable-next-line no-await-in-loop await this.addApp(currentApp, tmpAppId); } } else { @@ -546,7 +531,6 @@ class QlikSenseApps { const tmpAppId = `newapp-${currentApp.appCounter}`; this.appCounterIdMap.set(tmpAppId, uploadedAppId); - // eslint-disable-next-line no-await-in-loop await this.addApp(currentApp, tmpAppId); } } else { @@ -562,11 +546,11 @@ class QlikSenseApps { // Function to update tags, custom properties and owner of uploaded app async updateUploadedApp(newApp, uploadedAppId) { try { - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfigUploadedApp; if (this.options.authType === 'cert') { // Get info about just uploaded app - axiosConfigUploadedApp = setupQRSConnection(this.options, { + axiosConfigUploadedApp = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -574,7 +558,7 @@ class QlikSenseApps { path: `/qrs/app/${uploadedAppId}`, }); } else if (this.options.authType === 'jwt') { - axiosConfigUploadedApp = setupQRSConnection(this.options, { + axiosConfigUploadedApp = setupQrsConnection(this.options, { method: 'get', path: `/qrs/app/${uploadedAppId}`, }); @@ -604,11 +588,11 @@ class QlikSenseApps { `userDirectory eq '${newApp.appOwnerUserDirectory}' and userId eq '${newApp.appOwnerUserId}'` ); - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfigUser; if (this.options.authType === 'cert') { // Get info about just uploaded app - axiosConfigUser = setupQRSConnection(this.options, { + axiosConfigUser = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -617,7 +601,7 @@ class QlikSenseApps { queryParameters: [{ name: 'filter', value: filter }], }); } else if (this.options.authType === 'jwt') { - axiosConfigUser = setupQRSConnection(this.options, { + axiosConfigUser = setupQrsConnection(this.options, { method: 'get', path: '/qrs/user', queryParameters: [{ name: 'filter', value: filter }], @@ -653,11 +637,11 @@ class QlikSenseApps { // Pause for a while to let Sense repository catch up await sleep(1000); - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfig2; if (this.options.authType === 'cert') { // Uppdate app with tags, custom properties and app owner - axiosConfig2 = setupQRSConnection(this.options, { + axiosConfig2 = setupQrsConnection(this.options, { method: 'put', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -666,7 +650,7 @@ class QlikSenseApps { body: app, }); } else if (this.options.authType === 'jwt') { - axiosConfig2 = setupQRSConnection(this.options, { + axiosConfig2 = setupQrsConnection(this.options, { method: 'put', path: `/qrs/app/${app.id}`, body: app, @@ -875,11 +859,11 @@ class QlikSenseApps { { name: 'name', value: appName }, ]; - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfig; if (this.options.authType === 'cert') { // Build QRS query - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'put', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -888,7 +872,7 @@ class QlikSenseApps { queryParameters, }); } else if (this.options.authType === 'jwt') { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'put', path: `/qrs/app/${appId}/publish`, queryParameters, @@ -921,11 +905,11 @@ class QlikSenseApps { // Define query parameters const queryParameters = [{ name: 'app', value: targetAppId }]; - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfig; if (this.options.authType === 'cert') { // Build QRS query - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'put', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -934,7 +918,7 @@ class QlikSenseApps { queryParameters, }); } else if (this.options.authType === 'jwt') { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'put', path: `/qrs/app/${sourceAppId}/replace`, queryParameters, @@ -982,11 +966,11 @@ class QlikSenseApps { filter = encodeURIComponent(`stream.name eq '${streamName}' and name eq '${appName}'`); } - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfig; if (this.options.authType === 'cert') { // Build QRS query - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -995,7 +979,7 @@ class QlikSenseApps { queryParameters: [{ name: 'filter', value: filter }], }); } else if (this.options.authType === 'jwt') { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', path: `/qrs/app`, queryParameters: [{ name: 'filter', value: filter }], @@ -1029,11 +1013,11 @@ class QlikSenseApps { // Build QRS query const filter = encodeURIComponent(`stream.name eq '${streamName}' and name eq '${appName}'`); - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfig; if (this.options.authType === 'cert') { // Build QRS query - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1042,7 +1026,7 @@ class QlikSenseApps { queryParameters: [{ name: 'filter', value: filter }], }); } else if (this.options.authType === 'jwt') { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', path: `/qrs/app`, queryParameters: [{ name: 'filter', value: filter }], @@ -1093,10 +1077,10 @@ class QlikSenseApps { if (validate(uploadedAppInfo.appPublishToStream)) { // It's a valid GUID - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? if (this.options.authType === 'cert') { // Build QRS query - axiosConfigPublish = setupQRSConnection(this.options, { + axiosConfigPublish = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1104,7 +1088,7 @@ class QlikSenseApps { path: `/qrs/stream/${uploadedAppInfo.appPublishToStream}`, }); } else if (this.options.authType === 'jwt') { - axiosConfigPublish = setupQRSConnection(this.options, { + axiosConfigPublish = setupQrsConnection(this.options, { method: 'get', path: `/qrs/stream/${uploadedAppInfo.appPublishToStream}`, }); @@ -1115,17 +1099,16 @@ class QlikSenseApps { // Yes, the GUID represents a stream responsePublish = JSON.parse(resultPublish.data); - // eslint-disable-next-line prefer-destructuring return { streamId: responsePublish.id, streamName: responsePublish.name }; } } else { // Provided stream name is not a GUID, make sure only one stream exists with this name, then get its GUID const filter = encodeURIComponent(`name eq '${uploadedAppInfo.appPublishToStream}'`); - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? if (this.options.authType === 'cert') { // Build QRS query - axiosConfigPublish = setupQRSConnection(this.options, { + axiosConfigPublish = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1134,7 +1117,7 @@ class QlikSenseApps { queryParameters: [{ name: 'filter', value: filter }], }); } else if (this.options.authType === 'jwt') { - axiosConfigPublish = setupQRSConnection(this.options, { + axiosConfigPublish = setupQrsConnection(this.options, { method: 'get', path: '/qrs/stream', queryParameters: [{ name: 'filter', value: filter }], @@ -1192,7 +1175,7 @@ class QlikSenseApps { form.append('qvfFile', sourceFileBuffer, newApp.qvfName); // Build Axios config - const axiosConfig = setupQRSConnection(this.options, { + const axiosConfig = setupQrsConnection(this.options, { method: 'post', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1209,8 +1192,9 @@ class QlikSenseApps { ], }); + const retryCount = 5; axiosConfig.raxConfig = { - retry: 8, + retry: retryCount, noResponseRetries: 2, httpMethodsToRetry: ['GET', 'HEAD', 'OPTIONS', 'DELETE', 'PUT', 'POST'], statusCodesToRetry: [ @@ -1234,15 +1218,18 @@ class QlikSenseApps { if (status === 429) { logger.warn(`🔄 [${status}] QRS API rate limit reached. Pausing, then retry attempt #${cfg.currentRetryAttempt}`); } else { - logger.warn(`🔄 [${status}] Error from QRS API. Pausing, then retry attempt #${cfg.currentRetryAttempt}`); + logger.warn( + `🔄 [${status}] Error from QRS API. Pausing, then retry attempt #${cfg.currentRetryAttempt} of ${retryCount}. Error message: ${err.message}` + ); + } + + // Message when last attempt has failed + if (cfg.currentRetryAttempt === retryCount) { + logger.warn(`🔄 [${status}] Final attempt failed. Is Qlik Sense responding? Is the QVF file valid?`); } }, }; - // axiosConfig.baseURL = 'https://httpstat.us'; - // axiosConfig.url = '/429'; - // axiosConfig.method = 'get'; - const myAxiosInstance = axios.create(axiosConfig); myAxiosInstance.defaults.raxConfig = { @@ -1283,11 +1270,11 @@ class QlikSenseApps { const exportToken = uuidv4(); const excludeData = this.options.excludeAppData === 'true' ? 'true' : 'false'; - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfig; if (this.options.authType === 'cert') { // Build QRS query - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'post', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1296,7 +1283,7 @@ class QlikSenseApps { queryParameters: [{ name: 'skipData', value: excludeData }], }); } else if (this.options.authType === 'jwt') { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'post', path: `/qrs/app/${app.id}/export/${exportToken}`, queryParameters: [{ name: 'skipData', value: excludeData }], @@ -1396,11 +1383,11 @@ class QlikSenseApps { } else { writer = fs2.createWriteStream(fileName); - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? let axiosConfig; if (this.options.authType === 'cert') { // Build QRS query - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1409,7 +1396,7 @@ class QlikSenseApps { queryParameters: [{ name: paramName, value: paramValue }], }); } else if (this.options.authType === 'jwt') { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', path: urlPath, queryParameters: [{ name: paramName, value: paramValue }], diff --git a/src/lib/cli/qscloud-test-connection.js b/src/lib/cli/qscloud-test-connection.js new file mode 100644 index 0000000..c12cd85 --- /dev/null +++ b/src/lib/cli/qscloud-test-connection.js @@ -0,0 +1,27 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qscloudSharedParamAssertOptions } from '../util/qscloud/assert-options.js'; +import { qscloudTestConnection } from '../cmd/qscloud/testconnection.js'; + +export function setupQscloudTestConnectionCommand(qsCloud) { + qsCloud + .command('connection-test') + .description('test connection to Qlik Sense Cloud.') + .action(async (options) => { + try { + await qscloudSharedParamAssertOptions(options); + + await qscloudTestConnection(options); + } catch (err) { + catchLog('QS CLOUD CONNECTION TEST', err); + } + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + + .requiredOption('--tenant-host ', 'Host of Qlik Sense cloud tenant. Example: "tenant.eu.qlikcloud.com"') + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['apikey']).default('apikey')) + .requiredOption('--apikey ', 'API key used to access the Sense APIs'); +} diff --git a/src/lib/cli/qseow-cp-user-activity-bucket.js b/src/lib/cli/qseow-cp-user-activity-bucket.js new file mode 100644 index 0000000..45ae077 --- /dev/null +++ b/src/lib/cli/qseow-cp-user-activity-bucket.js @@ -0,0 +1,93 @@ +import { Option, InvalidArgumentError } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, userActivityBucketsCustomPropertyAssertOptions } from '../util/qseow/assert-options.js'; +import { createUserActivityBucketsCustomProperty } from '../cmd/qseow/createuseractivitycp.js'; + +// Function to parse update batch size +// Must be a number between 1 and 25 +function parseUpdateBatchSize(value) { + console.log('sdklfjsdlkfjsdlkfjsdlfkj'); + const parsedValue = parseInt(value, 10); + if (isNaN(parsedValue) || parsedValue < 1 || parsedValue > 25) { + throw new InvalidArgumentError('Must be a number between 1 and 25.'); + } + return parsedValue; +} + +export function setupQseowUserActivityCustomPropertyCommand(qseow) { + qseow + .command('user-activity-bucket-cp-create') + .description( + 'create custom property and populate it with values ("activity buckets") indicating how long ago users last logged into Sense' + ) + .action(async (options) => { + try { + const newOptions = options; + + await qseowSharedParamAssertOptions(newOptions); + await userActivityBucketsCustomPropertyAssertOptions(newOptions); + + const result = await createUserActivityBucketsCustomProperty(newOptions); + } catch (err) { + catchLog('USER ACTIVITY BUCKET CUSTOM PROPERTY', err); + } + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense repository API port', '4242') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .option('--auth-user-dir ', 'user directory for user to connect with', 'Internal') + .option('--auth-user-id ', 'user ID for user to connect with', 'sa_repository') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--jwt ', 'JSON Web Token (JWT) to use for authenticating with Qlik Sense', '') + + .option('--user-directory ', 'name of user directories whose users will be updated with activity info', '') + .addOption( + new Option( + '--license-type ', + 'license type(s) to consider when calculating user activity. Default is all license types.' + ) + .choices(['analyzer', 'analyzer-time', 'login', 'professional', 'user']) + .default(['analyzer', 'analyzer-time', 'login', 'professional', 'user']) + ) + + .requiredOption('--custom-property-name ', 'name of custom property that will hold user activity buckets') + .addOption( + new Option('--force', 'forcibly overwrite and replace custom property and its values if the custom property already exists') + ) + .option( + '--activity-buckets ', + 'custom property values/user activity buckets to be defined. A comma or space separated list of numbers, representing days since last login.', + ['1', '7', '14', '30', '90', '180', '365'] + ) + .option( + '--update-batch-size ', + 'number of users to update in each batch when writing user activity info back into Sense. Valid values are 1-25.', + parseUpdateBatchSize, + 25 + ) + .option( + '--update-batch-sleep ', + 'Wait this long before continuing after each batch of users has been updated in Sense. 0 = no wait.', + 3 + ) + .option( + '--update-user-sleep ', + 'Wait this long after updating each user in the Qlik Sense repository. 0 = no wait.', + 500 + ) + + .option('--dry-run', 'do a dry run, i.e. do not create or update anything - just show what would be done'); +} diff --git a/src/lib/cli/qseow-delete-master-dimension.js b/src/lib/cli/qseow-delete-master-dimension.js new file mode 100644 index 0000000..66739f2 --- /dev/null +++ b/src/lib/cli/qseow-delete-master-dimension.js @@ -0,0 +1,43 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, masterItemDimDeleteAssertOptions } from '../util/qseow/assert-options.js'; +import deleteMasterDimension from '../cmd/qseow/getdim.js'; + +export function setupQseowDeleteMasterDimensionCommand(qseow) { + qseow + .command('master-item-dim-delete') + .description('delete master dimension(s)') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + masterItemDimDeleteAssertOptions(options); + + deleteMasterDimension(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--app-id ', 'Qlik Sense app ID') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption(new Option('--id-type ', 'type of identifier passed in the --master-item option').choices(['id', 'name'])) + .option('--master-item ', 'names or IDs of master dimensions to be deleted. Multiple IDs should be space separated') + .option('--delete-all', 'delete all master dimensions') + .option('--dry-run', 'do a dry run, i.e. do not delete anything - just show what would be deleted'); +} diff --git a/src/lib/cli/qseow-delete-master-measure.js b/src/lib/cli/qseow-delete-master-measure.js new file mode 100644 index 0000000..cb5c045 --- /dev/null +++ b/src/lib/cli/qseow-delete-master-measure.js @@ -0,0 +1,43 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, masterItemMeasureDeleteAssertOptions } from '../util/qseow/assert-options.js'; +import deleteMasterMeasure from '../cmd/qseow/deletemeasure.js'; + +export function setupQseowDeleteMasterMeasureCommand(qseow) { + qseow + .command('master-item-measure-delete') + .description('delete master measure(s)') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + masterItemMeasureDeleteAssertOptions(options); + + deleteMasterMeasure(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--app-id ', 'Qlik Sense app ID') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption(new Option('--id-type ', 'type of identifier passed in the --master-item option').choices(['id', 'name'])) + .option('--master-item ', 'names or IDs of master measures to be deleted. Multiple IDs should be space separated') + .option('--delete-all', 'delete all master measures') + .option('--dry-run', 'do a dry run, i.e. do not delete anything - just show what would be deleted'); +} diff --git a/src/lib/cli/qseow-delete-proxy-session.js b/src/lib/cli/qseow-delete-proxy-session.js new file mode 100644 index 0000000..4b23249 --- /dev/null +++ b/src/lib/cli/qseow-delete-proxy-session.js @@ -0,0 +1,45 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, deleteSessionsAssertOptions } from '../util/qseow/assert-options.js'; +import deleteSessions from '../cmd/qseow/deletesessions.js'; + +export function setupQseowDeleteProxySessionsCommand(qseow) { + qseow + .command('session-delete') + .description('delete proxy session(s) on a specific virtual proxy and proxy service') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + await deleteSessionsAssertOptions(options); + + deleteSessions(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + + .requiredOption('--host ', 'Qlik Sense host (IP/FQDN) where Qlik Repository Service (QRS) is running') + .option('--qrs-port ', 'Qlik Sense repository service (QRS) port (usually 4242)', '4242') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix to access QRS via', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + + .option('--session-id ', 'session IDs to delete') + .requiredOption('--session-virtual-proxy ', 'Qlik Sense virtual proxy (prefix) to delete proxy session(s) on', '') + .requiredOption( + '--host-proxy ', + 'Qlik Sense proxy (IP/FQDN) where sessions should be deleted. Must match the host name of a Sense node' + ) + .option('--qps-port ', 'Qlik Sense proxy service (QPS) port (usually 4243)', '4243') + + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem'); +} diff --git a/src/lib/cli/qseow-delete-variable.js b/src/lib/cli/qseow-delete-variable.js new file mode 100644 index 0000000..6046f83 --- /dev/null +++ b/src/lib/cli/qseow-delete-variable.js @@ -0,0 +1,47 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, variableDeleteAssertOptions } from '../util/qseow/assert-options.js'; +import deleteVariable from '../cmd/qseow/deletevariable.js'; + +export function setupQseowDeleteVariableCommand(qseow) { + qseow + .command('variable-delete') + .description('delete one or more variables in one or more apps') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + variableDeleteAssertOptions(options); + + deleteVariable(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--engine-port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--qrs-port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .option('--app-id ', 'Qlik Sense app ID(s) to get variables from') + .option('--app-tag ', 'Qlik Sense app tag(s) to get variables') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption( + new Option('--id-type ', 'type of identifier passed in the --variable option').choices(['id', 'name']).default('name') + ) + .option('--variable ', 'variables to retrieve. If not specified all variables will be retrieved') + .option('--delete-all', 'delete all variables') + .option('--dry-run', 'do a dry run, i.e. do not delete anything - just show what would be deleted'); +} diff --git a/src/lib/cli/qseow-export-app-to-file.js b/src/lib/cli/qseow-export-app-to-file.js new file mode 100644 index 0000000..145646a --- /dev/null +++ b/src/lib/cli/qseow-export-app-to-file.js @@ -0,0 +1,73 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, appExportAssertOptions } from '../util/qseow/assert-options.js'; +import exportAppToFile from '../cmd/qseow/importtask.js'; + +export function setupQseowExportAppCommand(qseow) { + qseow + .command('app-export') + .description('export Qlik Sense apps to QVF files on disk.') + .action(async (options) => { + try { + await qseowSharedParamAssertOptions(options); + await appExportAssertOptions(options); + exportAppToFile(options); + } catch (err) { + catchLog('EXPORT APP', err); + } + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .option('--app-id ', 'use app IDs to select which apps to export') + .option('--app-tag ', 'use app tags to select which apps to export') + .requiredOption('--app-published', 'export all published apps ', false) + + .requiredOption('--output-dir ', 'relative or absolut path in which QVF files should be stored.', 'qvf-export') + .addOption( + new Option('--qvf-name-format ', 'structure of QVF file name format') + .choices(['app-id', 'app-name', 'export-date', 'export-time']) + .default(['app-name']) + ) + .addOption( + new Option('--qvf-name-separator ', 'character used to separate parts of the QVF file name') + .choices(['-', '--', '_', '__']) + .default('_') + ) + .option('--qvf-overwrite', 'overwrite existing QVF files without asking') + + .requiredOption('--exclude-app-data ', 'exclude or include app data in QVF file', true) + .requiredOption('--limit-export-count ', 'export at most x number of apps. Defaults to 0 = no limit', 0) + .requiredOption( + '--sleep-app-export ', + 'Wait this long before continuing after each app has been exported. Defaults to 1000 = 1 second', + 1000 + ) + + // Export of app metadata + .option('--metadata-file-create', 'create a separate file with information about all exported apps') + .addOption(new Option('--metadata-file-name ', 'file name to store app metadata in').default('app_export.xlsx')) + .addOption(new Option('--metadata-file-format ', 'file type/format').choices(['excel']).default('excel')) + .option('--metadata-file-overwrite', 'overwrite app metadata file without asking') + + .option('--dry-run', 'do a dry run, i.e. do not export any apps - just show what would be done'); +} diff --git a/src/lib/cli/qseow-get-bookmark.js b/src/lib/cli/qseow-get-bookmark.js new file mode 100644 index 0000000..1335766 --- /dev/null +++ b/src/lib/cli/qseow-get-bookmark.js @@ -0,0 +1,46 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, getBookmarkAssertOptions } from '../util/qseow/assert-options.js'; +import getBookmark from '../cmd/qseow/getbookmark.js'; + +export function setupQseowGetBookmarkCommand(qseow) { + qseow + .command('bookmark-get') + .description('get info about one or more bookmarks') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + getBookmarkAssertOptions(options); + + getBookmark(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--app-id ', 'Qlik Sense app ID') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption( + new Option('--id-type ', 'type of bookmark identifier passed in the --bookmark option') + .choices(['id', 'name']) + .default('name') + ) + .option('--bookmark ', 'bookmark to retrieve. If not specified all bookmarks will be retrieved') + .option('--output-format ', 'output format', 'json'); +} diff --git a/src/lib/cli/qseow-get-master-dimension.js b/src/lib/cli/qseow-get-master-dimension.js new file mode 100644 index 0000000..424c1cf --- /dev/null +++ b/src/lib/cli/qseow-get-master-dimension.js @@ -0,0 +1,43 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions } from '../util/qseow/assert-options.js'; +import getMasterDimension from '../cmd/qseow/getdim.js'; + +export function setupQseowGetMasterDimensionCommand(qseow) { + qseow + .command('master-item-dim-get') + .description('get info about one or more master dimensions') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + + getMasterDimension(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--app-id ', 'Qlik Sense app ID') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .requiredOption('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .requiredOption('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .requiredOption('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption( + new Option('--id-type ', 'type of identifier passed in the --master-item option').choices(['id', 'name']).default('name') + ) + .option('--master-item ', 'master dimension to retrieve. If not specified all dimensions will be retrieved') + .addOption(new Option('--output-format ', 'output format').choices(['json', 'table']).default('json')); +} diff --git a/src/lib/cli/qseow-get-master-measure.js b/src/lib/cli/qseow-get-master-measure.js new file mode 100644 index 0000000..0d229c8 --- /dev/null +++ b/src/lib/cli/qseow-get-master-measure.js @@ -0,0 +1,44 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, masterItemGetAssertOptions } from '../util/qseow/assert-options.js'; +import getMasterMeasure from '../cmd/qseow/getmeasure.js'; + +export function setupQseowGetMasterMeasureCommand(qseow) { + qseow + .command('master-item-measure-get') + .description('get info about one or more master measures') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + masterItemGetAssertOptions(options); + + getMasterMeasure(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--app-id ', 'Qlik Sense app ID') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption( + new Option('--id-type ', 'type of identifier passed in the --master-item option').choices(['id', 'name']).default('name') + ) + .option('--master-item ', 'master measure to retrieve. If not specified all measures will be retrieved') + .addOption(new Option('--output-format ', 'output format').choices(['json', 'table']).default('json')); +} diff --git a/src/lib/cli/qseow-get-proxy-session.js b/src/lib/cli/qseow-get-proxy-session.js new file mode 100644 index 0000000..766a0dd --- /dev/null +++ b/src/lib/cli/qseow-get-proxy-session.js @@ -0,0 +1,52 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, getSessionsAssertOptions } from '../util/qseow/assert-options.js'; +import getSessions from '../cmd/qseow/getsessions.js'; + +export function setupQseowGetProxySessionsCommand(qseow) { + qseow + .command('session-get') + .description('get info about proxy sessions on one or more virtual proxies') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + await getSessionsAssertOptions(options); + + getSessions(options, null); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + + .requiredOption('--host ', 'Qlik Sense host (IP/FQDN) where Qlik Repository Service (QRS) is running') + .option('--qrs-port ', 'Qlik Sense repository service (QRS) port (usually 4242)', '4242') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix to access QRS via', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + + .option('--session-virtual-proxy ', 'one or more Qlik Sense virtual proxies to get sessions for') + .option( + '--host-proxy ', + 'Qlik Sense hosts/proxies (IP/FQDN) to get sessions from. Must match the host names of the Sense nodes' + ) + .option('--qps-port ', 'Qlik Sense proxy service (QPS) port (usually 4243)', '4243') + + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + + .option('--output-format ', 'output format', 'json') + + .addOption( + new Option('-s, --sort-by ', 'column to sort output table by') + .choices(['prefix', 'proxyhost', 'proxyname', 'userdir', 'userid', 'username']) + .default('prefix') + ); +} diff --git a/src/lib/cli/qseow-get-script.js b/src/lib/cli/qseow-get-script.js new file mode 100644 index 0000000..d0a76d6 --- /dev/null +++ b/src/lib/cli/qseow-get-script.js @@ -0,0 +1,39 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, getScriptAssertOptions } from '../util/qseow/assert-options.js'; +import getScript from '../cmd/qseow/getscript.js'; + +export function setupGetScriptCommand(qseow) { + qseow + .command('script-get') + .description('get script from Qlik Sense app') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + getScriptAssertOptions(options); + + getScript(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--app-id ', 'Qlik Sense app ID') + .addOption(new Option('--open-without-data ', 'open app without data').choices(['true', 'false']).default('true')) + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server'); +} diff --git a/src/lib/cli/qseow-get-task.js b/src/lib/cli/qseow-get-task.js new file mode 100644 index 0000000..66d369b --- /dev/null +++ b/src/lib/cli/qseow-get-task.js @@ -0,0 +1,78 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, getTaskAssertOptions } from '../util/qseow/assert-options.js'; +import getTask from '../cmd/qseow/gettask.js'; + +export function setupGetTaskCommand(qseow) { + qseow + .command('task-get') + .description('get info about one or more tasks') + .action(async (options) => { + const newOptions = options; + // If options.tableDetails is true, it means --table-details was passed as options without any explicit value. + // This is allowed, but should be interpreted as "all" table details. + // Make options.tableDetails an array with all possible table details. + if (options.tableDetails === true) { + newOptions.tableDetails = ['common', 'lastexecution', 'tag', 'customproperty', 'schematrigger', 'compositetrigger']; + } + + await qseowSharedParamAssertOptions(newOptions); + getTaskAssertOptions(newOptions); + + // If --output-format=table and --task-type is not specified, default to ['reload', 'ext-program'] + if (newOptions.outputFormat === 'table' && !newOptions.taskType) { + newOptions.taskType = ['reload', 'ext-program']; + } + + getTask(newOptions); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption(new Option('--task-type ', 'type of tasks to include').choices(['reload', 'ext-program'])) + .option('--task-id ', 'use task IDs to select which tasks to retrieve. Only allowed when --output-format=table') + .option('--task-tag ', 'use tags to select which tasks to retrieve. Only allowed when --output-format=table') + + .addOption(new Option('--output-format ', 'output format').choices(['table', 'tree']).default('tree')) + .addOption(new Option('--output-dest ', 'where to send task info').choices(['screen', 'file']).default('screen')) + .addOption(new Option('--output-file-name ', 'file name to store task info in').default('')) + .addOption(new Option('--output-file-format ', 'file type/format').choices(['excel', 'csv', 'json']).default('excel')) + .option('--output-file-overwrite', 'overwrite output file without asking') + + .addOption(new Option('--text-color ', 'use colored text in task views').choices(['yes', 'no']).default('yes')) + + .option('--tree-icons', 'display task status icons in tree view') + .addOption( + new Option('--tree-details [detail...]', 'display details for each task in tree view') + .choices(['taskid', 'laststart', 'laststop', 'nextstart', 'appname', 'appstream']) + .default('') + ) + + .addOption( + new Option( + '--table-details [detail...]', + 'which aspects of tasks should be included in table view. Not choosing any details will show all' + ) + .choices(['common', 'lastexecution', 'tag', 'customproperty', 'schematrigger', 'compositetrigger']) + .default('') + ); +} diff --git a/src/lib/cli/qseow-get-variable.js b/src/lib/cli/qseow-get-variable.js new file mode 100644 index 0000000..8667ddc --- /dev/null +++ b/src/lib/cli/qseow-get-variable.js @@ -0,0 +1,46 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, variableGetAssertOptions } from '../util/qseow/assert-options.js'; +import getVariable from '../cmd/qseow/getvariable.js'; + +export function setpQseowGetVariableCommand(qseow) { + qseow + .command('variable-get') + .description('get variable definitions in one or more apps') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + variableGetAssertOptions(options); + + getVariable(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--engine-port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--qrs-port ', 'Qlik Sense repository service (QRS) port (usually 4747 for cert auth, 443 for jwt auth)', '4242') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .option('--app-id ', 'Qlik Sense app ID(s) to get variables from') + .option('--app-tag ', 'Qlik Sense app tag(s) to get variables') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption( + new Option('--id-type ', 'type of identifier passed in the --variable option').choices(['id', 'name']).default('name') + ) + .option('--variable ', 'variables to retrieve. If not specified all variables will be retrieved') + .addOption(new Option('--output-format ', 'output format').choices(['json', 'table']).default('json')); +} diff --git a/src/lib/cli/qseow-import-app-from-file.js b/src/lib/cli/qseow-import-app-from-file.js new file mode 100644 index 0000000..a6a1841 --- /dev/null +++ b/src/lib/cli/qseow-import-app-from-file.js @@ -0,0 +1,53 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, appImportAssertOptions } from '../util/qseow/assert-options.js'; +import importAppFromFile from '../cmd/qseow/importapp.js'; + +export function setupQseowImportAppFromFileCommand(qseow) { + qseow + .command('app-import') + .description('import apps/upload QVF files on disk to Sense based on definitions in Excel file.') + .action(async (options) => { + try { + await qseowSharedParamAssertOptions(options); + appImportAssertOptions(options); + importAppFromFile(options); + } catch (err) { + catchLog('IMPORT APP', err); + } + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption(new Option('-t, --file-type ', 'source file type').choices(['excel']).default('excel')) + .requiredOption('--file-name ', 'file containing app definitions') + .requiredOption('--sheet-name ', 'name of Excel sheet where app info is found') + + .requiredOption('--limit-import-count ', 'import at most x number of apps. Defaults to 0 = no limit', 0) + .requiredOption( + '--sleep-app-upload ', + 'Wait this long before continuing after each app has been uploaded to Sense. Defaults to 1000 = 1 second', + 1000 + ) + + .option('--dry-run', 'do a dry run, i.e. do not import any apps - just show what would be done'); +} diff --git a/src/lib/cli/qseow-import-task-from-file.js b/src/lib/cli/qseow-import-task-from-file.js new file mode 100644 index 0000000..f0e6214 --- /dev/null +++ b/src/lib/cli/qseow-import-task-from-file.js @@ -0,0 +1,62 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, taskImportAssertOptions } from '../util/qseow/assert-options.js'; +import importTaskFromFile from '../cmd/qseow/importtask.js'; + +export function setupQseowImportTaskFromFileCommand(qseow) { + qseow + .command('task-import') + .description('create tasks based on definitions in a file on disk, optionally also importing apps from QVF files.') + .action(async (options) => { + try { + await qseowSharedParamAssertOptions(options); + taskImportAssertOptions(options); + importTaskFromFile(options); + } catch (err) { + catchLog('IMPORT TASK 1', err); + } + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption(new Option('-t, --file-type ', 'source file type').choices(['excel', 'csv']).default('excel')) + .requiredOption('--file-name ', 'file containing task definitions') + .option('--sheet-name ', 'name of Excel sheet where task info is found') + + .addOption(new Option('--update-mode ', 'create new or update existing tasks').choices(['create']).default('create')) + + .requiredOption( + '--limit-import-count ', + 'import at most x number of tasks from the source file. Defaults to 0 = no limit', + 0 + ) + .requiredOption( + '--sleep-app-upload ', + 'Wait this long before continuing after each app has been uploaded to Sense. Defaults to 1000 = 1 second', + 1000 + ) + + .option('--import-app', 'import Sense app QVFs from specified directory') + .option('--import-app-sheet-name ', 'name of Excel sheet where app definitions are found') + + .option('--dry-run', 'do a dry run, i.e. do not create any reload tasks - just show what would be done'); +} diff --git a/src/lib/cli/qseow-master-item-import.js b/src/lib/cli/qseow-master-item-import.js new file mode 100644 index 0000000..44643b2 --- /dev/null +++ b/src/lib/cli/qseow-master-item-import.js @@ -0,0 +1,101 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, masterItemImportAssertOptions } from '../util/qseow/assert-options.js'; +import importMasterItemFromFile from '../cmd/qseow/import-masteritem-excel.js'; + +export function setupQseowMasterItemImportCommand(qseow) { + qseow + .command('master-item-import') + .description('create master items based on definitions in a file on disk') + .action(async (options) => { + try { + await qseowSharedParamAssertOptions(options); + masterItemImportAssertOptions(options); + importMasterItemFromFile(options); + } catch (err) { + catchLog('IMPORT EXCEL', err); + } + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--app-id ', 'Qlik Sense app ID') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption(new Option('-t, --file-type ', 'source file type').choices(['excel']).default('excel')) + .requiredOption('--file ', 'file containing master item definitions') + .requiredOption('--sheet ', 'name of Excel sheet where dim/measure flag column is found') + .addOption( + new Option( + '--col-ref-by ', + 'how to refer to columns in the source file. Options are by name or by position (zero based)' + ) + .choices(['name', 'position']) + .default('name') + ) + .requiredOption( + '--col-item-type ', + 'column where dim/measure flag is found. Use "dim-single" in that column to create dimension, "dim-drilldown" for drill-down dimension, "measure" for measure', + 'Master item type' + ) + .requiredOption( + '--col-master-item-name ', + 'column number (zero based) or name to use as master item name', + 'Master item name' + ) + .requiredOption( + '--col-master-item-descr ', + 'column number (zero based) or name to use as master item description', + 'Description' + ) + .requiredOption( + '--col-master-item-label ', + 'column number (zero based) or name to use as master item label', + 'Label' + ) + .requiredOption( + '--col-master-item-expr ', + 'column number (zero based) or name to use as master item expression', + 'Expression' + ) + .requiredOption( + '--col-master-item-tag ', + 'column number (zero based) or name to use as master item tags', + 'Tag' + ) + .requiredOption( + '--col-master-item-color ', + 'column number (zero based) or name to use as color for dimensions/measures', + 'Color' + ) + .requiredOption( + '--col-master-item-per-value-color ', + 'column number (zero based) or name to use as per-value/segment color for dimensions/measures', + 'Per value color' + ) + + .requiredOption('--sleep-between-imports ', 'sleep this many milliseconds between imports. Set to 0 to disable', 1000) + .requiredOption( + '--limit-import-count ', + 'import at most x number of master items from the Excel file. Defaults to 0 = no limit', + 0 + ) + .option('--dry-run', 'do a dry run, i.e. do not create or update anything - just show what would be done'); +} diff --git a/src/lib/cli/qseow-scramble-field.js b/src/lib/cli/qseow-scramble-field.js new file mode 100644 index 0000000..a29e286 --- /dev/null +++ b/src/lib/cli/qseow-scramble-field.js @@ -0,0 +1,40 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions } from '../util/qseow/assert-options.js'; +import scrambleField from '../cmd/qseow/scramblefield.js'; + +export function setupQseowScrambleFieldCommand(qseow) { + qseow + .command('field-scramble') + .description('scramble one or more fields in an app. A new app with the scrambled data is created.') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + + scrambleField(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense server engine port (usually 4747 for cert auth, 443 for jwt auth)', '4747') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--app-id ', 'Qlik Sense app ID') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .requiredOption('--field-name ', 'name of field(s) to be scrambled') + .requiredOption('--new-app-name ', 'name of new app that will contain scrambled data'); +} diff --git a/src/lib/cli/qseow-set-task-cp.js b/src/lib/cli/qseow-set-task-cp.js new file mode 100644 index 0000000..afa5f8d --- /dev/null +++ b/src/lib/cli/qseow-set-task-cp.js @@ -0,0 +1,55 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions, setTaskCustomPropertyAssertOptions } from '../util/qseow/assert-options.js'; +import setTaskCustomProperty from '../cmd/qseow/settaskcp.js'; + +export function setupQseowSetTaskCustomPropertyCommand(qseow) { + qseow + .command('task-custom-property-set') + .description('update a custom property of one or more tasks') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + setTaskCustomPropertyAssertOptions(options); + + await setTaskCustomProperty(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + .addOption( + new Option('--task-type ', 'type of tasks to list').choices(['reload']).default(['reload']) + // .choices(['reload', 'ext-program']) + // .default(['reload', 'ext-program']) + ) + .option('--task-id ', 'use task IDs to select which tasks to retrieve') + .option('--task-tag ', 'use tags to select which tasks to retrieve') + + .requiredOption('--custom-property-name ', 'name of custom property that will be updated') + .requiredOption('--custom-property-value ', 'one or more values name of custom property that will be updated') + .option('--overwrite', 'overwrite existing custom property values without asking') + .addOption( + new Option('--update-mode ', 'append or replace value(s) to existing custom property') + .choices(['append', 'replace']) + .default('append') + ) + .option('--dry-run', 'do a dry run, i.e. do not modify any reload tasks - just show what would be updated'); +} diff --git a/src/lib/cli/qseow-show-version.js b/src/lib/cli/qseow-show-version.js new file mode 100644 index 0000000..d5a159f --- /dev/null +++ b/src/lib/cli/qseow-show-version.js @@ -0,0 +1,13 @@ +import { Option } from 'commander'; + +export function setupQseowShowVersionCommand(qseow) { + qseow + .command('version') + .description('show version info') + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .action(async (options) => { + logger.verbose(`Version: ${appVersion}`); + }); +} diff --git a/src/lib/cli/qseow-test-connection.js b/src/lib/cli/qseow-test-connection.js new file mode 100644 index 0000000..41a8b86 --- /dev/null +++ b/src/lib/cli/qseow-test-connection.js @@ -0,0 +1,39 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions } from '../util/qseow/assert-options.js'; +import testConnection from '../cmd/qseow/testconnection.js'; + +export function setupQseowTestConnectionCommand(qseow) { + qseow + .command('connection-test') + .description('test connection to Qlik Sense server.') + .action(async (options) => { + try { + await qseowSharedParamAssertOptions(options); + await testConnection(options); + } catch (err) { + catchLog('CONNECTION TEST', err); + } + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense proxy service port', '4242') + .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + .option('--auth-user-dir ', 'user directory for user to connect with') + .option('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server'); +} diff --git a/src/lib/cli/qseow-visualise-task.js b/src/lib/cli/qseow-visualise-task.js new file mode 100644 index 0000000..1a9d8ff --- /dev/null +++ b/src/lib/cli/qseow-visualise-task.js @@ -0,0 +1,41 @@ +import { Option } from 'commander'; + +import { catchLog } from '../util/log.js'; +import { qseowSharedParamAssertOptions } from '../util/qseow/assert-options.js'; +import visTask from '../cmd/qseow/vistask.js'; + +export function setupQseowVisualiseTaskCommand(qseow) { + qseow + .command('task-vis') + .description('visualise task network') + .action(async (options) => { + await qseowSharedParamAssertOptions(options); + + await visTask(options); + }) + .addOption( + new Option('--log-level ', 'log level').choices(['error', 'warn', 'info', 'verbose', 'debug', 'silly']).default('info') + ) + .requiredOption('--host ', 'Qlik Sense server IP/FQDN') + .option('--port ', 'Qlik Sense repository service (QRS) port (usually 4242 for cert auth, 443 for jwt auth)', '4242') + // .option('--schema-version ', 'Qlik Sense engine schema version', '12.612.0') + .requiredOption('--virtual-proxy ', 'Qlik Sense virtual proxy prefix', '') + .requiredOption( + '--secure ', + 'https connection to Qlik Sense must use correct certificate. Invalid certificates will result in rejected/failed connection.', + true + ) + + .requiredOption('--auth-user-dir ', 'user directory for user to connect with') + .requiredOption('--auth-user-id ', 'user ID for user to connect with') + + .addOption(new Option('-a, --auth-type ', 'authentication type').choices(['cert', 'jwt']).default('cert')) + .option('--auth-cert-file ', 'Qlik Sense certificate file (exported from QMC)', './cert/client.pem') + .option('--auth-cert-key-file ', 'Qlik Sense certificate key file (exported from QMC)', './cert/client_key.pem') + .option('--auth-root-cert-file ', 'Qlik Sense root certificate file (exported from QMC)', './cert/root.pem') + .option('--auth-jwt ', 'JSON Web Token (JWT) to use for authentication with Qlik Sense server') + + // Options for visualisation host + .option('--vis-host ', 'host for visualisation server', 'localhost') + .option('--vis-port ', 'port for visualisation server', '3000'); +} diff --git a/src/lib/cmd/createuseractivitycp.js b/src/lib/cmd/createuseractivitycp.js deleted file mode 100644 index be0673a..0000000 --- a/src/lib/cmd/createuseractivitycp.js +++ /dev/null @@ -1,216 +0,0 @@ -import qrsInteract from 'qrs-interact'; -import path from 'path'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; - -import { - getUserActivityProfessional, - getUserActivityAnalyzer, - getUserActivityAnalyzerTime, - getUserActivityLogin, - getUserActivityUser, - getUsersLastActivity, -} from './useractivity.js'; -import { catchLog } from '../util/log.js'; - -const _MS_PER_DAY = 1000 * 60 * 60 * 24; - -// a and b are javascript Date objects -function dateDiffInDays(a, b) { - // Discard the time and time-zone information. - const utc1 = Date.UTC(a.getFullYear(), a.getMonth(), a.getDate()); - const utc2 = Date.UTC(b.getFullYear(), b.getMonth(), b.getDate()); - - return Math.floor((utc2 - utc1) / _MS_PER_DAY); -} - -/** - * - * @param {*} options - */ -const createUserActivityCustomProperty = async (options) => { - try { - // Set log level - setLoggingLevel(options.logLevel); - - logger.verbose(`Ctrl-Q was started as a stand-alone binary: ${isPkg}`); - logger.verbose(`Ctrl-Q was started from ${execPath}`); - - logger.info('Create custom property for tracking user activity in QMC'); - logger.debug(`Options: ${JSON.stringify(options, null, 2)}`); - - // Set up connection to Sense repository service - const certPath = path.resolve(process.cwd(), options.authCertFile); - const keyPath = path.resolve(process.cwd(), options.authCertKeyFile); - - // Verify cert files exist - - const configQRS = { - hostname: options.host, - portNumber: options.port, - certificates: { - certFile: certPath, - keyFile: keyPath, - }, - }; - - configQRS.headers = { - 'X-Qlik-User': 'UserDirectory=Internal; UserId=sa_repository', - 'Content-Type': 'application/json', - }; - - // eslint-disable-next-line new-cap - const qrsInteractInstance = new qrsInteract(configQRS); - let result; - - // Does CP already exist? - try { - result = await qrsInteractInstance.Get(`custompropertydefinition/full?filter=name eq '${options.customPropertyName}'`); - } catch (err) { - // Return error msg - catchLog(`USER ACTIVITY CP: Error getting user activity custom property`, err); - } - - if (result.statusCode === 200) { - if (result.body.length === 1) { - // CP exists - logger.debug(`USER ACTIVITY CP: Custom property name passed via command line exists`); - - // Does the existing CP have *exactly* the same values as passed in via comand line? - if (options.activityBuckets.length === result.body[0].choiceValues.length) { - // Same number of custom property values. Are they the same? - } else { - // Different number of values. Do nothing, unless the --force paramerer equals true - // eslint-disable-next-line no-lonely-if - if (options.force === 'false') { - // Don't force overwrite the existni custom property. - // Show warning and return - logger.warn( - `USER ACTIVITY CP: Custom property already exists, with existing values different from the ones pass in via command line. Aborting.` - ); - } else { - // - logger.verbose(`USER ACTIVITY CP: Replacing custom property ${options.customPropertyName}`); - } - } - } else if (result.body.length === 0) { - // CP does not exist - logger.debug(`USER ACTIVITY CP: Custom property name passed via command line does not exist`); - - // Create new CP - try { - result = await qrsInteractInstance.Post( - 'custompropertydefinition', - { - name: options.customPropertyName, - valueType: 'Text', - // choiceValues: ['1', '7', '14'], - choiceValues: options.activityBuckets, - objectTypes: ['User'], - description: 'Ctrl-Q user activity buckets', - }, - 'json' - ); - } catch (err) { - catchLog(`USER ACTIVITY CP: Error creating user activity custom property`, err); - } - - if (result.statusCode === 201) { - logger.verbose(`USER ACTIVITY CP: Created new custom property "${options.customPropertyName}"`); - } - } - - // User activity info will available in following format - // Array of objects: - // { - // id: "41e8464e-87ed-4ea3-9fc7-e09d2dc6781a", - // createdDate: "2021-11-19T12:23:58.850Z", - // modifiedDate: "2022-08-27T06:47:08.600Z", - // modifiedByUserName: "LAB\\testuser_2", - // user: { - // id: "9e403391-58a7-4442-ada7-c54dc8906016", - // userId: "testuser_2", - // userDirectory: "LAB", - // userDirectoryConnectorName: "LAB", - // name: "Testuser2", - // privileges: null, - // }, - // lastUsed: "2022-08-27T06:47:08.584Z", - // excess: false, - // quarantined: false, - // quarantineEnd: "1753-01-01T00:00:00.000Z", - // deletedUserId: "", - // deletedUserDirectory: "", - // privileges: null, - // schemaPath: "License.AnalyzerAccessType", - // } - - // Get user activity via QRS API, per license type - const activityProfessional = await getUserActivityProfessional(qrsInteractInstance); - logger.debug(`USER ACTIVITY CP: Professional licenses: ${JSON.stringify(activityProfessional)}`); - - const activityAnalyzer = await getUserActivityAnalyzer(qrsInteractInstance); - logger.debug(`USER ACTIVITY CP: Analyzer licenses: ${JSON.stringify(activityAnalyzer)}`); - - const activityAnalyzerTime = await getUserActivityAnalyzerTime(qrsInteractInstance); - logger.debug(`USER ACTIVITY CP: Analyzer time licenses: ${JSON.stringify(activityAnalyzerTime)}`); - - const activityLogin = await getUserActivityLogin(qrsInteractInstance); - logger.debug(`USER ACTIVITY CP: Login licenses: ${JSON.stringify(activityLogin)}`); - - const activityUser = await getUserActivityUser(qrsInteractInstance); - logger.debug(`USER ACTIVITY CP: User licenses: ${JSON.stringify(activityUser)}`); - - const usersLastActivity = await getUsersLastActivity( - activityProfessional, - activityAnalyzer, - activityAnalyzerTime, - activityLogin, - activityUser - ); - - // Assign users to activity buckets - // eslint-disable-next-line no-restricted-syntax - for (const user of usersLastActivity) { - // How many days ago was user active? Round down to nearest full day - const dateNow = new Date(); - const dateUserLastActivity = new Date(user.lastUsed); - // const diffTime = Math.abs(dateNow - dateUserLastActivity); - // const diffDays = Math.ceil(diffTime / (1000 * 60 * 60 * 24)); - const diffDays = dateDiffInDays(dateUserLastActivity, dateNow); - - // eslint-disable-next-line no-restricted-syntax - for (const bucket of options.activityBuckets) { - if (diffDays <= bucket) { - user.activityBucket = bucket; - break; - } - } - - // Set custom property for user - try { - // eslint-disable-next-line no-await-in-loop - result = await qrsInteractInstance.Post( - 'custompropertydefinition', - { - name: options.customPropertyName, - valueType: 'Text', - // choiceValues: ['1', '7', '14'], - choiceValues: options.activityBuckets, - objectTypes: ['User'], - description: 'Ctrl-Q user activity buckets', - }, - 'json' - ); - } catch (err) { - catchLog(`USER ACTIVITY CP: Error creating user activity custom property`, err); - } - } - logger.verbose(`USER ACTIVITY CP: Assigned activity buckets to users via custom property ${options.customPropertyName}`); - } - } catch (err) { - // Return error msg - catchLog(`USER ACTIVITY CP: Error creating user activity custom property`, err); - } -}; - -export default createUserActivityCustomProperty; diff --git a/src/lib/cmd/qscloud/testconnection.js b/src/lib/cmd/qscloud/testconnection.js new file mode 100644 index 0000000..c065058 --- /dev/null +++ b/src/lib/cmd/qscloud/testconnection.js @@ -0,0 +1,35 @@ +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { getQscloudCurrentUser } from '../../util/qscloud/user.js'; +import { catchLog } from '../../util/log.js'; + +export async function qscloudTestConnection(options) { + try { + // Set log level + setLoggingLevel(options.logLevel); + + logger.verbose(`Ctrl-Q was started as a stand-alone binary: ${isPkg}`); + logger.verbose(`Ctrl-Q was started from ${execPath}`); + + logger.info(`Testing connection to Qlik Sense Cloud tenant "${options.tenantUrl}"`); + logger.debug(`Options: ${JSON.stringify(options, null, 2)}`); + + // Get info about user associated with the auth JWT + const userInfo = await getQscloudCurrentUser(options); + + if (userInfo === false) { + logger.error(`Could not connect to Qlik Sense Cloud`); + return false; + } + + logger.info(`Successfully connected to Qlik Sense Cloud tenant "${options.host}"`); + logger.info(`Tenant ID : ${userInfo.tenantId}`); + logger.info(`User ID : ${userInfo.id}`); + logger.info(`User name : ${userInfo.name}`); + logger.info(`User email : ${userInfo.email}`); + logger.info(`User status: ${userInfo.status}`); + } catch (err) { + catchLog(`Error testing connection to Qlik Sense server ${options.host} on port ${options.port}`, err); + logger.error(`EXPORT APP: ${err.stack}`); + return false; + } +} diff --git a/src/lib/cmd/createdim.js b/src/lib/cmd/qseow/createdim.js similarity index 98% rename from src/lib/cmd/createdim.js rename to src/lib/cmd/qseow/createdim.js index f078d37..c6efd18 100644 --- a/src/lib/cmd/createdim.js +++ b/src/lib/cmd/qseow/createdim.js @@ -1,7 +1,7 @@ import enigma from 'enigma.js'; -import setupEnigmaConnection from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import setupEnigmaConnection from '../../util/qseow/enigma_util.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; /** * diff --git a/src/lib/cmd/qseow/createuseractivitycp.js b/src/lib/cmd/qseow/createuseractivitycp.js new file mode 100644 index 0000000..079394d --- /dev/null +++ b/src/lib/cmd/qseow/createuseractivitycp.js @@ -0,0 +1,590 @@ +import axios from 'axios'; + +import { logger, setLoggingLevel, isPkg, execPath, sleep } from '../../../globals.js'; +import { setupQrsConnection } from '../../util/qseow/qrs.js'; +import { catchLog } from '../../util/log.js'; +import { + getUserActivityProfessional, + getUserActivityAnalyzer, + getUserActivityAnalyzerTime, + getUserActivityLogin, + getUserActivityUser, + getUsersLastActivity, +} from './useractivity.js'; +import { getCustomPropertiesFromQseow, createCustomProperty, updateCustomProperty } from '../../util/qseow/customproperties.js'; + +const _MS_PER_DAY = 1000 * 60 * 60 * 24; + +// a and b are javascript Date objects +function dateDiffInDays(a, b) { + // Discard the time and time-zone information. + const utc1 = Date.UTC(a.getFullYear(), a.getMonth(), a.getDate()); + const utc2 = Date.UTC(b.getFullYear(), b.getMonth(), b.getDate()); + + return Math.floor((utc2 - utc1) / _MS_PER_DAY); +} + +/** + * Function to create custom property for tracking user activity in QMC. + * + * - User activity is tracked based on the number of days since the user last logged in. + * - This information is available via QRS API /license//full + * - Possible access license types are: + * - professionalaccesstype + * - analyzeraccesstype + * - analyzertimeaccesstype + * - loginaccesstype + * - useraccesstype + * - A custom property will be set to users based on the number of days since the user last logged in. + * - For example, if the user last logged in 3 days ago, the custom property will be set to 3. + * - The custom property will be created with the name passed in via the command line. + * - The custom property will have a set of allowed values, specified via the command line (or as default values if not specified). + * - Certificate or JWT authentication is used to connect to the Qlik Sense repository service (QRS). + * + * General steps: + * - Check if the custom property already exists in QMC + * - If it does not exist, create it + * - If it does exist, check if the allowed values are the same as the ones passed in via the command line + * - If they are different, show a warning and do nothing, unless the --force parameter equals true + * - If --force equals true, delete the existing custom property and create a new one using data from the command line + * - If they are the same, show an info message and continue + * - Get user activity for each access license type enabled via the command line option --license-type + * - Filter QRS call on + * - users' user directory, if specified via the command line + * - users' tag(s), if specified via the command line (future feature) + * - users' custom property value(s), if specified via the command line (future feature) + * - Get user activity for each user. How many days ago was the user last active? + * - Calculate activity buckets for all users (matching command line filters) before writing back to QRS + * - Update users in QRS with the user activity custom property value + * - Update batches of users using the QRS API endpoint POST /user/many + * - Batch size determined by command line parameter --update-batch-size + * - Wait for a short time between each batch, to avoid overloading the QRS. Delay is determined by command line parameter --update-batch-sleep + * + * If the process above fails at some point, show an error message and return with false. + * @param {*} options + */ +export async function createUserActivityBucketsCustomProperty(options) { + try { + // Set log level + setLoggingLevel(options.logLevel); + + logger.verbose(`Ctrl-Q was started as a stand-alone binary: ${isPkg}`); + logger.verbose(`Ctrl-Q was started from ${execPath}`); + + logger.info('== Step 1: Create custom property for tracking user activity in QMC'); + logger.debug(`Options: ${JSON.stringify(options, null, 2)}`); + + // Sort activity buckets passed via command line in ascending order + // When creating new or updating existing custom property, the allowed values should be sorted in ascending order + const activityBucketsSorted = options.activityBuckets.sort((a, b) => a - b); + + // Get custom properties from QSEoW + let customProperties = await getCustomPropertiesFromQseow(options); + logger.info(` Successfully retrieved ${customProperties.length} custom properties from QSEoW`); + + // Does the custom property already exist in QMC? + let customPropertyExisting = customProperties.find((cp) => cp.name === options.customPropertyName); + + if (customPropertyExisting) { + // A custom property with correct name already exists + + // Does the existing CP have *exactly* the same choice-values as passed in via command line? + if (activityBucketsSorted.length === customPropertyExisting.choiceValues.length) { + // Same number of custom property values. Are they the same and in same order? + let keepExistingCustomProperty = true; + + for (let i = 0; i < activityBucketsSorted.length; i++) { + if (activityBucketsSorted[i] !== customPropertyExisting.choiceValues[i]) { + keepExistingCustomProperty = false; + break; + } + } + + if (keepExistingCustomProperty) { + // Custom property already exists with the same allowed values, in the same order, as passed in via command line. + // Show info message and continue, no need to modify the existing custom property + logger.info( + ` Custom property "${options.customPropertyName}" already exists with the same allowed values as passed in via command line. No action needed.` + ); + } else { + logger.warn( + `Custom property already exists, but existing values are different from the ones passed in via command line.` + ); + logger.warn(`Allowed values for existing custom property: ${customPropertyExisting.choiceValues}`); + logger.warn(`Allowed values (sorted ascending) passed in via command line: ${activityBucketsSorted}`); + + // Do nothing, unless the --force parameter equals true + if (options.force === 'true' || options.force === true) { + // Force overwrite the existing custom property + logger.info( + ` Option "--force" specified, updating custom property ${options.customPropertyName} with new allowed values.` + ); + + // Update existing custom property + // First copy existing custom property to a new object, then replace the choiceValues with the new ones + const customPropertyDefinition = JSON.parse(JSON.stringify(customPropertyExisting)); + customPropertyDefinition.choiceValues = activityBucketsSorted; + + const result = await updateCustomProperty(options, customPropertyDefinition); + if (result) { + logger.verbose( + ` Updated existing custom property "${options.customPropertyName}" with new allowed values passed in via command line.` + ); + } else { + logger.error( + `Failed to update existing custom property "${options.customPropertyName}" with new allowed values.` + ); + return false; + } + } else { + // Don't force overwrite the existing custom property. + // Show warning and return + logger.warn(`"--force" option not specified. Aborting.`); + return false; + } + } + } else { + // Custom property exists, but has different number of values compared to command line options. + // Do nothing unless the --force paramerer equals true + if (options.force === 'false' || options.force === false || options.force === undefined) { + // Don't force overwrite the existni custom property. + // Show warning and return + logger.warn( + `Custom property "${options.customPropertyName}" already exists, but has different allowed values compared to the ones passed in via command line. Use the --force option to overwrite the existing custom property.` + ); + logger.warn(`Use the --force option to overwrite the existing custom property.`); + return false; + } else { + // Force replace the existing custom property + logger.verbose(` Replacing custom property ${options.customPropertyName}`); + + // Update existing custom property + // First copy existing custom property to a new object, then replace the choiceValues with the new ones + const customPropertyDefinition = JSON.parse(JSON.stringify(customPropertyExisting)); + customPropertyDefinition.choiceValues = activityBucketsSorted; + + const result = await updateCustomProperty(options, customPropertyDefinition); + if (result) { + logger.verbose( + ` Updated existing custom property "${options.customPropertyName}" with new allowed values passed in via command line.` + ); + } else { + logger.error(`Failed to update existing custom property "${options.customPropertyName}" with new allowed values.`); + return false; + } + } + } + } else { + // Custom property does not exist. Create it. + + // Create custom property definition/payload to QRS POST call + const customPropertyDefinition = { + valueType: 'Text', + schemaPath: 'CustomPropertyDefinition', + objectTypes: ['User'], + name: options.customPropertyName, + description: 'Ctrl-Q user activity bucket', + choiceValues: activityBucketsSorted, + }; + + const result = await createCustomProperty(options, customPropertyDefinition); + if (result) { + logger.verbose(` Created custom property "${options.customPropertyName}"`); + } else { + logger.error(`Failed to create custom property "${options.customPropertyName}"`); + return false; + } + } + + // Get custom property again, as it has potentially been created or updated + customProperties = await getCustomPropertiesFromQseow(options); + + // Does the custom property already exist in QMC? + customPropertyExisting = customProperties.find((cp) => cp.name === options.customPropertyName); + + // Get user activity for each access license type enabled via the command line option --license-type + // Filter QRS call on + // - users' user directory, if specified via the command line + // If user directory is not specified, no filtering on user directory will be done + + let activityAnalyzer = []; + let activityAnalyzerTime = []; + let activityLogin = []; + let activityProfessional = []; + let activityUser = []; + + logger.info(''); + logger.info(`== Step 2 : Getting user activity for each license type enabled via the command line...`); + + // Is "analyzer" license type enabled? + if (options.licenseType.includes('analyzer')) { + // Get user activity for analyzer license type + // Array of objects: + // { + // "privileges" : [ "privileges", "privileges" ], + // "quarantineEnd" : "2000-01-23T04:56:07.000+00:00", + // "schemaPath" : "schemaPath", + // "quarantined" : true, + // "deletedUserId" : "deletedUserId", + // "lastUsed" : "2000-01-23T04:56:07.000+00:00", + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "modifiedByUserName" : "modifiedByUserName", + // "deletedUserDirectory" : "deletedUserDirectory", + // "excess" : true, + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "user" : { + // "privileges" : [ "privileges", "privileges" ], + // "userDirectory" : "userDirectory", + // "userDirectoryConnectorName" : "userDirectoryConnectorName", + // "name" : "name", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "userId" : "userId" + // } + // } + activityAnalyzer = await getUserActivityAnalyzer(options); + logger.debug(` Analyzer licenses: ${JSON.stringify(activityAnalyzer)}`); + } + + // Is "analyzer-time" license type enabled? + if (options.licenseType.includes('analyzer-time')) { + // Get user activity for analyzer-time license type + // Array of objects: + // { + // "latestActivity" : "2000-01-23T04:56:07.000+00:00", + // "privileges" : [ "privileges", "privileges" ], + // "hostName" : "hostName", + // "sessions" : [ { + // "latestActivity" : "2000-01-23T04:56:07.000+00:00", + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "modifiedByUserName" : "modifiedByUserName", + // "schemaPath" : "schemaPath", + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "sessionID" : "sessionID", + // "serverNodeConfigurationId" : "serverNodeConfigurationId" + // }, { + // "latestActivity" : "2000-01-23T04:56:07.000+00:00", + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "modifiedByUserName" : "modifiedByUserName", + // "schemaPath" : "schemaPath", + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "sessionID" : "sessionID", + // "serverNodeConfigurationId" : "serverNodeConfigurationId" + // } ], + // "useStopTime" : "2000-01-23T04:56:07.000+00:00", + // "useStartTime" : "2000-01-23T04:56:07.000+00:00", + // "schemaPath" : "schemaPath", + // "serverNodeConfigurationId" : "serverNodeConfigurationId", + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "modifiedByUserName" : "modifiedByUserName", + // "analyzerTimeAccessType" : { + // "privileges" : [ "privileges", "privileges" ], + // "name" : "name", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91" + // }, + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "user" : { + // "privileges" : [ "privileges", "privileges" ], + // "userDirectory" : "userDirectory", + // "userDirectoryConnectorName" : "userDirectoryConnectorName", + // "name" : "name", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "userId" : "userId" + // } + // } + activityAnalyzerTime = await getUserActivityAnalyzerTime(options); + logger.debug(` Analyzer time licenses: ${JSON.stringify(activityAnalyzerTime)}`); + } + + // Is "login" license type enabled? + if (options.licenseType.includes('login')) { + // Get user activity for login license type + // Array of objects: + // { + // "latestActivity" : "2000-01-23T04:56:07.000+00:00", + // "privileges" : [ "privileges", "privileges" ], + // "hostName" : "hostName", + // "sessions" : [ { + // "latestActivity" : "2000-01-23T04:56:07.000+00:00", + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "modifiedByUserName" : "modifiedByUserName", + // "schemaPath" : "schemaPath", + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "sessionID" : "sessionID", + // "serverNodeConfigurationId" : "serverNodeConfigurationId" + // }, { + // "latestActivity" : "2000-01-23T04:56:07.000+00:00", + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "modifiedByUserName" : "modifiedByUserName", + // "schemaPath" : "schemaPath", + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "sessionID" : "sessionID", + // "serverNodeConfigurationId" : "serverNodeConfigurationId" + // } ], + // "useStopTime" : "2000-01-23T04:56:07.000+00:00", + // "useStartTime" : "2000-01-23T04:56:07.000+00:00", + // "schemaPath" : "schemaPath", + // "serverNodeConfigurationId" : "serverNodeConfigurationId", + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "loginAccessType" : { + // "privileges" : [ "privileges", "privileges" ], + // "name" : "name", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91" + // }, + // "modifiedByUserName" : "modifiedByUserName", + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "user" : { + // "privileges" : [ "privileges", "privileges" ], + // "userDirectory" : "userDirectory", + // "userDirectoryConnectorName" : "userDirectoryConnectorName", + // "name" : "name", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "userId" : "userId" + // } + // } + activityLogin = await getUserActivityLogin(options); + logger.debug(` Login licenses: ${JSON.stringify(activityLogin)}`); + } + + // Is "professional" license type enabled? + if (options.licenseType.includes('professional')) { + // Get user activity for professional license type + // Array of objects: + // { + // "privileges" : [ "privileges", "privileges" ], + // "quarantineEnd" : "2000-01-23T04:56:07.000+00:00", + // "schemaPath" : "schemaPath", + // "quarantined" : true, + // "deletedUserId" : "deletedUserId", + // "lastUsed" : "2000-01-23T04:56:07.000+00:00", + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "modifiedByUserName" : "modifiedByUserName", + // "deletedUserDirectory" : "deletedUserDirectory", + // "excess" : true, + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "user" : { + // "privileges" : [ "privileges", "privileges" ], + // "userDirectory" : "userDirectory", + // "userDirectoryConnectorName" : "userDirectoryConnectorName", + // "name" : "name", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "userId" : "userId" + // } + // } + activityProfessional = await getUserActivityProfessional(options); + logger.debug(` Professional licenses: ${JSON.stringify(activityProfessional)}`); + } + + // Is "user" license type enabled? + if (options.licenseType.includes('user')) { + // Get user activity for user license type + // Array of objects: + // { + // "lastUsed" : "2000-01-23T04:56:07.000+00:00", + // "privileges" : [ "privileges", "privileges" ], + // "createdDate" : "2000-01-23T04:56:07.000+00:00", + // "quarantineEnd" : "2000-01-23T04:56:07.000+00:00", + // "modifiedByUserName" : "modifiedByUserName", + // "deletedUserDirectory" : "deletedUserDirectory", + // "schemaPath" : "schemaPath", + // "modifiedDate" : "2000-01-23T04:56:07.000+00:00", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "quarantined" : true, + // "user" : { + // "privileges" : [ "privileges", "privileges" ], + // "userDirectory" : "userDirectory", + // "userDirectoryConnectorName" : "userDirectoryConnectorName", + // "name" : "name", + // "id" : "046b6c7f-0b8a-43b9-b35d-6489e6daee91", + // "userId" : "userId" + // }, + // "deletedUserId" : "deletedUserId" + // } + activityUser = await getUserActivityUser(options); + logger.debug(` User licenses: ${JSON.stringify(activityUser)}`); + } + + const usersLastActivity = await getUsersLastActivity( + activityAnalyzer, + activityAnalyzerTime, + activityLogin, + activityProfessional, + activityUser + ); + + // Assign users to activity buckets + logger.info(''); + logger.info(`== Step 3 : Calculate days since last activity for each user...`); + + for (const user of usersLastActivity) { + // How many days ago was user active? Round down to nearest full day + const dateNow = new Date(); + const dateUserLastActivity = new Date(user.lastUsed); + const diffDays = dateDiffInDays(dateUserLastActivity, dateNow); + + for (const bucket of activityBucketsSorted) { + // Assign user to activity bucket that is equal to or greater than the number of days since last activity + if (diffDays <= bucket) { + user.activityBucket = bucket; + break; + } + } + } + logger.verbose(` Assigned activity buckets to users via custom property ${options.customPropertyName}`); + + // Update data in QRS + // Batch updates to avoid overloading QRS by calling once for each user + // Batch size determined by command line parameter --update-batch-size + // Wait for a short time between each batch, to avoid overloading the QRS. Delay is determined by command line parameter --update-batch-sleep + const batchSize = options.updateBatchSize; + const batchSleep = options.updateBatchSleep * 1000; // Convert seconds to milliseconds + const totalBatches = Math.ceil(usersLastActivity.length / batchSize); + const outputUserArray = []; + + logger.info(''); + logger.info(`== Step 4 : Get user data from Sense, one batch at a time (each batch is ${batchSize} users)...`); + logger.info(` Total number of users to process: ${usersLastActivity.length}`); + logger.info(` Total number of batches: ${totalBatches}`); + + for (let i = 0; i < totalBatches; i++) { + const start = i * batchSize; + const end = start + batchSize; + const usersBatch = usersLastActivity.slice(start, end); + + logger.info(''); + logger.info(` >> Batch ${i + 1} of ${totalBatches} (users ${start + 1} to ${end})`); + logger.info(` Calculating activity buckets`); + + // Get full user data from QRS for users in this batch + // Users are identified by usersBartch[i].userSenseId + // This is the user ID in Qlik Sense + + // First create filter string for QRS call. Format is + // filter=userId eq 'user1.id' or userId eq 'user2.id' or userId eq 'user3.id' + const filter = usersBatch.map((user) => { + return `id eq ${user.userSenseId}`; + }); + const filterString = filter.join(' or '); + logger.debug(` Filter string for getting user batch ${i} from QRS: ${filterString}`); + + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: 'qrs/user/full', + queryParameters: [ + { + name: 'filter', + value: filterString, + }, + ], + }); + + // Get current info from QRS for this batch of users + const result = await axios.request(axiosConfig); + let currentUserInfo; + if (result.status === 200) { + currentUserInfo = JSON.parse(result.data); + logger.debug(` Response from QRS for getting user batch ${i}: ${result.status}`); + } else { + logger.error(`Error ${result.status} getting user batch ${i} from QRS`); + return false; + } + + // Update user objects with activity buckets + logger.info(` Preparing user activity custom property`); + + for (const currentUser of currentUserInfo) { + // user is a full user object from QRS + + const userActivityCp = currentUser.customProperties.find((cp) => cp.definition.name === options.customPropertyName); + + if (userActivityCp) { + // User activity custom property already exists + // Remove it + currentUser.customProperties = currentUser.customProperties.filter( + (cp) => cp.definition.name !== options.customPropertyName + ); + } + // Custom property does not exist for this user, add it + // Note that the custom property itself however exists, it's just not assigned to this user (yet) + currentUser.customProperties.push({ + definition: { + valueType: 'Text', + name: customPropertyExisting.name, + id: customPropertyExisting.id, + choiceValues: customPropertyExisting.choiceValues, + }, + value: usersBatch.find((user) => user.userSenseId === currentUser.id).activityBucket.toString(), + }); + + // Add user to output array + outputUserArray.push(currentUser); + } + + // Pause options.updateBatchSleep seconds before next batch + logger.info(` Pausing ${options.updateBatchSleep} seconds before next batch...`); + if (batchSleep > 0) await sleep(batchSleep); + } + + logger.info(''); + logger.info(`Done calculating activity buckets for all users. Proceeding to update user activity custom property in Qlik Sense.`); + + logger.info(''); + logger.info(`== Step 5 : Update user activity custom property in Qlik Sense.`); + + // Update user activity custom property in Qlik Sense + // Loop over the same buckets, using the same batch size. + // Data to be sent to QRS is the outputUserArray array + const totalOutputBatches = Math.ceil(outputUserArray.length / batchSize); + logger.info(` Number of batches to process: ${totalOutputBatches} of ${batchSize} users each.`); + + let userCounter = 1; + for (let i = 0; i < totalOutputBatches; i++) { + const start = i * batchSize; + const end = start + batchSize; + const usersBatch = outputUserArray.slice(start, end); + + logger.info(` Storing activity buckets for batch ${i + 1} of ${totalBatches} in Sense repository.`); + + // Loop over the users in the batch, writing the user activity custom property to QRS + for (const user of usersBatch) { + // Payload: array of user objects + const axiosConfig = setupQrsConnection(options, { + method: 'put', + path: `qrs/user/${user.id}`, + body: user, + }); + + const result = await axios.request(axiosConfig); + if (result.status === 200) { + logger.info( + ` Updated user ${userCounter} of ${outputUserArray.length}, "${user.userDirectory}\\${user.userId}" in batch ${ + i + 1 + } of ${totalBatches}` + ); + } else { + logger.error(`Error ${result.status} updating user activity custom property for batch ${i + 1} of ${totalBatches}`); + return false; + } + userCounter++; + + // Pause half a second between each user + if (options.updateUserSleep > 0) await sleep(options.updateUserSleep); + } + } + + logger.info(''); + logger.info(`Done updating user activity custom property in Qlik Sense.`); + + return true; + } catch (err) { + // Return error msg + catchLog(`Error creating user activity custom property`, err); + } +} diff --git a/src/lib/cmd/deletedim.js b/src/lib/cmd/qseow/deletedim.js similarity index 97% rename from src/lib/cmd/deletedim.js rename to src/lib/cmd/qseow/deletedim.js index 91761b6..f2d6f98 100644 --- a/src/lib/cmd/deletedim.js +++ b/src/lib/cmd/qseow/deletedim.js @@ -1,7 +1,7 @@ import enigma from 'enigma.js'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; // Variable to keep track of how many dimensions have been deleted let deleteCount = 0; diff --git a/src/lib/cmd/deletemeasure.js b/src/lib/cmd/qseow/deletemeasure.js similarity index 97% rename from src/lib/cmd/deletemeasure.js rename to src/lib/cmd/qseow/deletemeasure.js index 6bd61c4..d7e6991 100644 --- a/src/lib/cmd/deletemeasure.js +++ b/src/lib/cmd/qseow/deletemeasure.js @@ -1,7 +1,7 @@ import enigma from 'enigma.js'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; // Variable to keep track of how many measures have been deleted let deleteCount = 0; diff --git a/src/lib/cmd/deletesessions.js b/src/lib/cmd/qseow/deletesessions.js similarity index 89% rename from src/lib/cmd/deletesessions.js rename to src/lib/cmd/qseow/deletesessions.js index 2addfc8..417d76b 100644 --- a/src/lib/cmd/deletesessions.js +++ b/src/lib/cmd/qseow/deletesessions.js @@ -1,6 +1,6 @@ -import { deleteSessionsFromQSEoWIds } from '../util/session.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { deleteSessionsFromQSEoWIds } from '../../util/qseow/session.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; /** * Delete Qlik Sense proxy sessions diff --git a/src/lib/cmd/deletevariable.js b/src/lib/cmd/qseow/deletevariable.js similarity index 97% rename from src/lib/cmd/deletevariable.js rename to src/lib/cmd/qseow/deletevariable.js index a119e8b..d5f1f78 100644 --- a/src/lib/cmd/deletevariable.js +++ b/src/lib/cmd/qseow/deletevariable.js @@ -1,11 +1,9 @@ -/* eslint-disable no-restricted-syntax */ -/* eslint-disable no-await-in-loop */ import enigma from 'enigma.js'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { getApps } from '../util/app.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { getApps } from '../../util/qseow/app.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; /** * diff --git a/src/lib/cmd/exportapp.js b/src/lib/cmd/qseow/exportapp.js similarity index 96% rename from src/lib/cmd/exportapp.js rename to src/lib/cmd/qseow/exportapp.js index dc18d7c..8a13fc9 100644 --- a/src/lib/cmd/exportapp.js +++ b/src/lib/cmd/qseow/exportapp.js @@ -1,11 +1,11 @@ import xlsx from 'node-xlsx'; -import fs from 'fs'; -import path from 'path'; +import fs from 'node:fs'; +import path from 'node:path'; import yesno from 'yesno'; -import { logger, setLoggingLevel, isPkg, execPath, mergeDirFilePath, verifyFileExists, isNumeric, sleep } from '../../globals.js'; -import QlikSenseApps from '../app/class_allapps.js'; -import { catchLog } from '../util/log.js'; +import { logger, setLoggingLevel, isPkg, execPath, mergeDirFilePath, verifyFileExists, isNumeric, sleep } from '../../../globals.js'; +import QlikSenseApps from '../../app/class_allapps.js'; +import { catchLog } from '../../util/log.js'; const exportAppToFile = async (options) => { try { diff --git a/src/lib/cmd/getbookmark.js b/src/lib/cmd/qseow/getbookmark.js similarity index 98% rename from src/lib/cmd/getbookmark.js rename to src/lib/cmd/qseow/getbookmark.js index 09fb95c..3a43199 100644 --- a/src/lib/cmd/getbookmark.js +++ b/src/lib/cmd/qseow/getbookmark.js @@ -1,8 +1,8 @@ import enigma from 'enigma.js'; import { table } from 'table'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; const consoleTableConfig = { border: { diff --git a/src/lib/cmd/getdim.js b/src/lib/cmd/qseow/getdim.js similarity index 98% rename from src/lib/cmd/getdim.js rename to src/lib/cmd/qseow/getdim.js index c62d91e..8f8e7f4 100644 --- a/src/lib/cmd/getdim.js +++ b/src/lib/cmd/qseow/getdim.js @@ -1,11 +1,9 @@ -/* eslint-disable no-restricted-syntax */ -/* eslint-disable no-await-in-loop */ import enigma from 'enigma.js'; import { table } from 'table'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; const consoleTableConfig = { border: { diff --git a/src/lib/cmd/getmeasure.js b/src/lib/cmd/qseow/getmeasure.js similarity index 98% rename from src/lib/cmd/getmeasure.js rename to src/lib/cmd/qseow/getmeasure.js index d530ab0..589f733 100644 --- a/src/lib/cmd/getmeasure.js +++ b/src/lib/cmd/qseow/getmeasure.js @@ -1,10 +1,9 @@ -/* eslint-disable no-restricted-syntax */ import enigma from 'enigma.js'; import { table } from 'table'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; const consoleTableConfig = { border: { diff --git a/src/lib/cmd/getscript.js b/src/lib/cmd/qseow/getscript.js similarity index 93% rename from src/lib/cmd/getscript.js rename to src/lib/cmd/qseow/getscript.js index aaf9be6..34f4822 100644 --- a/src/lib/cmd/getscript.js +++ b/src/lib/cmd/qseow/getscript.js @@ -1,7 +1,8 @@ import enigma from 'enigma.js'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; + +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { catchLog } from '../../util/log.js'; /** * @@ -25,7 +26,7 @@ const getScript = async (options) => { let configEnigma; let session; try { - configEnigma = await setupEnigmaConnection(options, sessionId); + configEnigma = setupEnigmaConnection(options, sessionId); session = await enigma.create(configEnigma); logger.verbose(`Created session to server ${options.host}.`); } catch (err) { diff --git a/src/lib/cmd/getsessions.js b/src/lib/cmd/qseow/getsessions.js similarity index 98% rename from src/lib/cmd/getsessions.js rename to src/lib/cmd/qseow/getsessions.js index 5c61dd1..f0b4faf 100644 --- a/src/lib/cmd/getsessions.js +++ b/src/lib/cmd/qseow/getsessions.js @@ -1,7 +1,7 @@ import { table } from 'table'; -import { getSessionsFromQseow } from '../util/session.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { getSessionsFromQseow } from '../../util/qseow/session.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; const consoleTableConfig = { border: { diff --git a/src/lib/cmd/gettask.js b/src/lib/cmd/qseow/gettask.js similarity index 99% rename from src/lib/cmd/gettask.js rename to src/lib/cmd/qseow/gettask.js index 2fdbb6a..4869910 100644 --- a/src/lib/cmd/gettask.js +++ b/src/lib/cmd/qseow/gettask.js @@ -1,14 +1,14 @@ import tree from 'text-treeview'; import { table } from 'table'; -import { promises as Fs } from 'fs'; +import { promises as Fs } from 'node:fs'; import xlsx from 'node-xlsx'; import { stringify } from 'csv-stringify'; import yesno from 'yesno'; -import { logger, setLoggingLevel, isPkg, execPath, verifyFileExists } from '../../globals.js'; -import QlikSenseTasks from '../task/class_alltasks.js'; -import { mapEventType, mapIncrementOption, mapDaylightSavingTime, mapRuleState } from '../util/lookups.js'; -import { getTagsFromQseow } from '../util/tag.js'; -import { catchLog } from '../util/log.js'; +import { logger, setLoggingLevel, isPkg, execPath, verifyFileExists } from '../../../globals.js'; +import QlikSenseTasks from '../../task/class_alltasks.js'; +import { mapEventType, mapIncrementOption, mapDaylightSavingTime, mapRuleState } from '../../util/qseow/lookups.js'; +import { getTagsFromQseow } from '../../util/qseow/tag.js'; +import { catchLog } from '../../util/log.js'; const consoleTableConfig = { border: { diff --git a/src/lib/cmd/getvariable.js b/src/lib/cmd/qseow/getvariable.js similarity index 97% rename from src/lib/cmd/getvariable.js rename to src/lib/cmd/qseow/getvariable.js index d0df068..3ca271b 100644 --- a/src/lib/cmd/getvariable.js +++ b/src/lib/cmd/qseow/getvariable.js @@ -1,12 +1,10 @@ -/* eslint-disable no-restricted-syntax */ -/* eslint-disable no-await-in-loop */ import enigma from 'enigma.js'; import { table } from 'table'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { getApps } from '../util/app.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { getApps } from '../../util/qseow/app.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; const consoleTableConfig = { border: { diff --git a/src/lib/cmd/import-masteritem-excel.js b/src/lib/cmd/qseow/import-masteritem-excel.js similarity index 99% rename from src/lib/cmd/import-masteritem-excel.js rename to src/lib/cmd/qseow/import-masteritem-excel.js index 4cec13a..95c1f75 100644 --- a/src/lib/cmd/import-masteritem-excel.js +++ b/src/lib/cmd/qseow/import-masteritem-excel.js @@ -1,12 +1,10 @@ -/* eslint-disable no-console */ -/* eslint-disable no-await-in-loop */ import enigma from 'enigma.js'; - import xlsx from 'node-xlsx'; import { v4 as uuidCreate } from 'uuid'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath, verifyFileExists, sleep } from '../../globals.js'; -import { catchLog } from '../util/log.js'; + +import { logger, setLoggingLevel, isPkg, execPath, verifyFileExists, sleep } from '../../../globals.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { catchLog } from '../../util/log.js'; let importCount = 0; @@ -55,7 +53,7 @@ const createColorMap = async (app, colorMapId, newPerValueColorMap) => { // const newColorMapProperties = await newGenericColorMapRefModel.getProperties(); // const newColorMapPropertiesLayout = await newGenericColorMapRefModel.getLayout(); // 4. Set properties of created color map - const res = await newGenericColorMapRefModel.setProperties(newGenericColorMapRefProp); + const _res = await newGenericColorMapRefModel.setProperties(newGenericColorMapRefProp); // let res = await newGenericColorMapRefModel.setProperties({ // qInfo: { diff --git a/src/lib/cmd/importapp.js b/src/lib/cmd/qseow/importapp.js similarity index 87% rename from src/lib/cmd/importapp.js rename to src/lib/cmd/qseow/importapp.js index 7ad908e..9ff5830 100644 --- a/src/lib/cmd/importapp.js +++ b/src/lib/cmd/qseow/importapp.js @@ -1,13 +1,11 @@ -// import { csvParse } from 'csv-parse'; - import xlsx from 'node-xlsx'; -import { logger, setLoggingLevel, isPkg, execPath, verifyFileExists, isNumeric } from '../../globals.js'; -import QlikSenseApps from '../app/class_allapps.js'; -import { getAppColumnPosFromHeaderRow } from '../util/lookups.js'; -import { getTagsFromQseow } from '../util/tag.js'; -import { getCustomPropertiesFromQseow } from '../util/customproperties.js'; -import { catchLog } from '../util/log.js'; +import { logger, setLoggingLevel, isPkg, execPath, verifyFileExists, isNumeric } from '../../../globals.js'; +import QlikSenseApps from '../../app/class_allapps.js'; +import { getAppColumnPosFromHeaderRow } from '../../util/qseow/lookups.js'; +import { getTagsFromQseow } from '../../util/qseow/tag.js'; +import { getCustomPropertiesFromQseow } from '../../util/qseow/customproperties.js'; +import { catchLog } from '../../util/log.js'; const importAppFromFile = async (options) => { try { @@ -25,6 +23,7 @@ const importAppFromFile = async (options) => { // Get all custom properties const cpExisting = await getCustomPropertiesFromQseow(options); + logger.info(`Successfully retrieved ${cpExisting.length} custom properties from QSEoW`); // Verify file exists const appFileExists = await verifyFileExists(options.fileName); diff --git a/src/lib/cmd/importtask.js b/src/lib/cmd/qseow/importtask.js similarity index 95% rename from src/lib/cmd/importtask.js rename to src/lib/cmd/qseow/importtask.js index fc4b4a6..ff047cd 100644 --- a/src/lib/cmd/importtask.js +++ b/src/lib/cmd/qseow/importtask.js @@ -1,20 +1,14 @@ import xlsx from 'node-xlsx'; import { parse } from 'csv-parse'; +import fs from 'node:fs'; -// const { parse } = require('csv-parse/lib/sync'); -import fs from 'fs'; - -// const fsp = require('fs').promises; - -import { finished } from 'stream/promises'; - -import { logger, setLoggingLevel, isPkg, execPath, verifyFileExists, isNumeric } from '../../globals.js'; -import QlikSenseTasks from '../task/class_alltasks.js'; -import QlikSenseApps from '../app/class_allapps.js'; -import { getTaskColumnPosFromHeaderRow } from '../util/lookups.js'; -import { getTagsFromQseow } from '../util/tag.js'; -import { getCustomPropertiesFromQseow } from '../util/customproperties.js'; -import { catchLog } from '../util/log.js'; +import { logger, setLoggingLevel, isPkg, execPath, verifyFileExists, isNumeric } from '../../../globals.js'; +import QlikSenseTasks from '../../task/class_alltasks.js'; +import QlikSenseApps from '../../app/class_allapps.js'; +import { getTaskColumnPosFromHeaderRow } from '../../util/qseow/lookups.js'; +import { getTagsFromQseow } from '../../util/qseow/tag.js'; +import { getCustomPropertiesFromQseow } from '../../util/qseow/customproperties.js'; +import { catchLog } from '../../util/log.js'; const getHeaders = async (options) => { const records = []; @@ -27,7 +21,6 @@ const getHeaders = async (options) => { ); // Get the header row - // eslint-disable-next-line no-restricted-syntax for await (const record of parser) { if (record.info.lines === 1) { // Header row @@ -44,7 +37,6 @@ const processCsvFile = async (options) => { const headerRow = []; // Push all column headers to array - // eslint-disable-next-line no-restricted-syntax for (const record of headers[0]) { // Get each column header text headerRow.push(record); @@ -289,7 +281,6 @@ const processCsvFile = async (options) => { }) ); - // eslint-disable-next-line no-restricted-syntax for await (const record of parser) { // ALways add the header line if (record.info.lines === 1) { @@ -324,6 +315,7 @@ const importTaskFromFile = async (options) => { // Get all custom properties const cpExisting = await getCustomPropertiesFromQseow(options); + logger.info(`Successfully retrieved ${cpExisting.length} custom properties from QSEoW`); // Verify task definitions file exists const taskFileExists = await verifyFileExists(options.fileName); @@ -348,7 +340,6 @@ const importTaskFromFile = async (options) => { data: [], }; - // eslint-disable-next-line no-restricted-syntax for (const item of tmpTasksFromFile) { tasksFromFile.data.push(item); } diff --git a/src/lib/cmd/scramblefield.js b/src/lib/cmd/qseow/scramblefield.js similarity index 95% rename from src/lib/cmd/scramblefield.js rename to src/lib/cmd/qseow/scramblefield.js index 720c87d..babeff2 100644 --- a/src/lib/cmd/scramblefield.js +++ b/src/lib/cmd/qseow/scramblefield.js @@ -1,7 +1,7 @@ import enigma from 'enigma.js'; -import { setupEnigmaConnection, addTrafficLogging } from '../util/enigma.js'; -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import { catchLog } from '../util/log.js'; +import { setupEnigmaConnection, addTrafficLogging } from '../../util/qseow/enigma_util.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; /** * diff --git a/src/lib/cmd/settaskcp.js b/src/lib/cmd/qseow/settaskcp.js similarity index 88% rename from src/lib/cmd/settaskcp.js rename to src/lib/cmd/qseow/settaskcp.js index 556b7be..ea3a3fe 100644 --- a/src/lib/cmd/settaskcp.js +++ b/src/lib/cmd/qseow/settaskcp.js @@ -1,7 +1,7 @@ import yesno from 'yesno'; -import { logger } from '../../globals.js'; -import { getCustomProperty, getTasksFromQseow, updateReloadTask } from '../task/task_qrs.js'; -import { catchLog } from '../util/log.js'; +import { logger } from '../../../globals.js'; +import { getCustomProperty, getTasksFromQseow, updateReloadTask } from '../../task/task_qrs.js'; +import { catchLog } from '../../util/log.js'; const updateTask = async (options, customPropertyDef, task) => new Promise(async (resolve, reject) => { @@ -42,7 +42,6 @@ const updateTask = async (options, customPropertyDef, task) => // Should new values be replacing or appended to existing values? if (options.updateMode === 'append') { - // eslint-disable-next-line no-restricted-syntax for (const newCpValue of options.customPropertyValue) { // Don't append a value if it's already set for the custom property if (!newPayload.task?.customProperties?.find((item) => item.value === newCpValue)) @@ -52,14 +51,13 @@ const updateTask = async (options, customPropertyDef, task) => }); } } else if (options.updateMode === 'replace') { - // First clear the custom property, then add the values provided via --custom-property-value + // First copy all existing CP values that should not be replaced to the new payload const cp = newPayload.task?.customProperties.filter( (existingCustomProperty) => existingCustomProperty.definition.name !== options.customPropertyName ); newPayload.task.customProperties = cp; // Now add the new CP values - // eslint-disable-next-line no-restricted-syntax for (const newCpValue of options.customPropertyValue) { newPayload.task?.customProperties?.push({ definition: { id: customPropertyDef[0].id, name: customPropertyDef[0].name }, @@ -75,12 +73,10 @@ const updateTask = async (options, customPropertyDef, task) => let ok; logger.info(); if (options.updateMode === 'replace') { - // eslint-disable-next-line no-await-in-loop ok = await yesno({ question: ` Replace current values in custom property "${options.customPropertyName}" with new ones? (y/n)`, }); } else if (options.updateMode === 'append') { - // eslint-disable-next-line no-await-in-loop ok = await yesno({ question: ` Append new values to custom property "${options.customPropertyName}"? (y/n)`, }); @@ -92,7 +88,6 @@ const updateTask = async (options, customPropertyDef, task) => logger.debug(`SET RELOAD TASK CP: Update payload for task ${task.id}: ${JSON.stringify(newPayload, null, 2)}`); if (options.dryRun === undefined || options.dryRun === false) { - // eslint-disable-next-line no-await-in-loop updateResult = await updateReloadTask(options, newPayload); } else { logger.info(`DRY RUN: Update of task custom property ${task.customPropertyName} would happen here.`); @@ -101,7 +96,6 @@ const updateTask = async (options, customPropertyDef, task) => logger.info(`Did not update task "${task.name}"`); } } else if (options.dryRun === undefined || options.dryRun === false) { - // eslint-disable-next-line no-await-in-loop updateResult = await updateReloadTask(options, newPayload); } else { logger.info(`DRY RUN: Update of task custom property ${task.customPropertyName} would happen here.`); @@ -123,7 +117,6 @@ const updateTask = async (options, customPropertyDef, task) => // Custom property does NOT already have the custom property set for this task. // Add CP values to task - // eslint-disable-next-line no-restricted-syntax for (const newCpValue of options.customPropertyValue) { newPayload.task?.customProperties?.push({ definition: { id: customPropertyDef[0].id, name: customPropertyDef[0].name }, @@ -133,7 +126,6 @@ const updateTask = async (options, customPropertyDef, task) => // Update task logger.debug(`SET RELOAD TASK CP: Update payload for task ${task.id}: ${JSON.stringify(newPayload, null, 2)}`); - // eslint-disable-next-line no-await-in-loop const updateResult = await updateReloadTask(options, newPayload); if (updateResult) { @@ -177,7 +169,6 @@ const setTaskCustomProperty = async (options) => { } // Ensure that the new CP values are among the CP's choiceValues - // eslint-disable-next-line no-restricted-syntax for (const newCPValue of options.customPropertyValue) { if (!customPropertyDef[0].choiceValues.find((item) => item === newCPValue)) { // An invalud custom property value detected @@ -195,22 +186,15 @@ const setTaskCustomProperty = async (options) => { // Log which tasks will be processed logger.info(`Number of tasks that will be updated: ${taskList.length}`); - // const updateTasks = []; - // eslint-disable-next-line no-restricted-syntax for (const task of taskList) { - // updateTasks.push(updateTask(options, customPropertyDef, task)); - logger.info(``); logger.info(`-----------------------------------------------------------`); logger.info(`Processing task "${task.name}" with ID=${task.id}`); - // eslint-disable-next-line no-await-in-loop const res = await updateTask(options, customPropertyDef, task); logger.debug(`Custom property update result: ${res}`); } - // await Promise.all(updateTasks); - // logger.debug('Update task custom property: All promises resolved'); return true; } } catch (err) { diff --git a/src/lib/cmd/testconnection.js b/src/lib/cmd/qseow/testconnection.js similarity index 91% rename from src/lib/cmd/testconnection.js rename to src/lib/cmd/qseow/testconnection.js index 2588787..21b0ddb 100644 --- a/src/lib/cmd/testconnection.js +++ b/src/lib/cmd/qseow/testconnection.js @@ -1,6 +1,6 @@ -import { logger, setLoggingLevel, isPkg, execPath } from '../../globals.js'; -import getAboutFromQseow from '../util/about.js'; -import { catchLog } from '../util/log.js'; +import { logger, setLoggingLevel, isPkg, execPath } from '../../../globals.js'; +import getAboutFromQseow from '../../util/qseow/about.js'; +import { catchLog } from '../../util/log.js'; const testConnection = async (options) => { try { diff --git a/src/lib/cmd/qseow/useractivity.js b/src/lib/cmd/qseow/useractivity.js new file mode 100644 index 0000000..05c2383 --- /dev/null +++ b/src/lib/cmd/qseow/useractivity.js @@ -0,0 +1,241 @@ +import axios from 'axios'; +import path from 'node:path'; + +import { logger, execPath } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; +import { setupQrsConnection } from '../../util/qseow/qrs.js'; + +// Function to get user activity from QRS for license type "Analyzer" +export async function getUserActivityAnalyzer(options) { + logger.verbose(`Getting user activity for license type "Analyzer"...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/license/analyzeraccesstype/full', + }); + + const result = await axios.request(axiosConfig); + if (result.status === 200) { + const response = JSON.parse(result.data); + logger.info(` Successfully retrieved ${response.length} user activity records for license type "Analyzer" from QSEoW`); + + return response; + } + return false; + } catch (err) { + catchLog(`USER ACTIVITY ANALYZER: Error getting user activity info from QRS`, err); + return false; + } +} + +// Function to get user activity from QRS for license type "Analyzer Time" +export async function getUserActivityAnalyzerTime(options) { + logger.verbose(`Getting user activity for license type "Analyzer Time"...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/license/analyzertimeaccessusage/full', + }); + + const result = await axios.request(axiosConfig); + if (result.status === 200) { + const response = JSON.parse(result.data); + logger.info(` Successfully retrieved ${response.length} user activity records for license type "Analyzer Time" from QSEoW`); + + return response; + } + return false; + } catch (err) { + catchLog(`USER ACTIVITY ANALYZER TIME: Error getting user activity info from QRS`, err); + return false; + } +} + +// Function to get user activity from QRS for license type "Login" +export async function getUserActivityLogin(options) { + logger.verbose(`Getting user activity for license type "Login"...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/license/loginaccessusage/full', + }); + + const result = await axios.request(axiosConfig); + if (result.status === 200) { + const response = JSON.parse(result.data); + logger.info(` Successfully retrieved ${response.length} user activity records for license type "Login" from QSEoW`); + + return response; + } + return false; + } catch (err) { + catchLog(`USER ACTIVITY LOGIN: Error getting user activity info from QRS`, err); + return false; + } +} + +// Function to get user activity from QRS for license type "Professional" +export async function getUserActivityProfessional(options) { + logger.verbose(`Getting user activity for license type "Professional"...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/license/professionalaccesstype/full', + }); + + const result = await axios.request(axiosConfig); + if (result.status === 200) { + const response = JSON.parse(result.data); + logger.info(` Successfully retrieved ${response.length} user activity records for license type "Professional" from QSEoW`); + + return response; + } + return false; + } catch (err) { + catchLog(`USER ACTIVITY PROFESSIONAL: Error getting user activity info from QRS`, err); + return false; + } +} + +// Function to get user activity from QRS for license type "User" +export async function getUserActivityUser(options) { + logger.verbose(`Getting user activity for license type "User"...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/license/useraccessusage/full', + }); + + const result = await axios.request(axiosConfig); + if (result.status === 200) { + const response = JSON.parse(result.data); + logger.info(` Successfully retrieved ${response.length} user activity records for license type "User" from QSEoW`); + + return response; + } + return false; + } catch (err) { + catchLog(`USER ACTIVITY USER: Error getting user activity info from QRS`, err); + return false; + } +} + +// Function to extract the last activity date for the different license types. +// +// Return: +// An array of objects, each object containing +// - user directory +// - user ID +// - user name +// - last activity date +export async function getUsersLastActivity(activityAnalyzer, activityAnalyzerTime, activityLogin, activityProfessional, activityUser) { + const usersActivity = []; + + for (const user of activityAnalyzer) { + // Does this user already exist in user activity array? + if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { + // User ID has already been added (seems it appears in more than one activity type!) + // Pick the most recent last activity date + logger.debug( + ` USER ACTIVITY ANALYZER: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` + ); + } else { + // User ID has not been added yet. Add it! + usersActivity.push({ + userSenseId: user.user.id, + userId: user.user.userId, + userDirectory: user.user.userDirectory, + userName: user.user.name, + lastUsed: user.lastUsed, + }); + } + } + + for (const user of activityAnalyzerTime) { + // Does this user already exist in user activity array? + if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { + // User ID has already been added (seems it appears in more than one activity type!) + // Pick the most recent last activity date + logger.debug( + ` USER ACTIVITY ANALYZER TIME: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` + ); + } else { + // User ID has not been added yet. Add it! + usersActivity.push({ + userSenseId: user.user.id, + userId: user.user.userId, + userDirectory: user.user.userDirectory, + userName: user.user.name, + lastUsed: user.latestActivity, + }); + } + } + + for (const user of activityLogin) { + // Does this user already exist in user activity array? + if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { + // User ID has already been added (seems it appears in more than one activity type!) + // Pick the most recent last activity date + logger.debug( + ` USER ACTIVITY LOGIN: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` + ); + } else { + // User ID has not been added yet. Add it! + usersActivity.push({ + userSenseId: user.user.id, + userId: user.user.userId, + userDirectory: user.user.userDirectory, + userName: user.user.name, + lastUsed: user.latestActivity, + }); + } + } + + for (const user of activityProfessional) { + // Does this user already exist in user activity array? + if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { + // User ID has already been added (seems it appears in more than one activity type!) + // Pick the most recent last activity date + logger.debug( + ` USER ACTIVITY PROFESSIONAL: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` + ); + } else { + // User ID has not been added yet. Add it! + usersActivity.push({ + userSenseId: user.user.id, + userId: user.user.userId, + userDirectory: user.user.userDirectory, + userName: user.user.name, + lastUsed: user.lastUsed, + }); + } + } + + for (const user of activityUser) { + // Does this user already exist in user activity array? + if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { + // User ID has already been added (seems it appears in more than one activity type!) + // Pick the most recent last activity date + logger.debug( + ` USER ACTIVITY USER: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` + ); + } else { + // User ID has not been added yet. Add it! + usersActivity.push({ + userSenseId: user.user.id, + userId: user.user.userId, + userDirectory: user.user.userDirectory, + userName: user.user.name, + lastUsed: user.lastUsed, + }); + } + } + + logger.verbose(` USER ACTIVITY: Net list of user activity data consists of ${usersActivity.length} items.`); + return usersActivity; +} diff --git a/src/lib/cmd/vistask.js b/src/lib/cmd/qseow/vistask.js similarity index 98% rename from src/lib/cmd/vistask.js rename to src/lib/cmd/qseow/vistask.js index fc33b05..d11b58c 100644 --- a/src/lib/cmd/vistask.js +++ b/src/lib/cmd/qseow/vistask.js @@ -1,10 +1,10 @@ -import http from 'http'; -import path from 'path'; -import fs from 'fs'; +import http from 'node:http'; +import path from 'node:path'; +import fs from 'node:fs'; import handlebars from 'handlebars'; -import { Readable } from 'stream'; -import { appVersion, logger, setLoggingLevel, isPkg, execPath, verifyFileExists } from '../../globals.js'; -import QlikSenseTasks from '../task/class_alltasks.js'; +import { Readable } from 'node:stream'; +import { appVersion, logger, setLoggingLevel, isPkg, execPath, verifyFileExists } from '../../../globals.js'; +import QlikSenseTasks from '../../task/class_alltasks.js'; // js: 'application/javascript', const MIME_TYPES = { @@ -133,7 +133,7 @@ function getSchemaText(incrementOption, incrementDescription) { "2: daily", incrementDescription: Repeat after each '0 0 days 0 ' "3: weekly", - "4: monthly" + "4: monthly" */ if (incrementOption === 0) { diff --git a/src/lib/cmd/useractivity.js b/src/lib/cmd/useractivity.js deleted file mode 100644 index ea489bb..0000000 --- a/src/lib/cmd/useractivity.js +++ /dev/null @@ -1,187 +0,0 @@ -import { logger } from '../../globals.js'; -import { catchLog } from '../util/log.js'; - -export function getUserActivityProfessional(qrsInteractInstance) { - // eslint-disable-next-line no-unused-vars, no-async-promise-executor - return new Promise(async (resolve, _reject) => { - let result; - try { - result = await qrsInteractInstance.Get('license/professionalaccesstype/full'); - } catch (err) { - catchLog(`USER ACTIVITY PROFESSIONAL: Error getting user activity info from QRS`, err); - } - - resolve(result.body); - }); -} - -export function getUserActivityAnalyzer(qrsInteractInstance) { - // eslint-disable-next-line no-unused-vars, no-async-promise-executor - return new Promise(async (resolve, _reject) => { - let result; - try { - result = await qrsInteractInstance.Get('license/analyzeraccesstype/full'); - } catch (err) { - catchLog(`USER ACTIVITY ANALYZER: Error getting user activity info from QRS`, err); - } - - resolve(result.body); - }); -} - -export function getUserActivityAnalyzerTime(qrsInteractInstance) { - // eslint-disable-next-line no-unused-vars, no-async-promise-executor - return new Promise(async (resolve, _reject) => { - let result; - try { - result = await qrsInteractInstance.Get('license/analyzertimeaccesstype/full'); - } catch (err) { - catchLog(`USER ACTIVITY ANALYZER TIME: Error getting user activity info from QRS`, err); - } - - resolve(result.body); - }); -} - -export function getUserActivityLogin(qrsInteractInstance) { - // eslint-disable-next-line no-unused-vars, no-async-promise-executor - return new Promise(async (resolve, _reject) => { - let result; - try { - result = await qrsInteractInstance.Get('license/loginaccesstype/full'); - } catch (err) { - catchLog(`USER ACTIVITY LOGIN: Error getting user activity info from QRS`, err); - } - - resolve(result.body); - }); -} - -export function getUserActivityUser(qrsInteractInstance) { - // eslint-disable-next-line no-unused-vars, no-async-promise-executor - return new Promise(async (resolve, _reject) => { - let result; - try { - result = await qrsInteractInstance.Get('license/useraccesstype/full'); - } catch (err) { - catchLog(`USER ACTIVITY USER: Error getting user activity info from QRS`, err); - } - - resolve(result.body); - }); -} - -export function getUsersLastActivity(activityProfessional, activityAnalyzer, activityAnalyzerTime, activityLogin, activityUser) { - // eslint-disable-next-line no-unused-vars, no-async-promise-executor - return new Promise(async (resolve, _reject) => { - const usersActivity = []; - - // eslint-disable-next-line no-restricted-syntax - for (const user of activityProfessional) { - // Does this user already exist in user activity array? - if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { - // User ID has already been added (seems it appears in more than one activity type!) - // Pick the most recent last activity date - logger.debug( - `USER ACTIVITY PROFESSIONAL: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` - ); - } else { - // User ID has not been added yet. Add it! - usersActivity.push({ - userSenseId: user.user.id, - userId: user.user.userId, - userDirectory: user.user.userDirectory, - userName: user.user.name, - lastUsed: user.lastUsed, - }); - } - } - - // eslint-disable-next-line no-restricted-syntax - for (const user of activityAnalyzer) { - // Does this user already exist in user activity array? - if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { - // User ID has already been added (seems it appears in more than one activity type!) - // Pick the most recent last activity date - logger.debug( - `USER ACTIVITY ANALYZER: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` - ); - } else { - // User ID has not been added yet. Add it! - usersActivity.push({ - userSenseId: user.user.id, - userId: user.user.userId, - userDirectory: user.user.userDirectory, - userName: user.user.name, - lastUsed: user.lastUsed, - }); - } - } - - // eslint-disable-next-line no-restricted-syntax - for (const user of activityAnalyzerTime) { - // Does this user already exist in user activity array? - if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { - // User ID has already been added (seems it appears in more than one activity type!) - // Pick the most recent last activity date - logger.debug( - `USER ACTIVITY ANALYZER TIME: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` - ); - } else { - // User ID has not been added yet. Add it! - usersActivity.push({ - userSenseId: user.user.id, - userId: user.user.userId, - userDirectory: user.user.userDirectory, - userName: user.user.name, - lastUsed: user.lastUsed, - }); - } - } - - // eslint-disable-next-line no-restricted-syntax - for (const user of activityLogin) { - // Does this user already exist in user activity array? - if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { - // User ID has already been added (seems it appears in more than one activity type!) - // Pick the most recent last activity date - logger.debug( - `USER ACTIVITY LOGIN: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` - ); - } else { - // User ID has not been added yet. Add it! - usersActivity.push({ - userSenseId: user.user.id, - userId: user.user.userId, - userDirectory: user.user.userDirectory, - userName: user.user.name, - lastUsed: user.lastUsed, - }); - } - } - - // eslint-disable-next-line no-restricted-syntax - for (const user of activityUser) { - // Does this user already exist in user activity array? - if (usersActivity.find((findUser) => findUser.userSenseId === user.user.id) !== undefined) { - // User ID has already been added (seems it appears in more than one activity type!) - // Pick the most recent last activity date - logger.debug( - `USER ACTIVITY USER: User id ${user.user.id}, ${user.user.userDirectory}\\${user.user.userId} already exists in activity array. Will use entry with the most recent activity date.` - ); - } else { - // User ID has not been added yet. Add it! - usersActivity.push({ - userSenseId: user.user.id, - userId: user.user.userId, - userDirectory: user.user.userDirectory, - userName: user.user.name, - lastUsed: user.lastUsed, - }); - } - } - - logger.verbose(`USER ACTIVITY: Net list of user activity data consists of ${usersActivity.length} items.`); - resolve(usersActivity); - }); -} diff --git a/src/lib/task/class_allcompositeevents.js b/src/lib/task/class_allcompositeevents.js index f6dab0e..7e7e1a1 100644 --- a/src/lib/task/class_allcompositeevents.js +++ b/src/lib/task/class_allcompositeevents.js @@ -1,12 +1,13 @@ import axios from 'axios'; -import path from 'path'; +import path from 'node:path'; + import { logger, execPath, verifyFileExists } from '../../globals.js'; -import setupQRSConnection from '../util/qrs.js'; +import { setupQrsConnection } from '../util/qseow/qrs.js'; import QlikSenseCompositeEvent from './class_compositeevent.js'; import { catchLog } from '../util/log.js'; +import { getCertFilePaths } from '../util/qseow/cert.js'; class QlikSenseCompositeEvents { - // eslint-disable-next-line no-useless-constructor constructor() { // } @@ -16,11 +17,14 @@ class QlikSenseCompositeEvents { this.compositeEventList = []; this.options = options; + // Should certificates be used for authentication? if (this.options.authType === 'cert') { - // Make sure certificates exist - this.fileCert = path.resolve(execPath, options.authCertFile); - this.fileCertKey = path.resolve(execPath, options.authCertKeyFile); - this.fileCertCA = path.resolve(execPath, options.authRootCertFile); + // Get certificate paths + const { fileCert, fileCertKey, fileCertCA } = getCertFilePaths(options); + + this.fileCert = fileCert; + this.fileCertKey = fileCertKey; + this.fileCertCA = fileCertCA; } } catch (err) { catchLog(`GET COMPOSITE EVENT`, err); @@ -42,7 +46,7 @@ class QlikSenseCompositeEvents { try { logger.debug('GET SCHEMAEVENT: Starting get composite events from QSEoW'); - const axiosConfig = await setupQRSConnection(this.options, { + const axiosConfig = await setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, diff --git a/src/lib/task/class_allschemaevents.js b/src/lib/task/class_allschemaevents.js index b0a9866..80007d9 100644 --- a/src/lib/task/class_allschemaevents.js +++ b/src/lib/task/class_allschemaevents.js @@ -1,12 +1,11 @@ import axios from 'axios'; -import path from 'path'; -import { logger, execPath } from '../../globals.js'; -import setupQRSConnection from '../util/qrs.js'; +import { logger } from '../../globals.js'; +import { setupQrsConnection } from '../util/qseow/qrs.js'; import QlikSenseSchemaEvent from './class_schemaevent.js'; import { catchLog } from '../util/log.js'; +import { getCertFilePaths } from '../util/qseow/cert.js'; class QlikSenseSchemaEvents { - // eslint-disable-next-line no-useless-constructor constructor() { // } @@ -17,10 +16,12 @@ class QlikSenseSchemaEvents { this.options = options; if (this.options.authType === 'cert') { - // Make sure certificates exist - this.fileCert = path.resolve(execPath, options.authCertFile); - this.fileCertKey = path.resolve(execPath, options.authCertKeyFile); - this.fileCertCA = path.resolve(execPath, options.authRootCertFile); + // Get certificate paths + const { fileCert, fileCertKey, fileCertCA } = getCertFilePaths(options); + + this.fileCert = fileCert; + this.fileCertKey = fileCertKey; + this.fileCertCA = fileCertCA; } } catch (err) { catchLog(`GET SCHEMA EVENT INIT`, err); @@ -38,7 +39,6 @@ class QlikSenseSchemaEvents { } getSchemaEventsFromFile(schemaEvent) { - // eslint-disable-next-line no-async-promise-executor return new Promise(async (resolve, reject) => { try { logger.debug('GET SCHEMA EVENT: Starting get schema events from QSEoW'); @@ -54,12 +54,11 @@ class QlikSenseSchemaEvents { } getSchemaEventsFromQseow() { - // eslint-disable-next-line no-async-promise-executor return new Promise(async (resolve, reject) => { try { logger.debug('GET SCHEMA EVENT: Starting get schema events from QSEoW'); - const axiosConfig = await setupQRSConnection(this.options, { + const axiosConfig = await setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -75,7 +74,6 @@ class QlikSenseSchemaEvents { logger.verbose(`GET SCHEMA EVENT: Total number of schema events: ${schemaEvents.length}`); this.clear(); - // eslint-disable-next-line no-plusplus for (let i = 0; i < schemaEvents.length; i++) { this.addSchemaEvent(schemaEvents[i]); } diff --git a/src/lib/task/class_alltasks.js b/src/lib/task/class_alltasks.js index 5b40beb..2dccbde 100644 --- a/src/lib/task/class_alltasks.js +++ b/src/lib/task/class_alltasks.js @@ -1,8 +1,7 @@ import axios from 'axios'; -import path from 'path'; import { v4 as uuidv4, validate } from 'uuid'; -import { logger, execPath } from '../../globals.js'; -import setupQRSConnection from '../util/qrs.js'; +import { logger } from '../../globals.js'; +import { setupQrsConnection } from '../util/qseow/qrs.js'; import { mapTaskType, @@ -11,19 +10,19 @@ import { mapIncrementOption, mapRuleState, getTaskColumnPosFromHeaderRow, -} from '../util/lookups.js'; +} from '../util/qseow/lookups.js'; import QlikSenseTask from './class_task.js'; import QlikSenseSchemaEvents from './class_allschemaevents.js'; import QlikSenseCompositeEvents from './class_allcompositeevents.js'; -import { getTagIdByName } from '../util/tag.js'; -import { getCustomPropertyIdByName } from '../util/customproperties.js'; -import { getAppById } from '../util/app.js'; -import { taskExistById, getTaskById } from '../util/task.js'; +import { getTagIdByName } from '../util/qseow/tag.js'; +import { getCustomPropertyIdByName } from '../util/qseow/customproperties.js'; +import { getAppById } from '../util/qseow/app.js'; +import { taskExistById, getTaskById } from '../util/qseow/task.js'; import { catchLog } from '../util/log.js'; +import { getCertFilePaths } from '../util/qseow/cert.js'; class QlikSenseTasks { - // eslint-disable-next-line no-useless-constructor constructor() { // } @@ -43,10 +42,12 @@ class QlikSenseTasks { this.taskTreeCyclicVisited = new Set(); if (options.authType === 'cert') { - // Make sure certificates exist - this.fileCert = path.resolve(execPath, options.authCertFile); - this.fileCertKey = path.resolve(execPath, options.authCertKeyFile); - this.fileCertCA = path.resolve(execPath, options.authRootCertFile); + // Get certificate paths + const { fileCert, fileCertKey, fileCertCA } = getCertFilePaths(options); + + this.fileCert = fileCert; + this.fileCertKey = fileCertKey; + this.fileCertCA = fileCertCA; } this.qlikSenseSchemaEvents = new QlikSenseSchemaEvents(); @@ -166,7 +167,6 @@ class QlikSenseTasks { process.exit(1); } - // eslint-disable-next-line no-await-in-loop const app = await getAppById(appId, param?.options); if (!app) { @@ -178,7 +178,6 @@ class QlikSenseTasks { } else if (validate(appIdRaw)) { // App ID is a proper UUID. We don't know if the app actually exists though. - // eslint-disable-next-line no-await-in-loop const app = await getAppById(appIdRaw, param?.options); if (!app) { @@ -244,9 +243,7 @@ class QlikSenseTasks { .filter((item) => item.trim().length !== 0) .map((item) => item.trim()); - // eslint-disable-next-line no-restricted-syntax for (const item of tmpTags) { - // eslint-disable-next-line no-await-in-loop const tagId = await getTagIdByName(item, param.tagsExisting); currentTask.tags.push({ id: tagId, @@ -262,7 +259,6 @@ class QlikSenseTasks { .filter((item) => item.trim().length !== 0) .map((cp) => cp.trim()); - // eslint-disable-next-line no-restricted-syntax for (const item of tmpCustomProperties) { const tmpCustomProperty = item .split('=') @@ -271,8 +267,7 @@ class QlikSenseTasks { // Do we have two items in the array? First is the custom property name, second is the value if (tmpCustomProperty?.length === 2) { - // eslint-disable-next-line no-await-in-loop - const customPropertyId = await getCustomPropertyIdByName('ReloadTask', tmpCustomProperty[0], param.cpExisting); + const customPropertyId = getCustomPropertyIdByName('ReloadTask', tmpCustomProperty[0], param.cpExisting); // If previous call returned false, it means the custom property does not exist in Sense // or cannot be used with this task type. In that case, skip it. @@ -387,9 +382,7 @@ class QlikSenseTasks { .filter((item) => item.trim().length !== 0) .map((item) => item.trim()); - // eslint-disable-next-line no-restricted-syntax for (const item of tmpTags) { - // eslint-disable-next-line no-await-in-loop const tagId = await getTagIdByName(item, param.tagsExisting); currentTask.tags.push({ id: tagId, @@ -405,7 +398,6 @@ class QlikSenseTasks { .filter((item) => item.trim().length !== 0) .map((cp) => cp.trim()); - // eslint-disable-next-line no-restricted-syntax for (const item of tmpCustomProperties) { const tmpCustomProperty = item .split('=') @@ -414,8 +406,7 @@ class QlikSenseTasks { // Do we have two items in the array? First is the custom property name, second is the value if (tmpCustomProperty?.length === 2) { - // eslint-disable-next-line no-await-in-loop - const customPropertyId = await getCustomPropertyIdByName('ExternalProgramTask', tmpCustomProperty[0], param.cpExisting); + const customPropertyId = getCustomPropertyIdByName('ExternalProgramTask', tmpCustomProperty[0], param.cpExisting); // If previous call returned false, it means the custom property does not exist in Sense // or cannot be used with this task type. In that case, skip it. @@ -486,7 +477,6 @@ class QlikSenseTasks { ); // Add schema edges and start/trigger nodes - // eslint-disable-next-line no-restricted-syntax for (const schemaEventRow of schemaEventRows) { // Create object using same format that Sense uses for schema events const schemaEvent = { @@ -605,7 +595,6 @@ class QlikSenseTasks { ); // Loop over all composite events, adding them and their event rules - // eslint-disable-next-line no-restricted-syntax for (const compositeEventRow of compositeEventRows) { // Get value in "Event counter" column for this composite event, then get array of all associated event rules const compositeEventCounter = compositeEventRow[param.taskFileColumnHeaders.eventCounter.pos]; @@ -645,7 +634,6 @@ class QlikSenseTasks { } // Add rules - // eslint-disable-next-line no-restricted-syntax for (const rule of compositeEventRules) { // Does the upstream task pointed to by the composite rule exist? // If it *does* exist it means it's a real, existing task in QSEoW that should be used. @@ -654,7 +642,6 @@ class QlikSenseTasks { if (validate(rule[param.taskFileColumnHeaders.ruleTaskId.pos])) { // The rule points to an valid UUID. It should exist, otherwise it's an error - // eslint-disable-next-line no-await-in-loop const taskExists = await taskExistById(rule[param.taskFileColumnHeaders.ruleTaskId.pos], this.options); if (taskExists) { @@ -711,7 +698,6 @@ class QlikSenseTasks { upstreamTaskExistence = 'exists-in-source-file'; } else { - // eslint-disable-next-line no-await-in-loop upstreamTask = await getTaskById(rule[param.taskFileColumnHeaders.ruleTaskId.pos], param?.options); // Save upstream task in shared task list @@ -810,7 +796,6 @@ class QlikSenseTasks { param.nodesWithEvents.add(nodeId); // Add edges from upstream tasks to the new meta node - // eslint-disable-next-line no-restricted-syntax for (const rule of compositeEvent.compositeRules) { this.taskNetwork.edges.push({ from: rule.task.id, @@ -850,7 +835,6 @@ class QlikSenseTasks { // - cpExisting: Array of existing custom properties in QSEoW // - options: Options object passed on the command line async getTaskModelFromFile(tasksFromFile, tagsExisting, cpExisting, options) { - // eslint-disable-next-line no-async-promise-executor return new Promise(async (resolve, reject) => { try { logger.debug('PARSE TASKS FROM FILE: Starting get tasks from data in file'); @@ -947,7 +931,6 @@ class QlikSenseTasks { // Create a fake ID for this task. Used to associate task with schema/composite events const fakeTaskId = `reload-task-${uuidv4()}`; - // eslint-disable-next-line no-await-in-loop const res = await this.parseReloadTask({ taskRows, taskFileColumnHeaders, @@ -963,7 +946,6 @@ class QlikSenseTasks { // NB: A top level node is defined as: // 1. A task whose taskID does not show up in the "to" field of any edge. - // eslint-disable-next-line no-restricted-syntax this.taskNetwork.nodes.push({ id: res.currentTask.id, metaNode: false, @@ -1007,7 +989,6 @@ class QlikSenseTasks { if (this.options.updateMode === 'create') { // Create new task if (this.options.dryRun === false || this.options.dryRun === undefined) { - // eslint-disable-next-line no-await-in-loop const newTaskId = await this.createReloadTaskInQseow(res.currentTask, taskCounter); logger.info( `(${taskCounter}) Created new reload task "${res.currentTask.name}", new task id: ${newTaskId}.` @@ -1024,7 +1005,6 @@ class QlikSenseTasks { res.currentTask.idRef = res.currentTask.id; res.currentTask.id = newTaskId; - // eslint-disable-next-line no-await-in-loop await this.addTask('from_file', res.currentTask, false); } else { logger.info(`(${taskCounter}) DRY RUN: Creating reload task in QSEoW "${res.currentTask.name}"`); @@ -1078,7 +1058,6 @@ class QlikSenseTasks { // Create a fake ID for this task. Used to associate task with schema/composite events const fakeTaskId = `ext-pgm-task-${uuidv4()}`; - // eslint-disable-next-line no-await-in-loop const res = await this.parseExternalProgramTask({ taskRows, taskFileColumnHeaders, @@ -1094,7 +1073,6 @@ class QlikSenseTasks { // NB: A top level node is defined as: // 1. A task whose taskID does not show up in the "to" field of any edge. - // eslint-disable-next-line no-restricted-syntax this.taskNetwork.nodes.push({ id: res.currentTask.id, metaNode: false, @@ -1123,7 +1101,6 @@ class QlikSenseTasks { if (this.options.updateMode === 'create') { // Create new task if (this.options.dryRun === false || this.options.dryRun === undefined) { - // eslint-disable-next-line no-await-in-loop const newTaskId = await this.createExternalProgramTaskInQseow(res.currentTask, taskCounter); logger.info( `(${taskCounter}) Created new external program task "${res.currentTask.name}", new task id: ${newTaskId}.` @@ -1140,7 +1117,6 @@ class QlikSenseTasks { res.currentTask.idRef = res.currentTask.id; res.currentTask.id = newTaskId; - // eslint-disable-next-line no-await-in-loop await this.addTask('from_file', res.currentTask, false); } else { logger.info(`(${taskCounter}) DRY RUN: Creating external program task in QSEoW "${res.currentTask.name}"`); @@ -1211,7 +1187,6 @@ class QlikSenseTasks { taskType = task.taskType; // const { taskType } = this.taskNetwork.nodes.find((node) => node.id === id).completeTaskObject; } else if (b.upstreamTaskExistence === 'exists-in-sense') { - // eslint-disable-next-line no-await-in-loop const task = this.compositeEventUpstreamTask.find((item4) => item4.id === b.task.id); // Ensure we got a task back @@ -1256,10 +1231,8 @@ class QlikSenseTasks { logger.info('-------------------------------------------------------------------'); logger.info('Creating composite events for the just created tasks...'); - // eslint-disable-next-line no-restricted-syntax for (const { compositeEvent } of this.qlikSenseCompositeEvents.compositeEventList) { if (this.options.dryRun === false || this.options.dryRun === undefined) { - // eslint-disable-next-line no-await-in-loop await this.createCompositeEventInQseow(compositeEvent); } else { logger.info(`DRY RUN: Creating composite event "${compositeEvent.name}"`); @@ -1288,7 +1261,6 @@ class QlikSenseTasks { } createCompositeEventInQseow(newCompositeEvent) { - // eslint-disable-next-line no-async-promise-executor return new Promise(async (resolve, reject) => { try { logger.debug('CREATE COMPOSITE EVENT IN QSEOW: Starting'); @@ -1297,7 +1269,7 @@ class QlikSenseTasks { const body = newCompositeEvent; // Save task to QSEoW - const axiosConfig = setupQRSConnection(this.options, { + const axiosConfig = setupQrsConnection(this.options, { method: 'post', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1341,7 +1313,6 @@ class QlikSenseTasks { // Function to create new reload task in QSEoW createReloadTaskInQseow(newTask, taskCounter) { - // eslint-disable-next-line no-async-promise-executor return new Promise(async (resolve, reject) => { try { logger.debug(`(${taskCounter}) CREATE RELOAD TASK IN QSEOW: Starting`); @@ -1367,7 +1338,7 @@ class QlikSenseTasks { }; // Save task to QSEoW - const axiosConfig = setupQRSConnection(this.options, { + const axiosConfig = setupQrsConnection(this.options, { method: 'post', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1407,7 +1378,6 @@ class QlikSenseTasks { // - newTask: Object containing task data // - taskCounter: Task counter, unique for each task in the source file createExternalProgramTaskInQseow(newTask, taskCounter) { - // eslint-disable-next-line no-async-promise-executor return new Promise(async (resolve, reject) => { try { logger.debug(`(${taskCounter}) CREATE EXTERNAL PROGRAM TASK IN QSEOW: Starting`); @@ -1430,7 +1400,7 @@ class QlikSenseTasks { }; // Save task to QSEoW - const axiosConfig = setupQRSConnection(this.options, { + const axiosConfig = setupQrsConnection(this.options, { method: 'post', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1466,14 +1436,11 @@ class QlikSenseTasks { } saveTaskModelToQseow() { - // eslint-disable-next-line no-async-promise-executor return new Promise(async (resolve, reject) => { try { logger.debug('SAVE TASKS TO QSEOW: Starting save tasks to QSEoW'); - // eslint-disable-next-line no-restricted-syntax for (const task of this.taskList) { - // eslint-disable-next-line no-await-in-loop await new Promise((resolve2, reject2) => { // Build a body for the API call const body = { @@ -1497,7 +1464,7 @@ class QlikSenseTasks { }; // Save task to QSEoW - const axiosConfig = setupQRSConnection(this.options, { + const axiosConfig = setupQrsConnection(this.options, { method: 'post', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1533,7 +1500,6 @@ class QlikSenseTasks { } async getTasksFromQseow() { - // eslint-disable-next-line no-async-promise-executor, no-unused-vars return new Promise(async (resolve, reject) => { // try { logger.debug('GET TASKS FROM QSEOW: Starting get reload tasks from QSEoW'); @@ -1604,7 +1570,7 @@ class QlikSenseTasks { try { // Get reload tasks if (filter === '') { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1612,7 +1578,7 @@ class QlikSenseTasks { path: '/qrs/reloadtask/full', }); } else { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1634,7 +1600,7 @@ class QlikSenseTasks { try { // Get external program tasks if (filter === '') { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1642,7 +1608,7 @@ class QlikSenseTasks { path: '/qrs/externalprogramtask/full', }); } else { - axiosConfig = setupQRSConnection(this.options, { + axiosConfig = setupQrsConnection(this.options, { method: 'get', fileCert: this.fileCert, fileCertKey: this.fileCertKey, @@ -1702,7 +1668,6 @@ class QlikSenseTasks { const downstreamTasks = self.taskNetwork.edges.filter((edge) => edge.from === task.id); let kids = []; - // eslint-disable-next-line no-restricted-syntax for (const downstreamTask of downstreamTasks) { logger.debug( `GET TASK SUBTREE: Processing downstream task: ${downstreamTask.to}. Current/source task: ${downstreamTask.from}` @@ -1722,7 +1687,6 @@ class QlikSenseTasks { ]; } else { // Check for cyclic task tree - // eslint-disable-next-line no-lonely-if if (this.isTaskTreeCyclic(tmp)) { if (parentTask) { logger.warn( @@ -1896,7 +1860,6 @@ class QlikSenseTasks { const downstreamTasks = self.taskNetwork.edges.filter((edge) => edge.from === task.id); // console.log('downStreamTasks 1: ' + JSON.stringify(downstreamTasks)); let kids = []; - // eslint-disable-next-line no-restricted-syntax for (const downstreamTask of downstreamTasks) { if (downstreamTask.to !== undefined) { // Get downstream task object @@ -2019,7 +1982,6 @@ class QlikSenseTasks { // Add schema edges and start/trigger nodes logger.verbose('GET TASK MODEL FROM QSEOW: Adding schema edges and start/trigger nodes to internal task model'); - // eslint-disable-next-line no-restricted-syntax for (const schemaEvent of this.qlikSenseSchemaEvents.schemaEventList) { logger.silly(`Schema event contents: ${JSON.stringify(schemaEvent, null, 2)}`); // Schedule is associated with a reload task @@ -2081,7 +2043,6 @@ class QlikSenseTasks { // Add composite events logger.verbose('GET TASK MODEL FROM QSEOW: Adding composite events to internal task model'); - // eslint-disable-next-line no-restricted-syntax for (const compositeEvent of this.qlikSenseCompositeEvents.compositeEventList) { logger.silly(`Composite event contents: ${JSON.stringify(compositeEvent, null, 2)}`); @@ -2178,7 +2139,6 @@ class QlikSenseTasks { nodesWithEvents.add(nodeId); // Add edges from upstream tasks to the new meta node - // eslint-disable-next-line no-restricted-syntax for (const rule of compositeEvent.compositeEvent.compositeRules) { if (validate(rule?.reloadTask?.id)) { // Upstream task is a reload task @@ -2315,7 +2275,6 @@ class QlikSenseTasks { nodesWithEvents.add(nodeId); // Add edges from upstream tasks to the new meta node - // eslint-disable-next-line no-restricted-syntax for (const rule of compositeEvent.compositeEvent.compositeRules) { if (validate(rule?.reloadTask?.id)) { // Upstream task is a reload task @@ -2380,7 +2339,6 @@ class QlikSenseTasks { // NB: A top level node is defined as: // 1. A task whose taskID does not show up in the "to" field of any edge. - // eslint-disable-next-line no-restricted-syntax for (const node of this.taskList) { if (node.completeTaskObject.schemaPath === 'ReloadTask') { this.taskNetwork.nodes.push({ diff --git a/src/lib/task/class_task.js b/src/lib/task/class_task.js index a51bdfe..5e35cdf 100644 --- a/src/lib/task/class_task.js +++ b/src/lib/task/class_task.js @@ -1,7 +1,7 @@ import { Duration } from 'luxon'; // const { randomWords } = require('random-words'); import { logger } from '../../globals.js'; -import { mapTaskExecutionStatus } from '../util/lookups.js'; +import { mapTaskExecutionStatus } from '../util/qseow/lookups.js'; // const randomWords2 = (...args) => import('random-words').then(({ default: randomWords }) => randomWords(...args)); class QlikSenseTask { diff --git a/src/lib/task/task_qrs.js b/src/lib/task/task_qrs.js index 22209c3..d8bfe85 100644 --- a/src/lib/task/task_qrs.js +++ b/src/lib/task/task_qrs.js @@ -1,12 +1,11 @@ import axios from 'axios'; -import path from 'path'; // const { promises: Fs } = require('fs'); // const yesno = require('yesno'); -import { logger, execPath } from '../../globals.js'; +import { logger } from '../../globals.js'; +import { setupQrsConnection } from '../util/qseow/qrs.js'; +import { getCertFilePaths } from '../util/qseow/cert.js'; -import setupQRSConnection from '../util/qrs.js'; -import getCertFilePaths from '../util/cert.js'; import { catchLog } from '../util/log.js'; // const { QlikSenseTasks } = require('./class_alltasks'); // const { mapEventType, mapIncrementOption, mapDaylightSavingTime, mapRuleState } = require('../util/lookups'); @@ -26,12 +25,12 @@ export const getCustomProperty = async (options) => { try { // Get cert files - const certFilesFullPath = await getCertFilePaths(options); + const certFilesFullPath = getCertFilePaths(options); // Build QRS query string using custom property name const filter = encodeURIComponent(`name eq '${options.customPropertyName}'`); - const axiosConfig = await setupQRSConnection(options, { + const axiosConfig = await setupQrsConnection(options, { method: 'get', fileCert: certFilesFullPath.fileCert, fileCertKey: certFilesFullPath.fileCertKey, @@ -62,16 +61,11 @@ export const getCustomProperty = async (options) => { return cp; }; +// TODO Should this function support JWT auth too? export const getTasksFromQseow = async (options) => { let taskList; try { - // Get QRS certificates - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - // // Build QRS query string using task IDs let filter = ''; if (options.taskId && options?.taskId.length >= 1) { @@ -110,11 +104,8 @@ export const getTasksFromQseow = async (options) => { } logger.debug(`GET TASK: Final QRS query filter: ${filter}`); - const axiosConfig = await setupQRSConnection(options, { + const axiosConfig = await setupQrsConnection(options, { method: 'get', - fileCert, - fileCertKey, - fileCertCA, path: '/qrs/reloadtask/full', queryParameters: [{ name: 'filter', value: filter }], }); @@ -139,14 +130,9 @@ export const getTasksFromQseow = async (options) => { export const updateReloadTask = async (options, payload) => { try { - // Get cert files - const certFilesFullPath = await getCertFilePaths(options); - - const axiosConfig = await setupQRSConnection(options, { + // TODO Should be using PUT instead of POST if updating an existing task? + const axiosConfig = await setupQrsConnection(options, { method: 'post', - fileCert: certFilesFullPath.fileCert, - fileCertKey: certFilesFullPath.fileCertKey, - fileCertCA: certFilesFullPath.fileCertCA, path: '/qrs/reloadtask/update', body: payload, }); @@ -154,6 +140,8 @@ export const updateReloadTask = async (options, payload) => { // Update reload task const result = await axios.request(axiosConfig); logger.debug(`UPDATE RELOAD TASK CUSTOM PROPERTY: Result=${result.status}`); + + // TODO Check if task was updated successfully. Log error and return false if not. } catch (err) { catchLog(`UPDATE RELOAD TASK`, err); return false; diff --git a/src/lib/util/about.js b/src/lib/util/about.js deleted file mode 100644 index d0803f7..0000000 --- a/src/lib/util/about.js +++ /dev/null @@ -1,52 +0,0 @@ -import axios from 'axios'; -import path from 'path'; -import { logger, execPath } from '../../globals.js'; -import setupQRSConnection from './qrs.js'; -import { catchLog } from './log.js'; - -function getAboutFromQseow(options) { - return new Promise((resolve, reject) => { - logger.verbose(`Getting about info from QSEoW...`); - - // Should cerrificates be used for authentication? - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: '/qrs/about', - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: '/qrs/about', - }); - } - - logger.debug(`About to get about info from QSEoW`); - - axios - .request(axiosConfig) - .then((result) => { - if (result.status === 200) { - const response = JSON.parse(result.data); - logger.debug(`Successfully retrieved about info from QSEoW`); - // Yes, the tag exists - resolve(response); - } - resolve(false); - }) - .catch((err) => { - catchLog('GET ABOUT INFO', err); - }); - }); -} - -export default getAboutFromQseow; diff --git a/src/lib/util/customproperties.js b/src/lib/util/customproperties.js deleted file mode 100644 index 33a2d03..0000000 --- a/src/lib/util/customproperties.js +++ /dev/null @@ -1,254 +0,0 @@ -import axios from 'axios'; -import path from 'path'; -import { logger, execPath } from '../../globals.js'; -import setupQRSConnection from './qrs.js'; - -export function getCustomPropertiesFromQseow(options) { - return new Promise((resolve, _reject) => { - logger.verbose(`Getting custom properties from QSEoW...`); - - // Should cerrificates be used for authentication? - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: '/qrs/custompropertydefinition/full', - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: '/qrs/custompropertydefinition/full', - }); - } - - axios - .request(axiosConfig) - .then((result) => { - if (result.status === 200) { - const response = JSON.parse(result.data); - logger.info(`Successfully retrieved ${response.length} custom properties from QSEoW`); - - // Yes, the tag exists - resolve(response); - } - resolve(false); - }) - .catch((err) => { - logger.error(`GET CUSTOM PROPERTIES FROM QSEoW: ${err}`); - }); - }); -} - -export function getCustomPropertyIdByName(objectType, customPropertyName, cpExisting) { - return new Promise((resolve, _reject) => { - logger.debug(`Looking up ID for custom property named "${customPropertyName}" on object type "${objectType}"`); - - const cp = cpExisting.filter((item) => item.name === customPropertyName); - - if (cp.length === 1) { - // The custom property exists, but is it enabled for this object type (task, app etc)? - const correctObjectType = cp[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); - if (!correctObjectType) { - logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); - resolve(false); - } - - // Yes, the the custom property exists - logger.verbose(`Successfully found ID ${cp[0].id} for custom property named "${customPropertyName}"`); - resolve(cp[0].id); - } else if (cp.length === 0) { - logger.warn(`Custom property "${customPropertyName}" does not exist.`); - resolve(false); - } - }); -} -// function getCustomPropertyIdByName2(objectType, customPropertyName, options, fileCert, fileCertKey) { -// return new Promise((resolve, reject) => { -// logger.debug(`Looking up ID for custom property named "${customPropertyName}" on object type "${objectType}"`); - -// const axiosConfig = setupQRSConnection(options, { -// method: 'get', -// fileCert, -// fileCertKey, -// path: '/qrs/custompropertydefinition/full', -// queryParameters: [{ name: 'filter', value: encodeURI(`name eq '${customPropertyName}'`) }], -// }); - -// axios -// .request(axiosConfig) -// .then((result) => { -// if (result.status === 200 && result.data.length === 0) { -// logger.warn(`Custom property "${customPropertyName}" does not exist.`); -// resolve(false); -// } -// if (result.status === 200 && result.data.length === 1) { -// const correctObjectType = result.data[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); -// if (!correctObjectType) { -// logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); -// resolve(false); -// } - -// // Yes, the the custom property exists -// logger.verbose(`Successfully found ID ${result.data[0].id} for custom property named "${customPropertyName}"`); -// resolve(result.data[0].id); -// } -// resolve(false); -// }) -// .catch((err) => { -// logger.error(`CUSTOM PROPERTY ID BY NAME: ${err}`); -// }); -// }); -// } - -export function getCustomPropertyDefinitionByName(objectType, customPropertyName, cpExisting) { - return new Promise((resolve, _reject) => { - logger.debug(`Looking up definition for custom property named "${customPropertyName}" on object type "${objectType}"`); - - const cp = cpExisting.filter((item) => item.name === customPropertyName); - - if (cp.length === 1) { - // The custom property exists, but is it enabled for this object type (task, app etc)? - const correctObjectType = cp[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); - if (!correctObjectType) { - logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); - resolve(false); - } - - // Yes, the the custom property exists - logger.verbose(`Successfully found definition ${JSON.stringify(cp[0])} for custom property named "${customPropertyName}"`); - resolve(cp[0]); - } else if (cp.length === 0) { - logger.warn(`Custom property "${customPropertyName}" does not exist.`); - resolve(false); - } - }); -} -// function getCustomPropertyDefinitionByName2(objectType, customPropertyName, options, fileCert, fileCertKey) { -// return new Promise((resolve, reject) => { -// logger.debug(`Looking up definition for custom property named "${customPropertyName}" on object type "${objectType}"`); - -// const axiosConfig = setupQRSConnection(options, { -// method: 'get', -// fileCert, -// fileCertKey, -// path: '/qrs/custompropertydefinition/full', -// queryParameters: [{ name: 'filter', value: encodeURI(`name eq '${customPropertyName}'`) }], -// }); - -// axios -// .request(axiosConfig) -// .then((result) => { -// if (result.status === 200 && result.data.length === 0) { -// logger.warn(`Custom property "${customPropertyName}" does not exist.`); -// resolve(false); -// } -// if (result.status === 200 && result.data.length === 1) { -// const correctObjectType = result.data[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); -// if (!correctObjectType) { -// logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); -// resolve(false); -// } - -// // Yes, the the custom property exists -// logger.verbose(`Successfully found definition ${result.data[0]} for custom property named "${customPropertyName}"`); -// resolve(result.data[0]); -// } -// resolve(false); -// }) -// .catch((err) => { -// logger.error(`CUSTOM PROPERTY ID BY NAME: ${err}`); -// }); -// }); -// } - -export function doesCustomPropertyValueExist(objectType, customPropertyName, customPropertyValue, cpExisting) { - return new Promise((resolve, _reject) => { - logger.debug( - `Checking if value "${customPropertyValue}" is valid for custom property "${customPropertyName}" on object type "${objectType}"` - ); - - const cp = cpExisting.filter((item) => item.name === customPropertyName); - - if (cp.length === 1) { - // The custom property exists, but is it enabled for this object type (task, app etc)? - const correctObjectType = cp[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); - if (!correctObjectType) { - logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); - resolve(false); - } - - // Check if value is valid for this custom property - const valueExists = cp[0].choiceValues.find((item) => item === customPropertyValue); - if (!valueExists) { - logger.warn( - `"${customPropertyValue}" is not a valid value for custom property "${customPropertyName}", for object type "${objectType}".` - ); - resolve(false); - } - - // Yes, the the custom property exists - logger.verbose(`Successfully found ID ${cp[0].id} for custom property named "${customPropertyName}"`); - resolve(cp[0].id); - } else if (cp.length === 0) { - logger.warn(`Custom property "${customPropertyName}" does not exist.`); - resolve(false); - } - }); -} - -// function doesCustomPropertyValueExist2(objectType, customPropertyName, customPropertyValue, options, fileCert, fileCertKey) { -// return new Promise((resolve, reject) => { -// logger.debug( -// `Checking if value "${customPropertyValue}" is valid for custom property "${customPropertyName}" on object type "${objectType}"` -// ); - -// const axiosConfig = setupQRSConnection(options, { -// method: 'get', -// fileCert, -// fileCertKey, -// path: '/qrs/custompropertydefinition/full', -// queryParameters: [{ name: 'filter', value: encodeURI(`name eq '${customPropertyName}'`) }], -// }); - -// axios -// .request(axiosConfig) -// .then((result) => { -// if (result.status === 200 && result.data.length === 0) { -// logger.warn(`Custom property "${customPropertyName}" does not exist.`); -// resolve(false); -// } -// if (result.status === 200 && result.data.length === 1) { -// const correctObjectType = result.data[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); -// if (!correctObjectType) { -// logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); -// resolve(false); -// } - -// // Check if value is valid for this custom property -// const valueExists = result.data[0].choiceValues.find((item) => item === customPropertyValue); -// if (!valueExists) { -// logger.warn( -// `"${customPropertyValue}" is not a valid value for custom property "${customPropertyName}", for object type "${objectType}".` -// ); -// resolve(false); -// } - -// // Yes, the the custom property exists -// logger.verbose(`Successfully found ID ${result.data[0].id} for custom property named "${customPropertyName}"`); -// resolve(result.data[0].id); -// } -// resolve(false); -// }) -// .catch((err) => { -// logger.error(`CUSTOM PROPERTY ID BY NAME: ${err}`); -// }); -// }); -// } diff --git a/src/lib/util/import-meta-url.js b/src/lib/util/import-meta-url.js index 7d389f9..9844f01 100644 --- a/src/lib/util/import-meta-url.js +++ b/src/lib/util/import-meta-url.js @@ -1 +1,3 @@ +const { createRequire } = require('node:module'); +require = createRequire(__filename); export var import_meta_url = require('url').pathToFileURL(__filename); diff --git a/src/lib/util/qscloud/assert-options.js b/src/lib/util/qscloud/assert-options.js new file mode 100644 index 0000000..51e807d --- /dev/null +++ b/src/lib/util/qscloud/assert-options.js @@ -0,0 +1,22 @@ +import path from 'node:path'; +import { version as uuidVersion, validate as uuidValidate } from 'uuid'; +import { logger, execPath, verifyFileExists } from '../../../globals.js'; + +export const qscloudSharedParamAssertOptions = async (options) => { + // Ensure that parameters common to all QS Cloud commands are valid + if (options.authType === undefined || !options.authType) { + logger.error('Mandatory option --auth-type is missing. Use it to specify how authorization with Qlik Sense Cloud will be done.'); + process.exit(1); + } + + // Debug + logger.debug(`Auth type: ${options.authType}`); + logger.debug(`execPath: ${execPath}`); + logger.debug(`authCertFile: ${options.authCertFile}`); + logger.debug(`authCertKeyFile: ${options.authCertKeyFile}`); + + // API key authentication + if (options.authType === 'apikey') { + // + } +}; diff --git a/src/lib/util/qscloud/user.js b/src/lib/util/qscloud/user.js new file mode 100644 index 0000000..087f004 --- /dev/null +++ b/src/lib/util/qscloud/user.js @@ -0,0 +1,38 @@ +// const QlikSaas = require('./cloud-repo'); +import { auth, users } from '@qlik/api'; + +import { logger } from '../../../globals.js'; + +// Function to get info about user associated with the auth JWT being used +export async function getQscloudCurrentUser(options) { + const hostConfig = { + authType: options.authType, + host: options.tenantHost, + apiKey: options.apikey, + }; + + try { + // sets a default host config for every api request + // auth.setDefaultConfig(hostConfig); + auth.setDefaultHostConfig(hostConfig); + } catch (err) { + logger.error(`Error setting default authentication for Qlik Sense Cloud: ${err}`); + return false; + } + + try { + const { data, headers, status } = await users.getMyUser(); + + if (status !== 200) { + logger.error(`Error getting user info from Qlik Sense Cloud: ${status}`); + return false; + } + + logger.debug(`User info from Qlik Sense Cloud: ${JSON.stringify(data)}`); + + return data; + } catch (err) { + logger.error(`Error getting user info from Qlik Sense Cloud: ${err}`); + return false; + } +} diff --git a/src/lib/util/qseow/about.js b/src/lib/util/qseow/about.js new file mode 100644 index 0000000..844e1d9 --- /dev/null +++ b/src/lib/util/qseow/about.js @@ -0,0 +1,33 @@ +import axios from 'axios'; +import path from 'path'; +import { logger, execPath } from '../../../globals.js'; +import { setupQrsConnection } from './qrs.js'; +import { catchLog } from '../log.js'; + +async function getAboutFromQseow(options) { + logger.verbose(`Getting about info from QSEoW...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/about', + }); + + logger.debug(`About to get about info from QSEoW`); + + const result = await axios.request(axiosConfig); + + if (result.status === 200) { + const response = JSON.parse(result.data); + logger.debug(`Successfully retrieved about info from QSEoW`); + + return response; + } + return false; + } catch (err) { + catchLog('GET ABOUT INFO', err); + return false; + } +} + +export default getAboutFromQseow; diff --git a/src/lib/util/app.js b/src/lib/util/qseow/app.js similarity index 60% rename from src/lib/util/app.js rename to src/lib/util/qseow/app.js index bab6816..67b5df8 100644 --- a/src/lib/util/app.js +++ b/src/lib/util/qseow/app.js @@ -1,9 +1,9 @@ import axios from 'axios'; -import path from 'path'; +import path from 'node:path'; import { validate } from 'uuid'; -import { logger, execPath, getCliOptions } from '../../globals.js'; -import setupQRSConnection from './qrs.js'; -import { catchLog } from './log.js'; +import { logger, execPath, getCliOptions } from '../../../globals.js'; +import { setupQrsConnection } from './qrs.js'; +import { catchLog } from '../log.js'; export async function getApps(options, idArray, tagArray) { try { @@ -57,34 +57,17 @@ export async function getApps(options, idArray, tagArray) { } logger.debug(`GET APPS: QRS query filter (incl ids, tags): ${filter}`); - let axiosConfig; if (filter === '') { // No apps matching the provided app IDs and tags. Error! logger.error('GET APPS: No apps matching the provided app IDs and and tags. Exiting.'); process.exit(1); } - // Should cerrificates be used for authentication? - else if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: '/qrs/app/full', - queryParameters: [{ name: 'filter', value: filter }], - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: '/qrs/app/full', - queryParameters: [{ name: 'filter', value: filter }], - }); - } + + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/app/full', + queryParameters: [{ name: 'filter', value: filter }], + }); const result = await axios.request(axiosConfig); logger.debug(`GET APPS BY TAG: Result=result.status`); @@ -119,27 +102,11 @@ export async function getAppById(appId, optionsParam) { return false; } - // Should cerrificates be used for authentication? - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: `/qrs/app/${appId}`, - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: `/qrs/app/${appId}`, - }); - } + // Should certificates be used for authentication? + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: `/qrs/app/${appId}`, + }); const result = await axios.request(axiosConfig); logger.debug(`GET APP BY ID: Result=${result.status}`); @@ -175,31 +142,10 @@ export async function deleteAppById(appId, options) { try { logger.debug(`DELETE APP: Starting delete app from QSEoW for app id ${appId}`); - // Get CLI options - // const cliOptions = getCliOptions(); - - // Should cerrificates be used for authentication? - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'delete', - fileCert, - fileCertKey, - fileCertCA, - path: `/qrs/app/${appId}`, - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'delete', - path: `/qrs/app/${appId}`, - }); - } - + const axiosConfig = setupQrsConnection(options, { + method: 'delete', + path: `/qrs/app/${appId}`, + }); const result = await axios.request(axiosConfig); logger.debug(`DELETE APP: Result=result.status`); @@ -227,29 +173,11 @@ export async function appExistById(appId, options) { return false; } - // Should cerrificates be used for authentication? - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: '/qrs/app', - queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${appId}`) }], - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: '/qrs/app', - queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${appId}`) }], - }); - } + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/app', + queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${appId}`) }], + }); const result = await axios.request(axiosConfig); logger.debug(`APP EXIST BY ID: Result=${result.status}`); diff --git a/src/lib/util/assert-options.js b/src/lib/util/qseow/assert-options.js similarity index 92% rename from src/lib/util/assert-options.js rename to src/lib/util/qseow/assert-options.js index 498fd25..0e10c4c 100644 --- a/src/lib/util/assert-options.js +++ b/src/lib/util/qseow/assert-options.js @@ -1,8 +1,9 @@ -import path from 'path'; import { version as uuidVersion, validate as uuidValidate } from 'uuid'; -import { logger, execPath, verifyFileExists } from '../../globals.js'; -export const sharedParamAssertOptions = async (options) => { +import { logger, execPath, verifyFileExists } from '../../../globals.js'; +import { getCertFilePaths } from '../qseow/cert.js'; + +export const qseowSharedParamAssertOptions = async (options) => { // Ensure that parameters common to all commands are valid if (options.authType === undefined || !options.authType) { logger.error('Mandatory option --auth-type is missing. Use it to specify how authorization with Qlik Sense will be done.'); @@ -18,8 +19,9 @@ export const sharedParamAssertOptions = async (options) => { // If certificate authentication is used: certs and user dir/id must be present. if (options.authType === 'cert') { // Verify that certificate files exists (if specified) - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); + + // Get certificate paths + const { fileCert, fileCertKey, fileCertCA } = getCertFilePaths(options); const fileCertExists = await verifyFileExists(fileCert); if (fileCertExists === false) { @@ -36,6 +38,12 @@ export const sharedParamAssertOptions = async (options) => { } else { logger.verbose(`Certificate key file ${fileCertKey} found`); } + + const fileCertCAExists = await verifyFileExists(fileCertCA); + if (fileCertCAExists === false) { + logger.error(`Missing certificate CA file ${fileCertCA}. Aborting`); + process.exit(1); + } } else if (options.authType === 'jwt') { // Verify that --auth-jwt parameter is specified if (options.authJwt === undefined || !options.authJwt) { @@ -124,22 +132,18 @@ export const masterItemDimDeleteAssertOptions = (options) => { } }; -// eslint-disable-next-line no-unused-vars export const masterItemGetAssertOptions = (options) => { // }; -// eslint-disable-next-line no-unused-vars export const getScriptAssertOptions = (options) => { // }; -// eslint-disable-next-line no-unused-vars export const getBookmarkAssertOptions = (options) => { // }; -// eslint-disable-next-line no-unused-vars export const getTaskAssertOptions = (options) => { // ---task-id and --task-tag only allowed for task tables, not trees if (options.taskId || options.taskTag) { @@ -150,7 +154,6 @@ export const getTaskAssertOptions = (options) => { // Verify all task IDs are valid uuids if (options.taskId) { - // eslint-disable-next-line no-restricted-syntax for (const taskId of options.taskId) { if (!uuidValidate(taskId)) { logger.error(`Invalid format of task ID parameter "${taskId}". Exiting.`); @@ -201,12 +204,10 @@ export const getTaskAssertOptions = (options) => { } }; -// eslint-disable-next-line no-unused-vars export const setTaskCustomPropertyAssertOptions = (options) => { // }; -// eslint-disable-next-line no-unused-vars export const taskImportAssertOptions = (options) => { // If --import-app is specified, the import file type must be Excel if (options.importApp && options.fileType !== 'excel') { @@ -231,7 +232,6 @@ export const taskImportAssertOptions = (options) => { } }; -// eslint-disable-next-line no-unused-vars export const appImportAssertOptions = (options) => { // }; @@ -241,7 +241,6 @@ export const appImportAssertExcelSheet = (options) => { // }; -// eslint-disable-next-line no-unused-vars export const appExportAssertOptions = async (options) => { // Verify output directory exists // const outputDir = mergeDirFilePath([options.outputDir]); @@ -293,12 +292,18 @@ export const variableDeleteAssertOptions = (options) => { } }; -// eslint-disable-next-line no-unused-vars export const getSessionsAssertOptions = (options) => { // }; -// eslint-disable-next-line no-unused-vars export const deleteSessionsAssertOptions = (options) => { // }; + +export const userActivityBucketsCustomPropertyAssertOptions = (options) => { + // Verify that custom property name only contains letters, numbers and underscores + if (!/^[a-zA-Z0-9_]+$/.test(options.customPropertyName)) { + logger.error(`Invalid custom property name "${options.customPropertyName}". Only letters, numbers and underscores are allowed.`); + process.exit(1); + } +}; diff --git a/src/lib/util/cert.js b/src/lib/util/qseow/cert.js similarity index 56% rename from src/lib/util/cert.js rename to src/lib/util/qseow/cert.js index 4bcd021..e025361 100644 --- a/src/lib/util/cert.js +++ b/src/lib/util/qseow/cert.js @@ -1,22 +1,20 @@ -import path from 'path'; -import { logger, execPath } from '../../globals.js'; -import { catchLog } from './log.js'; +import path from 'node:path'; +import { logger, execPath, verifyFileExists } from '../../../globals.js'; +import { catchLog } from '../log.js'; -const getCertFilePaths = async (options) => { +export function getCertFilePaths(options) { let fileCert; let fileCertKey; let fileCertCA; try { - // Make sure QRS certificates exist + // Get cert paths from command line options fileCert = path.resolve(execPath, options.authCertFile); fileCertKey = path.resolve(execPath, options.authCertKeyFile); fileCertCA = path.resolve(execPath, options.authRootCertFile); } catch (err) { - catchLog('GET TASK QRS (ID). Exiting. ', err); + catchLog('GET CERT FILE PATHS', err); process.exit(1); } return { fileCert, fileCertKey, fileCertCA }; -}; - -export default getCertFilePaths; +} diff --git a/src/lib/util/qseow/customproperties.js b/src/lib/util/qseow/customproperties.js new file mode 100644 index 0000000..bad667f --- /dev/null +++ b/src/lib/util/qseow/customproperties.js @@ -0,0 +1,186 @@ +import axios from 'axios'; + +import { logger } from '../../../globals.js'; +import { catchLog } from '../../util/log.js'; +import { setupQrsConnection } from './qrs.js'; + +export async function getCustomPropertiesFromQseow(options) { + logger.verbose(`Getting custom properties from QSEoW...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/custompropertydefinition/full', + }); + + const result = await axios.request(axiosConfig); + if (result.status === 200) { + const response = JSON.parse(result.data); + + // Yes, the custom property exists + return response; + } + return false; + } catch (err) { + catchLog('GET CUSTOM PROPERTIES FROM QSEoW', err); + return false; + } +} + +export function getCustomPropertyIdByName(objectType, customPropertyName, cpExisting) { + logger.debug(`Looking up ID for custom property named "${customPropertyName}" on object type "${objectType}"`); + + try { + const cp = cpExisting.filter((item) => item.name === customPropertyName); + + if (cp.length === 1) { + // The custom property exists, but is it enabled for this object type (task, app etc)? + const correctObjectType = cp[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); + if (!correctObjectType) { + logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); + return false; + } + + // Yes, the the custom property exists + logger.verbose(`Successfully found ID ${cp[0].id} for custom property named "${customPropertyName}"`); + return cp[0].id; + } else if (cp.length === 0) { + logger.warn(`Custom property "${customPropertyName}" does not exist.`); + return false; + } + } catch (err) { + catchLog('GET CUSTOM PROPERTY ID BY NAME', err); + return false; + } +} + +export function getCustomPropertyDefinitionByName(objectType, customPropertyName, cpExisting) { + logger.debug(`Looking up definition for custom property named "${customPropertyName}" on object type "${objectType}"`); + + try { + const cp = cpExisting.filter((item) => item.name === customPropertyName); + + if (cp.length === 1) { + // The custom property exists, but is it enabled for this object type (task, app etc)? + const correctObjectType = cp[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); + if (!correctObjectType) { + logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); + return false; + } + + // Yes, the the custom property exists + logger.verbose(`Successfully found definition ${JSON.stringify(cp[0])} for custom property named "${customPropertyName}"`); + return cp[0]; + } else if (cp.length === 0) { + logger.warn(`Custom property "${customPropertyName}" does not exist.`); + return false; + } + } catch (err) { + catchLog('GET CUSTOM PROPERTY DEFINITION BY NAME', err); + return false; + } +} + +export function doesCustomPropertyValueExist(objectType, customPropertyName, customPropertyValue, cpExisting) { + logger.debug( + `Checking if value "${customPropertyValue}" is valid for custom property "${customPropertyName}" on object type "${objectType}"` + ); + + try { + const cp = cpExisting.filter((item) => item.name === customPropertyName); + + if (cp.length === 1) { + // The custom property exists, but is it enabled for this object type (task, app etc)? + const correctObjectType = cp[0].objectTypes.find((item) => objectType.toLowerCase() === item.toLowerCase()); + if (!correctObjectType) { + logger.warn(`Custom property "${customPropertyName}" is not valid for task type "${objectType}".`); + return false; + } + + // Check if value is valid for this custom property + const valueExists = cp[0].choiceValues.find((item) => item === customPropertyValue); + if (!valueExists) { + logger.warn( + `"${customPropertyValue}" is not a valid value for custom property "${customPropertyName}", for object type "${objectType}".` + ); + return false; + } + + // Yes, the the custom property exists + logger.verbose(`Successfully found ID ${cp[0].id} for custom property named "${customPropertyName}"`); + return cp[0].id; + } else if (cp.length === 0) { + logger.warn(`Custom property "${customPropertyName}" does not exist.`); + return false; + } + } catch (err) { + catchLog('CUSTOM PROPERTY ID BY NAME', err); + return false; + } +} + +// Function to create a custom property +// customPropertyDefinition has properties: +// - objectTypes (array of strings). Types of objects this custom property is valid for. +// - name (string) +// - choiceValues (array of strings). Possible values for this custom property. +// - description (string) +// - values (array of strings). Values that are actually set for this custom property. +export async function createCustomProperty(options, customPropertyDefinition) { + logger.verbose(`Creating custom property "${customPropertyDefinition.name}"...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'post', + path: '/qrs/custompropertydefinition', + }); + + // Set payload + axiosConfig.data = customPropertyDefinition; + + logger.debug(`About to create custom property "${customPropertyDefinition.name}"`); + const result = await axios.request(axiosConfig); + + if (result.status === 201) { + logger.info(`Successfully created custom property "${customPropertyDefinition.name}"`); + return true; + } + + logger.error(`Failed to create custom property "${customPropertyDefinition.name}"`); + return false; + } catch (err) { + catchLog('CREATE CUSTOM PROPERTY', err); + } +} + +// Function to update a custom property +// +// Parameters: +// options: Command line options +// customPropertyDefinition: The new/updated custom property definition. Object with properties: +// - objectTypes (array of strings). Types of objects this custom property is valid for. + +export async function updateCustomProperty(options, customPropertyDefinition) { + logger.verbose(`Updating custom property "${customPropertyDefinition.name}"...`); + + try { + const axiosConfig = setupQrsConnection(options, { + method: 'put', + path: `/qrs/custompropertydefinition/${customPropertyDefinition.id}`, + body: customPropertyDefinition, + }); + + // Update custom property + const result = await axios.request(axiosConfig); + if (result.status === 200) { + logger.info(`Successfully updated custom property "${customPropertyDefinition.name}"`); + return true; + } + + logger.error(`Failed to update custom property "${customPropertyDefinition.name}"`); + return false; + } catch (err) { + catchLog('UPDATE CUSTOM PROPERTY', err); + return false; + } +} diff --git a/src/lib/util/enigma.js b/src/lib/util/qseow/enigma_util.js similarity index 62% rename from src/lib/util/enigma.js rename to src/lib/util/qseow/enigma_util.js index f87f783..1889906 100644 --- a/src/lib/util/enigma.js +++ b/src/lib/util/qseow/enigma_util.js @@ -1,76 +1,97 @@ import SenseUtilities from 'enigma.js/sense-utilities.js'; import WebSocket from 'ws'; -import path from 'path'; -import { readFileSync } from 'fs'; -import { fileURLToPath } from 'url'; +import { readFileSync } from 'node:fs'; +import { fileURLToPath } from 'node:url'; import upath from 'upath'; -import { logger, execPath, readCert } from '../../globals.js'; +import sea from 'node:sea'; -export const setupEnigmaConnection = async (options, sessionId) => { - logger.debug('Prepping for Enigma connection...'); +import { logger, readCert } from '../../../globals.js'; +import { getCertFilePaths } from '../../util/qseow/cert.js'; - // Set up enigma.js configuration - let schemaFile; - let a; - let b; - let c; +// Function to get Enigma.js schema file +const getEnigmaSchema = (processPkgFlag, seaFlag, options) => { + // Array of supported schema versions + const supportedSchemaVersions = ['12.170.2', '12.612.0', '12.936.0', '12.1306.0', '12.1477.0', '12.1657.0', '12.1823.0', '12.2015.0']; - logger.debug(`Enigma.js schema version: ${options.schemaVersion}`); + let qixSchemaJson; + try { + // Check if the specified schema version is supported + if (!supportedSchemaVersions.includes(options.schemaVersion)) { + logger.error(`Unsupported schema version specified: ${options.schemaVersion}`); + + // Show supported schema versions + logger.error(`Supported schema versions: ${supportedSchemaVersions.join(', ')}`); + + logger.error(`Exiting...`); + process.exit(1); + } + + // Are we running as a packaged app? + if (processPkgFlag) { + const schemaFile = `./node_modules/enigma.js/schemas/${options.schemaVersion}.json`; + logger.debug(`Enigma.js schema file: ${schemaFile}`); + + // Yes, we are running as a packaged app + // Get path to JS file const + const a = process.pkg.defaultEntrypoint; + logger.debug(`APPDUMP schema path a: ${a}`); + + // Strip off the filename + const b = upath.dirname(a); + logger.debug(`APPDUMP schema path b: ${b}`); + + // Add path to schema file + const c = upath.join(b, schemaFile); + logger.debug(`APPDUMP schema path c: ${c}`); + + qixSchemaJson = readFileSync(c); + } else if (seaFlag) { + // Load schema file + qixSchemaJson = sea.getAsset(`enigma_schema_${options.schemaVersion}.json`, 'utf8'); + } else { + // No, we are running as native Node.js + const schemaFile = `../node_modules/enigma.js/schemas/${options.schemaVersion}.json`; + logger.debug(`Enigma.js schema file: ${schemaFile}`); - // Are we running as a packaged app? - if (process.pkg) { - schemaFile = `./node_modules/enigma.js/schemas/${options.schemaVersion}.json`; - logger.debug(`Enigma.js schema file: ${schemaFile}`); - - // Yes, we are running as a packaged app - // Get path to JS file const - a = process.pkg.defaultEntrypoint; - logger.debug(`APPDUMP schema path a: ${a}`); - - // Strip off the filename - b = upath.dirname(a); - logger.debug(`APPDUMP schema path b: ${b}`); - - // Add path to package.json file - c = upath.join(b, schemaFile); - logger.debug(`APPDUMP schema path c: ${c}`); - } else { - schemaFile = `../node_modules/enigma.js/schemas/${options.schemaVersion}.json`; - logger.debug(`Enigma.js schema file: ${schemaFile}`); - - // No, we are running as native Node.js - // Get path to JS file - a = fileURLToPath(import.meta.url); - logger.debug(`APPDUMP schema path a: ${a}`); - - // Strip off the filename - b = upath.dirname(a); - logger.debug(`APPDUMP schema path b: ${b}`); - - // Add path to package.json file - c = upath.join(b, '..', '..', schemaFile); - logger.debug(`APPDUMP schema path c: ${c}`); + // Get path to JS file + const a = fileURLToPath(import.meta.url); + logger.debug(`APPDUMP schema path a: ${a}`); + + // Strip off the filename + const b = upath.dirname(a); + logger.debug(`APPDUMP schema path b: ${b}`); + + // Add path to package.json file + const c = upath.join(b, '..', '..', '..', schemaFile); + logger.debug(`APPDUMP schema path c: ${c}`); + + qixSchemaJson = readFileSync(c); + } + } catch (err) { + logger.error(`Error when getting Enigma schema: ${err}`); + process.exit(1); } - logger.verbose(`APPDUMP: Using engine schema in file: ${c}`); - const qixSchema = JSON.parse(readFileSync(c)); + const qixSchema = JSON.parse(qixSchemaJson); + logger.debug(`Enigma.js schema: ${qixSchema}`); - // eslint-disable-next-line global-require, import/no-dynamic-require - // const qixSchema = require(`enigma.js/schemas/${options.schemaVersion}`); + return qixSchema; +}; + +export const setupEnigmaConnection = (options, sessionId) => { + logger.debug('Prepping for Enigma connection...'); + + // Set up enigma.js configuration + logger.debug(`Enigma.js schema version: ${options.schemaVersion}`); + const qixSchema = getEnigmaSchema(process.pkg, sea.isSea(), options); let enigmaConfig; // Should certificates be used for authentication? if (options.authType === 'cert') { logger.verbose(`Using certificates for authentication with Enigma`); - logger.verbose('Verify that cert files exists'); - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - - if (!fileCert || !fileCertKey) { - logger.error(`Certificate file(s) not found when setting up Enigma connection`); - process.exit(1); - } + // Get certificate paths + const { fileCert, fileCertKey, fileCertCA } = getCertFilePaths(options); // Set up Enigma configuration // buildUrl docs: https://github.com/qlik-oss/enigma.js/blob/master/docs/api.md#senseutilitiesbuildurlconfig @@ -88,8 +109,11 @@ export const setupEnigmaConnection = async (options, sessionId) => { new WebSocket(url, { key: readCert(fileCertKey), cert: readCert(fileCert), + ca: [readCert(fileCertCA)], headers: { - 'X-Qlik-User': `UserDirectory=${options.authUserDir};UserId=${options.authUserId}`, + 'X-Qlik-User': `UserDirectory=${encodeURIComponent(options.authUserDir)};UserId=${encodeURIComponent( + options.authUserId + )}`, }, rejectUnauthorized: false, }), diff --git a/src/lib/util/lookups.js b/src/lib/util/qseow/lookups.js similarity index 100% rename from src/lib/util/lookups.js rename to src/lib/util/qseow/lookups.js diff --git a/src/lib/util/proxy.js b/src/lib/util/qseow/proxy.js similarity index 59% rename from src/lib/util/proxy.js rename to src/lib/util/qseow/proxy.js index c7e76a8..2eaa563 100644 --- a/src/lib/util/proxy.js +++ b/src/lib/util/qseow/proxy.js @@ -1,22 +1,15 @@ import axios from 'axios'; -import path from 'path'; -import { logger, execPath } from '../../globals.js'; -import setupQRSConnection from './qrs.js'; -import { catchLog } from './log.js'; +import path from 'node:path'; +import { logger, execPath } from '../../../globals.js'; +import { setupQrsConnection } from './qrs.js'; +import { catchLog } from '../log.js'; const getProxiesFromQseow = async (options, _sessionCookie) => { logger.verbose(`Getting all proxies from QSEoW...`); - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - const axiosConfig = setupQRSConnection(options, { + // TODO Should support JWTs here too? + const axiosConfig = setupQrsConnection(options, { method: 'get', - fileCert, - fileCertKey, - fileCertCA, path: '/qrs/proxyservice/full', sessionCookie: null, }); diff --git a/src/lib/util/qps.js b/src/lib/util/qseow/qps.js similarity index 91% rename from src/lib/util/qps.js rename to src/lib/util/qseow/qps.js index ee9dd7b..efc7126 100644 --- a/src/lib/util/qps.js +++ b/src/lib/util/qseow/qps.js @@ -1,8 +1,7 @@ -import https from 'https'; -import { logger, generateXrfKey, readCert } from '../../globals.js'; +import https from 'node:https'; +import { logger, generateXrfKey, readCert } from '../../../globals.js'; const setupQPSConnection = (options, param) => { - // eslint-disable-next-line no-unused-vars // Ensure valid http method if (!param.method || (param.method.toLowerCase() !== 'get' && param.method.toLowerCase() !== 'delete')) { logger.error(`Setting up connection to QPS. Invalid http method '${param.method}'. Exiting.`); @@ -17,7 +16,7 @@ const setupQPSConnection = (options, param) => { let axiosConfig; - // Use cerrificates be used for authentication + // Use certificates be used for authentication if (options.authType === 'cert') { logger.debug(`Using certificates for authentication with QPS`); logger.debug(`QPS host: ${options.hostProxy}`); @@ -68,7 +67,6 @@ const setupQPSConnection = (options, param) => { // Add parameters (if any) if (param.queryParameters?.length > 0) { - // eslint-disable-next-line no-restricted-syntax for (const queryParam of param.queryParameters) { axiosConfig.url += `&${queryParam.name}=${queryParam.value}`; } diff --git a/src/lib/util/qrs.js b/src/lib/util/qseow/qrs.js similarity index 62% rename from src/lib/util/qrs.js rename to src/lib/util/qseow/qrs.js index 430d04f..7b46606 100644 --- a/src/lib/util/qrs.js +++ b/src/lib/util/qseow/qrs.js @@ -1,8 +1,43 @@ -import https from 'https'; -import { logger, generateXrfKey, readCert } from '../../globals.js'; +import https from 'node:https'; + +import { logger, generateXrfKey, readCert } from '../../../globals.js'; +import { getCertFilePaths } from '../qseow/cert.js'; + +// Function to sanitize virtual proxy +export function sanitizeVirtualProxy(virtualProxy) { + // - Should always start with a / + // - Should never end with a / + if (virtualProxy === '') { + virtualProxy = '/'; + } else { + if (!virtualProxy.startsWith('/')) { + virtualProxy = `/${virtualProxy}`; + } + + // Remove all trailing / + virtualProxy = virtualProxy.replace(/\/+$/, ''); + } + + return virtualProxy; +} + +// Function to set up connection to Qlik Sense Repository Service (QRS) +export function setupQrsConnection(options, param) { + // Ensure correct auth info is present + if (options.authType === 'cert') { + // options.authUserDir and options.authUserId should be set + if (!options.authUserDir || !options.authUserId) { + logger.error(`Setting up connection to QRS. Missing user directory and/or user ID. Exiting.`); + process.exit(1); + } + } else if (options.authType === 'jwt') { + // options.authJwt should be set + if (!options.authJwt) { + logger.error(`Setting up connection to QRS. Missing JWT. Exiting.`); + process.exit(1); + } + } -const setupQRSConnection = (options, param) => { - // eslint-disable-next-line no-unused-vars // Ensure valid http method if ( !param.method || @@ -41,17 +76,34 @@ const setupQRSConnection = (options, param) => { } let axiosConfig; - // Should cerrificates be used for authentication? + // Should certificates be used for authentication? if (options.authType === 'cert') { logger.debug(`Using certificates for authentication with QRS`); logger.debug(`QRS host: ${options.host}`); logger.debug(`Reject unauthorized certificate: ${options.secure}`); + // Get certificate paths + // If specified in the param object, use those paths + // Otherwise, use the paths from the command line options + let { fileCert, fileCertKey, fileCertCA } = getCertFilePaths(options); + + if (param.fileCert) { + fileCert = param.fileCert; + } + + if (param.fileCertKey) { + fileCertKey = param.fileCertKey; + } + + if (param.fileCertCA) { + fileCertCA = param.fileCertCA; + } + const httpsAgent = new https.Agent({ rejectUnauthorized: options.secure !== 'false', - cert: readCert(param.fileCert), - key: readCert(param.fileCertKey), - ca: readCert(param.fileCertCA), + cert: readCert(fileCert), + key: readCert(fileCertKey), + ca: readCert(fileCertCA), }); axiosConfig = { @@ -71,6 +123,7 @@ const setupQRSConnection = (options, param) => { logger.verbose(`Using JWT for authentication with QRS`); const httpsAgent = new https.Agent({ + // rejectUnauthorized: options.secure !== 'false', rejectUnauthorized: false, }); @@ -101,13 +154,10 @@ const setupQRSConnection = (options, param) => { // Add parameters (if any) if (param.queryParameters?.length > 0) { - // eslint-disable-next-line no-restricted-syntax for (const queryParam of param.queryParameters) { axiosConfig.url += `&${queryParam.name}=${queryParam.value}`; } } return axiosConfig; -}; - -export default setupQRSConnection; +} diff --git a/src/lib/util/session.js b/src/lib/util/qseow/session.js similarity index 91% rename from src/lib/util/session.js rename to src/lib/util/qseow/session.js index 0d481fc..5bd5532 100644 --- a/src/lib/util/session.js +++ b/src/lib/util/qseow/session.js @@ -1,11 +1,11 @@ import axios from 'axios'; -import path from 'path'; +import path from 'node:path'; import { table } from 'table'; import yesno from 'yesno'; -import { logger, execPath } from '../../globals.js'; +import { logger, execPath } from '../../../globals.js'; import setupQPSConnection from './qps.js'; -import setupQRSConnection from './qrs.js'; -import { catchLog } from './log.js'; +import { setupQrsConnection } from './qrs.js'; +import { catchLog } from '../log.js'; import getProxiesFromQseow from './proxy.js'; const consoleProxiesTableConfig = { @@ -66,17 +66,12 @@ const consoleProxiesTableConfig = { export const getSessionsFromQseow = async (options, sessionCookie) => { logger.verbose(`Getting sessions from QSEoW...`); - // Only cerrificates allowed for authentication + // Only certificates allowed for authentication if (options.authType !== 'cert') { logger.error(`Only certificates allowed for authentication with Qlik Proxy Service (QPS)`); return false; } - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - let axiosConfig; let virtualProxiesToProcess = []; @@ -89,21 +84,15 @@ export const getSessionsFromQseow = async (options, sessionCookie) => { // Virtual proxies are specified as an array of strings // Filter format is: id eq 'vpName1' or id eq 'vpName2' or id eq 'vpName3' const vpFilter = options.sessionVirtualProxy.map((vp) => `prefix eq '${vp}'`).join(' or '); - axiosConfig = setupQRSConnection(options, { + axiosConfig = setupQrsConnection(options, { method: 'get', - fileCert, - fileCertKey, - fileCertCA, path: '/qrs/virtualproxyconfig/full', queryParameters: [{ name: 'filter', value: encodeURI(vpFilter) }], }); } else { // No virtual proxies specified, get all of them from QRS - axiosConfig = setupQRSConnection(options, { + axiosConfig = setupQrsConnection(options, { method: 'get', - fileCert, - fileCertKey, - fileCertCA, path: '/qrs/virtualproxyconfig/full', }); } @@ -191,7 +180,6 @@ export const getSessionsFromQseow = async (options, sessionCookie) => { let sessions = []; // Loop over virtual proxies and get sessions for each, but only if the linked proxy is in the list of proxies to process - // eslint-disable-next-line no-restricted-syntax for (const vp of virtualProxiesToProcess) { // Is this virtual proxy linked to at least one proxy? const proxiesVirtualProxy = proxiesAvailable.filter((p) => p.settings.virtualProxies.find((q) => q.id === vp.id)); @@ -209,7 +197,6 @@ export const getSessionsFromQseow = async (options, sessionCookie) => { let sessionPerVirtualProxy = 0; // Loop over all proxies linked to this virtual proxy, get the proxy sessions for each one - // eslint-disable-next-line no-restricted-syntax for (const proxy of proxiesVirtualProxy) { // Is this proxy in list of proxies to process? if (proxiesToProcess.length > 0 && !proxiesToProcess.includes(proxy.serverNodeConfiguration.hostName)) { @@ -231,7 +218,6 @@ export const getSessionsFromQseow = async (options, sessionCookie) => { }); try { - // eslint-disable-next-line no-await-in-loop const result = await axios.request(axiosConfig); if (result.status === 200) { @@ -268,17 +254,12 @@ export const getSessionsFromQseow = async (options, sessionCookie) => { export const deleteSessionsFromQSEoWIds = async (options) => { logger.verbose(`Deleting proxy sessions from QSEoW...`); - // Only cerrificates allowed for authentication + // Only certificates allowed for authentication if (options.authType !== 'cert') { logger.error(`Only certificates allowed for authentication with Qlik Proxy Service (QPS)`); return false; } - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - try { const sessionDelete = []; @@ -320,9 +301,7 @@ export const deleteSessionsFromQSEoWIds = async (options) => { } } else { // Use session IDs specified on command line - // eslint-disable-next-line no-restricted-syntax for (const s of options.sessionId) { - // eslint-disable-next-line no-restricted-syntax for (const vp of vpWithSessions) { if (vp.sessions.find((x) => x.SessionId === s)) { const sessionObject = { @@ -350,7 +329,6 @@ export const deleteSessionsFromQSEoWIds = async (options) => { let deleteCounter = 0; // Loop over all session IDs and delete each one - // eslint-disable-next-line no-restricted-syntax for (const s of sessionDelete) { logger.verbose( `Deleting session ID "${s.sessionId}" on proxy "${options.hostProxy}", virtual proxy "${options.sessionVirtualProxy}"...` @@ -361,14 +339,10 @@ export const deleteSessionsFromQSEoWIds = async (options) => { const axiosConfig = setupQPSConnection(options, { hostProxy: options.hostProxy, method: 'delete', - fileCert, - fileCertKey, - fileCertCA, path: `/qps/${options.sessionVirtualProxy}/session/${s.sessionId}`, sessionCookie: null, }); - // eslint-disable-next-line no-await-in-loop const result = await axios.request(axiosConfig); if (result.status === 200) { diff --git a/src/lib/util/qseow/tag.js b/src/lib/util/qseow/tag.js new file mode 100644 index 0000000..7eeba36 --- /dev/null +++ b/src/lib/util/qseow/tag.js @@ -0,0 +1,53 @@ +import axios from 'axios'; +import path from 'node:path'; +import { logger, execPath } from '../../../globals.js'; +import { setupQrsConnection } from './qrs.js'; +import { catchLog } from '../log.js'; + +export function getTagsFromQseow(options) { + return new Promise((resolve, _reject) => { + logger.verbose(`Getting tags from QSEoW...`); + + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/tag/full', + }); + + logger.debug(`About to retrieve tags from QRS API.`); + + axios + .request(axiosConfig) + .then((result) => { + if (result.status === 200) { + const response = JSON.parse(result.data); + logger.info(`Successfully retrieved ${response.length} tags from QSEoW`); + // Yes, the tag exists + resolve(response); + } + resolve(false); + }) + .catch((err) => { + catchLog('GET TAGS FROM QSEoW', err); + }); + }); +} + +export function getTagIdByName(tagName, tagsExisting) { + return new Promise((resolve, _reject) => { + logger.debug(`Looking up ID for tag named "${tagName}"`); + + let tag; + if (typeof tagsExisting === 'string') { + tag = JSON.parse(tagsExisting).filter((item) => item.name === tagName); + } else { + tag = tagsExisting.filter((item) => item.name === tagName); + } + + if (tag.length === 1) { + // The tag exists + resolve(tag[0].id); + } else { + resolve(false); + } + }); +} diff --git a/src/lib/util/task.js b/src/lib/util/qseow/task.js similarity index 55% rename from src/lib/util/task.js rename to src/lib/util/qseow/task.js index f88ef9f..30d6484 100644 --- a/src/lib/util/task.js +++ b/src/lib/util/qseow/task.js @@ -1,10 +1,9 @@ import axios from 'axios'; -import fs from 'fs'; -import path from 'path'; +import fs from 'node:fs'; import { validate } from 'uuid'; -import { logger, execPath, getCliOptions } from '../../globals.js'; -import setupQRSConnection from './qrs.js'; -import { catchLog } from './log.js'; +import { logger, getCliOptions } from '../../../globals.js'; +import { setupQrsConnection } from './qrs.js'; +import { catchLog } from '../log.js'; // Check if a task with a given id exists // Look for all kinds of tasks, not just reload tasks @@ -30,30 +29,11 @@ export async function taskExistById(taskId, optionsParam) { return false; } - // Should cerrificates be used for authentication? - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - // const filter = encodeURI(`name eq '👍😎 updateSheetThumbnail'`); - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: '/qrs/task', - queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${taskId}`) }], - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: '/qrs/task', - queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${taskId}`) }], - }); - } + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/task', + queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${taskId}`) }], + }); const result = await axios.request(axiosConfig); logger.debug(`TASK EXIST BY ID: Result=${result.status}`); @@ -101,28 +81,11 @@ export async function getTaskByName(taskName, optionsParam) { options = optionsParam; } - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: '/qrs/task/full', - queryParameters: [{ name: 'filter', value: encodeURI(`name eq '${taskName}'`) }], - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: '/qrs/task/full', - queryParameters: [{ name: 'filter', value: encodeURI(`name eq '${taskName}'`) }], - }); - } + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: '/qrs/task/full', + queryParameters: [{ name: 'filter', value: encodeURI(`name eq '${taskName}'`) }], + }); const result = await axios.request(axiosConfig); logger.debug(`GET TASK BY NAME: Result=${result.status}`); @@ -173,28 +136,11 @@ export async function getTaskById(taskId, optionsParam) { logger.verbose(`GET TASK BY ID: Task ID ${taskId} is a valid GUID. Get associated task from QSEoW.`); - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: `/qrs/task/full`, - queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${taskId}`) }], - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: `/qrs/task/full`, - queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${taskId}`) }], - }); - } + const axiosConfig = setupQrsConnection(options, { + method: 'get', + path: `/qrs/task/full`, + queryParameters: [{ name: 'filter', value: encodeURI(`id eq ${taskId}`) }], + }); const result = await axios.request(axiosConfig); logger.debug(`GET TASK BY ID: Result=${result.status}`); @@ -247,38 +193,10 @@ export async function deleteReloadTaskById(taskId, optionsParam) { logger.verbose(`DELETE RELOAD TASK BY ID: Task ID ${taskId} is a valid GUID. Delete associated task from QSEoW.`); - let axiosConfig; - - if (optionsParam.authType === 'cert') { - // Expand cert file paths - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - // Make sure certificate files exist on disk - if (!fs.existsSync(fileCert)) { - logger.error(`DELETE RELOAD TASK BY ID: Certificate file ${fileCert} does not exist.`); - return false; - } - - if (!fs.existsSync(fileCertKey)) { - logger.error(`DELETE RELOAD TASK BY ID: Certificate key file ${fileCertKey} does not exist.`); - return false; - } - - axiosConfig = setupQRSConnection(options, { - method: 'delete', - fileCert, - fileCertKey, - fileCertCA, - path: `/qrs/reloadtask/${taskId}`, - }); - } else if (optionsParam.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'delete', - path: `/qrs/reloadtask/${taskId}`, - }); - } + const axiosConfig = setupQrsConnection(options, { + method: 'delete', + path: `/qrs/reloadtask/${taskId}`, + }); const result = await axios.request(axiosConfig); logger.debug(`DELETE RELOAD TASK BY ID: Result=${result.status}`); @@ -319,38 +237,10 @@ export async function deleteExternalProgramTaskById(taskId, optionsParam) { logger.verbose(`DELETE EXT PGM TASK BY ID: Task ID ${taskId} is a valid GUID. Delete associated task from QSEoW.`); - let axiosConfig; - - if (optionsParam.authType === 'cert') { - // Expand cert file paths - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - // Make sure certificate files exist on disk - if (!fs.existsSync(fileCert)) { - logger.error(`DELETE EXT PGM TASK BY ID: Certificate file ${fileCert} does not exist.`); - return false; - } - - if (!fs.existsSync(fileCertKey)) { - logger.error(`DELETE EXT PGM TASK BY ID: Certificate key file ${fileCertKey} does not exist.`); - return false; - } - - axiosConfig = setupQRSConnection(options, { - method: 'delete', - fileCert, - fileCertKey, - fileCertCA, - path: `/qrs/externalprogramtask/${taskId}`, - }); - } else if (optionsParam.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'delete', - path: `/qrs/externalprogramtask/${taskId}`, - }); - } + const axiosConfig = setupQrsConnection(options, { + method: 'delete', + path: `/qrs/externalprogramtask/${taskId}`, + }); const result = await axios.request(axiosConfig); logger.debug(`DELETE TASK BY ID: Result=${result.status}`); diff --git a/src/lib/util/tag.js b/src/lib/util/tag.js deleted file mode 100644 index bde341d..0000000 --- a/src/lib/util/tag.js +++ /dev/null @@ -1,99 +0,0 @@ -import axios from 'axios'; -import path from 'path'; -import { logger, execPath } from '../../globals.js'; -import setupQRSConnection from './qrs.js'; -import { catchLog } from './log.js'; - -export function getTagsFromQseow(options) { - return new Promise((resolve, _reject) => { - logger.verbose(`Getting tags from QSEoW...`); - - // Should cerrificates be used for authentication? - let axiosConfig; - if (options.authType === 'cert') { - // Make sure certificates exist - const fileCert = path.resolve(execPath, options.authCertFile); - const fileCertKey = path.resolve(execPath, options.authCertKeyFile); - const fileCertCA = path.resolve(execPath, options.authRootCertFile); - - axiosConfig = setupQRSConnection(options, { - method: 'get', - fileCert, - fileCertKey, - fileCertCA, - path: '/qrs/tag/full', - }); - } else if (options.authType === 'jwt') { - axiosConfig = setupQRSConnection(options, { - method: 'get', - path: '/qrs/tag/full', - }); - } - - logger.debug(`About to retrieve tags from QRS API.`); - - axios - .request(axiosConfig) - .then((result) => { - if (result.status === 200) { - const response = JSON.parse(result.data); - logger.info(`Successfully retrieved ${response.length} tags from QSEoW`); - // Yes, the tag exists - resolve(response); - } - resolve(false); - }) - .catch((err) => { - catchLog('GET TAGS FROM QSEoW', err); - }); - }); -} - -export function getTagIdByName(tagName, tagsExisting) { - return new Promise((resolve, _reject) => { - logger.debug(`Looking up ID for tag named "${tagName}"`); - - let tag; - if (typeof tagsExisting === 'string') { - tag = JSON.parse(tagsExisting).filter((item) => item.name === tagName); - } else { - tag = tagsExisting.filter((item) => item.name === tagName); - } - - if (tag.length === 1) { - // The tag exists - resolve(tag[0].id); - } else { - resolve(false); - } - }); -} - -// function getTagIdByName2(tagName, options, fileCert, fileCertKey) { -// return new Promise((resolve, reject) => { -// logger.debug(`Looking up ID for tag named "${tagName}"`); - -// // const filter = encodeURI(`name eq '👍😎 updateSheetThumbnail'`); -// const axiosConfig = setupQRSConnection(options, { -// method: 'get', -// fileCert, -// fileCertKey, -// path: '/qrs/tag', -// queryParameters: [{ name: 'filter', value: encodeURI(`name eq '${tagName}'`) }], -// }); - -// axios -// .request(axiosConfig) -// .then((result) => { -// if (result.data.length === 1) { -// logger.verbose(`Successfully found ID ${result.data[0].id} for tag named "${tagName}"`); -// // Yes, the tag exists -// resolve(result.data[0].id); -// } -// resolve(false); -// }) -// .catch((err) => { -// logger.error(`TAG ID BY NAME: ${err}`); -// }); -// }); -// } diff --git a/src/static/404.html b/src/static/404.html index 3b178f7..8f58182 100644 --- a/src/static/404.html +++ b/src/static/404.html @@ -1,4 +1,4 @@ - + 404 Not Found @@ -11,7 +11,9 @@ font-size: 50px; } body { - font: 20px Helvetica, sans-serif; + font: + 20px Helvetica, + sans-serif; color: #333; } article {