From 9fc2b084d3544801ff652ccfe2e135baae287b00 Mon Sep 17 00:00:00 2001 From: MVarshini Date: Mon, 1 Apr 2024 15:49:41 +0530 Subject: [PATCH 01/14] PANDA-361 Revamp CPT dashboard --- frontend/.eslintrc.cjs | 21 + frontend/.gitignore | 131 +- frontend/README.md | 74 +- frontend/index.html | 16 + frontend/package-lock.json | 7847 +++++++++++++++++ frontend/package.json | 70 +- frontend/public/index.html | 43 - frontend/public/manifest.json | 15 - frontend/public/robots.txt | 3 - frontend/src/App.js | 55 - frontend/src/App.jsx | 24 + frontend/src/App.less | 4 + .../assets/images/fa-external-link-alt.svg | 0 .../assets/images/grafana-icon.png | Bin .../assets/images/jenkins-icon.svg | 0 .../assets/images/prow-icon.png | Bin frontend/src/assets/logo.png | Bin 0 -> 11105 bytes frontend/src/common/fonts.css | 161 - .../commons/DisplayGrafana.js | 0 frontend/src/components/Home/HomeView.js | 85 - frontend/src/components/NavBar/NavBar.js | 29 - frontend/src/components/NavBar/ToolBar.js | 81 - .../src/components/OCP/BenchmarkResults.js | 27 - frontend/src/components/OCP/DisplayGraph.js | 38 - frontend/src/components/OCP/InstallCard.js | 97 - frontend/src/components/OCP/OCPHome.js | 119 - .../Accordion/AccordionView.js | 16 - .../PatternflyComponents/Card/CardView.js | 39 - .../Date/DatePickerView.js | 12 - .../Form/FormSelectView.js | 12 - .../PatternflyComponents/List/ListView.js | 14 - .../PatternflyComponents/Spinners/PuffLoad.js | 10 - .../PatternflyComponents/Split/SplitView.js | 17 - .../PatternflyComponents/Table/TableView.js | 143 - .../PatternflyComponents/Text/Text.js | 27 - .../src/components/Quay/BenchmarkResults.js | 36 - frontend/src/components/Quay/DisplayGraph.js | 40 - frontend/src/components/Quay/InstallCard.js | 90 - frontend/src/components/Quay/QuayHome.js | 93 - .../ReactGraphs/plotly/PlotlyView.js | 11 - frontend/src/components/css/NavBar.css | 9 - frontend/src/components/css/PlatformView.css | 12 - .../molecules/SideMenuOptions/index.jsx | 51 + .../src/components/organisms/Header/index.jsx | 54 + .../components/organisms/Header/index.less | 10 + .../organisms/LoadingComponent/index.jsx | 21 + .../organisms/LoadingComponent/index.less | 20 + .../components/organisms/Pagination/index.jsx | 18 + .../components/organisms/SideMenu/index.jsx | 17 + .../components/organisms/SideMenu/index.less | 0 .../organisms/ToastComponent/index.jsx | 5 + .../src/components/templates/Home/index.jsx | 5 + .../HomeLayout/DisplayTableDataLayout.js | 45 - .../templates/HomeLayout/SidebarLayout.js | 71 - .../templates/HomeLayout/TopLayout.js | 47 - .../components/templates/HomeLayout/index.js | 45 - frontend/src/containers/MainLayout/index.jsx | 26 + frontend/src/containers/MainLayout/index.less | 0 .../src/{store/reducers => helpers}/Utils.js | 0 frontend/src/index.css | 34 +- frontend/src/index.js | 19 - frontend/src/main.jsx | 16 + frontend/src/reportWebVitals.js | 13 - frontend/src/setupTests.js | 5 - frontend/src/store/Actions/ActionCreator.js | 280 - frontend/src/store/Shared.js | 16 - frontend/src/store/reducers/CPTJobsReducer.js | 58 - frontend/src/store/reducers/GraphReducer.js | 24 - frontend/src/store/reducers/InitialData.js | 136 - frontend/src/store/reducers/OCPJobsReducer.js | 82 - .../src/store/reducers/QuayGraphReducer.js | 24 - .../src/store/reducers/QuayJobsReducer.js | 64 - frontend/src/store/reducers/index.js | 14 - frontend/src/store/store.js | 13 +- frontend/src/utils/routeConstants.js | 2 + frontend/vite.config.js | 19 + 76 files changed, 8228 insertions(+), 2547 deletions(-) create mode 100644 frontend/.eslintrc.cjs create mode 100644 frontend/index.html create mode 100644 frontend/package-lock.json delete mode 100644 frontend/public/index.html delete mode 100644 frontend/public/manifest.json delete mode 100644 frontend/public/robots.txt delete mode 100644 frontend/src/App.js create mode 100644 frontend/src/App.jsx create mode 100644 frontend/src/App.less rename frontend/{public => src}/assets/images/fa-external-link-alt.svg (100%) rename frontend/{public => src}/assets/images/grafana-icon.png (100%) rename frontend/{public => src}/assets/images/jenkins-icon.svg (100%) rename frontend/{public => src}/assets/images/prow-icon.png (100%) create mode 100644 frontend/src/assets/logo.png delete mode 100644 frontend/src/common/fonts.css rename frontend/src/{components => }/commons/DisplayGrafana.js (100%) delete mode 100644 frontend/src/components/Home/HomeView.js delete mode 100644 frontend/src/components/NavBar/NavBar.js delete mode 100644 frontend/src/components/NavBar/ToolBar.js delete mode 100644 frontend/src/components/OCP/BenchmarkResults.js delete mode 100644 frontend/src/components/OCP/DisplayGraph.js delete mode 100644 frontend/src/components/OCP/InstallCard.js delete mode 100644 frontend/src/components/OCP/OCPHome.js delete mode 100644 frontend/src/components/PatternflyComponents/Accordion/AccordionView.js delete mode 100644 frontend/src/components/PatternflyComponents/Card/CardView.js delete mode 100644 frontend/src/components/PatternflyComponents/Date/DatePickerView.js delete mode 100644 frontend/src/components/PatternflyComponents/Form/FormSelectView.js delete mode 100644 frontend/src/components/PatternflyComponents/List/ListView.js delete mode 100644 frontend/src/components/PatternflyComponents/Spinners/PuffLoad.js delete mode 100644 frontend/src/components/PatternflyComponents/Split/SplitView.js delete mode 100644 frontend/src/components/PatternflyComponents/Table/TableView.js delete mode 100644 frontend/src/components/PatternflyComponents/Text/Text.js delete mode 100644 frontend/src/components/Quay/BenchmarkResults.js delete mode 100644 frontend/src/components/Quay/DisplayGraph.js delete mode 100644 frontend/src/components/Quay/InstallCard.js delete mode 100644 frontend/src/components/Quay/QuayHome.js delete mode 100644 frontend/src/components/ReactGraphs/plotly/PlotlyView.js delete mode 100644 frontend/src/components/css/NavBar.css delete mode 100644 frontend/src/components/css/PlatformView.css create mode 100644 frontend/src/components/molecules/SideMenuOptions/index.jsx create mode 100644 frontend/src/components/organisms/Header/index.jsx create mode 100644 frontend/src/components/organisms/Header/index.less create mode 100644 frontend/src/components/organisms/LoadingComponent/index.jsx create mode 100644 frontend/src/components/organisms/LoadingComponent/index.less create mode 100644 frontend/src/components/organisms/Pagination/index.jsx create mode 100644 frontend/src/components/organisms/SideMenu/index.jsx create mode 100644 frontend/src/components/organisms/SideMenu/index.less create mode 100644 frontend/src/components/organisms/ToastComponent/index.jsx create mode 100644 frontend/src/components/templates/Home/index.jsx delete mode 100644 frontend/src/components/templates/HomeLayout/DisplayTableDataLayout.js delete mode 100644 frontend/src/components/templates/HomeLayout/SidebarLayout.js delete mode 100644 frontend/src/components/templates/HomeLayout/TopLayout.js delete mode 100644 frontend/src/components/templates/HomeLayout/index.js create mode 100644 frontend/src/containers/MainLayout/index.jsx create mode 100644 frontend/src/containers/MainLayout/index.less rename frontend/src/{store/reducers => helpers}/Utils.js (100%) delete mode 100644 frontend/src/index.js create mode 100644 frontend/src/main.jsx delete mode 100644 frontend/src/reportWebVitals.js delete mode 100644 frontend/src/setupTests.js delete mode 100644 frontend/src/store/Actions/ActionCreator.js delete mode 100644 frontend/src/store/Shared.js delete mode 100644 frontend/src/store/reducers/CPTJobsReducer.js delete mode 100644 frontend/src/store/reducers/GraphReducer.js delete mode 100644 frontend/src/store/reducers/InitialData.js delete mode 100644 frontend/src/store/reducers/OCPJobsReducer.js delete mode 100644 frontend/src/store/reducers/QuayGraphReducer.js delete mode 100644 frontend/src/store/reducers/QuayJobsReducer.js delete mode 100644 frontend/src/store/reducers/index.js create mode 100644 frontend/src/utils/routeConstants.js create mode 100644 frontend/vite.config.js diff --git a/frontend/.eslintrc.cjs b/frontend/.eslintrc.cjs new file mode 100644 index 00000000..3e212e1d --- /dev/null +++ b/frontend/.eslintrc.cjs @@ -0,0 +1,21 @@ +module.exports = { + root: true, + env: { browser: true, es2020: true }, + extends: [ + 'eslint:recommended', + 'plugin:react/recommended', + 'plugin:react/jsx-runtime', + 'plugin:react-hooks/recommended', + ], + ignorePatterns: ['dist', '.eslintrc.cjs'], + parserOptions: { ecmaVersion: 'latest', sourceType: 'module' }, + settings: { react: { version: '18.2' } }, + plugins: ['react-refresh'], + rules: { + 'react/jsx-no-target-blank': 'off', + 'react-refresh/only-export-components': [ + 'warn', + { allowConstantExport: true }, + ], + }, +} diff --git a/frontend/.gitignore b/frontend/.gitignore index 06e31940..a547bf36 100644 --- a/frontend/.gitignore +++ b/frontend/.gitignore @@ -1,129 +1,24 @@ -yarn.lock - # Logs logs *.log npm-debug.log* yarn-debug.log* yarn-error.log* +pnpm-debug.log* lerna-debug.log* -package-lock.json -# Diagnostic reports (https://nodejs.org/api/report.html) -report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json - -# Runtime data -pids -*.pid -*.seed -*.pid.lock - -# Directory for instrumented libs generated by jscoverage/JSCover -lib-cov - -# Coverage directory used by tools like istanbul -coverage -*.lcov - -# nyc test coverage -.nyc_output - -# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) -.grunt - -# Bower dependency directory (https://bower.io/) -bower_components - -# node-waf configuration -.lock-wscript - -# Compiled binary addons (https://nodejs.org/api/addons.html) -build/Release - -# Dependency directories -node_modules/ -jspm_packages/ - -# TypeScript v1 declaration files -typings/ - -# TypeScript cache -*.tsbuildinfo - -# Optional npm cache directory -.npm - -# Optional eslint cache -.eslintcache - -# Microbundle cache -.rpt2_cache/ -.rts2_cache_cjs/ -.rts2_cache_es/ -.rts2_cache_umd/ - -# Optional REPL history -.node_repl_history - -# Output of 'npm pack' -*.tgz -# Yarn Integrity file -.yarn-integrity - -# dotenv environment variables file -.env -.env.test - -# parcel-bundler cache (https://parceljs.org/) -.cache - -# Next.js build output -.next - -# Nuxt.js build / generate output -.nuxt +node_modules dist +dist-ssr +*.local -# Gatsby files -.cache/ -# Comment in the public line in if your project uses Gatsby and *not* Next.js -# https://nextjs.org/blog/next-9-1#public-directory-support -# public - -# vuepress build output -.vuepress/dist - -# Serverless directories -.serverless/ - -# FuseBox cache -.fusebox/ - -# DynamoDB Local files -.dynamodb/ - -# TernJS port file -.tern-port -# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. - -# dependencies -/node_modules -/.pnp -.pnp.js - -# testing -/coverage - -# production -/build - -# misc +# Editor directories and files +.vscode/* +!.vscode/extensions.json +.idea .DS_Store -.env.local -.env.development.local -.env.test.local -.env.production.local - -npm-debug.log* -yarn-debug.log* -yarn-error.log* +*.suo +*.ntvs* +*.njsproj +*.sln +*.sw? diff --git a/frontend/README.md b/frontend/README.md index edb8f540..f768e33f 100644 --- a/frontend/README.md +++ b/frontend/README.md @@ -1,72 +1,8 @@ -# frontend +# React + Vite -## Development in a container +This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules. +Currently, two official plugins are available: - -## Available Scripts - -In the project directory, you can run: - -### `yarn start` - -Runs the app in the development mode.\ -Open [http://localhost:3000](http://localhost:3000) to view it in the browser. - -The page will reload if you make edits.\ -You will also see any lint errors in the console. - -### `yarn test` - -Launches the test runner in the interactive watch mode.\ -See the section about [running tests](https://facebook.github.io/create-react-app/docs/running-tests) for more information. - -### `yarn build` - -Builds the app for production to the `build` folder.\ -It correctly bundles React in production mode and optimizes the build for the best performance. - -The build is minified and the filenames include the hashes.\ -Your app is ready to be deployed! - -See the section about [deployment](https://facebook.github.io/create-react-app/docs/deployment) for more information. - -### `yarn eject` - -**Note: this is a one-way operation. Once you `eject`, you can’t go back!** - -If you aren’t satisfied with the build tool and configuration choices, you can `eject` at any time. This command will remove the single build dependency from your project. - -Instead, it will copy all the configuration files and the transitive dependencies (webpack, Babel, ESLint, etc) right into your project so you have full control over them. All of the commands except `eject` will still work, but they will point to the copied scripts so you can tweak them. At this point you’re on your own. - -You don’t have to ever use `eject`. The curated feature set is suitable for small and middle deployments, and you shouldn’t feel obligated to use this feature. However we understand that this tool wouldn’t be useful if you couldn’t customize it when you are ready for it. - -## Learn More - -You can learn more in the [Create React App documentation](https://facebook.github.io/create-react-app/docs/getting-started). - -To learn React, check out the [React documentation](https://reactjs.org/). - -### Code Splitting - -This section has moved here: [https://facebook.github.io/create-react-app/docs/code-splitting](https://facebook.github.io/create-react-app/docs/code-splitting) - -### Analyzing the Bundle Size - -This section has moved here: [https://facebook.github.io/create-react-app/docs/analyzing-the-bundle-size](https://facebook.github.io/create-react-app/docs/analyzing-the-bundle-size) - -### Making a Progressive Web App - -This section has moved here: [https://facebook.github.io/create-react-app/docs/making-a-progressive-web-app](https://facebook.github.io/create-react-app/docs/making-a-progressive-web-app) - -### Advanced Configuration - -This section has moved here: [https://facebook.github.io/create-react-app/docs/advanced-configuration](https://facebook.github.io/create-react-app/docs/advanced-configuration) - -### Deployment - -This section has moved here: [https://facebook.github.io/create-react-app/docs/deployment](https://facebook.github.io/create-react-app/docs/deployment) - -### `yarn build` fails to minify - -This section has moved here: [https://facebook.github.io/create-react-app/docs/troubleshooting#npm-run-build-fails-to-minify](https://facebook.github.io/create-react-app/docs/troubleshooting#npm-run-build-fails-to-minify) +- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react/README.md) uses [Babel](https://babeljs.io/) for Fast Refresh +- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh diff --git a/frontend/index.html b/frontend/index.html new file mode 100644 index 00000000..da34d8ff --- /dev/null +++ b/frontend/index.html @@ -0,0 +1,16 @@ + + + + + + + + OpenShift Performance and Scale - CI Dashboard + + + +
+ + + + \ No newline at end of file diff --git a/frontend/package-lock.json b/frontend/package-lock.json new file mode 100644 index 00000000..10007d03 --- /dev/null +++ b/frontend/package-lock.json @@ -0,0 +1,7847 @@ +{ + "name": "frontend", + "version": "0.0.0", + "lockfileVersion": 2, + "requires": true, + "packages": { + "": { + "name": "frontend", + "version": "0.0.0", + "dependencies": { + "@patternfly/react-core": "^5.2.3", + "@reduxjs/toolkit": "^2.2.3", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "react-redux": "^9.1.0", + "react-router-dom": "^6.22.3" + }, + "devDependencies": { + "@types/react": "^18.2.66", + "@types/react-dom": "^18.2.22", + "@vitejs/plugin-react": "^4.2.1", + "eslint": "^8.57.0", + "eslint-plugin-react": "^7.34.1", + "eslint-plugin-react-hooks": "^4.6.0", + "eslint-plugin-react-refresh": "^0.4.6", + "less": "^4.2.0", + "vite": "^5.2.0" + } + }, + "node_modules/@aashutoshrathi/word-wrap": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz", + "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@ampproject/remapping": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", + "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", + "dev": true, + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.24.2", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.24.2.tgz", + "integrity": "sha512-y5+tLQyV8pg3fsiln67BVLD1P13Eg4lh5RW9mF0zUuvLrv9uIQ4MCL+CRT+FTsBlBjcIan6PGsLcBN0m3ClUyQ==", + "dev": true, + "dependencies": { + "@babel/highlight": "^7.24.2", + "picocolors": "^1.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.24.1.tgz", + "integrity": "sha512-Pc65opHDliVpRHuKfzI+gSA4zcgr65O4cl64fFJIWEEh8JoHIHh0Oez1Eo8Arz8zq/JhgKodQaxEwUPRtZylVA==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.24.3", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.24.3.tgz", + "integrity": "sha512-5FcvN1JHw2sHJChotgx8Ek0lyuh4kCKelgMTTqhYJJtloNvUfpAFMeNQUtdlIaktwrSV9LtCdqwk48wL2wBacQ==", + "dev": true, + "dependencies": { + "@ampproject/remapping": "^2.2.0", + "@babel/code-frame": "^7.24.2", + "@babel/generator": "^7.24.1", + "@babel/helper-compilation-targets": "^7.23.6", + "@babel/helper-module-transforms": "^7.23.3", + "@babel/helpers": "^7.24.1", + "@babel/parser": "^7.24.1", + "@babel/template": "^7.24.0", + "@babel/traverse": "^7.24.1", + "@babel/types": "^7.24.0", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/generator": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.24.1.tgz", + "integrity": "sha512-DfCRfZsBcrPEHUfuBMgbJ1Ut01Y/itOs+hY2nFLgqsqXd52/iSiVq5TITtUasIUgm+IIKdY2/1I7auiQOEeC9A==", + "dev": true, + "dependencies": { + "@babel/types": "^7.24.0", + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.25", + "jsesc": "^2.5.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.23.6", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.23.6.tgz", + "integrity": "sha512-9JB548GZoQVmzrFgp8o7KxdgkTGm6xs9DW0o/Pim72UDjzr5ObUQ6ZzYPqA+g9OTS2bBQoctLJrky0RDCAWRgQ==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.23.5", + "@babel/helper-validator-option": "^7.23.5", + "browserslist": "^4.22.2", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-environment-visitor": { + "version": "7.22.20", + "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.22.20.tgz", + "integrity": "sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-function-name": { + "version": "7.23.0", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.23.0.tgz", + "integrity": "sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw==", + "dev": true, + "dependencies": { + "@babel/template": "^7.22.15", + "@babel/types": "^7.23.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-hoist-variables": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.22.5.tgz", + "integrity": "sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==", + "dev": true, + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.24.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.24.3.tgz", + "integrity": "sha512-viKb0F9f2s0BCS22QSF308z/+1YWKV/76mwt61NBzS5izMzDPwdq1pTrzf+Li3npBWX9KdQbkeCt1jSAM7lZqg==", + "dev": true, + "dependencies": { + "@babel/types": "^7.24.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.23.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.23.3.tgz", + "integrity": "sha512-7bBs4ED9OmswdfDzpz4MpWgSrV7FXlc3zIagvLFjS5H+Mk7Snr21vQ6QwrsoCGMfNC4e4LQPdoULEt4ykz0SRQ==", + "dev": true, + "dependencies": { + "@babel/helper-environment-visitor": "^7.22.20", + "@babel/helper-module-imports": "^7.22.15", + "@babel/helper-simple-access": "^7.22.5", + "@babel/helper-split-export-declaration": "^7.22.6", + "@babel/helper-validator-identifier": "^7.22.20" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.24.0", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.24.0.tgz", + "integrity": "sha512-9cUznXMG0+FxRuJfvL82QlTqIzhVW9sL0KjMPHhAOOvpQGL8QtdxnBKILjBqxlHyliz0yCa1G903ZXI/FuHy2w==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-simple-access": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.22.5.tgz", + "integrity": "sha512-n0H99E/K+Bika3++WNL17POvo4rKWZ7lZEp1Q+fStVbUi8nxPQEBOlTmCOxW/0JsS56SKKQ+ojAe2pHKJHN35w==", + "dev": true, + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-split-export-declaration": { + "version": "7.22.6", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.22.6.tgz", + "integrity": "sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==", + "dev": true, + "dependencies": { + "@babel/types": "^7.22.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.1.tgz", + "integrity": "sha512-2ofRCjnnA9y+wk8b9IAREroeUP02KHp431N2mhKniy2yKIDKpbrHv9eXwm8cBeWQYcJmzv5qKCu65P47eCF7CQ==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.22.20", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.20.tgz", + "integrity": "sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.23.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.23.5.tgz", + "integrity": "sha512-85ttAOMLsr53VgXkTbkx8oA6YTfT4q7/HzXSLEYmjcSTJPMPQtvq1BD79Byep5xMUYbGRzEpDsjUf3dyp54IKw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.24.1.tgz", + "integrity": "sha512-BpU09QqEe6ZCHuIHFphEFgvNSrubve1FtyMton26ekZ85gRGi6LrTF7zArARp2YvyFxloeiRmtSCq5sjh1WqIg==", + "dev": true, + "dependencies": { + "@babel/template": "^7.24.0", + "@babel/traverse": "^7.24.1", + "@babel/types": "^7.24.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/highlight": { + "version": "7.24.2", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.24.2.tgz", + "integrity": "sha512-Yac1ao4flkTxTteCDZLEvdxg2fZfz1v8M4QpaGypq/WPDqg3ijHYbDfs+LG5hvzSoqaSZ9/Z9lKSP3CjZjv+pA==", + "dev": true, + "dependencies": { + "@babel/helper-validator-identifier": "^7.22.20", + "chalk": "^2.4.2", + "js-tokens": "^4.0.0", + "picocolors": "^1.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.24.1.tgz", + "integrity": "sha512-Zo9c7N3xdOIQrNip7Lc9wvRPzlRtovHVE4lkz8WEDr7uYh/GMQhSiIgFxGIArRHYdJE5kxtZjAf8rT0xhdLCzg==", + "dev": true, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-self": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.24.1.tgz", + "integrity": "sha512-kDJgnPujTmAZ/9q2CN4m2/lRsUUPDvsG3+tSHWUJIzMGTt5U/b/fwWd3RO3n+5mjLrsBrVa5eKFRVSQbi3dF1w==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.24.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-react-jsx-source": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.24.1.tgz", + "integrity": "sha512-1v202n7aUq4uXAieRTKcwPzNyphlCuqHHDcdSNc+vdhoTEZcFMh+L5yZuCmGaIO7bs1nJUNfHB89TZyoL48xNA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.24.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/template": { + "version": "7.24.0", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.24.0.tgz", + "integrity": "sha512-Bkf2q8lMB0AFpX0NFEqSbx1OkTHf0f+0j82mkw+ZpzBnkk7e9Ql0891vlfgi+kHwOk8tQjiQHpqh4LaSa0fKEA==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.23.5", + "@babel/parser": "^7.24.0", + "@babel/types": "^7.24.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.24.1.tgz", + "integrity": "sha512-xuU6o9m68KeqZbQuDt2TcKSxUw/mrsvavlEqQ1leZ/B+C9tk6E4sRWy97WaXgvq5E+nU3cXMxv3WKOCanVMCmQ==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.24.1", + "@babel/generator": "^7.24.1", + "@babel/helper-environment-visitor": "^7.22.20", + "@babel/helper-function-name": "^7.23.0", + "@babel/helper-hoist-variables": "^7.22.5", + "@babel/helper-split-export-declaration": "^7.22.6", + "@babel/parser": "^7.24.1", + "@babel/types": "^7.24.0", + "debug": "^4.3.1", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.24.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.24.0.tgz", + "integrity": "sha512-+j7a5c253RfKh8iABBhywc8NSfP5LURe7Uh4qpsh6jc+aLJguvmIUBdjSdEMQv2bENrCR5MfRdjGo7vzS/ob7w==", + "dev": true, + "dependencies": { + "@babel/helper-string-parser": "^7.23.4", + "@babel/helper-validator-identifier": "^7.22.20", + "to-fast-properties": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.20.2.tgz", + "integrity": "sha512-D+EBOJHXdNZcLJRBkhENNG8Wji2kgc9AZ9KiPr1JuZjsNtyHzrsfLRrY0tk2H2aoFu6RANO1y1iPPUCDYWkb5g==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.20.2.tgz", + "integrity": "sha512-t98Ra6pw2VaDhqNWO2Oph2LXbz/EJcnLmKLGBJwEwXX/JAN83Fym1rU8l0JUWK6HkIbWONCSSatf4sf2NBRx/w==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.20.2.tgz", + "integrity": "sha512-mRzjLacRtl/tWU0SvD8lUEwb61yP9cqQo6noDZP/O8VkwafSYwZ4yWy24kan8jE/IMERpYncRt2dw438LP3Xmg==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.20.2.tgz", + "integrity": "sha512-btzExgV+/lMGDDa194CcUQm53ncxzeBrWJcncOBxuC6ndBkKxnHdFJn86mCIgTELsooUmwUm9FkhSp5HYu00Rg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.20.2.tgz", + "integrity": "sha512-4J6IRT+10J3aJH3l1yzEg9y3wkTDgDk7TSDFX+wKFiWjqWp/iCfLIYzGyasx9l0SAFPT1HwSCR+0w/h1ES/MjA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.20.2.tgz", + "integrity": "sha512-tBcXp9KNphnNH0dfhv8KYkZhjc+H3XBkF5DKtswJblV7KlT9EI2+jeA8DgBjp908WEuYll6pF+UStUCfEpdysA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.20.2.tgz", + "integrity": "sha512-d3qI41G4SuLiCGCFGUrKsSeTXyWG6yem1KcGZVS+3FYlYhtNoNgYrWcvkOoaqMhwXSMrZRl69ArHsGJ9mYdbbw==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.20.2.tgz", + "integrity": "sha512-d+DipyvHRuqEeM5zDivKV1KuXn9WeRX6vqSqIDgwIfPQtwMP4jaDsQsDncjTDDsExT4lR/91OLjRo8bmC1e+Cw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.20.2.tgz", + "integrity": "sha512-VhLPeR8HTMPccbuWWcEUD1Az68TqaTYyj6nfE4QByZIQEQVWBB8vup8PpR7y1QHL3CpcF6xd5WVBU/+SBEvGTg==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.20.2.tgz", + "integrity": "sha512-9pb6rBjGvTFNira2FLIWqDk/uaf42sSyLE8j1rnUpuzsODBq7FvpwHYZxQ/It/8b+QOS1RYfqgGFNLRI+qlq2A==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.20.2.tgz", + "integrity": "sha512-o10utieEkNPFDZFQm9CoP7Tvb33UutoJqg3qKf1PWVeeJhJw0Q347PxMvBgVVFgouYLGIhFYG0UGdBumROyiig==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.20.2.tgz", + "integrity": "sha512-PR7sp6R/UC4CFVomVINKJ80pMFlfDfMQMYynX7t1tNTeivQ6XdX5r2XovMmha/VjR1YN/HgHWsVcTRIMkymrgQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.20.2.tgz", + "integrity": "sha512-4BlTqeutE/KnOiTG5Y6Sb/Hw6hsBOZapOVF6njAESHInhlQAghVVZL1ZpIctBOoTFbQyGW+LsVYZ8lSSB3wkjA==", + "cpu": [ + "mips64el" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.20.2.tgz", + "integrity": "sha512-rD3KsaDprDcfajSKdn25ooz5J5/fWBylaaXkuotBDGnMnDP1Uv5DLAN/45qfnf3JDYyJv/ytGHQaziHUdyzaAg==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.20.2.tgz", + "integrity": "sha512-snwmBKacKmwTMmhLlz/3aH1Q9T8v45bKYGE3j26TsaOVtjIag4wLfWSiZykXzXuE1kbCE+zJRmwp+ZbIHinnVg==", + "cpu": [ + "riscv64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.20.2.tgz", + "integrity": "sha512-wcWISOobRWNm3cezm5HOZcYz1sKoHLd8VL1dl309DiixxVFoFe/o8HnwuIwn6sXre88Nwj+VwZUvJf4AFxkyrQ==", + "cpu": [ + "s390x" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.20.2.tgz", + "integrity": "sha512-1MdwI6OOTsfQfek8sLwgyjOXAu+wKhLEoaOLTjbijk6E2WONYpH9ZU2mNtR+lZ2B4uwr+usqGuVfFT9tMtGvGw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.20.2.tgz", + "integrity": "sha512-K8/DhBxcVQkzYc43yJXDSyjlFeHQJBiowJ0uVL6Tor3jGQfSGHNNJcWxNbOI8v5k82prYqzPuwkzHt3J1T1iZQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.20.2.tgz", + "integrity": "sha512-eMpKlV0SThJmmJgiVyN9jTPJ2VBPquf6Kt/nAoo6DgHAoN57K15ZghiHaMvqjCye/uU4X5u3YSMgVBI1h3vKrQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.20.2.tgz", + "integrity": "sha512-2UyFtRC6cXLyejf/YEld4Hajo7UHILetzE1vsRcGL3earZEW77JxrFjH4Ez2qaTiEfMgAXxfAZCm1fvM/G/o8w==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.20.2.tgz", + "integrity": "sha512-GRibxoawM9ZCnDxnP3usoUDO9vUkpAxIIZ6GQI+IlVmr5kP3zUq+l17xELTHMWTWzjxa2guPNyrpq1GWmPvcGQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.20.2.tgz", + "integrity": "sha512-HfLOfn9YWmkSKRQqovpnITazdtquEW8/SoHW7pWpuEeguaZI4QnCRW6b+oZTztdBnZOS2hqJ6im/D5cPzBTTlQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.20.2.tgz", + "integrity": "sha512-N49X4lJX27+l9jbLKSqZ6bKNjzQvHaT8IIFUy+YIqmXQdjYCToGWwOItDrfby14c78aDd5NHQl29xingXfCdLQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", + "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "dev": true, + "dependencies": { + "eslint-visitor-keys": "^3.3.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.10.0", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.10.0.tgz", + "integrity": "sha512-Cu96Sd2By9mCNTx2iyKOmq10v22jUVQv0lQnlGNy16oE9589yE+QADPbrMGCkA51cKZSg3Pu/aTJVTGfL/qjUA==", + "dev": true, + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", + "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "dev": true, + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/eslintrc/node_modules/globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "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==", + "dev": true, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.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==", + "dev": true, + "dependencies": { + "@humanwhocodes/object-schema": "^2.0.2", + "debug": "^4.3.1", + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=10.10.0" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "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/@jridgewell/gen-mapping": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", + "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", + "dev": true, + "dependencies": { + "@jridgewell/set-array": "^1.2.1", + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.24" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/set-array": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", + "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.4.15", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", + "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==", + "dev": true + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.25", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", + "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", + "dev": true, + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@patternfly/react-core": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/@patternfly/react-core/-/react-core-5.2.3.tgz", + "integrity": "sha512-MJeOLyJFbZPV+cj4LjL15nUuhJUwFuqLFiv6f2YubRqHl/+z05oM0byhwfm/qu2VnKByY6X6lu3Hp+hMTZcbOA==", + "dependencies": { + "@patternfly/react-icons": "^5.2.1", + "@patternfly/react-styles": "^5.2.1", + "@patternfly/react-tokens": "^5.2.1", + "focus-trap": "7.5.2", + "react-dropzone": "^14.2.3", + "tslib": "^2.5.0" + }, + "peerDependencies": { + "react": "^17 || ^18", + "react-dom": "^17 || ^18" + } + }, + "node_modules/@patternfly/react-icons": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/@patternfly/react-icons/-/react-icons-5.2.1.tgz", + "integrity": "sha512-aeJ0X+U2NDe8UmI5eQiT0iuR/wmUq97UkDtx3HoZcpRb9T6eUBfysllxjRqHS8rOOspdU8OWq+CUhQ/E2ZDibg==", + "peerDependencies": { + "react": "^17 || ^18", + "react-dom": "^17 || ^18" + } + }, + "node_modules/@patternfly/react-styles": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/@patternfly/react-styles/-/react-styles-5.2.1.tgz", + "integrity": "sha512-GT96hzI1QenBhq6Pfc51kxnj9aVLjL1zSLukKZXcYVe0HPOy0BFm90bT1Fo4e/z7V9cDYw4SqSX1XLc3O4jsTw==" + }, + "node_modules/@patternfly/react-tokens": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/@patternfly/react-tokens/-/react-tokens-5.2.1.tgz", + "integrity": "sha512-8GYz/jnJTGAWUJt5eRAW5dtyiHPKETeFJBPGHaUQnvi/t1ZAkoy8i4Kd/RlHsDC7ktiu813SKCmlzwBwldAHKg==" + }, + "node_modules/@reduxjs/toolkit": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/@reduxjs/toolkit/-/toolkit-2.2.3.tgz", + "integrity": "sha512-76dll9EnJXg4EVcI5YNxZA/9hSAmZsFqzMmNRHvIlzw2WS/twfcVX3ysYrWGJMClwEmChQFC4yRq74tn6fdzRA==", + "dependencies": { + "immer": "^10.0.3", + "redux": "^5.0.1", + "redux-thunk": "^3.1.0", + "reselect": "^5.0.1" + }, + "peerDependencies": { + "react": "^16.9.0 || ^17.0.0 || ^18", + "react-redux": "^7.2.1 || ^8.1.3 || ^9.0.0" + }, + "peerDependenciesMeta": { + "react": { + "optional": true + }, + "react-redux": { + "optional": true + } + } + }, + "node_modules/@remix-run/router": { + "version": "1.15.3", + "resolved": "https://registry.npmjs.org/@remix-run/router/-/router-1.15.3.tgz", + "integrity": "sha512-Oy8rmScVrVxWZVOpEF57ovlnhpZ8CCPlnIIumVcV9nFdiSIrus99+Lw78ekXyGvVDlIsFJbSfmSovJUhCWYV3w==", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/@rollup/rollup-android-arm-eabi": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.13.2.tgz", + "integrity": "sha512-3XFIDKWMFZrMnao1mJhnOT1h2g0169Os848NhhmGweEcfJ4rCi+3yMCOLG4zA61rbJdkcrM/DjVZm9Hg5p5w7g==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-android-arm64": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.13.2.tgz", + "integrity": "sha512-GdxxXbAuM7Y/YQM9/TwwP+L0omeE/lJAR1J+olu36c3LqqZEBdsIWeQ91KBe6nxwOnb06Xh7JS2U5ooWU5/LgQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ] + }, + "node_modules/@rollup/rollup-darwin-arm64": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.13.2.tgz", + "integrity": "sha512-mCMlpzlBgOTdaFs83I4XRr8wNPveJiJX1RLfv4hggyIVhfB5mJfN4P8Z6yKh+oE4Luz+qq1P3kVdWrCKcMYrrA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-darwin-x64": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.13.2.tgz", + "integrity": "sha512-yUoEvnH0FBef/NbB1u6d3HNGyruAKnN74LrPAfDQL3O32e3k3OSfLrPgSJmgb3PJrBZWfPyt6m4ZhAFa2nZp2A==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ] + }, + "node_modules/@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.13.2.tgz", + "integrity": "sha512-GYbLs5ErswU/Xs7aGXqzc3RrdEjKdmoCrgzhJWyFL0r5fL3qd1NPcDKDowDnmcoSiGJeU68/Vy+OMUluRxPiLQ==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.13.2.tgz", + "integrity": "sha512-L1+D8/wqGnKQIlh4Zre9i4R4b4noxzH5DDciyahX4oOz62CphY7WDWqJoQ66zNR4oScLNOqQJfNSIAe/6TPUmQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-arm64-musl": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.13.2.tgz", + "integrity": "sha512-tK5eoKFkXdz6vjfkSTCupUzCo40xueTOiOO6PeEIadlNBkadH1wNOH8ILCPIl8by/Gmb5AGAeQOFeLev7iZDOA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.13.2.tgz", + "integrity": "sha512-zvXvAUGGEYi6tYhcDmb9wlOckVbuD+7z3mzInCSTACJ4DQrdSLPNUeDIcAQW39M3q6PDquqLWu7pnO39uSMRzQ==", + "cpu": [ + "ppc64le" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.13.2.tgz", + "integrity": "sha512-C3GSKvMtdudHCN5HdmAMSRYR2kkhgdOfye4w0xzyii7lebVr4riCgmM6lRiSCnJn2w1Xz7ZZzHKuLrjx5620kw==", + "cpu": [ + "riscv64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-s390x-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.13.2.tgz", + "integrity": "sha512-l4U0KDFwzD36j7HdfJ5/TveEQ1fUTjFFQP5qIt9gBqBgu1G8/kCaq5Ok05kd5TG9F8Lltf3MoYsUMw3rNlJ0Yg==", + "cpu": [ + "s390x" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.13.2.tgz", + "integrity": "sha512-xXMLUAMzrtsvh3cZ448vbXqlUa7ZL8z0MwHp63K2IIID2+DeP5iWIT6g1SN7hg1VxPzqx0xZdiDM9l4n9LRU1A==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-linux-x64-musl": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.13.2.tgz", + "integrity": "sha512-M/JYAWickafUijWPai4ehrjzVPKRCyDb1SLuO+ZyPfoXgeCEAlgPkNXewFZx0zcnoIe3ay4UjXIMdXQXOZXWqA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ] + }, + "node_modules/@rollup/rollup-win32-arm64-msvc": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.13.2.tgz", + "integrity": "sha512-2YWwoVg9KRkIKaXSh0mz3NmfurpmYoBBTAXA9qt7VXk0Xy12PoOP40EFuau+ajgALbbhi4uTj3tSG3tVseCjuA==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-ia32-msvc": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.13.2.tgz", + "integrity": "sha512-2FSsE9aQ6OWD20E498NYKEQLneShWes0NGMPQwxWOdws35qQXH+FplabOSP5zEe1pVjurSDOGEVCE2agFwSEsw==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@rollup/rollup-win32-x64-msvc": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.13.2.tgz", + "integrity": "sha512-7h7J2nokcdPePdKykd8wtc8QqqkqxIrUz7MHj6aNr8waBRU//NLDVnNjQnqQO6fqtjrtCdftpbTuOKAyrAQETQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ] + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.6.8", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.8.tgz", + "integrity": "sha512-ASsj+tpEDsEiFr1arWrlN6V3mdfjRMZt6LtK/Vp/kreFLnr5QH5+DhvD5nINYZXzwJvXeGq+05iUXcAzVrqWtw==", + "dev": true, + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.20.5.tgz", + "integrity": "sha512-WXCyOcRtH37HAUkpXhUduaxdm82b4GSlyTqajXviN4EfiuPgNYR109xMCKvpl6zPIpua0DGlMEDCq+g8EdoheQ==", + "dev": true, + "dependencies": { + "@babel/types": "^7.20.7" + } + }, + "node_modules/@types/estree": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", + "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", + "dev": true + }, + "node_modules/@types/prop-types": { + "version": "15.7.12", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.12.tgz", + "integrity": "sha512-5zvhXYtRNRluoE/jAp4GVsSduVUzNWKkOZrCDBWYtE7biZywwdC2AcEzg+cSMLFRfVgeAFqpfNabiPjxFddV1Q==", + "devOptional": true + }, + "node_modules/@types/react": { + "version": "18.2.73", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.73.tgz", + "integrity": "sha512-XcGdod0Jjv84HOC7N5ziY3x+qL0AfmubvKOZ9hJjJ2yd5EE+KYjWhdOjt387e9HPheHkdggF9atTifMRtyAaRA==", + "devOptional": true, + "dependencies": { + "@types/prop-types": "*", + "csstype": "^3.0.2" + } + }, + "node_modules/@types/react-dom": { + "version": "18.2.23", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.2.23.tgz", + "integrity": "sha512-ZQ71wgGOTmDYpnav2knkjr3qXdAFu0vsk8Ci5w3pGAIdj7/kKAyn+VsQDhXsmzzzepAiI9leWMmubXz690AI/A==", + "dev": true, + "dependencies": { + "@types/react": "*" + } + }, + "node_modules/@types/use-sync-external-store": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/@types/use-sync-external-store/-/use-sync-external-store-0.0.3.tgz", + "integrity": "sha512-EwmlvuaxPNej9+T4v5AuBPJa2x2UOJVdjCtDHgcDqitUeOtjnJKJ+apYjVcAoBEMjKW1VVFGZLUb5+qqa09XFA==" + }, + "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/@vitejs/plugin-react": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.2.1.tgz", + "integrity": "sha512-oojO9IDc4nCUUi8qIR11KoQm0XFFLIwsRBwHRR4d/88IWghn1y6ckz/bJ8GHDCsYEJee8mDzqtJxh15/cisJNQ==", + "dev": true, + "dependencies": { + "@babel/core": "^7.23.5", + "@babel/plugin-transform-react-jsx-self": "^7.23.3", + "@babel/plugin-transform-react-jsx-source": "^7.23.3", + "@types/babel__core": "^7.20.5", + "react-refresh": "^0.14.0" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "peerDependencies": { + "vite": "^4.2.0 || ^5.0.0" + } + }, + "node_modules/acorn": { + "version": "8.11.3", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.3.tgz", + "integrity": "sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/argparse": { + "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.1", + "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.1.tgz", + "integrity": "sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "is-array-buffer": "^3.0.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array-includes": { + "version": "3.1.8", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.8.tgz", + "integrity": "sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.4", + "is-string": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.findlast": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/array.prototype.findlast/-/array.prototype.findlast-1.2.5.tgz", + "integrity": "sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "es-shim-unscopables": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/array.prototype.flat": { + "version": "1.3.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/array.prototype.toreversed": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/array.prototype.toreversed/-/array.prototype.toreversed-1.1.2.tgz", + "integrity": "sha512-wwDCoT4Ck4Cz7sLtgUmzR5UV3YF5mFHUlbChCzZBQZ+0m2cl/DH3tKgvphv1nKgFsJ48oCSg6p91q2Vm0I/ZMA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "es-shim-unscopables": "^1.0.0" + } + }, + "node_modules/array.prototype.tosorted": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/array.prototype.tosorted/-/array.prototype.tosorted-1.1.3.tgz", + "integrity": "sha512-/DdH4TiTmOKzyQbp/eadcCVexiCb36xJg7HshYOYJnNZFDj33GEv0P7GxsynpShhq4OLYJzbGcBDkLsDt7MnNg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "es-abstract": "^1.22.3", + "es-errors": "^1.1.0", + "es-shim-unscopables": "^1.0.2" + } + }, + "node_modules/arraybuffer.prototype.slice": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.3.tgz", + "integrity": "sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==", + "dev": true, + "dependencies": { + "array-buffer-byte-length": "^1.0.1", + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "es-abstract": "^1.22.3", + "es-errors": "^1.2.1", + "get-intrinsic": "^1.2.3", + "is-array-buffer": "^3.0.4", + "is-shared-array-buffer": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/attr-accept": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/attr-accept/-/attr-accept-2.2.2.tgz", + "integrity": "sha512-7prDjvt9HmqiZ0cl5CRjtS84sEyhsHP2coDkaZKRKVfCDo9s7iw7ChVmar78Gu9pC4SoR/28wFu/G5JJhTnqEg==", + "engines": { + "node": ">=4" + } + }, + "node_modules/available-typed-arrays": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", + "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", + "dev": true, + "dependencies": { + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/browserslist": { + "version": "4.23.0", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.23.0.tgz", + "integrity": "sha512-QW8HiM1shhT2GuzkvklfjcKDiWFXHOeFCIA/huJPwHsslwcydgk7X+z2zXpEijP98UCY7HbubZt5J2Zgvf0CaQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "caniuse-lite": "^1.0.30001587", + "electron-to-chromium": "^1.4.668", + "node-releases": "^2.0.14", + "update-browserslist-db": "^1.0.13" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/call-bind": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", + "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", + "dev": true, + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "set-function-length": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001603", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001603.tgz", + "integrity": "sha512-iL2iSS0eDILMb9n5yKQoTBim9jMZ0Yrk8g0N9K7UzYyWnfIKzXBZD5ngpM37ZcL/cv0Mli8XtVMRYMQAfFpi5Q==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ] + }, + "node_modules/chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "dependencies": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "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/copy-anything": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/copy-anything/-/copy-anything-2.0.6.tgz", + "integrity": "sha512-1j20GZTsvKNkc4BY3NpMOM8tt///wY3FpIzozTOFO2ffuZcV61nojHXVKIy3WM+7ADCy5FVhdZYHYDdgTU0yJw==", + "dev": true, + "dependencies": { + "is-what": "^3.14.1" + }, + "funding": { + "url": "https://github.com/sponsors/mesqueeb" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/csstype": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", + "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", + "devOptional": true + }, + "node_modules/data-view-buffer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.1.tgz", + "integrity": "sha512-0lht7OugA5x3iJLOWFhWK/5ehONdprk0ISXqVFn/NFrDu+cuc8iADFrGQz5BnRK7LLU3JmkbXSxaqX+/mXYtUA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/data-view-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.1.tgz", + "integrity": "sha512-4J7wRJD3ABAzr8wP+OcIcqq2dlUKp4DVflx++hs5h5ZKydWMI6/D/fAot+yh6g2tHh8fLFTvNOaVN357NvSrOQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/data-view-byte-offset": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.0.tgz", + "integrity": "sha512-t/Ygsytq+R995EJ5PZlD4Cu56sWa8InXySaViRzw9apusqsOO2bQP+SbYzAhR0pFKoB+43lYy8rWban9JSuXnA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/deep-is": { + "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 + }, + "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, + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/define-properties": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", + "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", + "dev": true, + "dependencies": { + "define-data-property": "^1.0.1", + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/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.722", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.722.tgz", + "integrity": "sha512-5nLE0TWFFpZ80Crhtp4pIp8LXCztjYX41yUcV6b+bKR2PqzjskTMOOlBi1VjBHlvHwS+4gar7kNKOrsbsewEZQ==", + "dev": true + }, + "node_modules/errno": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.8.tgz", + "integrity": "sha512-dJ6oBr5SQ1VSd9qkk7ByRgb/1SH4JZjCHSW/mr63/QcXO9zLVxvJ6Oy13nio03rxpSnVDDjFor75SjVeZWPW/A==", + "dev": true, + "optional": true, + "dependencies": { + "prr": "~1.0.1" + }, + "bin": { + "errno": "cli.js" + } + }, + "node_modules/es-abstract": { + "version": "1.23.3", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.23.3.tgz", + "integrity": "sha512-e+HfNH61Bj1X9/jLc5v1owaLYuHdeHHSQlkhCBiTK8rBvKaULl/beGMxwrMXjpYrv4pz22BlY570vVePA2ho4A==", + "dev": true, + "dependencies": { + "array-buffer-byte-length": "^1.0.1", + "arraybuffer.prototype.slice": "^1.0.3", + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "data-view-buffer": "^1.0.1", + "data-view-byte-length": "^1.0.1", + "data-view-byte-offset": "^1.0.0", + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "es-set-tostringtag": "^2.0.3", + "es-to-primitive": "^1.2.1", + "function.prototype.name": "^1.1.6", + "get-intrinsic": "^1.2.4", + "get-symbol-description": "^1.0.2", + "globalthis": "^1.0.3", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2", + "has-proto": "^1.0.3", + "has-symbols": "^1.0.3", + "hasown": "^2.0.2", + "internal-slot": "^1.0.7", + "is-array-buffer": "^3.0.4", + "is-callable": "^1.2.7", + "is-data-view": "^1.0.1", + "is-negative-zero": "^2.0.3", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.3", + "is-string": "^1.0.7", + "is-typed-array": "^1.1.13", + "is-weakref": "^1.0.2", + "object-inspect": "^1.13.1", + "object-keys": "^1.1.1", + "object.assign": "^4.1.5", + "regexp.prototype.flags": "^1.5.2", + "safe-array-concat": "^1.1.2", + "safe-regex-test": "^1.0.3", + "string.prototype.trim": "^1.2.9", + "string.prototype.trimend": "^1.0.8", + "string.prototype.trimstart": "^1.0.8", + "typed-array-buffer": "^1.0.2", + "typed-array-byte-length": "^1.0.1", + "typed-array-byte-offset": "^1.0.2", + "typed-array-length": "^1.0.6", + "unbox-primitive": "^1.0.2", + "which-typed-array": "^1.1.15" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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, + "dependencies": { + "get-intrinsic": "^1.2.4" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "dev": true, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-iterator-helpers": { + "version": "1.0.18", + "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.0.18.tgz", + "integrity": "sha512-scxAJaewsahbqTYrGKJihhViaM6DDZDDoucfvzNbK0pOren1g/daDQ3IAhzn+1G14rBG7w+i5N+qul60++zlKA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.0", + "es-errors": "^1.3.0", + "es-set-tostringtag": "^2.0.3", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "globalthis": "^1.0.3", + "has-property-descriptors": "^1.0.2", + "has-proto": "^1.0.3", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.7", + "iterator.prototype": "^1.1.2", + "safe-array-concat": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-object-atoms": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.0.0.tgz", + "integrity": "sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==", + "dev": true, + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-set-tostringtag": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.3.tgz", + "integrity": "sha512-3T8uNMC3OQTHkFUsFq8r/BwAXLHvU/9O9mE0fBc/MY5iq/8H7ncvO947LmYA6ldWw9Uh8Yhf25zu6n7nML5QWQ==", + "dev": true, + "dependencies": { + "get-intrinsic": "^1.2.4", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.1" + }, + "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==", + "dev": true, + "dependencies": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/esbuild": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.20.2.tgz", + "integrity": "sha512-WdOOppmUNU+IbZ0PaDiTst80zjnrOkyJNHoKupIcVyU8Lvla3Ugx94VzkQ32Ijqd7UhHJy75gNWDMUekcrSJ6g==", + "dev": true, + "hasInstallScript": true, + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.20.2", + "@esbuild/android-arm": "0.20.2", + "@esbuild/android-arm64": "0.20.2", + "@esbuild/android-x64": "0.20.2", + "@esbuild/darwin-arm64": "0.20.2", + "@esbuild/darwin-x64": "0.20.2", + "@esbuild/freebsd-arm64": "0.20.2", + "@esbuild/freebsd-x64": "0.20.2", + "@esbuild/linux-arm": "0.20.2", + "@esbuild/linux-arm64": "0.20.2", + "@esbuild/linux-ia32": "0.20.2", + "@esbuild/linux-loong64": "0.20.2", + "@esbuild/linux-mips64el": "0.20.2", + "@esbuild/linux-ppc64": "0.20.2", + "@esbuild/linux-riscv64": "0.20.2", + "@esbuild/linux-s390x": "0.20.2", + "@esbuild/linux-x64": "0.20.2", + "@esbuild/netbsd-x64": "0.20.2", + "@esbuild/openbsd-x64": "0.20.2", + "@esbuild/sunos-x64": "0.20.2", + "@esbuild/win32-arm64": "0.20.2", + "@esbuild/win32-ia32": "0.20.2", + "@esbuild/win32-x64": "0.20.2" + } + }, + "node_modules/escalade": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.2.tgz", + "integrity": "sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/eslint": { + "version": "8.57.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.0.tgz", + "integrity": "sha512-dZ6+mexnaTIbSBZWgou51U6OmzIhYM2VcNdtiTtI7qPNZm35Akpr0f6vtw3w1Kmn5PYo+tZVfh13WrhpS6oLqQ==", + "dev": 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", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "@ungap/structured-clone": "^1.2.0", + "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", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "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" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-plugin-react": { + "version": "7.34.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.34.1.tgz", + "integrity": "sha512-N97CxlouPT1AHt8Jn0mhhN2RrADlUAsk1/atcT2KyA/l9Q/E6ll7OIGwNumFmWfZ9skV3XXccYS19h80rHtgkw==", + "dev": true, + "dependencies": { + "array-includes": "^3.1.7", + "array.prototype.findlast": "^1.2.4", + "array.prototype.flatmap": "^1.3.2", + "array.prototype.toreversed": "^1.1.2", + "array.prototype.tosorted": "^1.1.3", + "doctrine": "^2.1.0", + "es-iterator-helpers": "^1.0.17", + "estraverse": "^5.3.0", + "jsx-ast-utils": "^2.4.1 || ^3.0.0", + "minimatch": "^3.1.2", + "object.entries": "^1.1.7", + "object.fromentries": "^2.0.7", + "object.hasown": "^1.1.3", + "object.values": "^1.1.7", + "prop-types": "^15.8.1", + "resolve": "^2.0.0-next.5", + "semver": "^6.3.1", + "string.prototype.matchall": "^4.0.10" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": "^3 || ^4 || ^5 || ^6 || ^7 || ^8" + } + }, + "node_modules/eslint-plugin-react-hooks": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-4.6.0.tgz", + "integrity": "sha512-oFc7Itz9Qxh2x4gNHStv3BqJq54ExXmfC+a1NjAta66IAN87Wu0R/QArgIS9qKzX3dXKPI9H5crl9QchNMY9+g==", + "dev": true, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0" + } + }, + "node_modules/eslint-plugin-react-refresh": { + "version": "0.4.6", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.6.tgz", + "integrity": "sha512-NjGXdm7zgcKRkKMua34qVO9doI7VOxZ6ancSvBELJSSoX97jyndXcSoa8XBh69JoB31dNz3EEzlMcizZl7LaMA==", + "dev": true, + "peerDependencies": { + "eslint": ">=7" + } + }, + "node_modules/eslint-plugin-react/node_modules/doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/eslint-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==", + "dev": true, + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/eslint/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/eslint/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/eslint/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/eslint/node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint/node_modules/globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "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==", + "dev": true, + "dependencies": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esquery": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", + "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "dev": true, + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "node_modules/fast-levenshtein": { + "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.17.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.17.1.tgz", + "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", + "dev": true, + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "dependencies": { + "flat-cache": "^3.0.4" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/file-selector": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/file-selector/-/file-selector-0.6.0.tgz", + "integrity": "sha512-QlZ5yJC0VxHxQQsQhXvBaC7VRJ2uaxTf+Tfpu4Z/OcVQJVpZO+DGU0rkoVW5ce2SccxugvpBJoMvUs59iILYdw==", + "dependencies": { + "tslib": "^2.4.0" + }, + "engines": { + "node": ">= 12" + } + }, + "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, + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "dev": true, + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/flatted": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz", + "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", + "dev": true + }, + "node_modules/focus-trap": { + "version": "7.5.2", + "resolved": "https://registry.npmjs.org/focus-trap/-/focus-trap-7.5.2.tgz", + "integrity": "sha512-p6vGNNWLDGwJCiEjkSK6oERj/hEyI9ITsSwIUICBoKLlWiTWXJRfQibCwcoi50rTZdbi87qDtUlMCmQwsGSgPw==", + "dependencies": { + "tabbable": "^6.2.0" + } + }, + "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/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/function.prototype.name": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.6.tgz", + "integrity": "sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "functions-have-names": "^1.2.3" + }, + "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", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-intrinsic": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", + "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", + "dev": true, + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3", + "hasown": "^2.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-symbol-description": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.2.tgz", + "integrity": "sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/globalthis": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.3.tgz", + "integrity": "sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==", + "dev": true, + "dependencies": { + "define-properties": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gopd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz", + "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==", + "dev": true, + "dependencies": { + "get-intrinsic": "^1.1.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true, + "optional": true + }, + "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/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": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/has-property-descriptors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", + "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", + "dev": true, + "dependencies": { + "es-define-property": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-proto": { + "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, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-tostringtag": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", + "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", + "dev": true, + "dependencies": { + "has-symbols": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "optional": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ignore": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.1.tgz", + "integrity": "sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/image-size": { + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/image-size/-/image-size-0.5.5.tgz", + "integrity": "sha512-6TDAlDPZxUFCv+fuOkIoXT/V/f3Qbq8e37p+YOiYrUv3v9cc3/6x78VdfPgFVaB9dZYeLUfKgHRebpkm/oP2VQ==", + "dev": true, + "optional": true, + "bin": { + "image-size": "bin/image-size.js" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/immer": { + "version": "10.0.4", + "resolved": "https://registry.npmjs.org/immer/-/immer-10.0.4.tgz", + "integrity": "sha512-cuBuGK40P/sk5IzWa9QPUaAdvPHjkk1c+xYsd9oZw+YQQEV+10G0P5uMpGctZZKnyQ+ibRO08bD25nWLmYi2pw==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/immer" + } + }, + "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, + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dev": true, + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "node_modules/internal-slot": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.7.tgz", + "integrity": "sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g==", + "dev": true, + "dependencies": { + "es-errors": "^1.3.0", + "hasown": "^2.0.0", + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-array-buffer": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.4.tgz", + "integrity": "sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-async-function": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.0.0.tgz", + "integrity": "sha512-Y1JXKrfykRJGdlDwdKlLpLyMIiWqWvuSd17TvZk68PLAOGOoF4Xyav1z0Xhoi+gCYjZVeC5SI+hYFOfvXmGRCA==", + "dev": true, + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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-data-view": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.1.tgz", + "integrity": "sha512-AHkaJrsUVW6wq6JS8y3JnM/GJF/9cf+k20+iDzlSaJrinEo5+7vRiteOSwBhHRiAyQATN1AmY4hwzxJKPmYf+w==", + "dev": true, + "dependencies": { + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "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==", + "dev": true, + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-finalizationregistry": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.0.2.tgz", + "integrity": "sha512-0by5vtUJs8iFQb5TYUHHPudOR+qXYIMKtiUzvLIZITZUjknFmziyBJuLhVRc+Ds0dREFlskDNJKYIdIzu/9pfw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-generator-function": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz", + "integrity": "sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==", + "dev": true, + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-map": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", + "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-negative-zero": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", + "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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-set": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", + "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-shared-array-buffer": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.3.tgz", + "integrity": "sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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.13", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.13.tgz", + "integrity": "sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==", + "dev": true, + "dependencies": { + "which-typed-array": "^1.1.14" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-weakmap": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", + "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", + "dev": true, + "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-weakset": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.3.tgz", + "integrity": "sha512-LvIm3/KWzS9oRFHugab7d+M/GcBXuXX5xZkzPmN+NxihdQlZUQ4dWuSV1xR/sq6upL1TJEDrfBgRepHFdBtSNQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "get-intrinsic": "^1.2.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-what": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/is-what/-/is-what-3.14.1.tgz", + "integrity": "sha512-sNxgpk9793nzSs7bA6JQJGeIuRBQhAaNGG77kzYQgMkrID+lS6SlK07K5LaptscDlSaIgH+GPFzf+d75FVxozA==", + "dev": true + }, + "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", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "node_modules/iterator.prototype": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/iterator.prototype/-/iterator.prototype-1.1.2.tgz", + "integrity": "sha512-DR33HMMr8EzwuRL8Y9D3u2BMj8+RqSE850jfGu59kS7tbmPLzGkZmVSfyCFSDxuZiEY6Rzt3T2NA/qU+NwVj1w==", + "dev": true, + "dependencies": { + "define-properties": "^1.2.1", + "get-intrinsic": "^1.2.1", + "has-symbols": "^1.0.3", + "reflect.getprototypeof": "^1.0.4", + "set-function-name": "^2.0.1" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" + }, + "node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "dev": true, + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "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 + }, + "node_modules/json-schema-traverse": { + "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 + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/jsx-ast-utils": { + "version": "3.3.5", + "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-3.3.5.tgz", + "integrity": "sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==", + "dev": true, + "dependencies": { + "array-includes": "^3.1.6", + "array.prototype.flat": "^1.3.1", + "object.assign": "^4.1.4", + "object.values": "^1.1.6" + }, + "engines": { + "node": ">=4.0" + } + }, + "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, + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/less": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/less/-/less-4.2.0.tgz", + "integrity": "sha512-P3b3HJDBtSzsXUl0im2L7gTO5Ubg8mEN6G8qoTS77iXxXX4Hvu4Qj540PZDvQ8V6DmX6iXo98k7Md0Cm1PrLaA==", + "dev": true, + "dependencies": { + "copy-anything": "^2.0.1", + "parse-node-version": "^1.0.1", + "tslib": "^2.3.0" + }, + "bin": { + "lessc": "bin/lessc" + }, + "engines": { + "node": ">=6" + }, + "optionalDependencies": { + "errno": "^0.1.1", + "graceful-fs": "^4.1.2", + "image-size": "~0.5.0", + "make-dir": "^2.1.0", + "mime": "^1.4.1", + "needle": "^3.1.0", + "source-map": "~0.6.0" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "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, + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "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 + }, + "node_modules/loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "dependencies": { + "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/make-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", + "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", + "dev": true, + "optional": true, + "dependencies": { + "pify": "^4.0.1", + "semver": "^5.6.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/make-dir/node_modules/semver": { + "version": "5.7.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz", + "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==", + "dev": true, + "optional": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "dev": true, + "optional": true, + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/nanoid": { + "version": "3.3.7", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz", + "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "node_modules/needle": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/needle/-/needle-3.3.1.tgz", + "integrity": "sha512-6k0YULvhpw+RoLNiQCRKOl09Rv1dPLr8hHnVjHqdolKwDrdNyk+Hmrthi4lIGPPz3r39dLx0hsF5s40sZ3Us4Q==", + "dev": true, + "optional": true, + "dependencies": { + "iconv-lite": "^0.6.3", + "sax": "^1.2.4" + }, + "bin": { + "needle": "bin/needle" + }, + "engines": { + "node": ">= 4.4.x" + } + }, + "node_modules/node-releases": { + "version": "2.0.14", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.14.tgz", + "integrity": "sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw==", + "dev": true + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-inspect": { + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.1.tgz", + "integrity": "sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==", + "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", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.assign": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.5.tgz", + "integrity": "sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "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.8", + "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.8.tgz", + "integrity": "sha512-cmopxi8VwRIAw/fkijJohSfpef5PdN0pMQJN6VC/ZKvn0LIknWD8KtgY6KlQdEc4tIjcQ3HxSMmnvtzIscdaYQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.fromentries": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", + "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.hasown": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/object.hasown/-/object.hasown-1.1.4.tgz", + "integrity": "sha512-FZ9LZt9/RHzGySlBARE3VF+gE26TxR38SdmqOqliuTnl9wrKulaQs+4dee1V+Io8VfxqzAfHu6YuRgUy8OHoTg==", + "dev": true, + "dependencies": { + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/object.values": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.0.tgz", + "integrity": "sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/optionator": { + "version": "0.9.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", + "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==", + "dev": true, + "dependencies": { + "@aashutoshrathi/word-wrap": "^1.2.3", + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-node-version": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parse-node-version/-/parse-node-version-1.0.1.tgz", + "integrity": "sha512-3YHlOa/JgH6Mnpr05jP9eDG254US9ek25LyIxZlDItp2iJtwyaXQb57lBYLdT3MowkUFYEV2XXNAYIPlESvJlA==", + "dev": true, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "node_modules/picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", + "dev": true + }, + "node_modules/pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true, + "optional": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/possible-typed-array-names": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", + "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", + "dev": true, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/postcss": { + "version": "8.4.38", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.38.tgz", + "integrity": "sha512-Wglpdk03BSfXkHoQa3b/oulrotAkwrlLDRSOb9D0bN86FdRyE9lppSp33aHNPgBa0JKCoB+drFLZkQoRRYae5A==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "nanoid": "^3.3.7", + "picocolors": "^1.0.0", + "source-map-js": "^1.2.0" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/prop-types": { + "version": "15.8.1", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz", + "integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==", + "dependencies": { + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.13.1" + } + }, + "node_modules/prr": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", + "integrity": "sha512-yPw4Sng1gWghHQWj0B3ZggWUm4qVbPwPFcRG8KyxiU7J2OHFSoEHKS+EZ3fv5l1t9CyCiop6l/ZYeWbrgoQejw==", + "dev": true, + "optional": true + }, + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/react": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react/-/react-18.2.0.tgz", + "integrity": "sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==", + "dependencies": { + "loose-envify": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-dom": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.2.0.tgz", + "integrity": "sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==", + "dependencies": { + "loose-envify": "^1.1.0", + "scheduler": "^0.23.0" + }, + "peerDependencies": { + "react": "^18.2.0" + } + }, + "node_modules/react-dropzone": { + "version": "14.2.3", + "resolved": "https://registry.npmjs.org/react-dropzone/-/react-dropzone-14.2.3.tgz", + "integrity": "sha512-O3om8I+PkFKbxCukfIR3QAGftYXDZfOE2N1mr/7qebQJHs7U+/RSL/9xomJNpRg9kM5h9soQSdf0Gc7OHF5Fug==", + "dependencies": { + "attr-accept": "^2.2.2", + "file-selector": "^0.6.0", + "prop-types": "^15.8.1" + }, + "engines": { + "node": ">= 10.13" + }, + "peerDependencies": { + "react": ">= 16.8 || 18.0.0" + } + }, + "node_modules/react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==" + }, + "node_modules/react-redux": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/react-redux/-/react-redux-9.1.0.tgz", + "integrity": "sha512-6qoDzIO+gbrza8h3hjMA9aq4nwVFCKFtY2iLxCtVT38Swyy2C/dJCGBXHeHLtx6qlg/8qzc2MrhOeduf5K32wQ==", + "dependencies": { + "@types/use-sync-external-store": "^0.0.3", + "use-sync-external-store": "^1.0.0" + }, + "peerDependencies": { + "@types/react": "^18.2.25", + "react": "^18.0", + "react-native": ">=0.69", + "redux": "^5.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "react-native": { + "optional": true + }, + "redux": { + "optional": true + } + } + }, + "node_modules/react-refresh": { + "version": "0.14.0", + "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.14.0.tgz", + "integrity": "sha512-wViHqhAd8OHeLS/IRMJjTSDHF3U9eWi62F/MledQGPdJGDhodXJ9PBLNGr6WWL7qlH12Mt3TyTpbS+hGXMjCzQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/react-router": { + "version": "6.22.3", + "resolved": "https://registry.npmjs.org/react-router/-/react-router-6.22.3.tgz", + "integrity": "sha512-dr2eb3Mj5zK2YISHK++foM9w4eBnO23eKnZEDs7c880P6oKbrjz/Svg9+nxqtHQK+oMW4OtjZca0RqPglXxguQ==", + "dependencies": { + "@remix-run/router": "1.15.3" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "react": ">=16.8" + } + }, + "node_modules/react-router-dom": { + "version": "6.22.3", + "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.22.3.tgz", + "integrity": "sha512-7ZILI7HjcE+p31oQvwbokjk6OA/bnFxrhJ19n82Ex9Ph8fNAq+Hm/7KchpMGlTgWhUxRHMMCut+vEtNpWpowKw==", + "dependencies": { + "@remix-run/router": "1.15.3", + "react-router": "6.22.3" + }, + "engines": { + "node": ">=14.0.0" + }, + "peerDependencies": { + "react": ">=16.8", + "react-dom": ">=16.8" + } + }, + "node_modules/redux": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/redux/-/redux-5.0.1.tgz", + "integrity": "sha512-M9/ELqF6fy8FwmkpnF0S3YKOqMyoWJ4+CS5Efg2ct3oY9daQvd/Pc71FpGZsVsbl3Cpb+IIcjBDUnnyBdQbq4w==" + }, + "node_modules/redux-thunk": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/redux-thunk/-/redux-thunk-3.1.0.tgz", + "integrity": "sha512-NW2r5T6ksUKXCabzhL9z+h206HQw/NJkcLm1GPImRQ8IzfXwRGqjVhKJGauHirT0DAuyy6hjdnMZaRoAcy0Klw==", + "peerDependencies": { + "redux": "^5.0.0" + } + }, + "node_modules/reflect.getprototypeof": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.6.tgz", + "integrity": "sha512-fmfw4XgoDke3kdI6h4xcUz1dG8uaiv5q9gcEwLS4Pnth2kxT+GZ7YehS1JTMGBQmtV7Y4GFGbs2re2NqhdozUg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.1", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", + "globalthis": "^1.0.3", + "which-builtin-type": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/regexp.prototype.flags": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.2.tgz", + "integrity": "sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.6", + "define-properties": "^1.2.1", + "es-errors": "^1.3.0", + "set-function-name": "^2.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/reselect": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/reselect/-/reselect-5.1.0.tgz", + "integrity": "sha512-aw7jcGLDpSgNDyWBQLv2cedml85qd95/iszJjN988zX1t7AVRJi19d9kto5+W7oCfQ94gyo40dVbT6g2k4/kXg==" + }, + "node_modules/resolve": { + "version": "2.0.0-next.5", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-2.0.0-next.5.tgz", + "integrity": "sha512-U7WjGVG9sH8tvjW5SmGbQuui75FiyjAX72HX15DwBBwF9dNiQZRQAg9nnPhYy+TUnE0+VcrttuvNI8oSxZcocA==", + "dev": true, + "dependencies": { + "is-core-module": "^2.13.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true, + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/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/rollup": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.13.2.tgz", + "integrity": "sha512-MIlLgsdMprDBXC+4hsPgzWUasLO9CE4zOkj/u6j+Z6j5A4zRY+CtiXAdJyPtgCsc42g658Aeh1DlrdVEJhsL2g==", + "dev": true, + "dependencies": { + "@types/estree": "1.0.5" + }, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=18.0.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "@rollup/rollup-android-arm-eabi": "4.13.2", + "@rollup/rollup-android-arm64": "4.13.2", + "@rollup/rollup-darwin-arm64": "4.13.2", + "@rollup/rollup-darwin-x64": "4.13.2", + "@rollup/rollup-linux-arm-gnueabihf": "4.13.2", + "@rollup/rollup-linux-arm64-gnu": "4.13.2", + "@rollup/rollup-linux-arm64-musl": "4.13.2", + "@rollup/rollup-linux-powerpc64le-gnu": "4.13.2", + "@rollup/rollup-linux-riscv64-gnu": "4.13.2", + "@rollup/rollup-linux-s390x-gnu": "4.13.2", + "@rollup/rollup-linux-x64-gnu": "4.13.2", + "@rollup/rollup-linux-x64-musl": "4.13.2", + "@rollup/rollup-win32-arm64-msvc": "4.13.2", + "@rollup/rollup-win32-ia32-msvc": "4.13.2", + "@rollup/rollup-win32-x64-msvc": "4.13.2", + "fsevents": "~2.3.2" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/safe-array-concat": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.2.tgz", + "integrity": "sha512-vj6RsCsWBCf19jIeHEfkRMw8DPiBb+DMXklQ/1SGDHOMlHdPUkZXFQ2YdplS23zESTijAcurb1aSgJA3AgMu1Q==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "get-intrinsic": "^1.2.4", + "has-symbols": "^1.0.3", + "isarray": "^2.0.5" + }, + "engines": { + "node": ">=0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safe-regex-test": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.3.tgz", + "integrity": "sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-regex": "^1.1.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "dev": true, + "optional": true + }, + "node_modules/sax": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.3.0.tgz", + "integrity": "sha512-0s+oAmw9zLl1V1cS9BtZN7JAd0cW5e0QH4W3LWEK6a4LaLEA2OTpGYWDY+6XasBLtz6wkm3u1xRw95mRuJ59WA==", + "dev": true, + "optional": true + }, + "node_modules/scheduler": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.0.tgz", + "integrity": "sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==", + "dependencies": { + "loose-envify": "^1.1.0" + } + }, + "node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/set-function-length": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", + "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", + "dev": true, + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/set-function-name": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", + "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", + "dev": true, + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "functions-have-names": "^1.2.3", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/side-channel": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz", + "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", + "object-inspect": "^1.13.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-js": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.0.tgz", + "integrity": "sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/string.prototype.matchall": { + "version": "4.0.11", + "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.11.tgz", + "integrity": "sha512-NUdh0aDavY2og7IbBPenWqR9exH+E26Sv8e0/eTe1tltDGZL+GtBkDAnnyBtmekfK6/Dq3MkcGtzXFEd1LQrtg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.7", + "regexp.prototype.flags": "^1.5.2", + "set-function-name": "^2.0.2", + "side-channel": "^1.0.6" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trim": { + "version": "1.2.9", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.9.tgz", + "integrity": "sha512-klHuCNxiMZ8MlsOihJhJEBJAiMVqU3Z2nEXWfWnIqjN0gEFS9J9+IxKozWWtQGcgoa1WUZzLjKPTr4ZHNFTFxw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.0", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trimend": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.8.tgz", + "integrity": "sha512-p73uL5VCHCO2BZZ6krwwQE3kCzM7NKmis8S//xEC6fQonchbum4eP6kR4DLEjQFO3Wnj3Fuo8NM0kOSjVdHjZQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/string.prototype.trimstart": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", + "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/tabbable": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/tabbable/-/tabbable-6.2.0.tgz", + "integrity": "sha512-Cat63mxsVJlzYvN51JmVXIgNoUokrIaT2zLclCXjRd8boZ0004U4KCs/sToJ75C6sdlByWxpYnb5Boif1VSFew==" + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "node_modules/to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/tslib": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", + "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==" + }, + "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, + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/typed-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.2.tgz", + "integrity": "sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/typed-array-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.1.tgz", + "integrity": "sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-byte-offset": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.2.tgz", + "integrity": "sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==", + "dev": true, + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-length": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.6.tgz", + "integrity": "sha512-/OxDN6OtAk5KBpGb28T+HZc2M+ADtvRxXrKKbUwtsLgdoxgX13hyy7ek6bFRl5+aBs2yZzB0c4CnQfAtVypW/g==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13", + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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/update-browserslist-db": { + "version": "1.0.13", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.0.13.tgz", + "integrity": "sha512-xebP81SNcPuNpPP3uzeW1NYXxI3rxyJzF3pD6sH4jE7o/IX+WtSpwnVU+qIsDPyk0d3hmFQ7mjqc6AtV604hbg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "escalade": "^3.1.1", + "picocolors": "^1.0.0" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/use-sync-external-store": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.2.0.tgz", + "integrity": "sha512-eEgnFxGQ1Ife9bzYs6VLi8/4X6CObHMw9Qr9tPY43iKwsPw8xE8+EFsf/2cFZ5S3esXgpWgtSCtLNS41F+sKPA==", + "peerDependencies": { + "react": "^16.8.0 || ^17.0.0 || ^18.0.0" + } + }, + "node_modules/vite": { + "version": "5.2.7", + "resolved": "https://registry.npmjs.org/vite/-/vite-5.2.7.tgz", + "integrity": "sha512-k14PWOKLI6pMaSzAuGtT+Cf0YmIx12z9YGon39onaJNy8DLBfBJrzg9FQEmkAM5lpHBZs9wksWAsyF/HkpEwJA==", + "dev": true, + "dependencies": { + "esbuild": "^0.20.1", + "postcss": "^8.4.38", + "rollup": "^4.13.0" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^18.0.0 || >=20.0.0" + }, + "funding": { + "url": "https://github.com/vitejs/vite?sponsor=1" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + }, + "peerDependencies": { + "@types/node": "^18.0.0 || >=20.0.0", + "less": "*", + "lightningcss": "^1.21.0", + "sass": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.4.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "less": { + "optional": true + }, + "lightningcss": { + "optional": true + }, + "sass": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + } + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "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-builtin-type": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.1.3.tgz", + "integrity": "sha512-YmjsSMDBYsM1CaFiayOVT06+KJeXf0o5M/CAd4o1lTadFAtacTUM49zoYxr/oroopFDfhvN6iEcBxUyc3gvKmw==", + "dev": true, + "dependencies": { + "function.prototype.name": "^1.1.5", + "has-tostringtag": "^1.0.0", + "is-async-function": "^2.0.0", + "is-date-object": "^1.0.5", + "is-finalizationregistry": "^1.0.2", + "is-generator-function": "^1.0.10", + "is-regex": "^1.1.4", + "is-weakref": "^1.0.2", + "isarray": "^2.0.5", + "which-boxed-primitive": "^1.0.2", + "which-collection": "^1.0.1", + "which-typed-array": "^1.1.9" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-collection": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", + "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", + "dev": true, + "dependencies": { + "is-map": "^2.0.3", + "is-set": "^2.0.3", + "is-weakmap": "^2.0.2", + "is-weakset": "^2.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/which-typed-array": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.15.tgz", + "integrity": "sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==", + "dev": true, + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-tostringtag": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + }, + "dependencies": { + "@aashutoshrathi/word-wrap": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@aashutoshrathi/word-wrap/-/word-wrap-1.2.6.tgz", + "integrity": "sha512-1Yjs2SvM8TflER/OD3cOjhWWOZb58A2t7wpE2S9XfBYTiIl+XFhQG2bjy4Pu1I+EAlCNUzRDYDdFwFYUKvXcIA==", + "dev": true + }, + "@ampproject/remapping": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", + "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", + "dev": true, + "requires": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "@babel/code-frame": { + "version": "7.24.2", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.24.2.tgz", + "integrity": "sha512-y5+tLQyV8pg3fsiln67BVLD1P13Eg4lh5RW9mF0zUuvLrv9uIQ4MCL+CRT+FTsBlBjcIan6PGsLcBN0m3ClUyQ==", + "dev": true, + "requires": { + "@babel/highlight": "^7.24.2", + "picocolors": "^1.0.0" + } + }, + "@babel/compat-data": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.24.1.tgz", + "integrity": "sha512-Pc65opHDliVpRHuKfzI+gSA4zcgr65O4cl64fFJIWEEh8JoHIHh0Oez1Eo8Arz8zq/JhgKodQaxEwUPRtZylVA==", + "dev": true + }, + "@babel/core": { + "version": "7.24.3", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.24.3.tgz", + "integrity": "sha512-5FcvN1JHw2sHJChotgx8Ek0lyuh4kCKelgMTTqhYJJtloNvUfpAFMeNQUtdlIaktwrSV9LtCdqwk48wL2wBacQ==", + "dev": true, + "requires": { + "@ampproject/remapping": "^2.2.0", + "@babel/code-frame": "^7.24.2", + "@babel/generator": "^7.24.1", + "@babel/helper-compilation-targets": "^7.23.6", + "@babel/helper-module-transforms": "^7.23.3", + "@babel/helpers": "^7.24.1", + "@babel/parser": "^7.24.1", + "@babel/template": "^7.24.0", + "@babel/traverse": "^7.24.1", + "@babel/types": "^7.24.0", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + } + }, + "@babel/generator": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.24.1.tgz", + "integrity": "sha512-DfCRfZsBcrPEHUfuBMgbJ1Ut01Y/itOs+hY2nFLgqsqXd52/iSiVq5TITtUasIUgm+IIKdY2/1I7auiQOEeC9A==", + "dev": true, + "requires": { + "@babel/types": "^7.24.0", + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.25", + "jsesc": "^2.5.1" + } + }, + "@babel/helper-compilation-targets": { + "version": "7.23.6", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.23.6.tgz", + "integrity": "sha512-9JB548GZoQVmzrFgp8o7KxdgkTGm6xs9DW0o/Pim72UDjzr5ObUQ6ZzYPqA+g9OTS2bBQoctLJrky0RDCAWRgQ==", + "dev": true, + "requires": { + "@babel/compat-data": "^7.23.5", + "@babel/helper-validator-option": "^7.23.5", + "browserslist": "^4.22.2", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + } + }, + "@babel/helper-environment-visitor": { + "version": "7.22.20", + "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.22.20.tgz", + "integrity": "sha512-zfedSIzFhat/gFhWfHtgWvlec0nqB9YEIVrpuwjruLlXfUSnA8cJB0miHKwqDnQ7d32aKo2xt88/xZptwxbfhA==", + "dev": true + }, + "@babel/helper-function-name": { + "version": "7.23.0", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.23.0.tgz", + "integrity": "sha512-OErEqsrxjZTJciZ4Oo+eoZqeW9UIiOcuYKRJA4ZAgV9myA+pOXhhmpfNCKjEH/auVfEYVFJ6y1Tc4r0eIApqiw==", + "dev": true, + "requires": { + "@babel/template": "^7.22.15", + "@babel/types": "^7.23.0" + } + }, + "@babel/helper-hoist-variables": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.22.5.tgz", + "integrity": "sha512-wGjk9QZVzvknA6yKIUURb8zY3grXCcOZt+/7Wcy8O2uctxhplmUPkOdlgoNhmdVee2c92JXbf1xpMtVNbfoxRw==", + "dev": true, + "requires": { + "@babel/types": "^7.22.5" + } + }, + "@babel/helper-module-imports": { + "version": "7.24.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.24.3.tgz", + "integrity": "sha512-viKb0F9f2s0BCS22QSF308z/+1YWKV/76mwt61NBzS5izMzDPwdq1pTrzf+Li3npBWX9KdQbkeCt1jSAM7lZqg==", + "dev": true, + "requires": { + "@babel/types": "^7.24.0" + } + }, + "@babel/helper-module-transforms": { + "version": "7.23.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.23.3.tgz", + "integrity": "sha512-7bBs4ED9OmswdfDzpz4MpWgSrV7FXlc3zIagvLFjS5H+Mk7Snr21vQ6QwrsoCGMfNC4e4LQPdoULEt4ykz0SRQ==", + "dev": true, + "requires": { + "@babel/helper-environment-visitor": "^7.22.20", + "@babel/helper-module-imports": "^7.22.15", + "@babel/helper-simple-access": "^7.22.5", + "@babel/helper-split-export-declaration": "^7.22.6", + "@babel/helper-validator-identifier": "^7.22.20" + } + }, + "@babel/helper-plugin-utils": { + "version": "7.24.0", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.24.0.tgz", + "integrity": "sha512-9cUznXMG0+FxRuJfvL82QlTqIzhVW9sL0KjMPHhAOOvpQGL8QtdxnBKILjBqxlHyliz0yCa1G903ZXI/FuHy2w==", + "dev": true + }, + "@babel/helper-simple-access": { + "version": "7.22.5", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.22.5.tgz", + "integrity": "sha512-n0H99E/K+Bika3++WNL17POvo4rKWZ7lZEp1Q+fStVbUi8nxPQEBOlTmCOxW/0JsS56SKKQ+ojAe2pHKJHN35w==", + "dev": true, + "requires": { + "@babel/types": "^7.22.5" + } + }, + "@babel/helper-split-export-declaration": { + "version": "7.22.6", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.22.6.tgz", + "integrity": "sha512-AsUnxuLhRYsisFiaJwvp1QF+I3KjD5FOxut14q/GzovUe6orHLesW2C7d754kRm53h5gqrz6sFl6sxc4BVtE/g==", + "dev": true, + "requires": { + "@babel/types": "^7.22.5" + } + }, + "@babel/helper-string-parser": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.1.tgz", + "integrity": "sha512-2ofRCjnnA9y+wk8b9IAREroeUP02KHp431N2mhKniy2yKIDKpbrHv9eXwm8cBeWQYcJmzv5qKCu65P47eCF7CQ==", + "dev": true + }, + "@babel/helper-validator-identifier": { + "version": "7.22.20", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.20.tgz", + "integrity": "sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A==", + "dev": true + }, + "@babel/helper-validator-option": { + "version": "7.23.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.23.5.tgz", + "integrity": "sha512-85ttAOMLsr53VgXkTbkx8oA6YTfT4q7/HzXSLEYmjcSTJPMPQtvq1BD79Byep5xMUYbGRzEpDsjUf3dyp54IKw==", + "dev": true + }, + "@babel/helpers": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.24.1.tgz", + "integrity": "sha512-BpU09QqEe6ZCHuIHFphEFgvNSrubve1FtyMton26ekZ85gRGi6LrTF7zArARp2YvyFxloeiRmtSCq5sjh1WqIg==", + "dev": true, + "requires": { + "@babel/template": "^7.24.0", + "@babel/traverse": "^7.24.1", + "@babel/types": "^7.24.0" + } + }, + "@babel/highlight": { + "version": "7.24.2", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.24.2.tgz", + "integrity": "sha512-Yac1ao4flkTxTteCDZLEvdxg2fZfz1v8M4QpaGypq/WPDqg3ijHYbDfs+LG5hvzSoqaSZ9/Z9lKSP3CjZjv+pA==", + "dev": true, + "requires": { + "@babel/helper-validator-identifier": "^7.22.20", + "chalk": "^2.4.2", + "js-tokens": "^4.0.0", + "picocolors": "^1.0.0" + } + }, + "@babel/parser": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.24.1.tgz", + "integrity": "sha512-Zo9c7N3xdOIQrNip7Lc9wvRPzlRtovHVE4lkz8WEDr7uYh/GMQhSiIgFxGIArRHYdJE5kxtZjAf8rT0xhdLCzg==", + "dev": true + }, + "@babel/plugin-transform-react-jsx-self": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.24.1.tgz", + "integrity": "sha512-kDJgnPujTmAZ/9q2CN4m2/lRsUUPDvsG3+tSHWUJIzMGTt5U/b/fwWd3RO3n+5mjLrsBrVa5eKFRVSQbi3dF1w==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.24.0" + } + }, + "@babel/plugin-transform-react-jsx-source": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.24.1.tgz", + "integrity": "sha512-1v202n7aUq4uXAieRTKcwPzNyphlCuqHHDcdSNc+vdhoTEZcFMh+L5yZuCmGaIO7bs1nJUNfHB89TZyoL48xNA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.24.0" + } + }, + "@babel/template": { + "version": "7.24.0", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.24.0.tgz", + "integrity": "sha512-Bkf2q8lMB0AFpX0NFEqSbx1OkTHf0f+0j82mkw+ZpzBnkk7e9Ql0891vlfgi+kHwOk8tQjiQHpqh4LaSa0fKEA==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.23.5", + "@babel/parser": "^7.24.0", + "@babel/types": "^7.24.0" + } + }, + "@babel/traverse": { + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.24.1.tgz", + "integrity": "sha512-xuU6o9m68KeqZbQuDt2TcKSxUw/mrsvavlEqQ1leZ/B+C9tk6E4sRWy97WaXgvq5E+nU3cXMxv3WKOCanVMCmQ==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.24.1", + "@babel/generator": "^7.24.1", + "@babel/helper-environment-visitor": "^7.22.20", + "@babel/helper-function-name": "^7.23.0", + "@babel/helper-hoist-variables": "^7.22.5", + "@babel/helper-split-export-declaration": "^7.22.6", + "@babel/parser": "^7.24.1", + "@babel/types": "^7.24.0", + "debug": "^4.3.1", + "globals": "^11.1.0" + } + }, + "@babel/types": { + "version": "7.24.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.24.0.tgz", + "integrity": "sha512-+j7a5c253RfKh8iABBhywc8NSfP5LURe7Uh4qpsh6jc+aLJguvmIUBdjSdEMQv2bENrCR5MfRdjGo7vzS/ob7w==", + "dev": true, + "requires": { + "@babel/helper-string-parser": "^7.23.4", + "@babel/helper-validator-identifier": "^7.22.20", + "to-fast-properties": "^2.0.0" + } + }, + "@esbuild/aix-ppc64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.20.2.tgz", + "integrity": "sha512-D+EBOJHXdNZcLJRBkhENNG8Wji2kgc9AZ9KiPr1JuZjsNtyHzrsfLRrY0tk2H2aoFu6RANO1y1iPPUCDYWkb5g==", + "dev": true, + "optional": true + }, + "@esbuild/android-arm": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.20.2.tgz", + "integrity": "sha512-t98Ra6pw2VaDhqNWO2Oph2LXbz/EJcnLmKLGBJwEwXX/JAN83Fym1rU8l0JUWK6HkIbWONCSSatf4sf2NBRx/w==", + "dev": true, + "optional": true + }, + "@esbuild/android-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.20.2.tgz", + "integrity": "sha512-mRzjLacRtl/tWU0SvD8lUEwb61yP9cqQo6noDZP/O8VkwafSYwZ4yWy24kan8jE/IMERpYncRt2dw438LP3Xmg==", + "dev": true, + "optional": true + }, + "@esbuild/android-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.20.2.tgz", + "integrity": "sha512-btzExgV+/lMGDDa194CcUQm53ncxzeBrWJcncOBxuC6ndBkKxnHdFJn86mCIgTELsooUmwUm9FkhSp5HYu00Rg==", + "dev": true, + "optional": true + }, + "@esbuild/darwin-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.20.2.tgz", + "integrity": "sha512-4J6IRT+10J3aJH3l1yzEg9y3wkTDgDk7TSDFX+wKFiWjqWp/iCfLIYzGyasx9l0SAFPT1HwSCR+0w/h1ES/MjA==", + "dev": true, + "optional": true + }, + "@esbuild/darwin-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.20.2.tgz", + "integrity": "sha512-tBcXp9KNphnNH0dfhv8KYkZhjc+H3XBkF5DKtswJblV7KlT9EI2+jeA8DgBjp908WEuYll6pF+UStUCfEpdysA==", + "dev": true, + "optional": true + }, + "@esbuild/freebsd-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.20.2.tgz", + "integrity": "sha512-d3qI41G4SuLiCGCFGUrKsSeTXyWG6yem1KcGZVS+3FYlYhtNoNgYrWcvkOoaqMhwXSMrZRl69ArHsGJ9mYdbbw==", + "dev": true, + "optional": true + }, + "@esbuild/freebsd-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.20.2.tgz", + "integrity": "sha512-d+DipyvHRuqEeM5zDivKV1KuXn9WeRX6vqSqIDgwIfPQtwMP4jaDsQsDncjTDDsExT4lR/91OLjRo8bmC1e+Cw==", + "dev": true, + "optional": true + }, + "@esbuild/linux-arm": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.20.2.tgz", + "integrity": "sha512-VhLPeR8HTMPccbuWWcEUD1Az68TqaTYyj6nfE4QByZIQEQVWBB8vup8PpR7y1QHL3CpcF6xd5WVBU/+SBEvGTg==", + "dev": true, + "optional": true + }, + "@esbuild/linux-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.20.2.tgz", + "integrity": "sha512-9pb6rBjGvTFNira2FLIWqDk/uaf42sSyLE8j1rnUpuzsODBq7FvpwHYZxQ/It/8b+QOS1RYfqgGFNLRI+qlq2A==", + "dev": true, + "optional": true + }, + "@esbuild/linux-ia32": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.20.2.tgz", + "integrity": "sha512-o10utieEkNPFDZFQm9CoP7Tvb33UutoJqg3qKf1PWVeeJhJw0Q347PxMvBgVVFgouYLGIhFYG0UGdBumROyiig==", + "dev": true, + "optional": true + }, + "@esbuild/linux-loong64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.20.2.tgz", + "integrity": "sha512-PR7sp6R/UC4CFVomVINKJ80pMFlfDfMQMYynX7t1tNTeivQ6XdX5r2XovMmha/VjR1YN/HgHWsVcTRIMkymrgQ==", + "dev": true, + "optional": true + }, + "@esbuild/linux-mips64el": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.20.2.tgz", + "integrity": "sha512-4BlTqeutE/KnOiTG5Y6Sb/Hw6hsBOZapOVF6njAESHInhlQAghVVZL1ZpIctBOoTFbQyGW+LsVYZ8lSSB3wkjA==", + "dev": true, + "optional": true + }, + "@esbuild/linux-ppc64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.20.2.tgz", + "integrity": "sha512-rD3KsaDprDcfajSKdn25ooz5J5/fWBylaaXkuotBDGnMnDP1Uv5DLAN/45qfnf3JDYyJv/ytGHQaziHUdyzaAg==", + "dev": true, + "optional": true + }, + "@esbuild/linux-riscv64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.20.2.tgz", + "integrity": "sha512-snwmBKacKmwTMmhLlz/3aH1Q9T8v45bKYGE3j26TsaOVtjIag4wLfWSiZykXzXuE1kbCE+zJRmwp+ZbIHinnVg==", + "dev": true, + "optional": true + }, + "@esbuild/linux-s390x": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.20.2.tgz", + "integrity": "sha512-wcWISOobRWNm3cezm5HOZcYz1sKoHLd8VL1dl309DiixxVFoFe/o8HnwuIwn6sXre88Nwj+VwZUvJf4AFxkyrQ==", + "dev": true, + "optional": true + }, + "@esbuild/linux-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.20.2.tgz", + "integrity": "sha512-1MdwI6OOTsfQfek8sLwgyjOXAu+wKhLEoaOLTjbijk6E2WONYpH9ZU2mNtR+lZ2B4uwr+usqGuVfFT9tMtGvGw==", + "dev": true, + "optional": true + }, + "@esbuild/netbsd-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.20.2.tgz", + "integrity": "sha512-K8/DhBxcVQkzYc43yJXDSyjlFeHQJBiowJ0uVL6Tor3jGQfSGHNNJcWxNbOI8v5k82prYqzPuwkzHt3J1T1iZQ==", + "dev": true, + "optional": true + }, + "@esbuild/openbsd-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.20.2.tgz", + "integrity": "sha512-eMpKlV0SThJmmJgiVyN9jTPJ2VBPquf6Kt/nAoo6DgHAoN57K15ZghiHaMvqjCye/uU4X5u3YSMgVBI1h3vKrQ==", + "dev": true, + "optional": true + }, + "@esbuild/sunos-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.20.2.tgz", + "integrity": "sha512-2UyFtRC6cXLyejf/YEld4Hajo7UHILetzE1vsRcGL3earZEW77JxrFjH4Ez2qaTiEfMgAXxfAZCm1fvM/G/o8w==", + "dev": true, + "optional": true + }, + "@esbuild/win32-arm64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.20.2.tgz", + "integrity": "sha512-GRibxoawM9ZCnDxnP3usoUDO9vUkpAxIIZ6GQI+IlVmr5kP3zUq+l17xELTHMWTWzjxa2guPNyrpq1GWmPvcGQ==", + "dev": true, + "optional": true + }, + "@esbuild/win32-ia32": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.20.2.tgz", + "integrity": "sha512-HfLOfn9YWmkSKRQqovpnITazdtquEW8/SoHW7pWpuEeguaZI4QnCRW6b+oZTztdBnZOS2hqJ6im/D5cPzBTTlQ==", + "dev": true, + "optional": true + }, + "@esbuild/win32-x64": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.20.2.tgz", + "integrity": "sha512-N49X4lJX27+l9jbLKSqZ6bKNjzQvHaT8IIFUy+YIqmXQdjYCToGWwOItDrfby14c78aDd5NHQl29xingXfCdLQ==", + "dev": true, + "optional": true + }, + "@eslint-community/eslint-utils": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", + "integrity": "sha512-1/sA4dwrzBAyeUoQ6oxahHKmrZvsnLCg4RfxW3ZFGGmQkSNQPFNLV9CUEFQP1x9EYXHTo5p6xdhZM1Ne9p/AfA==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^3.3.0" + } + }, + "@eslint-community/regexpp": { + "version": "4.10.0", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.10.0.tgz", + "integrity": "sha512-Cu96Sd2By9mCNTx2iyKOmq10v22jUVQv0lQnlGNy16oE9589yE+QADPbrMGCkA51cKZSg3Pu/aTJVTGfL/qjUA==", + "dev": true + }, + "@eslint/eslintrc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", + "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "dev": true, + "requires": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "dependencies": { + "globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "requires": { + "type-fest": "^0.20.2" + } + } + } + }, + "@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==", + "dev": true + }, + "@humanwhocodes/config-array": { + "version": "0.11.14", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.14.tgz", + "integrity": "sha512-3T8LkOmg45BV5FICb15QQMsyUSWrQ8AygVfC7ZG32zOalnqrilm018ZVCw0eapXux8FtA33q8PSRSstjee3jSg==", + "dev": true, + "requires": { + "@humanwhocodes/object-schema": "^2.0.2", + "debug": "^4.3.1", + "minimatch": "^3.0.5" + } + }, + "@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true + }, + "@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 + }, + "@jridgewell/gen-mapping": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", + "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", + "dev": true, + "requires": { + "@jridgewell/set-array": "^1.2.1", + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true + }, + "@jridgewell/set-array": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", + "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", + "dev": true + }, + "@jridgewell/sourcemap-codec": { + "version": "1.4.15", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", + "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==", + "dev": true + }, + "@jridgewell/trace-mapping": { + "version": "0.3.25", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", + "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", + "dev": true, + "requires": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "requires": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + } + }, + "@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 + }, + "@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "requires": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + } + }, + "@patternfly/react-core": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/@patternfly/react-core/-/react-core-5.2.3.tgz", + "integrity": "sha512-MJeOLyJFbZPV+cj4LjL15nUuhJUwFuqLFiv6f2YubRqHl/+z05oM0byhwfm/qu2VnKByY6X6lu3Hp+hMTZcbOA==", + "requires": { + "@patternfly/react-icons": "^5.2.1", + "@patternfly/react-styles": "^5.2.1", + "@patternfly/react-tokens": "^5.2.1", + "focus-trap": "7.5.2", + "react-dropzone": "^14.2.3", + "tslib": "^2.5.0" + } + }, + "@patternfly/react-icons": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/@patternfly/react-icons/-/react-icons-5.2.1.tgz", + "integrity": "sha512-aeJ0X+U2NDe8UmI5eQiT0iuR/wmUq97UkDtx3HoZcpRb9T6eUBfysllxjRqHS8rOOspdU8OWq+CUhQ/E2ZDibg==", + "requires": {} + }, + "@patternfly/react-styles": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/@patternfly/react-styles/-/react-styles-5.2.1.tgz", + "integrity": "sha512-GT96hzI1QenBhq6Pfc51kxnj9aVLjL1zSLukKZXcYVe0HPOy0BFm90bT1Fo4e/z7V9cDYw4SqSX1XLc3O4jsTw==" + }, + "@patternfly/react-tokens": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/@patternfly/react-tokens/-/react-tokens-5.2.1.tgz", + "integrity": "sha512-8GYz/jnJTGAWUJt5eRAW5dtyiHPKETeFJBPGHaUQnvi/t1ZAkoy8i4Kd/RlHsDC7ktiu813SKCmlzwBwldAHKg==" + }, + "@reduxjs/toolkit": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/@reduxjs/toolkit/-/toolkit-2.2.3.tgz", + "integrity": "sha512-76dll9EnJXg4EVcI5YNxZA/9hSAmZsFqzMmNRHvIlzw2WS/twfcVX3ysYrWGJMClwEmChQFC4yRq74tn6fdzRA==", + "requires": { + "immer": "^10.0.3", + "redux": "^5.0.1", + "redux-thunk": "^3.1.0", + "reselect": "^5.0.1" + } + }, + "@remix-run/router": { + "version": "1.15.3", + "resolved": "https://registry.npmjs.org/@remix-run/router/-/router-1.15.3.tgz", + "integrity": "sha512-Oy8rmScVrVxWZVOpEF57ovlnhpZ8CCPlnIIumVcV9nFdiSIrus99+Lw78ekXyGvVDlIsFJbSfmSovJUhCWYV3w==" + }, + "@rollup/rollup-android-arm-eabi": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.13.2.tgz", + "integrity": "sha512-3XFIDKWMFZrMnao1mJhnOT1h2g0169Os848NhhmGweEcfJ4rCi+3yMCOLG4zA61rbJdkcrM/DjVZm9Hg5p5w7g==", + "dev": true, + "optional": true + }, + "@rollup/rollup-android-arm64": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.13.2.tgz", + "integrity": "sha512-GdxxXbAuM7Y/YQM9/TwwP+L0omeE/lJAR1J+olu36c3LqqZEBdsIWeQ91KBe6nxwOnb06Xh7JS2U5ooWU5/LgQ==", + "dev": true, + "optional": true + }, + "@rollup/rollup-darwin-arm64": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.13.2.tgz", + "integrity": "sha512-mCMlpzlBgOTdaFs83I4XRr8wNPveJiJX1RLfv4hggyIVhfB5mJfN4P8Z6yKh+oE4Luz+qq1P3kVdWrCKcMYrrA==", + "dev": true, + "optional": true + }, + "@rollup/rollup-darwin-x64": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.13.2.tgz", + "integrity": "sha512-yUoEvnH0FBef/NbB1u6d3HNGyruAKnN74LrPAfDQL3O32e3k3OSfLrPgSJmgb3PJrBZWfPyt6m4ZhAFa2nZp2A==", + "dev": true, + "optional": true + }, + "@rollup/rollup-linux-arm-gnueabihf": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.13.2.tgz", + "integrity": "sha512-GYbLs5ErswU/Xs7aGXqzc3RrdEjKdmoCrgzhJWyFL0r5fL3qd1NPcDKDowDnmcoSiGJeU68/Vy+OMUluRxPiLQ==", + "dev": true, + "optional": true + }, + "@rollup/rollup-linux-arm64-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.13.2.tgz", + "integrity": "sha512-L1+D8/wqGnKQIlh4Zre9i4R4b4noxzH5DDciyahX4oOz62CphY7WDWqJoQ66zNR4oScLNOqQJfNSIAe/6TPUmQ==", + "dev": true, + "optional": true + }, + "@rollup/rollup-linux-arm64-musl": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.13.2.tgz", + "integrity": "sha512-tK5eoKFkXdz6vjfkSTCupUzCo40xueTOiOO6PeEIadlNBkadH1wNOH8ILCPIl8by/Gmb5AGAeQOFeLev7iZDOA==", + "dev": true, + "optional": true + }, + "@rollup/rollup-linux-powerpc64le-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.13.2.tgz", + "integrity": "sha512-zvXvAUGGEYi6tYhcDmb9wlOckVbuD+7z3mzInCSTACJ4DQrdSLPNUeDIcAQW39M3q6PDquqLWu7pnO39uSMRzQ==", + "dev": true, + "optional": true + }, + "@rollup/rollup-linux-riscv64-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.13.2.tgz", + "integrity": "sha512-C3GSKvMtdudHCN5HdmAMSRYR2kkhgdOfye4w0xzyii7lebVr4riCgmM6lRiSCnJn2w1Xz7ZZzHKuLrjx5620kw==", + "dev": true, + "optional": true + }, + "@rollup/rollup-linux-s390x-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.13.2.tgz", + "integrity": "sha512-l4U0KDFwzD36j7HdfJ5/TveEQ1fUTjFFQP5qIt9gBqBgu1G8/kCaq5Ok05kd5TG9F8Lltf3MoYsUMw3rNlJ0Yg==", + "dev": true, + "optional": true + }, + "@rollup/rollup-linux-x64-gnu": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.13.2.tgz", + "integrity": "sha512-xXMLUAMzrtsvh3cZ448vbXqlUa7ZL8z0MwHp63K2IIID2+DeP5iWIT6g1SN7hg1VxPzqx0xZdiDM9l4n9LRU1A==", + "dev": true, + "optional": true + }, + "@rollup/rollup-linux-x64-musl": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.13.2.tgz", + "integrity": "sha512-M/JYAWickafUijWPai4ehrjzVPKRCyDb1SLuO+ZyPfoXgeCEAlgPkNXewFZx0zcnoIe3ay4UjXIMdXQXOZXWqA==", + "dev": true, + "optional": true + }, + "@rollup/rollup-win32-arm64-msvc": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.13.2.tgz", + "integrity": "sha512-2YWwoVg9KRkIKaXSh0mz3NmfurpmYoBBTAXA9qt7VXk0Xy12PoOP40EFuau+ajgALbbhi4uTj3tSG3tVseCjuA==", + "dev": true, + "optional": true + }, + "@rollup/rollup-win32-ia32-msvc": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.13.2.tgz", + "integrity": "sha512-2FSsE9aQ6OWD20E498NYKEQLneShWes0NGMPQwxWOdws35qQXH+FplabOSP5zEe1pVjurSDOGEVCE2agFwSEsw==", + "dev": true, + "optional": true + }, + "@rollup/rollup-win32-x64-msvc": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.13.2.tgz", + "integrity": "sha512-7h7J2nokcdPePdKykd8wtc8QqqkqxIrUz7MHj6aNr8waBRU//NLDVnNjQnqQO6fqtjrtCdftpbTuOKAyrAQETQ==", + "dev": true, + "optional": true + }, + "@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "requires": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "@types/babel__generator": { + "version": "7.6.8", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.8.tgz", + "integrity": "sha512-ASsj+tpEDsEiFr1arWrlN6V3mdfjRMZt6LtK/Vp/kreFLnr5QH5+DhvD5nINYZXzwJvXeGq+05iUXcAzVrqWtw==", + "dev": true, + "requires": { + "@babel/types": "^7.0.0" + } + }, + "@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "requires": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "@types/babel__traverse": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.20.5.tgz", + "integrity": "sha512-WXCyOcRtH37HAUkpXhUduaxdm82b4GSlyTqajXviN4EfiuPgNYR109xMCKvpl6zPIpua0DGlMEDCq+g8EdoheQ==", + "dev": true, + "requires": { + "@babel/types": "^7.20.7" + } + }, + "@types/estree": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", + "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", + "dev": true + }, + "@types/prop-types": { + "version": "15.7.12", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.12.tgz", + "integrity": "sha512-5zvhXYtRNRluoE/jAp4GVsSduVUzNWKkOZrCDBWYtE7biZywwdC2AcEzg+cSMLFRfVgeAFqpfNabiPjxFddV1Q==", + "devOptional": true + }, + "@types/react": { + "version": "18.2.73", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.73.tgz", + "integrity": "sha512-XcGdod0Jjv84HOC7N5ziY3x+qL0AfmubvKOZ9hJjJ2yd5EE+KYjWhdOjt387e9HPheHkdggF9atTifMRtyAaRA==", + "devOptional": true, + "requires": { + "@types/prop-types": "*", + "csstype": "^3.0.2" + } + }, + "@types/react-dom": { + "version": "18.2.23", + "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.2.23.tgz", + "integrity": "sha512-ZQ71wgGOTmDYpnav2knkjr3qXdAFu0vsk8Ci5w3pGAIdj7/kKAyn+VsQDhXsmzzzepAiI9leWMmubXz690AI/A==", + "dev": true, + "requires": { + "@types/react": "*" + } + }, + "@types/use-sync-external-store": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/@types/use-sync-external-store/-/use-sync-external-store-0.0.3.tgz", + "integrity": "sha512-EwmlvuaxPNej9+T4v5AuBPJa2x2UOJVdjCtDHgcDqitUeOtjnJKJ+apYjVcAoBEMjKW1VVFGZLUb5+qqa09XFA==" + }, + "@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 + }, + "@vitejs/plugin-react": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.2.1.tgz", + "integrity": "sha512-oojO9IDc4nCUUi8qIR11KoQm0XFFLIwsRBwHRR4d/88IWghn1y6ckz/bJ8GHDCsYEJee8mDzqtJxh15/cisJNQ==", + "dev": true, + "requires": { + "@babel/core": "^7.23.5", + "@babel/plugin-transform-react-jsx-self": "^7.23.3", + "@babel/plugin-transform-react-jsx-source": "^7.23.3", + "@types/babel__core": "^7.20.5", + "react-refresh": "^0.14.0" + } + }, + "acorn": { + "version": "8.11.3", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.3.tgz", + "integrity": "sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==", + "dev": true + }, + "acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "requires": {} + }, + "ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true + }, + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "argparse": { + "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 + }, + "array-buffer-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.1.tgz", + "integrity": "sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==", + "dev": true, + "requires": { + "call-bind": "^1.0.5", + "is-array-buffer": "^3.0.4" + } + }, + "array-includes": { + "version": "3.1.8", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.8.tgz", + "integrity": "sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.4", + "is-string": "^1.0.7" + } + }, + "array.prototype.findlast": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/array.prototype.findlast/-/array.prototype.findlast-1.2.5.tgz", + "integrity": "sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "es-shim-unscopables": "^1.0.2" + } + }, + "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, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "es-shim-unscopables": "^1.0.0" + } + }, + "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, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "es-shim-unscopables": "^1.0.0" + } + }, + "array.prototype.toreversed": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/array.prototype.toreversed/-/array.prototype.toreversed-1.1.2.tgz", + "integrity": "sha512-wwDCoT4Ck4Cz7sLtgUmzR5UV3YF5mFHUlbChCzZBQZ+0m2cl/DH3tKgvphv1nKgFsJ48oCSg6p91q2Vm0I/ZMA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "es-shim-unscopables": "^1.0.0" + } + }, + "array.prototype.tosorted": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/array.prototype.tosorted/-/array.prototype.tosorted-1.1.3.tgz", + "integrity": "sha512-/DdH4TiTmOKzyQbp/eadcCVexiCb36xJg7HshYOYJnNZFDj33GEv0P7GxsynpShhq4OLYJzbGcBDkLsDt7MnNg==", + "dev": true, + "requires": { + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "es-abstract": "^1.22.3", + "es-errors": "^1.1.0", + "es-shim-unscopables": "^1.0.2" + } + }, + "arraybuffer.prototype.slice": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.3.tgz", + "integrity": "sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==", + "dev": true, + "requires": { + "array-buffer-byte-length": "^1.0.1", + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "es-abstract": "^1.22.3", + "es-errors": "^1.2.1", + "get-intrinsic": "^1.2.3", + "is-array-buffer": "^3.0.4", + "is-shared-array-buffer": "^1.0.2" + } + }, + "attr-accept": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/attr-accept/-/attr-accept-2.2.2.tgz", + "integrity": "sha512-7prDjvt9HmqiZ0cl5CRjtS84sEyhsHP2coDkaZKRKVfCDo9s7iw7ChVmar78Gu9pC4SoR/28wFu/G5JJhTnqEg==" + }, + "available-typed-arrays": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", + "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", + "dev": true, + "requires": { + "possible-typed-array-names": "^1.0.0" + } + }, + "balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "browserslist": { + "version": "4.23.0", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.23.0.tgz", + "integrity": "sha512-QW8HiM1shhT2GuzkvklfjcKDiWFXHOeFCIA/huJPwHsslwcydgk7X+z2zXpEijP98UCY7HbubZt5J2Zgvf0CaQ==", + "dev": true, + "requires": { + "caniuse-lite": "^1.0.30001587", + "electron-to-chromium": "^1.4.668", + "node-releases": "^2.0.14", + "update-browserslist-db": "^1.0.13" + } + }, + "call-bind": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", + "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", + "dev": true, + "requires": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "set-function-length": "^1.2.1" + } + }, + "callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true + }, + "caniuse-lite": { + "version": "1.0.30001603", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001603.tgz", + "integrity": "sha512-iL2iSS0eDILMb9n5yKQoTBim9jMZ0Yrk8g0N9K7UzYyWnfIKzXBZD5ngpM37ZcL/cv0Mli8XtVMRYMQAfFpi5Q==", + "dev": true + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "requires": { + "color-name": "1.1.3" + } + }, + "color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "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 + }, + "copy-anything": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/copy-anything/-/copy-anything-2.0.6.tgz", + "integrity": "sha512-1j20GZTsvKNkc4BY3NpMOM8tt///wY3FpIzozTOFO2ffuZcV61nojHXVKIy3WM+7ADCy5FVhdZYHYDdgTU0yJw==", + "dev": true, + "requires": { + "is-what": "^3.14.1" + } + }, + "cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "csstype": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", + "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", + "devOptional": true + }, + "data-view-buffer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.1.tgz", + "integrity": "sha512-0lht7OugA5x3iJLOWFhWK/5ehONdprk0ISXqVFn/NFrDu+cuc8iADFrGQz5BnRK7LLU3JmkbXSxaqX+/mXYtUA==", + "dev": true, + "requires": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + } + }, + "data-view-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.1.tgz", + "integrity": "sha512-4J7wRJD3ABAzr8wP+OcIcqq2dlUKp4DVflx++hs5h5ZKydWMI6/D/fAot+yh6g2tHh8fLFTvNOaVN357NvSrOQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + } + }, + "data-view-byte-offset": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.0.tgz", + "integrity": "sha512-t/Ygsytq+R995EJ5PZlD4Cu56sWa8InXySaViRzw9apusqsOO2bQP+SbYzAhR0pFKoB+43lYy8rWban9JSuXnA==", + "dev": true, + "requires": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + } + }, + "debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "deep-is": { + "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 + }, + "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, + "requires": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" + } + }, + "define-properties": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", + "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", + "dev": true, + "requires": { + "define-data-property": "^1.0.1", + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + } + }, + "doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "electron-to-chromium": { + "version": "1.4.722", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.722.tgz", + "integrity": "sha512-5nLE0TWFFpZ80Crhtp4pIp8LXCztjYX41yUcV6b+bKR2PqzjskTMOOlBi1VjBHlvHwS+4gar7kNKOrsbsewEZQ==", + "dev": true + }, + "errno": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.8.tgz", + "integrity": "sha512-dJ6oBr5SQ1VSd9qkk7ByRgb/1SH4JZjCHSW/mr63/QcXO9zLVxvJ6Oy13nio03rxpSnVDDjFor75SjVeZWPW/A==", + "dev": true, + "optional": true, + "requires": { + "prr": "~1.0.1" + } + }, + "es-abstract": { + "version": "1.23.3", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.23.3.tgz", + "integrity": "sha512-e+HfNH61Bj1X9/jLc5v1owaLYuHdeHHSQlkhCBiTK8rBvKaULl/beGMxwrMXjpYrv4pz22BlY570vVePA2ho4A==", + "dev": true, + "requires": { + "array-buffer-byte-length": "^1.0.1", + "arraybuffer.prototype.slice": "^1.0.3", + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "data-view-buffer": "^1.0.1", + "data-view-byte-length": "^1.0.1", + "data-view-byte-offset": "^1.0.0", + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "es-set-tostringtag": "^2.0.3", + "es-to-primitive": "^1.2.1", + "function.prototype.name": "^1.1.6", + "get-intrinsic": "^1.2.4", + "get-symbol-description": "^1.0.2", + "globalthis": "^1.0.3", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2", + "has-proto": "^1.0.3", + "has-symbols": "^1.0.3", + "hasown": "^2.0.2", + "internal-slot": "^1.0.7", + "is-array-buffer": "^3.0.4", + "is-callable": "^1.2.7", + "is-data-view": "^1.0.1", + "is-negative-zero": "^2.0.3", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.3", + "is-string": "^1.0.7", + "is-typed-array": "^1.1.13", + "is-weakref": "^1.0.2", + "object-inspect": "^1.13.1", + "object-keys": "^1.1.1", + "object.assign": "^4.1.5", + "regexp.prototype.flags": "^1.5.2", + "safe-array-concat": "^1.1.2", + "safe-regex-test": "^1.0.3", + "string.prototype.trim": "^1.2.9", + "string.prototype.trimend": "^1.0.8", + "string.prototype.trimstart": "^1.0.8", + "typed-array-buffer": "^1.0.2", + "typed-array-byte-length": "^1.0.1", + "typed-array-byte-offset": "^1.0.2", + "typed-array-length": "^1.0.6", + "unbox-primitive": "^1.0.2", + "which-typed-array": "^1.1.15" + } + }, + "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, + "requires": { + "get-intrinsic": "^1.2.4" + } + }, + "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 + }, + "es-iterator-helpers": { + "version": "1.0.18", + "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.0.18.tgz", + "integrity": "sha512-scxAJaewsahbqTYrGKJihhViaM6DDZDDoucfvzNbK0pOren1g/daDQ3IAhzn+1G14rBG7w+i5N+qul60++zlKA==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.0", + "es-errors": "^1.3.0", + "es-set-tostringtag": "^2.0.3", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "globalthis": "^1.0.3", + "has-property-descriptors": "^1.0.2", + "has-proto": "^1.0.3", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.7", + "iterator.prototype": "^1.1.2", + "safe-array-concat": "^1.1.2" + } + }, + "es-object-atoms": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.0.0.tgz", + "integrity": "sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==", + "dev": true, + "requires": { + "es-errors": "^1.3.0" + } + }, + "es-set-tostringtag": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.3.tgz", + "integrity": "sha512-3T8uNMC3OQTHkFUsFq8r/BwAXLHvU/9O9mE0fBc/MY5iq/8H7ncvO947LmYA6ldWw9Uh8Yhf25zu6n7nML5QWQ==", + "dev": true, + "requires": { + "get-intrinsic": "^1.2.4", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.1" + } + }, + "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, + "requires": { + "hasown": "^2.0.0" + } + }, + "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==", + "dev": true, + "requires": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + } + }, + "esbuild": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.20.2.tgz", + "integrity": "sha512-WdOOppmUNU+IbZ0PaDiTst80zjnrOkyJNHoKupIcVyU8Lvla3Ugx94VzkQ32Ijqd7UhHJy75gNWDMUekcrSJ6g==", + "dev": true, + "requires": { + "@esbuild/aix-ppc64": "0.20.2", + "@esbuild/android-arm": "0.20.2", + "@esbuild/android-arm64": "0.20.2", + "@esbuild/android-x64": "0.20.2", + "@esbuild/darwin-arm64": "0.20.2", + "@esbuild/darwin-x64": "0.20.2", + "@esbuild/freebsd-arm64": "0.20.2", + "@esbuild/freebsd-x64": "0.20.2", + "@esbuild/linux-arm": "0.20.2", + "@esbuild/linux-arm64": "0.20.2", + "@esbuild/linux-ia32": "0.20.2", + "@esbuild/linux-loong64": "0.20.2", + "@esbuild/linux-mips64el": "0.20.2", + "@esbuild/linux-ppc64": "0.20.2", + "@esbuild/linux-riscv64": "0.20.2", + "@esbuild/linux-s390x": "0.20.2", + "@esbuild/linux-x64": "0.20.2", + "@esbuild/netbsd-x64": "0.20.2", + "@esbuild/openbsd-x64": "0.20.2", + "@esbuild/sunos-x64": "0.20.2", + "@esbuild/win32-arm64": "0.20.2", + "@esbuild/win32-ia32": "0.20.2", + "@esbuild/win32-x64": "0.20.2" + } + }, + "escalade": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.2.tgz", + "integrity": "sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA==", + "dev": true + }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true + }, + "eslint": { + "version": "8.57.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.0.tgz", + "integrity": "sha512-dZ6+mexnaTIbSBZWgou51U6OmzIhYM2VcNdtiTtI7qPNZm35Akpr0f6vtw3w1Kmn5PYo+tZVfh13WrhpS6oLqQ==", + "dev": true, + "requires": { + "@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", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "@ungap/structured-clone": "^1.2.0", + "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", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "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" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true + }, + "globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "requires": { + "type-fest": "^0.20.2" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "eslint-plugin-react": { + "version": "7.34.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.34.1.tgz", + "integrity": "sha512-N97CxlouPT1AHt8Jn0mhhN2RrADlUAsk1/atcT2KyA/l9Q/E6ll7OIGwNumFmWfZ9skV3XXccYS19h80rHtgkw==", + "dev": true, + "requires": { + "array-includes": "^3.1.7", + "array.prototype.findlast": "^1.2.4", + "array.prototype.flatmap": "^1.3.2", + "array.prototype.toreversed": "^1.1.2", + "array.prototype.tosorted": "^1.1.3", + "doctrine": "^2.1.0", + "es-iterator-helpers": "^1.0.17", + "estraverse": "^5.3.0", + "jsx-ast-utils": "^2.4.1 || ^3.0.0", + "minimatch": "^3.1.2", + "object.entries": "^1.1.7", + "object.fromentries": "^2.0.7", + "object.hasown": "^1.1.3", + "object.values": "^1.1.7", + "prop-types": "^15.8.1", + "resolve": "^2.0.0-next.5", + "semver": "^6.3.1", + "string.prototype.matchall": "^4.0.10" + }, + "dependencies": { + "doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + } + } + }, + "eslint-plugin-react-hooks": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-4.6.0.tgz", + "integrity": "sha512-oFc7Itz9Qxh2x4gNHStv3BqJq54ExXmfC+a1NjAta66IAN87Wu0R/QArgIS9qKzX3dXKPI9H5crl9QchNMY9+g==", + "dev": true, + "requires": {} + }, + "eslint-plugin-react-refresh": { + "version": "0.4.6", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.6.tgz", + "integrity": "sha512-NjGXdm7zgcKRkKMua34qVO9doI7VOxZ6ancSvBELJSSoX97jyndXcSoa8XBh69JoB31dNz3EEzlMcizZl7LaMA==", + "dev": true, + "requires": {} + }, + "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==", + "dev": true, + "requires": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + } + }, + "eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true + }, + "espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "requires": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + } + }, + "esquery": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", + "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "dev": true, + "requires": { + "estraverse": "^5.1.0" + } + }, + "esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "requires": { + "estraverse": "^5.2.0" + } + }, + "estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true + }, + "esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true + }, + "fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "fastq": { + "version": "1.17.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.17.1.tgz", + "integrity": "sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==", + "dev": true, + "requires": { + "reusify": "^1.0.4" + } + }, + "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==", + "dev": true, + "requires": { + "flat-cache": "^3.0.4" + } + }, + "file-selector": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/file-selector/-/file-selector-0.6.0.tgz", + "integrity": "sha512-QlZ5yJC0VxHxQQsQhXvBaC7VRJ2uaxTf+Tfpu4Z/OcVQJVpZO+DGU0rkoVW5ce2SccxugvpBJoMvUs59iILYdw==", + "requires": { + "tslib": "^2.4.0" + } + }, + "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, + "requires": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + } + }, + "flat-cache": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "dev": true, + "requires": { + "flatted": "^3.2.9", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + } + }, + "flatted": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz", + "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", + "dev": true + }, + "focus-trap": { + "version": "7.5.2", + "resolved": "https://registry.npmjs.org/focus-trap/-/focus-trap-7.5.2.tgz", + "integrity": "sha512-p6vGNNWLDGwJCiEjkSK6oERj/hEyI9ITsSwIUICBoKLlWiTWXJRfQibCwcoi50rTZdbi87qDtUlMCmQwsGSgPw==", + "requires": { + "tabbable": "^6.2.0" + } + }, + "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, + "requires": { + "is-callable": "^1.1.3" + } + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "optional": true + }, + "function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "dev": true + }, + "function.prototype.name": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.6.tgz", + "integrity": "sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "functions-have-names": "^1.2.3" + } + }, + "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 + }, + "gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true + }, + "get-intrinsic": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", + "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", + "dev": true, + "requires": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3", + "hasown": "^2.0.0" + } + }, + "get-symbol-description": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.2.tgz", + "integrity": "sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==", + "dev": true, + "requires": { + "call-bind": "^1.0.5", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4" + } + }, + "glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "requires": { + "is-glob": "^4.0.3" + } + }, + "globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true + }, + "globalthis": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.3.tgz", + "integrity": "sha512-sFdI5LyBiNTHjRd7cGPWapiHWMOXKyuBNX/cWJ3NfzrZQVa8GI/8cofCl74AOVqq9W5kNmguTIzJ/1s2gyI9wA==", + "dev": true, + "requires": { + "define-properties": "^1.1.3" + } + }, + "gopd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz", + "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==", + "dev": true, + "requires": { + "get-intrinsic": "^1.1.3" + } + }, + "graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true, + "optional": true + }, + "graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true + }, + "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 + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "dev": true + }, + "has-property-descriptors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", + "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", + "dev": true, + "requires": { + "es-define-property": "^1.0.0" + } + }, + "has-proto": { + "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 + }, + "has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true + }, + "has-tostringtag": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", + "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", + "dev": true, + "requires": { + "has-symbols": "^1.0.3" + } + }, + "hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dev": true, + "requires": { + "function-bind": "^1.1.2" + } + }, + "iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "optional": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + } + }, + "ignore": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.1.tgz", + "integrity": "sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==", + "dev": true + }, + "image-size": { + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/image-size/-/image-size-0.5.5.tgz", + "integrity": "sha512-6TDAlDPZxUFCv+fuOkIoXT/V/f3Qbq8e37p+YOiYrUv3v9cc3/6x78VdfPgFVaB9dZYeLUfKgHRebpkm/oP2VQ==", + "dev": true, + "optional": true + }, + "immer": { + "version": "10.0.4", + "resolved": "https://registry.npmjs.org/immer/-/immer-10.0.4.tgz", + "integrity": "sha512-cuBuGK40P/sk5IzWa9QPUaAdvPHjkk1c+xYsd9oZw+YQQEV+10G0P5uMpGctZZKnyQ+ibRO08bD25nWLmYi2pw==" + }, + "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, + "requires": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + } + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true + }, + "internal-slot": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.7.tgz", + "integrity": "sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g==", + "dev": true, + "requires": { + "es-errors": "^1.3.0", + "hasown": "^2.0.0", + "side-channel": "^1.0.4" + } + }, + "is-array-buffer": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.4.tgz", + "integrity": "sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.2.1" + } + }, + "is-async-function": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.0.0.tgz", + "integrity": "sha512-Y1JXKrfykRJGdlDwdKlLpLyMIiWqWvuSd17TvZk68PLAOGOoF4Xyav1z0Xhoi+gCYjZVeC5SI+hYFOfvXmGRCA==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-bigint": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", + "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==", + "dev": true, + "requires": { + "has-bigints": "^1.0.1" + } + }, + "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, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + }, + "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 + }, + "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, + "requires": { + "hasown": "^2.0.0" + } + }, + "is-data-view": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.1.tgz", + "integrity": "sha512-AHkaJrsUVW6wq6JS8y3JnM/GJF/9cf+k20+iDzlSaJrinEo5+7vRiteOSwBhHRiAyQATN1AmY4hwzxJKPmYf+w==", + "dev": true, + "requires": { + "is-typed-array": "^1.1.13" + } + }, + "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==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true + }, + "is-finalizationregistry": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.0.2.tgz", + "integrity": "sha512-0by5vtUJs8iFQb5TYUHHPudOR+qXYIMKtiUzvLIZITZUjknFmziyBJuLhVRc+Ds0dREFlskDNJKYIdIzu/9pfw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "is-generator-function": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz", + "integrity": "sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-map": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", + "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", + "dev": true + }, + "is-negative-zero": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", + "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", + "dev": true + }, + "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, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "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 + }, + "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, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + }, + "is-set": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", + "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", + "dev": true + }, + "is-shared-array-buffer": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.3.tgz", + "integrity": "sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==", + "dev": true, + "requires": { + "call-bind": "^1.0.7" + } + }, + "is-string": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "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, + "requires": { + "has-symbols": "^1.0.2" + } + }, + "is-typed-array": { + "version": "1.1.13", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.13.tgz", + "integrity": "sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==", + "dev": true, + "requires": { + "which-typed-array": "^1.1.14" + } + }, + "is-weakmap": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", + "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", + "dev": true + }, + "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, + "requires": { + "call-bind": "^1.0.2" + } + }, + "is-weakset": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.3.tgz", + "integrity": "sha512-LvIm3/KWzS9oRFHugab7d+M/GcBXuXX5xZkzPmN+NxihdQlZUQ4dWuSV1xR/sq6upL1TJEDrfBgRepHFdBtSNQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "get-intrinsic": "^1.2.4" + } + }, + "is-what": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/is-what/-/is-what-3.14.1.tgz", + "integrity": "sha512-sNxgpk9793nzSs7bA6JQJGeIuRBQhAaNGG77kzYQgMkrID+lS6SlK07K5LaptscDlSaIgH+GPFzf+d75FVxozA==", + "dev": true + }, + "isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "dev": true + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "iterator.prototype": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/iterator.prototype/-/iterator.prototype-1.1.2.tgz", + "integrity": "sha512-DR33HMMr8EzwuRL8Y9D3u2BMj8+RqSE850jfGu59kS7tbmPLzGkZmVSfyCFSDxuZiEY6Rzt3T2NA/qU+NwVj1w==", + "dev": true, + "requires": { + "define-properties": "^1.2.1", + "get-intrinsic": "^1.2.1", + "has-symbols": "^1.0.3", + "reflect.getprototypeof": "^1.0.4", + "set-function-name": "^2.0.1" + } + }, + "js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" + }, + "js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "requires": { + "argparse": "^2.0.1" + } + }, + "jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "dev": true + }, + "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 + }, + "json-schema-traverse": { + "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 + }, + "json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true + }, + "jsx-ast-utils": { + "version": "3.3.5", + "resolved": "https://registry.npmjs.org/jsx-ast-utils/-/jsx-ast-utils-3.3.5.tgz", + "integrity": "sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==", + "dev": true, + "requires": { + "array-includes": "^3.1.6", + "array.prototype.flat": "^1.3.1", + "object.assign": "^4.1.4", + "object.values": "^1.1.6" + } + }, + "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, + "requires": { + "json-buffer": "3.0.1" + } + }, + "less": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/less/-/less-4.2.0.tgz", + "integrity": "sha512-P3b3HJDBtSzsXUl0im2L7gTO5Ubg8mEN6G8qoTS77iXxXX4Hvu4Qj540PZDvQ8V6DmX6iXo98k7Md0Cm1PrLaA==", + "dev": true, + "requires": { + "copy-anything": "^2.0.1", + "errno": "^0.1.1", + "graceful-fs": "^4.1.2", + "image-size": "~0.5.0", + "make-dir": "^2.1.0", + "mime": "^1.4.1", + "needle": "^3.1.0", + "parse-node-version": "^1.0.1", + "source-map": "~0.6.0", + "tslib": "^2.3.0" + } + }, + "levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + } + }, + "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, + "requires": { + "p-locate": "^5.0.0" + } + }, + "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 + }, + "loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "requires": { + "js-tokens": "^3.0.0 || ^4.0.0" + } + }, + "lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "requires": { + "yallist": "^3.0.2" + } + }, + "make-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", + "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", + "dev": true, + "optional": true, + "requires": { + "pify": "^4.0.1", + "semver": "^5.6.0" + }, + "dependencies": { + "semver": { + "version": "5.7.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz", + "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==", + "dev": true, + "optional": true + } + } + }, + "mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "dev": true, + "optional": true + }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "nanoid": { + "version": "3.3.7", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz", + "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==", + "dev": true + }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "needle": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/needle/-/needle-3.3.1.tgz", + "integrity": "sha512-6k0YULvhpw+RoLNiQCRKOl09Rv1dPLr8hHnVjHqdolKwDrdNyk+Hmrthi4lIGPPz3r39dLx0hsF5s40sZ3Us4Q==", + "dev": true, + "optional": true, + "requires": { + "iconv-lite": "^0.6.3", + "sax": "^1.2.4" + } + }, + "node-releases": { + "version": "2.0.14", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.14.tgz", + "integrity": "sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw==", + "dev": true + }, + "object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==" + }, + "object-inspect": { + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.1.tgz", + "integrity": "sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==", + "dev": true + }, + "object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "dev": true + }, + "object.assign": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.5.tgz", + "integrity": "sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "has-symbols": "^1.0.3", + "object-keys": "^1.1.1" + } + }, + "object.entries": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.8.tgz", + "integrity": "sha512-cmopxi8VwRIAw/fkijJohSfpef5PdN0pMQJN6VC/ZKvn0LIknWD8KtgY6KlQdEc4tIjcQ3HxSMmnvtzIscdaYQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + } + }, + "object.fromentries": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", + "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" + } + }, + "object.hasown": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/object.hasown/-/object.hasown-1.1.4.tgz", + "integrity": "sha512-FZ9LZt9/RHzGySlBARE3VF+gE26TxR38SdmqOqliuTnl9wrKulaQs+4dee1V+Io8VfxqzAfHu6YuRgUy8OHoTg==", + "dev": true, + "requires": { + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" + } + }, + "object.values": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.0.tgz", + "integrity": "sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "optionator": { + "version": "0.9.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", + "integrity": "sha512-JjCoypp+jKn1ttEFExxhetCKeJt9zhAgAve5FXHixTvFDW/5aEktX9bufBKLRRMdU7bNtpLfcGu94B3cdEJgjg==", + "dev": true, + "requires": { + "@aashutoshrathi/word-wrap": "^1.2.3", + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0" + } + }, + "p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "requires": { + "yocto-queue": "^0.1.0" + } + }, + "p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "requires": { + "p-limit": "^3.0.2" + } + }, + "parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "requires": { + "callsites": "^3.0.0" + } + }, + "parse-node-version": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parse-node-version/-/parse-node-version-1.0.1.tgz", + "integrity": "sha512-3YHlOa/JgH6Mnpr05jP9eDG254US9ek25LyIxZlDItp2iJtwyaXQb57lBYLdT3MowkUFYEV2XXNAYIPlESvJlA==", + "dev": true + }, + "path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", + "dev": true + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true, + "optional": true + }, + "possible-typed-array-names": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", + "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", + "dev": true + }, + "postcss": { + "version": "8.4.38", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.38.tgz", + "integrity": "sha512-Wglpdk03BSfXkHoQa3b/oulrotAkwrlLDRSOb9D0bN86FdRyE9lppSp33aHNPgBa0JKCoB+drFLZkQoRRYae5A==", + "dev": true, + "requires": { + "nanoid": "^3.3.7", + "picocolors": "^1.0.0", + "source-map-js": "^1.2.0" + } + }, + "prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true + }, + "prop-types": { + "version": "15.8.1", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz", + "integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==", + "requires": { + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.13.1" + } + }, + "prr": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", + "integrity": "sha512-yPw4Sng1gWghHQWj0B3ZggWUm4qVbPwPFcRG8KyxiU7J2OHFSoEHKS+EZ3fv5l1t9CyCiop6l/ZYeWbrgoQejw==", + "dev": true, + "optional": true + }, + "punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true + }, + "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 + }, + "react": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react/-/react-18.2.0.tgz", + "integrity": "sha512-/3IjMdb2L9QbBdWiW5e3P2/npwMBaU9mHCSCUzNln0ZCYbcfTsGbTJrU/kGemdH2IWmB2ioZ+zkxtmq6g09fGQ==", + "requires": { + "loose-envify": "^1.1.0" + } + }, + "react-dom": { + "version": "18.2.0", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.2.0.tgz", + "integrity": "sha512-6IMTriUmvsjHUjNtEDudZfuDQUoWXVxKHhlEGSk81n4YFS+r/Kl99wXiwlVXtPBtJenozv2P+hxDsw9eA7Xo6g==", + "requires": { + "loose-envify": "^1.1.0", + "scheduler": "^0.23.0" + } + }, + "react-dropzone": { + "version": "14.2.3", + "resolved": "https://registry.npmjs.org/react-dropzone/-/react-dropzone-14.2.3.tgz", + "integrity": "sha512-O3om8I+PkFKbxCukfIR3QAGftYXDZfOE2N1mr/7qebQJHs7U+/RSL/9xomJNpRg9kM5h9soQSdf0Gc7OHF5Fug==", + "requires": { + "attr-accept": "^2.2.2", + "file-selector": "^0.6.0", + "prop-types": "^15.8.1" + } + }, + "react-is": { + "version": "16.13.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", + "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==" + }, + "react-redux": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/react-redux/-/react-redux-9.1.0.tgz", + "integrity": "sha512-6qoDzIO+gbrza8h3hjMA9aq4nwVFCKFtY2iLxCtVT38Swyy2C/dJCGBXHeHLtx6qlg/8qzc2MrhOeduf5K32wQ==", + "requires": { + "@types/use-sync-external-store": "^0.0.3", + "use-sync-external-store": "^1.0.0" + } + }, + "react-refresh": { + "version": "0.14.0", + "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.14.0.tgz", + "integrity": "sha512-wViHqhAd8OHeLS/IRMJjTSDHF3U9eWi62F/MledQGPdJGDhodXJ9PBLNGr6WWL7qlH12Mt3TyTpbS+hGXMjCzQ==", + "dev": true + }, + "react-router": { + "version": "6.22.3", + "resolved": "https://registry.npmjs.org/react-router/-/react-router-6.22.3.tgz", + "integrity": "sha512-dr2eb3Mj5zK2YISHK++foM9w4eBnO23eKnZEDs7c880P6oKbrjz/Svg9+nxqtHQK+oMW4OtjZca0RqPglXxguQ==", + "requires": { + "@remix-run/router": "1.15.3" + } + }, + "react-router-dom": { + "version": "6.22.3", + "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.22.3.tgz", + "integrity": "sha512-7ZILI7HjcE+p31oQvwbokjk6OA/bnFxrhJ19n82Ex9Ph8fNAq+Hm/7KchpMGlTgWhUxRHMMCut+vEtNpWpowKw==", + "requires": { + "@remix-run/router": "1.15.3", + "react-router": "6.22.3" + } + }, + "redux": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/redux/-/redux-5.0.1.tgz", + "integrity": "sha512-M9/ELqF6fy8FwmkpnF0S3YKOqMyoWJ4+CS5Efg2ct3oY9daQvd/Pc71FpGZsVsbl3Cpb+IIcjBDUnnyBdQbq4w==" + }, + "redux-thunk": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/redux-thunk/-/redux-thunk-3.1.0.tgz", + "integrity": "sha512-NW2r5T6ksUKXCabzhL9z+h206HQw/NJkcLm1GPImRQ8IzfXwRGqjVhKJGauHirT0DAuyy6hjdnMZaRoAcy0Klw==", + "requires": {} + }, + "reflect.getprototypeof": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.6.tgz", + "integrity": "sha512-fmfw4XgoDke3kdI6h4xcUz1dG8uaiv5q9gcEwLS4Pnth2kxT+GZ7YehS1JTMGBQmtV7Y4GFGbs2re2NqhdozUg==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.1", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", + "globalthis": "^1.0.3", + "which-builtin-type": "^1.1.3" + } + }, + "regexp.prototype.flags": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.2.tgz", + "integrity": "sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==", + "dev": true, + "requires": { + "call-bind": "^1.0.6", + "define-properties": "^1.2.1", + "es-errors": "^1.3.0", + "set-function-name": "^2.0.1" + } + }, + "reselect": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/reselect/-/reselect-5.1.0.tgz", + "integrity": "sha512-aw7jcGLDpSgNDyWBQLv2cedml85qd95/iszJjN988zX1t7AVRJi19d9kto5+W7oCfQ94gyo40dVbT6g2k4/kXg==" + }, + "resolve": { + "version": "2.0.0-next.5", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-2.0.0-next.5.tgz", + "integrity": "sha512-U7WjGVG9sH8tvjW5SmGbQuui75FiyjAX72HX15DwBBwF9dNiQZRQAg9nnPhYy+TUnE0+VcrttuvNI8oSxZcocA==", + "dev": true, + "requires": { + "is-core-module": "^2.13.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + } + }, + "resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true + }, + "reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true + }, + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "rollup": { + "version": "4.13.2", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.13.2.tgz", + "integrity": "sha512-MIlLgsdMprDBXC+4hsPgzWUasLO9CE4zOkj/u6j+Z6j5A4zRY+CtiXAdJyPtgCsc42g658Aeh1DlrdVEJhsL2g==", + "dev": true, + "requires": { + "@rollup/rollup-android-arm-eabi": "4.13.2", + "@rollup/rollup-android-arm64": "4.13.2", + "@rollup/rollup-darwin-arm64": "4.13.2", + "@rollup/rollup-darwin-x64": "4.13.2", + "@rollup/rollup-linux-arm-gnueabihf": "4.13.2", + "@rollup/rollup-linux-arm64-gnu": "4.13.2", + "@rollup/rollup-linux-arm64-musl": "4.13.2", + "@rollup/rollup-linux-powerpc64le-gnu": "4.13.2", + "@rollup/rollup-linux-riscv64-gnu": "4.13.2", + "@rollup/rollup-linux-s390x-gnu": "4.13.2", + "@rollup/rollup-linux-x64-gnu": "4.13.2", + "@rollup/rollup-linux-x64-musl": "4.13.2", + "@rollup/rollup-win32-arm64-msvc": "4.13.2", + "@rollup/rollup-win32-ia32-msvc": "4.13.2", + "@rollup/rollup-win32-x64-msvc": "4.13.2", + "@types/estree": "1.0.5", + "fsevents": "~2.3.2" + } + }, + "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, + "requires": { + "queue-microtask": "^1.2.2" + } + }, + "safe-array-concat": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.2.tgz", + "integrity": "sha512-vj6RsCsWBCf19jIeHEfkRMw8DPiBb+DMXklQ/1SGDHOMlHdPUkZXFQ2YdplS23zESTijAcurb1aSgJA3AgMu1Q==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "get-intrinsic": "^1.2.4", + "has-symbols": "^1.0.3", + "isarray": "^2.0.5" + } + }, + "safe-regex-test": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.3.tgz", + "integrity": "sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==", + "dev": true, + "requires": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-regex": "^1.1.4" + } + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "dev": true, + "optional": true + }, + "sax": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.3.0.tgz", + "integrity": "sha512-0s+oAmw9zLl1V1cS9BtZN7JAd0cW5e0QH4W3LWEK6a4LaLEA2OTpGYWDY+6XasBLtz6wkm3u1xRw95mRuJ59WA==", + "dev": true, + "optional": true + }, + "scheduler": { + "version": "0.23.0", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.0.tgz", + "integrity": "sha512-CtuThmgHNg7zIZWAXi3AsyIzA3n4xx7aNyjwC2VJldO2LMVDhFK+63xGqq6CsJH4rTAt6/M+N4GhZiDYPx9eUw==", + "requires": { + "loose-envify": "^1.1.0" + } + }, + "semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true + }, + "set-function-length": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", + "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", + "dev": true, + "requires": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.2" + } + }, + "set-function-name": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", + "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", + "dev": true, + "requires": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "functions-have-names": "^1.2.3", + "has-property-descriptors": "^1.0.2" + } + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true + }, + "side-channel": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz", + "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", + "object-inspect": "^1.13.1" + } + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "optional": true + }, + "source-map-js": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.0.tgz", + "integrity": "sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg==", + "dev": true + }, + "string.prototype.matchall": { + "version": "4.0.11", + "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.11.tgz", + "integrity": "sha512-NUdh0aDavY2og7IbBPenWqR9exH+E26Sv8e0/eTe1tltDGZL+GtBkDAnnyBtmekfK6/Dq3MkcGtzXFEd1LQrtg==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.7", + "regexp.prototype.flags": "^1.5.2", + "set-function-name": "^2.0.2", + "side-channel": "^1.0.6" + } + }, + "string.prototype.trim": { + "version": "1.2.9", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.9.tgz", + "integrity": "sha512-klHuCNxiMZ8MlsOihJhJEBJAiMVqU3Z2nEXWfWnIqjN0gEFS9J9+IxKozWWtQGcgoa1WUZzLjKPTr4ZHNFTFxw==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.0", + "es-object-atoms": "^1.0.0" + } + }, + "string.prototype.trimend": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.8.tgz", + "integrity": "sha512-p73uL5VCHCO2BZZ6krwwQE3kCzM7NKmis8S//xEC6fQonchbum4eP6kR4DLEjQFO3Wnj3Fuo8NM0kOSjVdHjZQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + } + }, + "string.prototype.trimstart": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", + "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + } + }, + "strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.1" + } + }, + "strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + }, + "supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true + }, + "tabbable": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/tabbable/-/tabbable-6.2.0.tgz", + "integrity": "sha512-Cat63mxsVJlzYvN51JmVXIgNoUokrIaT2zLclCXjRd8boZ0004U4KCs/sToJ75C6sdlByWxpYnb5Boif1VSFew==" + }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==", + "dev": true + }, + "tslib": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", + "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==" + }, + "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, + "requires": { + "prelude-ls": "^1.2.1" + } + }, + "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 + }, + "typed-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.2.tgz", + "integrity": "sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-typed-array": "^1.1.13" + } + }, + "typed-array-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.1.tgz", + "integrity": "sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + } + }, + "typed-array-byte-offset": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.2.tgz", + "integrity": "sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==", + "dev": true, + "requires": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + } + }, + "typed-array-length": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.6.tgz", + "integrity": "sha512-/OxDN6OtAk5KBpGb28T+HZc2M+ADtvRxXrKKbUwtsLgdoxgX13hyy7ek6bFRl5+aBs2yZzB0c4CnQfAtVypW/g==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13", + "possible-typed-array-names": "^1.0.0" + } + }, + "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, + "requires": { + "call-bind": "^1.0.2", + "has-bigints": "^1.0.2", + "has-symbols": "^1.0.3", + "which-boxed-primitive": "^1.0.2" + } + }, + "update-browserslist-db": { + "version": "1.0.13", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.0.13.tgz", + "integrity": "sha512-xebP81SNcPuNpPP3uzeW1NYXxI3rxyJzF3pD6sH4jE7o/IX+WtSpwnVU+qIsDPyk0d3hmFQ7mjqc6AtV604hbg==", + "dev": true, + "requires": { + "escalade": "^3.1.1", + "picocolors": "^1.0.0" + } + }, + "uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "requires": { + "punycode": "^2.1.0" + } + }, + "use-sync-external-store": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.2.0.tgz", + "integrity": "sha512-eEgnFxGQ1Ife9bzYs6VLi8/4X6CObHMw9Qr9tPY43iKwsPw8xE8+EFsf/2cFZ5S3esXgpWgtSCtLNS41F+sKPA==", + "requires": {} + }, + "vite": { + "version": "5.2.7", + "resolved": "https://registry.npmjs.org/vite/-/vite-5.2.7.tgz", + "integrity": "sha512-k14PWOKLI6pMaSzAuGtT+Cf0YmIx12z9YGon39onaJNy8DLBfBJrzg9FQEmkAM5lpHBZs9wksWAsyF/HkpEwJA==", + "dev": true, + "requires": { + "esbuild": "^0.20.1", + "fsevents": "~2.3.3", + "postcss": "^8.4.38", + "rollup": "^4.13.0" + } + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "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, + "requires": { + "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" + } + }, + "which-builtin-type": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.1.3.tgz", + "integrity": "sha512-YmjsSMDBYsM1CaFiayOVT06+KJeXf0o5M/CAd4o1lTadFAtacTUM49zoYxr/oroopFDfhvN6iEcBxUyc3gvKmw==", + "dev": true, + "requires": { + "function.prototype.name": "^1.1.5", + "has-tostringtag": "^1.0.0", + "is-async-function": "^2.0.0", + "is-date-object": "^1.0.5", + "is-finalizationregistry": "^1.0.2", + "is-generator-function": "^1.0.10", + "is-regex": "^1.1.4", + "is-weakref": "^1.0.2", + "isarray": "^2.0.5", + "which-boxed-primitive": "^1.0.2", + "which-collection": "^1.0.1", + "which-typed-array": "^1.1.9" + } + }, + "which-collection": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", + "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", + "dev": true, + "requires": { + "is-map": "^2.0.3", + "is-set": "^2.0.3", + "is-weakmap": "^2.0.2", + "is-weakset": "^2.0.3" + } + }, + "which-typed-array": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.15.tgz", + "integrity": "sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==", + "dev": true, + "requires": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-tostringtag": "^1.0.2" + } + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + }, + "yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true + } + } +} diff --git a/frontend/package.json b/frontend/package.json index 085e7c1c..0161e7f6 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -1,55 +1,31 @@ { - "name": "ocp_perf_dashboard", - "version": "0.1.0", + "name": "frontend", "private": true, - "dependencies": { - "@patternfly/react-core": "^5.0.1", - "@patternfly/react-styles": "^5.0.1", - "@patternfly/react-table": "^5.0.1", - "@reduxjs/toolkit": "^1.9.5", - "@testing-library/jest-dom": "^5.11.4", - "@testing-library/react": "^11.1.0", - "@testing-library/user-event": "^12.1.10", - "axios": "^1.5.0", - "plotly.js": "^2.26.0", - "prop-types": "^15.8.1", - "react": "^17.0.1", - "react-dom": "^17.0.1", - "react-icons": "^4.2.0", - "react-loading-icons": "^1.1.0", - "react-plotly.js": "^2.6.0", - "react-redux": "^8.1.2", - "react-router-dom": "^5.2.0", - "react-scripts": "^5.0.1", - "react-table-column-resizer": "1.1.7", - "web-vitals": "^0.2.4" - }, + "version": "0.0.0", + "type": "module", "scripts": { - "start": "react-scripts start", - "build": "react-scripts build", - "test": "react-scripts test", - "eject": "react-scripts eject" + "dev": "vite", + "build": "vite build", + "lint": "eslint . --ext js,jsx --report-unused-disable-directives --max-warnings 0", + "preview": "vite preview" }, - "eslintConfig": { - "extends": [ - "react-app", - "react-app/jest" - ] - }, - "browserslist": { - "production": [ - ">0.2%", - "not dead", - "not op_mini all" - ], - "development": [ - "last 1 chrome version", - "last 1 firefox version", - "last 1 safari version" - ] + "dependencies": { + "@patternfly/react-core": "^5.2.3", + "@reduxjs/toolkit": "^2.2.3", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "react-redux": "^9.1.0", + "react-router-dom": "^6.22.3" }, "devDependencies": { - "@babel/plugin-proposal-private-property-in-object": "^7.21.11", - "redux-logger": "^3.0.6" + "@types/react": "^18.2.66", + "@types/react-dom": "^18.2.22", + "@vitejs/plugin-react": "^4.2.1", + "eslint": "^8.57.0", + "eslint-plugin-react": "^7.34.1", + "eslint-plugin-react-hooks": "^4.6.0", + "eslint-plugin-react-refresh": "^0.4.6", + "less": "^4.2.0", + "vite": "^5.2.0" } } diff --git a/frontend/public/index.html b/frontend/public/index.html deleted file mode 100644 index ea0a0892..00000000 --- a/frontend/public/index.html +++ /dev/null @@ -1,43 +0,0 @@ - - - - - - - - - - - - - OpenShift Performance and Scale - CI Dashboard - - - -
- - - diff --git a/frontend/public/manifest.json b/frontend/public/manifest.json deleted file mode 100644 index cd1e2df6..00000000 --- a/frontend/public/manifest.json +++ /dev/null @@ -1,15 +0,0 @@ -{ - "short_name": "OCP PerfScale", - "name": "OpenShift Performance and Scale CI Dashboard", - "icons": [ - { - "src": "logo.png", - "type": "image/png", - "sizes": "192x192" - } - ], - "start_url": ".", - "display": "standalone", - "theme_color": "#000000", - "background_color": "#ffffff" -} diff --git a/frontend/public/robots.txt b/frontend/public/robots.txt deleted file mode 100644 index e9e57dc4..00000000 --- a/frontend/public/robots.txt +++ /dev/null @@ -1,3 +0,0 @@ -# https://www.robotstxt.org/robotstxt.html -User-agent: * -Disallow: diff --git a/frontend/src/App.js b/frontend/src/App.js deleted file mode 100644 index 2715590a..00000000 --- a/frontend/src/App.js +++ /dev/null @@ -1,55 +0,0 @@ -import React, {useEffect} from 'react'; -import '@patternfly/react-core/dist/styles/base.css'; - -import { - Page, - PageSection, - PageSectionVariants, -} from '@patternfly/react-core'; -import {fetchOCPJobsData, fetchCPTJobsData, fetchQuayJobsData} from "./store/Actions/ActionCreator"; -import {useDispatch} from "react-redux"; -import {Route, Switch, BrowserRouter as Router} from "react-router-dom"; -import {NavBar} from "./components/NavBar/NavBar"; -import {HomeView} from "./components/Home/HomeView"; -import {OCPHome} from './components/OCP/OCPHome'; -import {QuayHome} from './components/Quay/QuayHome'; - - -export const App = () => { - const dispatch = useDispatch() - - useEffect(() => { - const fetchData = async () =>{ - await dispatch(fetchOCPJobsData()) - await dispatch(fetchCPTJobsData()) - await dispatch(fetchQuayJobsData()) - } - fetchData() - }, [dispatch]) - - - - - return ( - - } - groupProps={{ - stickyOnBreakpoint: { default: 'top' }, - sticky: 'top' - }} - > - - - - - - - - - - - ); -}; - -export default App diff --git a/frontend/src/App.jsx b/frontend/src/App.jsx new file mode 100644 index 00000000..3d11ca8c --- /dev/null +++ b/frontend/src/App.jsx @@ -0,0 +1,24 @@ +import "./App.less"; + +import * as APP_ROUTES from "./utils/routeConstants"; + +import { BrowserRouter, Route, Routes } from "react-router-dom"; + +import Home from "./components/templates/Home"; +import MainLayout from "./containers/MainLayout"; + +function App() { + return ( +
+ + + }> + } /> + + + +
+ ); +} + +export default App; diff --git a/frontend/src/App.less b/frontend/src/App.less new file mode 100644 index 00000000..ff178b31 --- /dev/null +++ b/frontend/src/App.less @@ -0,0 +1,4 @@ +#root { + +} + diff --git a/frontend/public/assets/images/fa-external-link-alt.svg b/frontend/src/assets/images/fa-external-link-alt.svg similarity index 100% rename from frontend/public/assets/images/fa-external-link-alt.svg rename to frontend/src/assets/images/fa-external-link-alt.svg diff --git a/frontend/public/assets/images/grafana-icon.png b/frontend/src/assets/images/grafana-icon.png similarity index 100% rename from frontend/public/assets/images/grafana-icon.png rename to frontend/src/assets/images/grafana-icon.png diff --git a/frontend/public/assets/images/jenkins-icon.svg b/frontend/src/assets/images/jenkins-icon.svg similarity index 100% rename from frontend/public/assets/images/jenkins-icon.svg rename to frontend/src/assets/images/jenkins-icon.svg diff --git a/frontend/public/assets/images/prow-icon.png b/frontend/src/assets/images/prow-icon.png similarity index 100% rename from frontend/public/assets/images/prow-icon.png rename to frontend/src/assets/images/prow-icon.png diff --git a/frontend/src/assets/logo.png b/frontend/src/assets/logo.png new file mode 100644 index 0000000000000000000000000000000000000000..f5d844bb2f19123cce453a70fef1e357785c28e7 GIT binary patch literal 11105 zcmXYXWmFtp(=8g@-N$fu4el^_f`s6N;O+zn5Zv7f?gR@2cL)Q)-Q67$9PT{tch~AZ zQ>Uu-u3e{Aul_L|p`oULgGq^rfPjFbq$sO}fPlmdzY)-p;U1?m>?!yUij@>p3IU-m z4(rJj6~0Gw)l!f~s2QU^L_k1B&`{NpQ*csUhHxw}@GJq@Ct2B^5fGNaT(1ZS3ygB} z5UwRK=Q0DwGMIgag=G=Uu@2%`1#zxG*q1>Z3lPqE2xJkCg>WwO^Dlrnm%v;rKz2A} z9l*6g&Akj^U!&((1FaI5)v;|7R@1o$zuHuK%L{F?0Us zfpE?LPZgYD0SveQ^K8(wu0uEuz)VXZ&i~f`qlF9muVr`@4!9V&{{YMg-!A;04Pu)4 zkCS5;!VH(R3uJ&N%>UN~yeQ{BhzU;KN5=z~2lwxTnGe7W2M{;|Qx6RvT;4A5Kl=dm z-wKcZuM5sS5ZwNsAIJdL+)KyT3E=Ca|$*4l6;e-%iy1|-rA;A;i)&w;r* zf&7c`S-_hD@{WLb`hk4YVD46c;BOG`1c;{>#QO)t(+&`501A(Sc$$DhJplem5ce;j z&^U-^56s*FFgB;qq9KMQ{_mfW<|v=IPAk61{;h{!MR zaY)__(%^EjpQCLapw#!_!^ki|OuTaxkWdEWQ|)Ls@g~ow1k5LNgVJ_2$(0DZN&sx#nL8^gmVXn?i$3YedCvcGH`f2!%v0yTC_Eb9`fJ3s+yq)= zo~yzhuCA+(y#y@%_6GFtEhNwPw;gI%cHE{FLxVbdW{z7+)2_a}8`p4bJj>E#|MV=f zmQq*sJmZ+s5S6)pm|b?$x!&{e=$6==(EKpE79x;cyn21$7yV$=clGFR9W`oK|Iauf zDogYR0Re!ZBrB!kwRDp6K~-m&Vn~p86q~o#iNb_h-od4dh{8nHAt=yr5Hs0Db1Xp~ zx11ag@01&gi8osj5>_JXyM}#)aVTtN=8~8zFdJ6k$T%ea*D?X;jfPaXlccq(!6CK} zN(jZj+(qBhpZ{L`&mj%-w^m=${#HA3W5bUafV9^Qu~#zEJWn|5m?gD^+8=+GwNRF^ z(uYla_KR}ATUnD?74G;O+B&gesK-W6n4^)lBI?5NBDOV_k)LYoxqQ|5$Y1M^2~hr! zDWK;{axeYuZ9ow_=NR34?bHJ#tYgtDaDvmpX)vx#14Zh4dC(;ulm#P1 zu7viTl4FS;XfdRtO`H$meU=u|mfDg$fjI1nhF-Jwq)|9~p`efff_UkP^PSWNouj%+ zdMaP=pEYodp7yoj)C2OiMCQ521lt9JZPoV;fsCyhjU78AG2USk+^a}&h**dJje&!k zHbF_*TOFS~9N&xxj6a;NmK>;8WVN17k`Pzkrh(R!$TK}WGAf^J;%JK$tfUiMeas=l zmp`tG$_bH@&nEeLux(?&n~uvUVsD9BKx`|)*JKi<#?c7K)j6;t;9lmQ0o-%>SmLGY5cfWwDjKCv0Za&T<8N3X5;MC>CI+UJjJQOJ zj6PI4dMGnWYK)1TJ|Ec9^~k%nnPjfyjWZBF{mVGGA%+ChD*&4Q;#_|N7GcMhmUpih z6>>L%c1>WBZ<2h6Oj)%^3B0-Fd$7#jBwW?n^jwHtdQvMy_Ipr9k0a**k(g6znvekq zH6ic7^yZw17)~G-Mk?lPsj?a;C$VB?^aJIvwy-u!u~|TpyTO(uHZ3~&F;dKv8+r1Z zoJEHS{vb&KZUeN$h5P8zYNVOE6pw?vm9G<&8iW5H#O+w=h|V;EIXnu$Wc_s+dj=CB zR0dH}v{d}bgS_9-n=SAp?RVc+6avuV^qwRk9GGE}SZxy~2pL?T!f2CgMRy}P`|hRc zVyG@LyGSiQBj7j{@#^=8U>qSJUR+8>p8j65LOi}Mkk1#^a3)#EWh|kk$J|BAM!Qm2 zFZos1w0Uac@k6Mn*3DZ2GPI^uS!S}U2hm%xt)bP9)kw_m>n58~F0VRx=w1*{EDL`e zkyiLllV;slQaPZi&Z|gI36i!G!Mn?sf-63 z6G{DPhzi;^rXS`E=Qs}xA^p8m{SH&IY;7gFkcDSc)d*r`q;%{k>?h@C9{eZ={Z%Sn z8B}Aj*w7J0CV|@0aEm=yNtZo-6*J2$&tWhY4CO;)v>Yjs_h?vmVu3k!g4CQV(h0iO zR}oApt^8F);qK6M@)FT1C>oX-IBzeFg0jb-yoFY_TI0$j9QVB)VlrkifGRpHt<5c$ zjf@U$dUFpDBYoM18QM3l^bkTwg7+yYNbCsLcLOP%uZ)>$u#gokOG%dE<85Glw^^d5 z+%!1!O>J7p%#^D^S+S^ucuh&WnO98e+!8tLYMZCcxpr04qKp+6`dN#{>D>>vy^EE% zEL{`PK|2Ls!C}#9hf`{zG$cX+HiHz{ICHSBzjxJkws{C2kR48o(M#X`?SywMl8~XCr?`@o+{@?SVQhH5gfF8_MkYR+qL%u%y43r_K>`jcWc z{DH$(Bz4leRRhg=08voWezXsbt?gI4K7-mE=@b_xeMI9;9Rur|?ftM~g1u0iX5W?g z4{?hr1G~c!_H2JPv+8M~a{(#LjM$!HjvPYDI!E&zKg-B9uJ*?lX)No{0u1V6WEhSJ z_55@B(kPLJ4A2d}O+mX(_HNTwM*4RH_MGnXXkAE|c(_9ouyf|*RV%&s)t7C)^U~)U z`SWf@6rY%hy{h&L9WGgvTaBa8L3RU~*4!-!{I_3pr+>U1=^wCTkCpjM!z`mUM#HcP znC(i47vFN(n`{o)042GxK|nZeO&sR?#+LmJiLVB`RPYQF0DcFe8j_;btvB!WCph4>S0TfikzfA(1|+5 z+M%{IY{$n@lluF3^eT0DX=)2*Xc7Kv1n6#1#>x1vh3ca^j>nINi|A%mhE$`fS~gM9 zB=DZu&JsN-jBC=i0Oyo8l`!0=GJ7;+yt2Z{_V#i+_Kw}=1cH*-N5Kgq^v|u3>4?_2 znjzH0&;d*}qw7^D!^-de0mDUq%|yw=PdOjVC}0zOnCr)XZ+5ZPc)S~gLstiRbvOox zsEVeB@BJy)B{4f0oSb+%_)6Ayu$*Ky8jQ7vfjh^QYkDIpMuC^1;xG{sHk(jA>ZJ^e-XEUXeoGeW8qs4vq zwCT|}PKNtZ5mq*dPgVw~?&mQwvGSJVDyepWGARb7|?XUDjKjnp58$-iY`t zPn+yDnf&&TF+)_9?&w%H;YCMiI_A@J-1C*Ut5|42yx`9eIfiqA6rW9Tl2_zl0_W)H zi~Y91pZa87mw$D_9!p&GYYPdVDC)W66G$?}Y3b-PNRL+3IU`B>&f;Itf0lo)6?|UA z<#OrkzVvEosNY{LW?yAz)JHWReja+X!dgMRNU_C{94MH<;1H{?8vycROCZDu5wFZ z@eQ{3Ok4sta4;t*>1YSvGr;LO%O1y0J>*)1L}46L&&_sFwPrY3KlnJu}G#S{EgNbGxRI88?Cpfzf%2Uk5Y#au8~!b*Gxm0CZ^I;?69LhP(xX4 ziE#-~(nlV=$n?gkk3CP@taP6HXOz$nJ*)mS+Db|Mv->5oAJ?!$_nF}G0;Ask=Hi9h zU}~{s8HLlhhfB~Cen6hS=bGr{@H_tqoO8P+ zk7dv{6r5}DmBeTx@XrcRX7g~m`l@000fven^u7G5O}gqQ`iI28-Z`pVwZJ&GY+7%znO4lIABpBy=gPZuuGK33jX$3GJ zpa4j738PA{#(}*1Y7On@d2o_+#Iq-H?6S$KiPy_hZ31B>FV56$5~(RZtQ@~qG{EiE z?G2;$>ph(f1=!Rn9JjBn#;2^vOYUg>Mf7ZE+8F=R)`_n~sQch)@w!sti@;0ndcs98 zSG%PJfAFezf$-Dt_>kZo7NP{R279YUD?n;b>4Z*#X1&)6rCr@c?unhAMs*x*mNS62 zGe*7W_2X8YdnMv}U(R>q*MGO$uU6+Pg|@9%LjLAOx{53;DBut;33-hI+IKP5?~W%g z1!z>wifIS%TXppMD^qp}3JZ%=-xO9DX&>>Y^WYh1vzPET%)Q-T6Wayiv1CS2q(uBu zI_qe;&2AW9U3VpWyuBB-U0v6IOlk;t^z`?26db*{Z^0kmn18#;_hOKizv-H)NV~9e zym7Iwk4KN(IO*~HW|;R`h=&_3lFk+_{r%MI0|sPo?=UG3pZi{oMZ`_*~_K-3?#)X?&90E*a@i z#RDT9-fT@}6-C|Q=Y`PTPX^te_qYLEGB;-4EQ(NuZ^dtV%-a!3iZd=U zLyDv0lJ2&H-FsCuv~vkR5M8gKUJ{~gKUyD4-`RdCf#t&UR>k8kuqk^>7Rvo zyiEgUECl) z$3D3xOfM224rlr^0HG92^~^7HK~Qe5zteb3FrhlU+>hb@<>Ur04pDx^H)QQcP3U&Y z^$F_J7)-FJf_1}4c<|vAZP6Lv_aloUr~HoMj_ddJ_+z9)Br}0s^Ba%O7?e&ee=Cq#ZAo+t--%KyGL%BPARUxu@=pKrLQ)m7nz-AGG3v0McJ%Ts zvRFe@jaFHe?w`QSM2i{M#-bFw%qaCi5el!?=XrjrnArDgcHG79F2>Ydur+(?Th8x8H2+{8 zhlG{T8K?YN^HqravEZ$<#j7=Ezcjoi&!4l}a-)iJ*2I7ZWr0ANx za~@W0Cb7K50S_V?dPJw3&%@XzR)=yWeMNH&yjt$pvX|t|zA$@2(~rw|g^! zP7-A4aD>P$Bd5L`|Mj3zY*@5Y_{K|5yikBAPF={smVdf&Dd?GZtnbi(VU5Y7nGqQy zhV)5U_f~@3>t7aMR!-7ZMm>#{vLS@eWd$gTv-)H(k(+dD$4`wvp3G!Y5Y*9)=V7xG zXR}^W`w;YHu6!%Vt3(z_6t7k=heIY~s(&}^sj2m|Ru2XuD@%9v2}UN;MF8)^LoU<5 zZc{W*eot5yYifVfQZ~^9@kWg{`}4hduC0LvLqM@Pm6PZbcO(2|ZzlMpBB>q7o|rZ3 z-=_BJ+@kM->P@qVED^#}tT3B+*Uu=RZbvea>}1TI6uf5R@smlW2C~n1(xzRgS)y8% ze77LDD?pX8C5i69sgT_}*|izgw7bCX{W+38UHHO*N&9_<^s6`DpOWzKlJr2)#kXHN z^46ZGBX|10QaAHsTjt7VxZF#tXHgEx@}YVOeP1SlT7(&sSqzWu=q;SnI`HS%_zH*yi2F;1b7CJv0_-uWE&rOIAu~-GkVwJ7m}M9 zbmMGiWM{_OWx(&p9YtfuSuh@ngLlebyr8uPglb zdBuTJNy^%xl%JIUQ8v-novz)|zt`Qvu#U{Qp5J*~PZ^F*elPg#J5=ed(b=CW?SWo!ml;Tf%_OBfG z3Tu^O!RKKmkFQl}(s!bk_UarBYW$Vo_$)Ewb|B5FbL!&0xxHqvv}lzrSWpMg{5QVc z@cCgS>(mnx5mgL%dPb54wbpkj4bqv3COTiCJ;ZzO>YL8R3+MfNMUGeMZKOu=B)Hdh zYK1B4M|+a@@z)cxe)wz(EF0|aAy*3Yv+Xk;{V_e?znUnTN+uZ9to&PgleG%7pmYv1 zoomrHn^modFF81lYd1DzxCTx0WC`00Q>sbx;@^+&1N z$Ld3BoyBW-)nsHX{i()I8U7ijRfuVaS<*g4{a1lj%zGCn%r-lhWuks=xV|22AyrIZ z{*_%OwI!=QTKGuwjL%wR=6-^-tJmdc>cWTj9}S5nles5;1T3i@{-Jh1eH+4wOHO=r zr2zb9oo$uF5^_>B+a9;=-x|Fc8k<%1@s?iVxwQ@Jbgiqp%|v}<$KcM|PHo8#6B3zi zPccUQb6Ot_Dlhx%wKDCz>}UVc_SbkC7M!16^}zi6L)&OZxdBoYA=epxvRXWUAZ+`7 zx6h#ho>rU|Sn{O$E{Wc?moAkt7Kv{d_Qy|X=`^E|*3_8rwZyZi*VtWcU+fsY<4)7N zG`i#0rDg#+?8~I~`}iM^L0cRt&A#rsEtypTU+4B90{*Y|zKGCEZB?~TZ*-qm9mXfKHVYJ8Hl*SGlQ#`}+UKDIJ!trWTq8`=eJV+iRA*(4_DW}e0;;LdG& z{vvHNAY0oOJJ!>F68O;G&t%yGxg9?dOjD)Gr0B_gQs(Ug(34`F6`Bc|R^^g@02XR>G<5_+yIo_BP2a^sV75 z60XXYw%f8beFW}s#Qxuq`)I*T;+DPIF}JU8S1!yPlaf@d4&JDU{u4L2g=0Us`zGdQ zl~2JIsQx*zpZl~xYGi_RlD@}h3poYCrK%F29X5H_L0;Mk7B`bi?Nr}$dSptg9cQ?^ zT=AbaeU9;M-D^Gm`&S9)v>KFr;WhU=`h5<9cVvBfAZ zbV+5dwkiAIR|%H1M5xafEJ~{%88!Ch+GWS8)wl5p3j5Yo^XDa+dp#WP>QyjzJ=6yf zSfUJA4DG5NQ8gb88>xVws+?eTLPDGl)cVygG`0U}0k-7QD=CT2b5$#$g_V_%y_3}^ z!i_`qz*Ow>mOVmPd)^$^S8#cY-Llxo4~=uKo;(<<*fe5$yP~;+y*NRZ2$gu?%3N>ZMW%$!V?}{V1{_g>HkFJgDHXT4 z<72-jN2$aIjV}?ygQ`(06(peW;xk9?QoIZ+!8V1O!_a;f*Y!_PFvCp7)=i!Yvy+Gv zv_7-tr{^b)s~DQwcPJgI8CP_e9%APBEnQ;qIhDe`b=L8vYcNPbH)qqr|pp@NT7Y=1zae7<$CxkkIGAQTjwv%a!l`F#gM1?Ds9X!P8Z@p>ZLiDJ6s#%m!N5}a3Pif$#*Q}(U#d2U(0wWB-c-~UB^3aA3LB=?!8aJich~{z95f3p z?$Y;D{947iPOeXFWcHuxo;M15!VRjdU_mXuzu(VN%)FvUA!B5i!lLWKS&D){DbT|% zD$3HfUIn6c5=*=|WxOs4%o*8X>FX4y5GC!gZYay$Z?Jgrodn@OvVtVc?7i+#puyx$ zpYvlxrVDoGWKa~l;&941^`R{EaxD~pJAX)VB^j6tG-R}aVT5zjvV=zk@py$gMU*ik zUtB5BKm`#WL5@jK2?YR*9KA6~usGBaUrKgY zPHQNO2_P5zojH|-l8_p~oY2lBeKwDALSG5{i2hU6hAEto)gtP5S_mVFkQYl_;2X>m z9V`lrMS<%1M}#1DG!<4<%_<7%(5(f9xn$Scdr90=Vp#&8h69G)B+Qjv`rv4P%94c; zoKUw1itwO76?&%VP^LPh=CF7*38yO^5g9&N0)Kt(&Dps5wl~roF=HUg0%D9B$U40} zEX?V<3?Z&)3bUo`MWAx1QYBJYGX12y9C4%CUARZQz%G+vd*5eQ58SwL;1}5?;ybZ0 zlq1T+bkH7^sg#vs77s#R9kI%9Z;^tXy2w!}O}6d!R4~ zBfOZucqWo|nc%8~`rGO}QiNAjDwCLkY8rSJ>pm>Qa-77bfLM`yjT^H}oc>{(d~%FQMZ|Ch|S4;|7E5iuqP$={$g)%$xL;gvaUeA+js|)FwF!yCP;xml29% zIgun7*SaRR%Ko0MI^)on^gC!psvI+~7Rhe6gK)4r_OzyD>G~C)d5us8g>mfqkLenw z)Atnx=xwVemK>AZ1HK*uM%DRWG@ZBl^6U*rb^#yNHPk}dFH7B`K>Am88L+?1@2Fheh7$E-+`IHGA()&HeG6#%bikvT}o`qq3#j=7u5PUIXTz`An9 z>ti6xZuC%k&@e<6UzE1J{q9jbP>clvbwS5tcj5 zGb|*I=Nl7mO$Q==uxLZ+moAEmB03q%=*h=7;wA`p2UIo(A{^U1V1*HL{*elZni>=( zlVRG25iB^CM`Xo31{j+1buq7q-cTdeOHVZ~oqIBuO1~EBeVtySnecNJ z5haUrh1XjwNV(AEyIsbcriyj)HzCZR$cy7w6sL?oTm5PJ7Hm3>eRHJM(M;w-aSF2d z4}0U|JW)i(xuE3u0yT9|X1w&q7n*=1Rh-9z$6@CxRq%{);UKu`oky{(^Y)kciCYSG zwbZgJF9(BRjVwm`f%nC-5$-1br0scO5BCEj<&1ay8T-T8ENBli?!0=lHbvyYzvu>C zQWxP?;l26gX`R6qY7!qhzE%?|#a!W)Zgdd}Q`||Ac^m zhV$RI0AreCM8qmv1{*Z#kx5oWkhd8bU!EQm!aKi*66-XRGAnUJSSiz`e#{1F4Gd@9 zhN#oAFAe@oyEppK@=jAT^S9WXTxe9%ebu#v&EX4mi3EgLbG5jP@t{3;H2B+@S+kIS zygR9>JMSZX+myX9%qOU-zcc4LVW;YP8hf#d_xbZcUSVbWaMGFcDIK8Ln&H-qvB;hQ z@F9EDGH1v?Al{$Rb>zFyeEL={KpndU4LeO^W{G52M7)NN*PUJ3y6vO4`UGMDKm31t zkNsWj5k{B=O&+ydN(OF#lM$QQSC9IJl60(hp`JQ_91#^?!=&|xO8uyYt;{~2oS5_y z340c!ZzWHtopzUasu;aO&!V;Thr;@KihZ5;mOuhxQXXiBR&OCE(lt?7=fdj`*>+)2 z*=rH)9$t&Dt5gK6)kBN+fk-$_f(7gNe`C#(sTf#bZ}S0;-?;k&mvxLKcLht_*u4$~2Iy>ZQw-)BOLD_(EcV`*F1 zFD7L7_`v#BExQ`0-*i-?tVSx%7(~s;+P)SJAD_2i`#m#m<{jg<^7N_A>4%0!jv8is zX(@t@vQCCuQ!;s?o9%Upc|Mo?5^N0h9oL76j3jDLKIQ87Qu1)k&KeR#r4Jb>wWU}H zdFVa~a``@d)BTB8ZjqCcI9c}SrvI@u(9dGYq=}?e;oZh>R!m`cE};>Vxyy!V2PJA} zU9!%kwlV4q5?V7SSrjy?sXpF7EwOhC-zeOyBX@|?Bs=7FjhwKM|OUk>l0W~Vw+ z0Q6X(ee+~mzCu&kusxqcgdoJ;9y?@#Xl@h6{g*Rbm zCJH}a7=cj?L4)?$H3!k((p6YBuY_~zsMgq3fOg2GqROlIo|Ak-&OSv{ktEJ;J1!!{ zW4M!K4MJ+0Zc&I$Rp?eKOr_>*U*x_A0t#G(bCLpR+3*v+k%U&w1J}eQ2J!(gYJN0H z)UL)8R3#$EC|PjNv`_-2;M*%PY)a=2{XUW(sn)UHo35J~R1W@V4p zh}6Zs03nc^^@s^)MqCK~Ly-RX>2nDLUs6b_6)F|))o->?F!X`Iz;GgoL_O}TNlkzn zJET^{ET^V(+w@tCMw){}rODrjn@m=!eHY(Hif%|tK@>RWf+1G@Y^kvz63&ZDEkl#} z4^l^K7X2sX?~=RaHlUxZk`>LhQ9rM{$@l56a_-((y#Dqdd|kqROWTSrGrJ&n?jEBR z(`qlm(-7#bMZ{9Yw-35gOZkrKxi`}+oO)HLzNW1FG(jv_GroIVtwP8k#f0P|^>@NK zjj1Wp7feJ3a}`Yy#q_+WTP5h5UsxV~IczJ^+r_<$Z~dV&+4eKGb34$4Ip^D(<)(%g z8%9UL{)ia)uIkj)a~1E<_fjF`AmQdGp(DGD6^Bz$rmoWVTOriy$ zr#c@M`kP~lA=Bl;i!@xSEnC2TSxD8H%Qpd=TV3j^pf>Z_0hXgCH)7(<{FV|mB)c&b zmOrVWxHeH9bbqIJL}ZJn|L|9)%|3qYtR||^{c}M0F4Y_B(Yt>k@{=x&Vu2J$ { - - const dispatch = useDispatch() - const history = useHistory(); - const {search} = useLocation(); - const searchParams = new URLSearchParams(search); - - const cptJobs = useSelector(state => state.cptJobs) - - const topHeadersData = [ - {loading: cptJobs.waitForUpdate, title: 'No. Jobs', value: cptJobs.total}, - {loading: cptJobs.waitForUpdate, title: 'Success', value: cptJobs.success}, - {loading: cptJobs.waitForUpdate, title: 'Failure', value: cptJobs.failure}, - {loading: cptJobs.waitForUpdate, title: 'Others', value: cptJobs.others}, - ] - - const [ciSystem, setCiSystem] = useState(searchParams.get("ciSystem") || cptJobs.selectedCiSystem) - const [product, setProduct] = useState(searchParams.get("product") || cptJobs.selectedProduct) - const [testName, setTestName] = useState(searchParams.get("testName") || cptJobs.selectedTestName) - const [jobStatus, setJobStatus] = useState(searchParams.get("jobStatus") || cptJobs.selectedJobStatus) - const [releaseStream, setReleaseStream] = useState(searchParams.get("releaseStream") || cptJobs.selectedReleaseStream) - const [startDate, setStartDate] = useState(searchParams.get("startDate") || cptJobs.startDate) || "" - const [endDate, setEndDate] = useState(searchParams.get("endDate") || cptJobs.endDate) || "" - - // Changing the URL Route Path - useEffect(() => { - let buildParams = '' - if(ciSystem !== '') buildParams += `&ciSystem=${ciSystem}` - if(product !== '') buildParams += `&product=${product}` - if(testName !== '') buildParams += `&testName=${testName}` - if(jobStatus !== '') buildParams += `&jobStatus=${jobStatus}` - if(releaseStream !== '') buildParams += `&releaseStream=${releaseStream}` - if(startDate !== '') buildParams += `&startDate=${startDate}` - if(endDate !== '') buildParams += `&endDate=${endDate}` - history.push(`/home?${buildParams.substring(1)}`, { replace: true }); - }, [history, ciSystem, product, testName, jobStatus, releaseStream, startDate, endDate]) - - // Update the Components Data - useEffect( ()=>{ - dispatch(updateCPTDataFilter({ciSystem, product, testName, jobStatus, releaseStream})) - }, [ ciSystem, product, testName, jobStatus, releaseStream, dispatch ]) - - // Fetch the data when startDate/ endDate changes - useEffect(() => { - if(startDate || endDate){ - dispatch(fetchCPTJobsData(startDate, endDate)) - } - }, [startDate, endDate, dispatch]) - - useEffect(() => { - if(endDate === ""){ - setEndDate(cptJobs.endDate) - setStartDate(cptJobs.startDate) - } - }, [endDate, setEndDate, setStartDate, cptJobs.startDate, cptJobs.endDate]) - - const sidebarComponents = [ - {name: "DateComponent", options: [], onChange: null, value: null, startDate: startDate, endDate: endDate, setStartDate: setStartDate, setEndDate: setEndDate}, - {name: "CiSystem", options: cptJobs.ciSystems, onChange: setCiSystem, value: ciSystem}, - {name: "Product", options: cptJobs.products, onChange: setProduct, value: product}, - {name: "Test Name", options: cptJobs.testNames, onChange: setTestName, value: testName}, - {name: "Job Status", options: cptJobs.jobStatuses, onChange: setJobStatus, value: jobStatus}, - {name: "Release Stream", options: cptJobs.releaseStreams, onChange: setReleaseStream, value: releaseStream}, - ] - - return ( - - ); -} diff --git a/frontend/src/components/NavBar/NavBar.js b/frontend/src/components/NavBar/NavBar.js deleted file mode 100644 index be562597..00000000 --- a/frontend/src/components/NavBar/NavBar.js +++ /dev/null @@ -1,29 +0,0 @@ -import { - Brand, - Icon, - Masthead, - MastheadBrand, - MastheadContent, - MastheadMain, -} from "@patternfly/react-core"; - -import "../css/NavBar.css" -import {ToolBar} from "./ToolBar"; -import React from "react"; - -export const NavBar = ({fixed=false}) => { - return <> - - - - - - - - - - - - - -} diff --git a/frontend/src/components/NavBar/ToolBar.js b/frontend/src/components/NavBar/ToolBar.js deleted file mode 100644 index 3179a916..00000000 --- a/frontend/src/components/NavBar/ToolBar.js +++ /dev/null @@ -1,81 +0,0 @@ -import { Toolbar, ToolbarGroup, ToolbarItem } from "@patternfly/react-core"; -import React, { useEffect, useState } from "react"; -import { Text4 } from "../PatternflyComponents/Text/Text"; -import { useSelector } from "react-redux"; -import { Link } from "react-router-dom"; - -export const ToolBar = () => { - const [active, setActive] = useState("/home"); - - useEffect(() => { - const path = window.location.href.split("/"); - let pathName = path[path.length - 1]; - if (pathName === "") pathName = "/home"; - setActive(pathName); - }, []); - - const linkStyle = (value) => { - const styles = {}; - if (active === value) { - styles["color"] = "white"; - styles["borderBottom"] = "2px solid red"; - } - return styles; - }; - - // Selectors for both ocpJobs and quayJobs - const ocpJobResults = useSelector((state) => state.ocpJobs); - const quayJobResults = useSelector((state) => state.quayJobs); - - const NavItems = ( - <> - - setActive("/home")} - /> - - - setActive("/ocp")} - /> - - {/* New Quay ToolbarItem */} - - setActive("/quay")} - /> - - - ); - - return ( - <> - - - - - - {NavItems} - - {/* Displaying the updated time for OCP or Quay based on the active path */} - - - - - - ); -}; diff --git a/frontend/src/components/OCP/BenchmarkResults.js b/frontend/src/components/OCP/BenchmarkResults.js deleted file mode 100644 index 4d7da958..00000000 --- a/frontend/src/components/OCP/BenchmarkResults.js +++ /dev/null @@ -1,27 +0,0 @@ -import {Grid, GridItem} from "@patternfly/react-core"; -import InstallCard from "./InstallCard"; -import React from "react"; -import {DisplayGraph} from "./DisplayGraph"; - - -export const BenchmarkResults = ({dataset, isExpanded}) => { - return ( - <> { - ( isExpanded && - - - - - - - - - - ) || <>NO Data - } - - ) -} diff --git a/frontend/src/components/OCP/DisplayGraph.js b/frontend/src/components/OCP/DisplayGraph.js deleted file mode 100644 index a9e18244..00000000 --- a/frontend/src/components/OCP/DisplayGraph.js +++ /dev/null @@ -1,38 +0,0 @@ -import {PlotlyView} from "../ReactGraphs/plotly/PlotlyView"; -import React, {useEffect} from "react"; -import {useDispatch, useSelector} from "react-redux"; -import CardView from "../PatternflyComponents/Card/CardView"; -import {Text6} from "../PatternflyComponents/Text/Text"; -import {SplitView} from "../PatternflyComponents/Split/SplitView"; -import {Spinner} from "@patternfly/react-core"; -import {fetchGraphData} from "../../store/Actions/ActionCreator"; - - -export const DisplayGraph = ({uuid, benchmark}) => { - - const [isExpanded, setExpanded] = React.useState(true) - const onExpand = () => setExpanded(!isExpanded) - - const dispatch = useDispatch() - const job_results = useSelector(state => state.graph) - const graphData = job_results.uuid_results[uuid] - - useEffect(() => { - dispatch(fetchGraphData(uuid)) - }, [dispatch, uuid]) - - const getGraphBody = () => { - return (job_results.graphError && ) || - (graphData && ) || - , ]} /> - } - - return <> - } - body={ getGraphBody() } - isExpanded={isExpanded} - expandView={true} - onExpand={onExpand} - /> - -} diff --git a/frontend/src/components/OCP/InstallCard.js b/frontend/src/components/OCP/InstallCard.js deleted file mode 100644 index 402ec84e..00000000 --- a/frontend/src/components/OCP/InstallCard.js +++ /dev/null @@ -1,97 +0,0 @@ -import React from 'react'; -import { Card, CardTitle, CardBody, CardFooter, CardHeader, CardExpandableContent } from '@patternfly/react-core'; -import { Grid, GridItem } from '@patternfly/react-core'; -import { Spinner } from '@patternfly/react-core'; -import { formatTime } from '../../helpers/Formatters' -import { FaCheck, FaExclamationCircle, FaExclamationTriangle } from "react-icons/fa"; -import { DisplayGrafana } from "../commons/DisplayGrafana"; - - -export default function InstallCard(props) { - let config = props.data - const [isExpanded, setExpanded] = React.useState([true, null]) - - - const onExpand = () => { - setExpanded(!isExpanded) - }; - - const icons = { - "failed": <>, - "success": <>, - "upstream_failed": <>, - - } - - if (config) { - return ( - - - Configs - - - - - - - Cluster Metadata -
    -
  • Release Binary: {config.cluster_version && config.cluster_version || config.ocpVersion}
  • -
  • Cluster Name: {config.cluster_name && config.cluster_name || config.clusterName}
  • -
  • Cluster Type: {config.cluster_type && config.cluster_type || config.clusterType}
  • -
  • Network Type: {config.network_type && config.network_type || config.networkType}
  • -
  • Benchmark Status: {icons[config.job_status && config.job_status || config.jobStatus] || config.job_status && config.job_status || config.jobStatus}
  • -
  • Duration: {formatTime(config.job_duration && config.job_duration || config.jobDuration)}
  • -
  • Test ID: {config.uuid}
  • -
  • Has IPSEC: {config.ipsec}
  • -
  • FIPS Enabled: {config.fips}
  • -
  • Is Encrypted: {config.encrypted}
  • -
  • Encryption Type: {config.encryptionType}
  • -
  • Control Plane Access: {config.publish}
  • -
  • Control Plane Arch.: {config.controlPlaneArch}
  • -
  • Compute Nodes Arch.: {config.computeArch}
  • -
-
- - - Node Types -
    -
  • Master: {config.master_type && config.master_type || config.masterNodesType}
  • -
  • Worker: {config.worker_type && config.worker_type || config.workerNodesType}
  • -
  • Workload: {config.workload_type && config.workload_type || config.benchmark}
  • -
  • Infra: {config.infra_type && config.infra_type || config.infraNodesType}
  • -
-
- Node Counts -
    -
  • Master: {config.master_count && config.master_count || config.masterNodesCount}
  • -
  • Worker: {config.worker_count && config.worker_count || config.workerNodesCount}
  • -
  • Infra: {config.infra_count && config.infra_count || config.infraNodesCount}
  • -
  • Total Nodes: {config.workload_count && config.workload_count || config.totalNodesCount}
  • -
-
-
-
-
- - -
) - } else { - return ( - - Install Configuration -
Awaiting Results
- -
- ) - } - -} diff --git a/frontend/src/components/OCP/OCPHome.js b/frontend/src/components/OCP/OCPHome.js deleted file mode 100644 index dadca488..00000000 --- a/frontend/src/components/OCP/OCPHome.js +++ /dev/null @@ -1,119 +0,0 @@ -import '../css/PlatformView.css'; -import "@patternfly/react-core/dist/styles/base.css"; -import React, {useEffect, useState} from 'react'; - -import {HomeLayout} from "../templates/HomeLayout"; -import {useDispatch, useSelector} from "react-redux"; -import {useHistory, useLocation} from "react-router-dom"; -import {updateOCPDataFilter} from "../../store/reducers/OCPJobsReducer"; -import {fetchOCPJobsData} from "../../store/Actions/ActionCreator"; -import {BenchmarkResults} from "./BenchmarkResults"; - - -export function OCPHome() { - - const dispatch = useDispatch() - const {search} = useLocation(); - const searchParams = new URLSearchParams(search); - const history = useHistory(); - const ocpJobs = useSelector(state => state.ocpJobs) - - const topHeadersData = [ - {loading: ocpJobs.waitForUpdate, title: 'No. Jobs', value: ocpJobs.total}, - {loading: ocpJobs.waitForUpdate, title: 'Success', value: ocpJobs.success}, - {loading: ocpJobs.waitForUpdate, title: 'Failure', value: ocpJobs.failure}, - {loading: ocpJobs.waitForUpdate, title: 'Others', value: ocpJobs.others}, - {loading: ocpJobs.waitForUpdate, title: 'Duration Running', value: ocpJobs.duration} - ] - - const [ciSystem, setCiSystem] = useState(searchParams.get("ciSystem") || ocpJobs.selectedCiSystem) - const [platform, setPlatform] = useState(searchParams.get("platform") || ocpJobs.selectedPlatform) - const [benchmark, setBenchmark] = useState(searchParams.get("benchmark") || ocpJobs.selectedBenchmark) - const [workerCount, setWorkerCount] = useState(searchParams.get("workerCount") || ocpJobs.selectedWorkerCount) - const [networkType, setNetworkType] = useState(searchParams.get("networkType") || ocpJobs.selectedNetworkType) - const [version, setVersion] = useState(searchParams.get("version") || ocpJobs.selectedVersion) - const [jobType, setJobType] = useState(searchParams.get("jobType") || ocpJobs.selectedJobType) - const [isRehearse, setRehearse] = useState(searchParams.get("isRehearse") || ocpJobs.selectedRehearse) - const [ipsec, setIpsec] = useState(searchParams.get("ipsec") || ocpJobs.selectedIpsec) - const [fips, setFips] = useState(searchParams.get("fips") || ocpJobs.selectedFips) - const [encrypted, setEncrypted] = useState(searchParams.get("encrypted") || ocpJobs.selectedEncrypted) - const [encryptionType, setEncryptionType] = useState(searchParams.get("encryptionType") || ocpJobs.selectedEncryptionType) - const [publish, setPublish] = useState(searchParams.get("publish") || ocpJobs.selectedPublish) - const [computeArch, setComputeArch] = useState(searchParams.get("computeArch") || ocpJobs.selectedComputeArch) - const [controlPlaneArch, setControlPlaneArch] = useState(searchParams.get("controlPlaneArch") || ocpJobs.selectedControlPlaneArch) - const [startDate, setStartDate] = useState(searchParams.get("startDate") || ocpJobs.startDate) || "" - const [endDate, setEndDate] = useState(searchParams.get("endDate") || ocpJobs.endDate) || "" - - const sidebarComponents = [ - {name: "DateComponent", options: [], onChange: null, value: null, startDate: startDate, endDate: endDate, setStartDate: setStartDate, setEndDate: setEndDate}, - {name: "Ci System", onChange: setCiSystem, value: ciSystem, options: ocpJobs.ciSystems}, - {name: "Platform", onChange: setPlatform, value: platform, options: ocpJobs.platforms}, - {name: "Benchmark", onChange: setBenchmark, value: benchmark, options:ocpJobs.benchmarks }, - {name: "Workers Count", onChange: setWorkerCount, value: workerCount, options:ocpJobs.workers }, - {name: "Network Type", onChange: setNetworkType, value: networkType, options:ocpJobs.networkTypes }, - {name: "Versions", onChange: setVersion, value: version, options: ocpJobs.versions}, - {name: "Job Type", onChange: setJobType, value: jobType, options: ocpJobs.jobTypes}, - {name: "Rehearse", onChange: setRehearse, value: isRehearse, options: ocpJobs.rehearses}, - {name: "Has IPSEC", onChange: setIpsec, value: ipsec, options: ocpJobs.allIpsec}, - {name: "FIPS Enabled", onChange: setFips, value: fips, options: ocpJobs.allFips}, - {name: "Is Encrypted", onChange: setEncrypted, value: encrypted, options: ocpJobs.allEncrypted}, - {name: "Encryption Type", onChange: setEncryptionType, value: encryptionType, options: ocpJobs.encryptionTypes}, - {name: "Control Plane Access", onChange: setPublish, value: publish, options: ocpJobs.allPublish}, - {name: "Compute Architecture", onChange: setComputeArch, value: computeArch, options: ocpJobs.computeArchs}, - {name: "Control Plane Architecture", onChange: setControlPlaneArch, value: controlPlaneArch, options: ocpJobs.controlPlaneArchs}, - ] - - useEffect(() => { - let buildParams = '' - if(ciSystem !== '') buildParams += `&ciSystem=${ciSystem}` - if(platform !== '') buildParams += `&platform=${platform}` - if(benchmark !== '') buildParams += `&benchmark=${benchmark}` - if(version !== '') buildParams += `&version=${version}` - if(workerCount !== '') buildParams += `&workerCount=${workerCount}` - if(networkType !== '') buildParams += `&networkType=${networkType}` - if(jobType !== '') buildParams += `&jobType=${jobType}` - if(isRehearse !== '') buildParams += `&isRehearse=${isRehearse}` - if(ipsec !== '') buildParams += `&ipsec=${ipsec}` - if(fips !== '') buildParams += `&fips=${fips}` - if(encrypted !== '') buildParams += `&encrypted=${encrypted}` - if(encryptionType !== '') buildParams += `&encryptionType=${encryptionType}` - if(publish !== '') buildParams += `&publish=${publish}` - if(computeArch !== '') buildParams += `&computeArch=${computeArch}` - if(controlPlaneArch !== '') buildParams += `&controlPlaneArch=${controlPlaneArch}` - if(startDate !== '') buildParams += `&startDate=${startDate}` - if(endDate !== '') buildParams += `&endDate=${endDate}` - history.push(`/ocp?${buildParams.substring(1)}`, { replace: true }); - - }, [history, ciSystem, platform, benchmark, version, workerCount, networkType, - jobType, isRehearse, ipsec, fips, encrypted, encryptionType, publish, - computeArch, controlPlaneArch, startDate, endDate]) - - useEffect( ()=>{ - dispatch(updateOCPDataFilter({ciSystem, platform, benchmark, version, workerCount, networkType, jobType, isRehearse, - ipsec, fips, encrypted, encryptionType, publish, computeArch, controlPlaneArch})) - }, [ ciSystem, platform, benchmark, version, workerCount, networkType, jobType, isRehearse, - ipsec, fips, encrypted, encryptionType, publish, computeArch, controlPlaneArch, dispatch ]) - - useEffect(() => { - if(startDate || endDate){ - dispatch(fetchOCPJobsData(startDate, endDate)) - } - }, [startDate, endDate, dispatch]) - - useEffect(() => { - if(endDate === ""){ - setEndDate(ocpJobs.endDate) - setStartDate(ocpJobs.startDate) - } - }, [endDate, setEndDate, setStartDate, ocpJobs.startDate, ocpJobs.endDate]) - - - return ( - - ); -} diff --git a/frontend/src/components/PatternflyComponents/Accordion/AccordionView.js b/frontend/src/components/PatternflyComponents/Accordion/AccordionView.js deleted file mode 100644 index 4cd745ae..00000000 --- a/frontend/src/components/PatternflyComponents/Accordion/AccordionView.js +++ /dev/null @@ -1,16 +0,0 @@ -import {Accordion, AccordionContent, AccordionItem} from "@patternfly/react-core"; -import React from "react"; - - -export const AccordionView = ({childComponent}) =>{ - return ( - - - - {childComponent} - - - - ) -} -export default AccordionView diff --git a/frontend/src/components/PatternflyComponents/Card/CardView.js b/frontend/src/components/PatternflyComponents/Card/CardView.js deleted file mode 100644 index 0abb03f8..00000000 --- a/frontend/src/components/PatternflyComponents/Card/CardView.js +++ /dev/null @@ -1,39 +0,0 @@ -import { - Card, - CardBody, CardExpandableContent, - CardHeader, - CardTitle -} from "@patternfly/react-core"; -import React from "react"; -import {Puff} from "react-loading-icons"; - - -export const CardView = ({header, body, initialState=false, isSelectableRaised=true, - isExpanded=false, onExpand, expandView=false}) => { - - const cardBody = - {!initialState && body } - {initialState && <> Loading....} - - - const toggleButtonProps = expandView ? - { - id: 'toggle-button', - 'aria-label': 'Details', - 'aria-labelledby': 'titleId toggle-button', - 'aria-expanded': isExpanded - } : {} - - return ( - - - {header && {header}} - - { - expandView && || cardBody - } - - ) -} - -export default CardView; diff --git a/frontend/src/components/PatternflyComponents/Date/DatePickerView.js b/frontend/src/components/PatternflyComponents/Date/DatePickerView.js deleted file mode 100644 index 5bcb59e1..00000000 --- a/frontend/src/components/PatternflyComponents/Date/DatePickerView.js +++ /dev/null @@ -1,12 +0,0 @@ -import {DatePicker} from "@patternfly/react-core"; -import React from "react"; - - -export const DatePickerView = ({defaultDate, setDate, ...props}) => { - return document.body} - value={defaultDate} - onChange={(_e, value)=>setDate(value)} - placement="right" - /> -} diff --git a/frontend/src/components/PatternflyComponents/Form/FormSelectView.js b/frontend/src/components/PatternflyComponents/Form/FormSelectView.js deleted file mode 100644 index 4a46c35a..00000000 --- a/frontend/src/components/PatternflyComponents/Form/FormSelectView.js +++ /dev/null @@ -1,12 +0,0 @@ -import {FormSelect, FormSelectOption} from "@patternfly/react-core"; - - -export const FormSelectView = ({onChange, options, selectedValue}) =>{ - return <> - onChange(e.target.value)} aria-label="FormSelect Input" ouiaId="BasicFormSelect"> - {options && options.map((value, index) => ( - - ))} - - -} diff --git a/frontend/src/components/PatternflyComponents/List/ListView.js b/frontend/src/components/PatternflyComponents/List/ListView.js deleted file mode 100644 index 27f80041..00000000 --- a/frontend/src/components/PatternflyComponents/List/ListView.js +++ /dev/null @@ -1,14 +0,0 @@ -import {List, ListItem} from "@patternfly/react-core"; - -export const ListView = ({list_view = [], isPlain=false}) => { - return ( - - {list_view && - list_view.map( - (value, index) => {value} - ) - } - - ) -} -export default ListView; diff --git a/frontend/src/components/PatternflyComponents/Spinners/PuffLoad.js b/frontend/src/components/PatternflyComponents/Spinners/PuffLoad.js deleted file mode 100644 index ad30a385..00000000 --- a/frontend/src/components/PatternflyComponents/Spinners/PuffLoad.js +++ /dev/null @@ -1,10 +0,0 @@ -import {Puff} from "react-loading-icons"; -import React from "react"; - - -export const PuffLoad = ({height=12, stroke= "#0000FF", strokeOpacity=.125} ) => { - return -} - -export default PuffLoad; - diff --git a/frontend/src/components/PatternflyComponents/Split/SplitView.js b/frontend/src/components/PatternflyComponents/Split/SplitView.js deleted file mode 100644 index 212b6525..00000000 --- a/frontend/src/components/PatternflyComponents/Split/SplitView.js +++ /dev/null @@ -1,17 +0,0 @@ -import {Split, SplitItem} from "@patternfly/react-core"; - - -export const SplitView = ({splitValues}) => { - return ( - <> - - { - splitValues && - splitValues.map( (value, index) => - - ) - } - - - ) -} diff --git a/frontend/src/components/PatternflyComponents/Table/TableView.js b/frontend/src/components/PatternflyComponents/Table/TableView.js deleted file mode 100644 index 8e208964..00000000 --- a/frontend/src/components/PatternflyComponents/Table/TableView.js +++ /dev/null @@ -1,143 +0,0 @@ - -import {Table, Thead, Tr, Th, Tbody, Td, ExpandableRowContent} from '@patternfly/react-table'; -import {Puff} from "react-loading-icons"; -import React, {useEffect, useState} from "react"; -import {getFormattedDate} from "../../../helpers/Formatters"; - - - -export const TableView = ({columns , rows = [], initialState = true, stickyHeader=false, - addExpandableRows = false, expandableComponent: ExpandableComponent, ...props }) => { - - const [expand, setExpand] = useState( - Object.assign({}, ...Object.keys(rows).map( (items, idx) => ({[idx]: false}))) - ) - - useEffect(()=>{ - if(rows) - setExpand( - Object.assign({}, ...Object.keys(rows).map( (items, idx) => ({[idx]: false}))) - ) - }, [rows]) - // - const handleToggle = (event, index) => { - setExpand({ - ...expand, [index]: !expand[index] - }) - } - - - const NonExpandableTableRows = () => { - return - { rows && - rows.map( (item, index)=> - {item.tableRows.map( (value, idx) => { columns[idx] === "Start Date" || columns[idx] === "End Date" ? getFormattedDate(value) : value } )} - - ) - } - - } - - const ExpandableRows = () => { - return( - <> - { rows && rows.map( (item, index)=> - - - {item.tableRows.map( (value, idx) => { columns[idx] === "Start Date" || columns[idx] === "End Date" ? getFormattedDate(value) : value } )} - - - - - { ExpandableComponent && - - } - - - - ) - } - - ) - } - - // index of the currently active column - const [activeSortIndex, setActiveSortIndex] = React.useState(-1); - // sort direction of the currently active column - const [activeSortDirection, setActiveSortDirection] = React.useState('none'); - - const onSort = (event, index, direction) => { - setActiveSortIndex(index); - setActiveSortDirection(direction); - // sorts the rows - rows = rows.sort((a, b) => { - a = a.tableRows - b = b.tableRows - if (typeof a[index].props.value === 'number') { - // numeric sort - if (direction === 'asc') { - return a[index].props.value - b[index].props.value; - } - return b[index].props.value - a[index].props.value; - } else if (new Date(a[index].props.value).toString() !== "Invalid Date") { - if (direction === 'asc') { - return new Date(a[index].props.value) - new Date(b[index].props.value) - } - return new Date(b[index].props.value) - new Date(a[index].props.value) - } else { - // string sort - if (direction === 'asc') { - return a[index].props.value.localeCompare(b[index].props.value); - } - return b[index].props.value.localeCompare(a[index].props.value); - } - }); - }; - - const onSortStatus = (event, index, direction) => { - setActiveSortIndex(index); - setActiveSortDirection(direction); - // sorts the rows - rows = rows.sort((a, b) => { - a = a.tableRows - b = b.tableRows - // string sort - if (direction === 'asc') { - return a[index].props.children.localeCompare(b[index].props.children); - } - return b[index].props.children.localeCompare(a[index].props.children); - }); - }; - - const getSortParams = (columnIndex, item) => ({ - sortBy: { - index: activeSortIndex, - direction: activeSortDirection - }, - onSort: ((item === 'Status') ? onSortStatus : onSort), - columnIndex - }); - - return <> - {initialState && <> Loading....} - {!initialState && - - - - {addExpandableRows && )} - - - { - (addExpandableRows && ExpandableRows()) || NonExpandableTableRows() - } -
} - { columns && columns.map( (item, index) => - {item}
} - -} diff --git a/frontend/src/components/PatternflyComponents/Text/Text.js b/frontend/src/components/PatternflyComponents/Text/Text.js deleted file mode 100644 index 1c3509a6..00000000 --- a/frontend/src/components/PatternflyComponents/Text/Text.js +++ /dev/null @@ -1,27 +0,0 @@ -import { - Text, TextVariants -} from "@patternfly/react-core"; - -export const Text1 = ({value, ...props}) => { - return <> -} - -export const Text2 = ({value, ...props}) => { - return <> -} - -export const Text3 = ({value, ...props}) => { - return <> -} - -export const Text4 = ({value, ...props}) => { - return <> -} - -export const Text5 = ({value, ...props}) => { - return <> -} - -export const Text6 = ({value, ...props}) => { - return <> -} diff --git a/frontend/src/components/Quay/BenchmarkResults.js b/frontend/src/components/Quay/BenchmarkResults.js deleted file mode 100644 index 82c49a76..00000000 --- a/frontend/src/components/Quay/BenchmarkResults.js +++ /dev/null @@ -1,36 +0,0 @@ -import {Grid, GridItem} from "@patternfly/react-core"; -import InstallCard from "./InstallCard"; -import React from "react"; -import {DisplayGraph} from "./DisplayGraph"; - - -export const BenchmarkResults = ({dataset, isExpanded}) => { - return ( - <> { - ( isExpanded && - - - - - - - - - - - - - - - ) || <>NO Data - } - - ) -} diff --git a/frontend/src/components/Quay/DisplayGraph.js b/frontend/src/components/Quay/DisplayGraph.js deleted file mode 100644 index c342f984..00000000 --- a/frontend/src/components/Quay/DisplayGraph.js +++ /dev/null @@ -1,40 +0,0 @@ -import {PlotlyView} from "../ReactGraphs/plotly/PlotlyView"; -import React, {useEffect} from "react"; -import {useDispatch, useSelector} from "react-redux"; -import CardView from "../PatternflyComponents/Card/CardView"; -import {Text6} from "../PatternflyComponents/Text/Text"; -import {SplitView} from "../PatternflyComponents/Split/SplitView"; -import {Spinner} from "@patternfly/react-core"; -import {fetchQuayGraphData} from "../../store/Actions/ActionCreator"; - - -export const DisplayGraph = ({uuid, resultKey, heading}) => { - - const [isExpanded, setExpanded] = React.useState(true) - const onExpand = () => setExpanded(!isExpanded) - - const dispatch = useDispatch() - const job_results = useSelector(state => state.quayGraph) - const graphData = job_results.uuid_results[uuid] - console.log(graphData) - - useEffect(() => { - dispatch(fetchQuayGraphData(uuid)) - }, [dispatch, uuid]) - - const getGraphBody = () => { - const dataForKey = graphData && graphData[resultKey]; - return (job_results.graphError && ) || - (dataForKey && ) || - , ]} /> - } - - return <> - } - body={ getGraphBody() } - isExpanded={isExpanded} - expandView={true} - onExpand={onExpand} - /> - -} diff --git a/frontend/src/components/Quay/InstallCard.js b/frontend/src/components/Quay/InstallCard.js deleted file mode 100644 index 2ca34b4b..00000000 --- a/frontend/src/components/Quay/InstallCard.js +++ /dev/null @@ -1,90 +0,0 @@ -import React from 'react'; -import { Card, CardTitle, CardBody, CardFooter, CardHeader, CardExpandableContent } from '@patternfly/react-core'; -import { Grid, GridItem } from '@patternfly/react-core'; -import { Spinner } from '@patternfly/react-core'; -import { formatTime } from '../../helpers/Formatters' -import { FaCheck, FaExclamationCircle, FaExclamationTriangle } from "react-icons/fa"; -import { DisplayGrafana } from "../commons/DisplayGrafana"; - - -export default function InstallCard(props) { - let config = props.data - const [isExpanded, setExpanded] = React.useState([true, null]) - - - const onExpand = () => { - setExpanded(!isExpanded) - }; - - const icons = { - "failed": <>, - "success": <>, - "upstream_failed": <>, - - } - - if (config) { - return ( - - - Configs - - - - - - - Cluster Metadata -
    -
  • Release Binary: {config.cluster_version && config.cluster_version || config.ocpVersion}
  • -
  • Cluster Name: {config.cluster_name && config.cluster_name || config.clusterName}
  • -
  • Cluster Type: {config.cluster_type && config.cluster_type || config.clusterType}
  • -
  • Network Type: {config.network_type && config.network_type || config.networkType}
  • -
  • Benchmark Status: {icons[config.job_status && config.job_status || config.jobStatus] || config.job_status && config.job_status || config.jobStatus}
  • -
  • Duration: {formatTime(config.job_duration && config.job_duration || config.jobDuration)}
  • -
  • Test ID: {config.uuid}
  • -
-
- - - Node Types -
    -
  • Master: {config.master_type && config.master_type || config.masterNodesType}
  • -
  • Worker: {config.worker_type && config.worker_type || config.workerNodesType}
  • -
  • Workload: {config.workload_type && config.workload_type || config.benchmark}
  • -
  • Infra: {config.infra_type && config.infra_type || config.infraNodesType}
  • -
-
- Node Counts -
    -
  • Master: {config.master_count && config.master_count || config.masterNodesCount}
  • -
  • Worker: {config.worker_count && config.worker_count || config.workerNodesCount}
  • -
  • Infra: {config.infra_count && config.infra_count || config.infraNodesCount}
  • -
  • Total Nodes: {config.workload_count && config.workload_count || config.totalNodesCount}
  • -
-
-
-
-
- - -
) - } else { - return ( - - Install Configuration -
Awaiting Results
- -
- ) - } - -} diff --git a/frontend/src/components/Quay/QuayHome.js b/frontend/src/components/Quay/QuayHome.js deleted file mode 100644 index fef13fb3..00000000 --- a/frontend/src/components/Quay/QuayHome.js +++ /dev/null @@ -1,93 +0,0 @@ -import '../css/PlatformView.css'; -import "@patternfly/react-core/dist/styles/base.css"; -import React, {useEffect, useState} from 'react'; - -import {HomeLayout} from "../templates/HomeLayout"; -import {useDispatch, useSelector} from "react-redux"; -import {useHistory, useLocation} from "react-router-dom"; -import {updateQuayDataFilter} from "../../store/reducers/QuayJobsReducer"; -import {fetchQuayJobsData} from "../../store/Actions/ActionCreator"; -import {BenchmarkResults} from "./BenchmarkResults"; - - -export function QuayHome() { - - const dispatch = useDispatch() - const {search} = useLocation(); - const searchParams = new URLSearchParams(search); - const history = useHistory(); - const quayJobs = useSelector(state => state.quayJobs) - - const topHeadersData = [ - {loading: quayJobs.waitForUpdate, title: 'No. Jobs', value: quayJobs.total}, - {loading: quayJobs.waitForUpdate, title: 'Success', value: quayJobs.success}, - {loading: quayJobs.waitForUpdate, title: 'Failure', value: quayJobs.failure}, - {loading: quayJobs.waitForUpdate, title: 'Others', value: quayJobs.others}, - {loading: quayJobs.waitForUpdate, title: 'Duration Running', value: quayJobs.duration} - ] - - const [ciSystem, setCiSystem] = useState(searchParams.get("ciSystem") || quayJobs.selectedCiSystem) - const [platform, setPlatform] = useState(searchParams.get("platform") || quayJobs.selectedPlatform) - const [benchmark, setBenchmark] = useState(searchParams.get("benchmark") || quayJobs.selectedBenchmark) - const [workerCount, setWorkerCount] = useState(searchParams.get("workerCount") || quayJobs.selectedWorkerCount) - const [releaseStream, setReleaseStream] = useState(searchParams.get("releaseStream") || quayJobs.selectedReleaseStream) - const [hitSize, setHitSize] = useState(searchParams.get("hitSize") || quayJobs.selectedHitSize) - const [concurrency, setConcurrency] = useState(searchParams.get('concurrency') || quayJobs.selectedConcurrency) - const [imagePushPulls, setImagePushPulls] = useState(searchParams.get('imagePushPulls') || quayJobs.selectedImagePushPulls) - const [startDate, setStartDate] = useState(searchParams.get("startDate") || quayJobs.startDate) || "" - const [endDate, setEndDate] = useState(searchParams.get("endDate") || quayJobs.endDate) || "" - - const sidebarComponents = [ - {name: "DateComponent", options: [], onChange: null, value: null, startDate: startDate, endDate: endDate, setStartDate: setStartDate, setEndDate: setEndDate}, - {name: "Ci System", onChange: setCiSystem, value: ciSystem, options: quayJobs.ciSystems}, - {name: "Platform", onChange: setPlatform, value: platform, options: quayJobs.platforms}, - {name: "Benchmark", onChange: setBenchmark, value: benchmark, options:quayJobs.benchmarks }, - {name: "Release Streams", onChange: setReleaseStream, value: releaseStream, options: quayJobs.releaseStreams}, - {name: "Workers Count", onChange: setWorkerCount, value: workerCount, options:quayJobs.workers }, - {name: "Hit Size", onChange: setHitSize, value: hitSize, options:quayJobs.hitSizes }, - {name: "Concurrency", onChange: setConcurrency, value: concurrency, options:quayJobs.concurrencies }, - {name: "Image Push/Pulls", onChange: setImagePushPulls, value: imagePushPulls, options:quayJobs.imagePushPulls } - ] - - useEffect(() => { - let buildParams = '' - if(ciSystem !== '') buildParams += `&ciSystem=${ciSystem}` - if(platform !== '') buildParams += `&platform=${platform}` - if(benchmark !== '') buildParams += `&benchmark=${benchmark}` - if(releaseStream !== '') buildParams += `&releaseStream=${releaseStream}` - if(workerCount !== '') buildParams += `&workerCount=${workerCount}` - if(hitSize !== '') buildParams += `&hitSize=${hitSize}` - if(concurrency !== '') buildParams += `&concurrency=${concurrency}` - if(imagePushPulls !== '') buildParams += `&imagePushPulls=${imagePushPulls}` - if(startDate !== '') buildParams += `&startDate=${startDate}` - if(endDate !== '') buildParams += `&endDate=${endDate}` - history.push(`/quay?${buildParams.substring(1)}`, { replace: true }); - - }, [history, ciSystem, platform, benchmark, releaseStream, workerCount, hitSize, concurrency, imagePushPulls, startDate, endDate]) - - useEffect( ()=>{ - dispatch(updateQuayDataFilter({ciSystem, platform, benchmark, releaseStream, workerCount, hitSize, concurrency, imagePushPulls})) - }, [ ciSystem, platform, benchmark, releaseStream, workerCount, hitSize, concurrency, imagePushPulls, dispatch ]) - - useEffect(() => { - if(startDate || endDate){ - dispatch(fetchQuayJobsData(startDate, endDate)) - } - }, [startDate, endDate, dispatch]) - - useEffect(() => { - if(endDate === ""){ - setEndDate(quayJobs.endDate) - setStartDate(quayJobs.startDate) - } - }, [endDate, setEndDate, setStartDate, quayJobs.startDate, quayJobs.endDate]) - - - return ( - - ); -} diff --git a/frontend/src/components/ReactGraphs/plotly/PlotlyView.js b/frontend/src/components/ReactGraphs/plotly/PlotlyView.js deleted file mode 100644 index e3f51e52..00000000 --- a/frontend/src/components/ReactGraphs/plotly/PlotlyView.js +++ /dev/null @@ -1,11 +0,0 @@ -import Plotly from "react-plotly.js"; -import React from "react"; - - -export const PlotlyView = ({data, width = "100%", height = "100%"}) => { - return -} diff --git a/frontend/src/components/css/NavBar.css b/frontend/src/components/css/NavBar.css deleted file mode 100644 index e923acbb..00000000 --- a/frontend/src/components/css/NavBar.css +++ /dev/null @@ -1,9 +0,0 @@ - -.fixed-navbar { - position: fixed; - top: 0; - left: 0; - right: 0; - box-shadow: 0px 4px 6px rgba(0, 0, 0, 0.1); /* Add a shadow if desired */ - z-index: 1000; /* Ensure it appears above other content */ -} diff --git a/frontend/src/components/css/PlatformView.css b/frontend/src/components/css/PlatformView.css deleted file mode 100644 index afed3c56..00000000 --- a/frontend/src/components/css/PlatformView.css +++ /dev/null @@ -1,12 +0,0 @@ -.PlatformView { - text-align: center; - background-color: white !important; -} - -.PlatformView-link { - color: #61dafb; -} - -.list-item-key { - font-weight: bold; -} diff --git a/frontend/src/components/molecules/SideMenuOptions/index.jsx b/frontend/src/components/molecules/SideMenuOptions/index.jsx new file mode 100644 index 00000000..ac6693f4 --- /dev/null +++ b/frontend/src/components/molecules/SideMenuOptions/index.jsx @@ -0,0 +1,51 @@ +import { Nav, NavItem, NavList } from "@patternfly/react-core"; + +import React from "react"; + +const sideMenuOptions = [ + { + id: 0, + key: "home", + displayName: "Home", + }, + { + id: 1, + key: "ocp", + displayName: "OCP", + }, + { + id: 2, + key: "quay", + displayName: "Quay", + }, +]; + +const MenuOptions = () => { + const [activeItem, setActiveItem] = React.useState(0); + const onSelect = (_event, itemId) => { + const item = itemId; + setActiveItem(item); + }; + + return ( + <> + + + ); +}; + +export default MenuOptions; diff --git a/frontend/src/components/organisms/Header/index.jsx b/frontend/src/components/organisms/Header/index.jsx new file mode 100644 index 00000000..3474c352 --- /dev/null +++ b/frontend/src/components/organisms/Header/index.jsx @@ -0,0 +1,54 @@ +import "./index.less"; + +import { + Brand, + Masthead, + MastheadBrand, + MastheadContent, + MastheadMain, + MastheadToggle, + PageToggleButton, +} from "@patternfly/react-core"; +import { useDispatch, useSelector } from "react-redux"; + +import { BarsIcon } from "@patternfly/react-icons"; +import logo from "@/assets//logo.png"; + +// import { toggleSideMenu } from "../../../reducer/HeaderReducer"; + +const Header = () => { + const isSideMenuOpen = true; + + // const dispatch = useDispatch(); + const onSidebarToggle = () => { + // dispatch(toggleSideMenu()); + }; + return ( + + + + + + + + + + + + + CPT Dashboard +
+
Last Updated: 21/02/2024 13:26:30
+
+
+
+ ); +}; + +export default Header; diff --git a/frontend/src/components/organisms/Header/index.less b/frontend/src/components/organisms/Header/index.less new file mode 100644 index 00000000..296358b3 --- /dev/null +++ b/frontend/src/components/organisms/Header/index.less @@ -0,0 +1,10 @@ +.header-logo { + width: 40px; + height: 40px; +} + +.last-updated-box { + display: flex; + justify-content: flex-end; + width: 90%; +} \ No newline at end of file diff --git a/frontend/src/components/organisms/LoadingComponent/index.jsx b/frontend/src/components/organisms/LoadingComponent/index.jsx new file mode 100644 index 00000000..388361f7 --- /dev/null +++ b/frontend/src/components/organisms/LoadingComponent/index.jsx @@ -0,0 +1,21 @@ +import "./index.less"; + +import { Spinner } from "@patternfly/react-core"; + +const LoadingComponent = ({ children }) => { + // const parentClass = isLoading ? "main-with-spinner" : ""; + const isLoading = false; + const parentClass = ""; + return ( +
+ {children} + {isLoading && ( +
+ +
+ )} +
+ ); +}; + +export default LoadingComponent; diff --git a/frontend/src/components/organisms/LoadingComponent/index.less b/frontend/src/components/organisms/LoadingComponent/index.less new file mode 100644 index 00000000..9aec0168 --- /dev/null +++ b/frontend/src/components/organisms/LoadingComponent/index.less @@ -0,0 +1,20 @@ +.main-page-container { + position: relative; + height: 100%; + .spinner-container { + width: 15%; + height: 15%; + position: absolute; + top: 50%; + left: 45%; + z-index: 9999; + svg { + width: 45px; + height: 45px; + } + } +} +.main-with-spinner { + pointer-events: none; + height: 100%; +} diff --git a/frontend/src/components/organisms/Pagination/index.jsx b/frontend/src/components/organisms/Pagination/index.jsx new file mode 100644 index 00000000..0f34dd3b --- /dev/null +++ b/frontend/src/components/organisms/Pagination/index.jsx @@ -0,0 +1,18 @@ +import { Pagination, PaginationVariant } from "@patternfly/react-core"; + +const RenderPagination = (props) => { + return ( + + ); +}; + +export default RenderPagination; diff --git a/frontend/src/components/organisms/SideMenu/index.jsx b/frontend/src/components/organisms/SideMenu/index.jsx new file mode 100644 index 00000000..2857205a --- /dev/null +++ b/frontend/src/components/organisms/SideMenu/index.jsx @@ -0,0 +1,17 @@ +import { PageSidebar, PageSidebarBody } from "@patternfly/react-core"; + +import MenuOptions from "@/components/molecules/SideMenuOptions/index"; +import { useSelector } from "react-redux"; + +const SideMenu = () => { + const isSideMenuOpen = true; + return ( + + + + + + ); +}; + +export default SideMenu; diff --git a/frontend/src/components/organisms/SideMenu/index.less b/frontend/src/components/organisms/SideMenu/index.less new file mode 100644 index 00000000..e69de29b diff --git a/frontend/src/components/organisms/ToastComponent/index.jsx b/frontend/src/components/organisms/ToastComponent/index.jsx new file mode 100644 index 00000000..a894dc24 --- /dev/null +++ b/frontend/src/components/organisms/ToastComponent/index.jsx @@ -0,0 +1,5 @@ +const ToastComponent = () => { + return <>hello; +}; + +export default ToastComponent; diff --git a/frontend/src/components/templates/Home/index.jsx b/frontend/src/components/templates/Home/index.jsx new file mode 100644 index 00000000..0b0a9e6a --- /dev/null +++ b/frontend/src/components/templates/Home/index.jsx @@ -0,0 +1,5 @@ +const Home = () => { + return <>Test; +}; + +export default Home; diff --git a/frontend/src/components/templates/HomeLayout/DisplayTableDataLayout.js b/frontend/src/components/templates/HomeLayout/DisplayTableDataLayout.js deleted file mode 100644 index e78eff51..00000000 --- a/frontend/src/components/templates/HomeLayout/DisplayTableDataLayout.js +++ /dev/null @@ -1,45 +0,0 @@ - -import {Badge, Page} from "@patternfly/react-core"; -import React from "react"; -import {InnerScrollContainer} from "@patternfly/react-table"; -import {TableView} from "../../PatternflyComponents/Table/TableView"; -import {Text6} from "../../PatternflyComponents/Text/Text"; - - -export const DisplayTableDataLayout = ({initialState, tableMetaData, tableData, addExpandableRows= false, - expandableComponent}) => { - - const getRows = () => { - console.log(tableData) - return tableData && tableData.map( items => { - const tableRows = tableMetaData.map( metadata => { - console.log(items[metadata.value]) - if(metadata.name === 'Status') - if(items[metadata.value].toLowerCase() === "success") - return - else if(items[metadata.value].toLowerCase() === "failure") - return - else - return - if(metadata.name === 'Build URL') - return Job - - - return - }) - return {dataset: items, tableRows} - }) - } - const tableColumns = tableMetaData.map(item => item.name) - return - } - /> - -} diff --git a/frontend/src/components/templates/HomeLayout/SidebarLayout.js b/frontend/src/components/templates/HomeLayout/SidebarLayout.js deleted file mode 100644 index 8a6d6913..00000000 --- a/frontend/src/components/templates/HomeLayout/SidebarLayout.js +++ /dev/null @@ -1,71 +0,0 @@ -import {Split, Stack, StackItem} from "@patternfly/react-core"; -import {Text4} from "../../PatternflyComponents/Text/Text"; -import {DatePickerView} from "../../PatternflyComponents/Date/DatePickerView"; -import CardView from "../../PatternflyComponents/Card/CardView"; -import {FormSelectView} from "../../PatternflyComponents/Form/FormSelectView"; -import PropTypes from "prop-types"; - - -export const SidebarLayout = ({sidebarComponents}) => { - - const DisplayDate = ({startDate, endDate, setStartDate, setEndDate}) => { - console.log(startDate) - const dateView = (name, dateValue, onChange) => { - return - }/> - }/> - - } - - return <> - - { dateView("Start Date", startDate, setStartDate) } - { dateView("End Date", endDate, setEndDate) } - - - } - - return <> - - { - - sidebarComponents && sidebarComponents.map( (component, index) => { - if(component.name === "DateComponent") { - return } - /> - } /> - } - else{ - return } - body={} - />} /> - } - }) - } - - -} - - -SidebarLayout.propTypes = { - sidebarComponents: PropTypes.arrayOf( - PropTypes.shape({ - name: PropTypes.string.isRequired, - options: PropTypes.array.isRequired, - onChange: PropTypes.func, - value: PropTypes.string, - startDate: PropTypes.string, - endDate: PropTypes.string, - setStartDate: PropTypes.func, - setEndDate: PropTypes.func - }) - ).isRequired -} diff --git a/frontend/src/components/templates/HomeLayout/TopLayout.js b/frontend/src/components/templates/HomeLayout/TopLayout.js deleted file mode 100644 index a7c8235a..00000000 --- a/frontend/src/components/templates/HomeLayout/TopLayout.js +++ /dev/null @@ -1,47 +0,0 @@ -import { - Grid, - GridItem -} from "@patternfly/react-core"; -import React from "react"; -import PuffLoad from "../../PatternflyComponents/Spinners/PuffLoad"; -import {Text4} from "../../PatternflyComponents/Text/Text"; -import CardView from "../../PatternflyComponents/Card/CardView"; -import PropTypes from "prop-types"; -import { formatTime } from "../../../helpers/Formatters"; - - -export const TopLayout = ({topHeadersData = []}) => { - - const displayText = (title, textValue, loading) => { - const load = loading? : "" - let value = {textValue} {load} - if (title === 'Duration Running') { - value = {formatTime(textValue)} {load} - } - return - } - - return <> - - { - topHeadersData && topHeadersData.map( (item, index) => { - return } - body={displayText(item.title, item.value, item.loading )} />} /> - } ) - } - - -} - - -TopLayout.propTypes = { - topHeadersData: PropTypes.arrayOf( - PropTypes.shape({ - loading: PropTypes.bool.isRequired, - title: PropTypes.string.isRequired, - value: PropTypes.bool.isRequired, - }) - ).isRequired -} diff --git a/frontend/src/components/templates/HomeLayout/index.js b/frontend/src/components/templates/HomeLayout/index.js deleted file mode 100644 index 17ec3377..00000000 --- a/frontend/src/components/templates/HomeLayout/index.js +++ /dev/null @@ -1,45 +0,0 @@ -import '../../css/PlatformView.css'; -import "@patternfly/react-core/dist/styles/base.css"; -import React from 'react'; -import { - Page, - PageSection, - PageSectionVariants, - PageSidebar, - PageSidebarBody -} from "@patternfly/react-core"; -import {TopLayout} from "./TopLayout"; -import {SidebarLayout} from "./SidebarLayout"; -import {DisplayTableDataLayout} from "./DisplayTableDataLayout"; - -export function HomeLayout({topHeadersData, sidebarComponents, initialState, tableData, tableMetaData, - addExpandableRows=false, expandableComponent}) { - - const sidebar = ( - - - - ); - - return ( - } - isBreadcrumbGrouped - groupProps={{ - stickyOnBreakpoint: { default: 'top' }, - sticky: 'top' - }} - > - - - - - ); -} diff --git a/frontend/src/containers/MainLayout/index.jsx b/frontend/src/containers/MainLayout/index.jsx new file mode 100644 index 00000000..7beaadad --- /dev/null +++ b/frontend/src/containers/MainLayout/index.jsx @@ -0,0 +1,26 @@ +import "./index.less"; + +import { Outlet, useNavigate } from "react-router-dom"; +import { Page, PageSection, PageSectionVariants } from "@patternfly/react-core"; + +import Header from "@/components/organisms/Header"; +import LoadingComponent from "@/components/organisms/LoadingComponent"; +import SideMenu from "@/components/organisms/SideMenu"; + +const MainLayout = () => { + const navigate = useNavigate(); + + return ( + <> + } sidebar={}> + + + + + + + + ); +}; + +export default MainLayout; diff --git a/frontend/src/containers/MainLayout/index.less b/frontend/src/containers/MainLayout/index.less new file mode 100644 index 00000000..e69de29b diff --git a/frontend/src/store/reducers/Utils.js b/frontend/src/helpers/Utils.js similarity index 100% rename from frontend/src/store/reducers/Utils.js rename to frontend/src/helpers/Utils.js diff --git a/frontend/src/index.css b/frontend/src/index.css index 361d3739..95f84114 100644 --- a/frontend/src/index.css +++ b/frontend/src/index.css @@ -1,31 +1,13 @@ -body { - margin: 0; - font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen', - 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', - sans-serif; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; -} - -code { - font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', - monospace; +:root { + font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif; + line-height: 1.5; } -html { - background-color: #282c34; +body { + margin: 0; + padding: 0; + min-height: 100vh; + font-size: 16px; } -.OCPPerformance-header { - background-color: #282c34 !important; - min-height: 10vh; - padding-bottom: 20px; - color: white; -} -.OCPPerformance-logo { - height: 40vmin; - pointer-events: none; - max-height: 100px; - margin: 10px; -} diff --git a/frontend/src/index.js b/frontend/src/index.js deleted file mode 100644 index 27f0b133..00000000 --- a/frontend/src/index.js +++ /dev/null @@ -1,19 +0,0 @@ -import React from 'react'; -import ReactDOM from 'react-dom'; -import './index.css'; -import './common/fonts.css' -import { Provider } from 'react-redux' - -import App from './App'; -import store from "./store/store" - - -ReactDOM.render( - - - - - - , - document.getElementById('root') -); diff --git a/frontend/src/main.jsx b/frontend/src/main.jsx new file mode 100644 index 00000000..4226ca31 --- /dev/null +++ b/frontend/src/main.jsx @@ -0,0 +1,16 @@ +import "@patternfly/react-core/dist/styles/base.css"; +import "./index.css"; + +import App from "./App.jsx"; +import { Provider } from "react-redux"; +import React from "react"; +import ReactDOM from "react-dom/client"; +import store from "./store/store"; + +ReactDOM.createRoot(document.getElementById("root")).render( + + + + + +); diff --git a/frontend/src/reportWebVitals.js b/frontend/src/reportWebVitals.js deleted file mode 100644 index 5253d3ad..00000000 --- a/frontend/src/reportWebVitals.js +++ /dev/null @@ -1,13 +0,0 @@ -const reportWebVitals = onPerfEntry => { - if (onPerfEntry && onPerfEntry instanceof Function) { - import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => { - getCLS(onPerfEntry); - getFID(onPerfEntry); - getFCP(onPerfEntry); - getLCP(onPerfEntry); - getTTFB(onPerfEntry); - }); - } -}; - -export default reportWebVitals; diff --git a/frontend/src/setupTests.js b/frontend/src/setupTests.js deleted file mode 100644 index 8f2609b7..00000000 --- a/frontend/src/setupTests.js +++ /dev/null @@ -1,5 +0,0 @@ -// jest-dom adds custom jest matchers for asserting on DOM nodes. -// allows you to do things like: -// expect(element).toHaveTextContent(/react/i) -// learn more: https://github.com/testing-library/jest-dom -import '@testing-library/jest-dom'; diff --git a/frontend/src/store/Actions/ActionCreator.js b/frontend/src/store/Actions/ActionCreator.js deleted file mode 100644 index a6112761..00000000 --- a/frontend/src/store/Actions/ActionCreator.js +++ /dev/null @@ -1,280 +0,0 @@ - -import {BASE_URL, OCP_GRAPH_API_V1, OCP_JOBS_API_V1, CPT_JOBS_API_V1, QUAY_JOBS_API_V1, QUAY_GRAPH_API_V1} from "../Shared"; -import axios from "axios"; -import { - errorOCPCall, - getOCPJobsData, - setWaitForOCPUpdate, - updateOCPMetaData -} from "../reducers/OCPJobsReducer"; -import { - errorCPTCall, - getCPTJobsData, - setWaitForCPTUpdate, - updateCPTMetaData -} from "../reducers/CPTJobsReducer"; -import { - errorQuayCall, - getQuayJobsData, - setWaitForQuayUpdate, - updateQuayMetaData -} from "../reducers/QuayJobsReducer"; -import {getUuidResults, setGraphError} from "../reducers/GraphReducer"; -import {getQuayUuidResults, setQuayGraphError} from "../reducers/QuayGraphReducer"; - -export const fetchAPI = async (url, requestOptions = {}) => { - const response = await axios(url, requestOptions) - return response.data -} - -export const fetchGraphData = (uuid) => async dispatch =>{ - try { - let buildUrl = `${BASE_URL}${OCP_GRAPH_API_V1}/${uuid}` - const api_data = await fetchAPI(buildUrl) - if(api_data) dispatch(getUuidResults({ [uuid]: api_data })) - } - catch (error){ - if (axios.isAxiosError(error)) { - console.error('Axios Error:', error); - console.error('Request:', error.request); - console.error('Response:', error.response); - } else { - console.error('Axios Error:', error); - dispatch(setGraphError({error: error.response.data.details})) - } - } -} - -export const fetchQuayGraphData = (uuid) => async dispatch =>{ - try { - let buildUrl = `${BASE_URL}${QUAY_GRAPH_API_V1}/${uuid}` - const api_data = await fetchAPI(buildUrl) - if(api_data) dispatch(getQuayUuidResults({ [uuid]: api_data })) - } - catch (error){ - if (axios.isAxiosError(error)) { - console.error('Axios Error:', error); - console.error('Request:', error.request); - console.error('Response:', error.response); - } else { - console.error('Axios Error:', error); - dispatch(setQuayGraphError({error: error.response.data.details})) - } - } -} - -export const fetchOCPJobsData = (startDate = '', endDate='') => async dispatch => { - let buildUrl = `${BASE_URL}${OCP_JOBS_API_V1}` - dispatch(setWaitForOCPUpdate({waitForUpdate:true})) - if(startDate !== '' && endDate !== '') { - buildUrl += `?start_date=${startDate}&end_date=${endDate}` - } - try{ - let api_data = await fetchAPI(buildUrl) - dispatch(setWaitForOCPUpdate({waitForUpdate:false})) - api_data = JSON.parse(api_data) - if(api_data){ - const results = api_data.results - if(results){ - const benchmarks = GetBenchmarks(results) - const versions = GetVersions(results) - const platforms = GetPlatforms(results) - const workers = GetWorkers(results) - const networkTypes = GetNetworkTypes(results) - const ciSystems = GetCiSystems(results) - const jobTypes = GetJobType(results) - const rehearses = ["TRUE", "FALSE"] - const allIpsec = ["TRUE", "FALSE"] - const allFips = ["TRUE", "FALSE"] - const allEncrypted = ["TRUE", "FALSE"] - const encryptionTypes = GetEncryptionTypes(results) - const allPublish = GetPublish(results) - const computeArchs = GetComputeArchs(results) - const controlPlaneArchs = GetControlPlaneArchs(results) - const updatedTime = new Date().toLocaleString().replace(', ', ' ').toString(); - await dispatch(getOCPJobsData({ - data: results, benchmarks, versions, waitForUpdate: false, platforms, workers, networkTypes, - updatedTime, ciSystems, jobTypes, rehearses, allIpsec, allFips, allEncrypted, encryptionTypes, - allPublish, computeArchs, controlPlaneArchs, startDate: api_data.startDate, endDate: api_data.endDate - })) - await dispatch(updateOCPMetaData({data: results})) - } - } - } - catch (e) { - const error = e - if(error.response){ - await dispatch(errorOCPCall({error: error.response.data.error})) - alert(error.response.data.error) - } - else{ - console.log(error) - } - dispatch(setWaitForOCPUpdate({waitForUpdate:false})) - } -} - -export const fetchQuayJobsData = (startDate = '', endDate='') => async dispatch => { - let buildUrl = `${BASE_URL}${QUAY_JOBS_API_V1}` - dispatch(setWaitForQuayUpdate({waitForUpdate:true})) - if(startDate !== '' && endDate !== '') { - buildUrl += `?start_date=${startDate}&end_date=${endDate}` - } - try{ - let api_data = await fetchAPI(buildUrl) - dispatch(setWaitForQuayUpdate({waitForUpdate:false})) - api_data = JSON.parse(api_data) - if(api_data){ - const results = api_data.results - if(results){ - const benchmarks = GetBenchmarks(results) - const releaseStreams = GetReleaseStreams(results) - const platforms = GetPlatforms(results) - const workers = GetWorkers(results) - const hitSizes = GetHitSizes(results) - const concurrencies = GetConcurrencies(results) - const imagePushPulls = GetImagePushPulls(results) - const ciSystems = GetCiSystems(results) - const updatedTime = new Date().toLocaleString().replace(', ', ' ').toString(); - await dispatch(getQuayJobsData({ - data: results, benchmarks, releaseStreams, waitForUpdate: false, platforms, workers, - hitSizes, concurrencies, imagePushPulls, updatedTime, ciSystems, startDate: api_data.startDate, - endDate: api_data.endDate - })) - await dispatch(updateQuayMetaData({data: results})) - } - } - } - catch (e) { - const error = e - if(error.response){ - await dispatch(errorQuayCall({error: error.response.data.error})) - alert(error.response.data.error) - } - else{ - console.log(error) - } - dispatch(setWaitForQuayUpdate({waitForUpdate:false})) - } -} - -export const fetchCPTJobsData = (startDate = '', endDate='') => async dispatch => { - let buildUrl = `${BASE_URL}${CPT_JOBS_API_V1}` - dispatch(setWaitForCPTUpdate({waitForUpdate:true})) - if(startDate !== '' && endDate !== '') { - buildUrl += `?start_date=${startDate}&end_date=${endDate}` - } - try{ - let api_data = await fetchAPI(buildUrl) - dispatch(setWaitForCPTUpdate({waitForUpdate:false})) - api_data = JSON.parse(api_data) - if(api_data){ - const results = api_data.results - if(results){ - const testNames = GetTestNames(results) - const products = GetProducts(results) - const ciSystems = GetCiSystems(results) - const jobStatuses = GetStatuses(results) - const releaseStreams = GetReleaseStreams(results) - const updatedTime = new Date().toLocaleString().replace(', ', ' ').toString(); - await dispatch(getCPTJobsData({ - data: results, testNames, products, waitForUpdate: false, - jobStatuses, releaseStreams, updatedTime, ciSystems, startDate: api_data.startDate, endDate: api_data.endDate - })) - await dispatch(updateCPTMetaData({data: results})) - } - } - } - catch (e) { - const error = e - if(error.response){ - await dispatch(errorCPTCall({error: error.response.data.error})) - alert(error.response.data.error) - } - else{ - console.log(error) - } - dispatch(setWaitForCPTUpdate({waitForUpdate:false})) - } -} - -const GetCiSystems = (api_data) => { - return Array.from(new Set(api_data.map(item => { - if(item.ciSystem === null) return '' - else return item.ciSystem.toUpperCase().trim() - }))).sort() -} - -export const GetVersions = (api_data) => { - return Array.from(new Set(api_data.map(item => item.shortVersion.toLowerCase().trim()))).sort() -} - -export const GetBenchmarks = (api_data) => { - return Array.from(new Set(api_data.map(item => { - if(item.benchmark === null) return '' - else return item.benchmark.toLowerCase().trim() - }))).sort() -} - -const GetPlatforms = (api_data) => { - return Array.from(new Set(api_data.map(item => item.platform.toUpperCase().trim()))).sort() -} - -const GetWorkers = (api_data) => { - return Array.from(new Set(api_data.map(item => parseInt(item.workerNodesCount)))).sort((a, b) => a-b) -} - -const GetHitSizes = (api_data) => { - return Array.from(new Set(api_data.map(item => parseInt(item.hitSize)))).sort((a, b) => a-b) -} - -const GetConcurrencies = (api_data) => { - return Array.from(new Set(api_data.map(item => parseInt(item.concurrency)))).sort((a, b) => a-b) -} - -const GetImagePushPulls = (api_data) => { - return Array.from(new Set(api_data.map(item => parseInt(item.imagePushPulls)))).sort((a, b) => a-b) -} - -const GetNetworkTypes = (api_data) => { - return Array.from(new Set(api_data.map(item => item.networkType.toUpperCase().trim()))).sort() -} - -const GetProducts = (api_data) => { - return Array.from(new Set(api_data.map(item => item.product.toUpperCase().trim()))).sort() -} - -const GetStatuses = (api_data) => { - return Array.from(new Set(api_data.map(item => item.jobStatus.toUpperCase().trim()))).sort() -} - -const GetReleaseStreams = (api_data) => { - return Array.from(new Set(api_data.map(item => item.releaseStream.toUpperCase().trim()))).sort() -} - -const GetTestNames = (api_data) => { - return Array.from(new Set(api_data.map(item => { - if(item.testName === null) return '' - else return item.testName.toLowerCase().trim() - }))).sort() -} - -const GetJobType = (api_data) => { - return Array.from(new Set(api_data.map(item => item.jobType.toUpperCase().trim()))).sort() -} - -const GetEncryptionTypes = (api_data) => { - return Array.from(new Set(api_data.map(item => item.encryptionType.toUpperCase().trim()))).sort() -} - -const GetPublish = (api_data) => { - return Array.from(new Set(api_data.map(item => item.publish.toUpperCase().trim()))).sort() -} - -const GetComputeArchs = (api_data) => { - return Array.from(new Set(api_data.map(item => item.computeArch.toUpperCase().trim()))).sort() -} - -const GetControlPlaneArchs = (api_data) => { - return Array.from(new Set(api_data.map(item => item.controlPlaneArch.toUpperCase().trim()))).sort() -} \ No newline at end of file diff --git a/frontend/src/store/Shared.js b/frontend/src/store/Shared.js deleted file mode 100644 index 31a078c9..00000000 --- a/frontend/src/store/Shared.js +++ /dev/null @@ -1,16 +0,0 @@ - - -export const getUrl = () => { - const {hostname, protocol} = window.location - return (hostname === "localhost") ? "http://localhost:8000":`${protocol}//${hostname}` -} - -export const BASE_URL = getUrl() - -export const OCP_JOBS_API_V1 = "/api/v1/ocp/jobs" -export const OCP_GRAPH_API_V1 = "/api/v1/ocp/graph" - -export const CPT_JOBS_API_V1 = "/api/v1/cpt/jobs" - -export const QUAY_JOBS_API_V1 = "/api/v1/quay/jobs" -export const QUAY_GRAPH_API_V1 = "/api/v1/quay/graph" \ No newline at end of file diff --git a/frontend/src/store/reducers/CPTJobsReducer.js b/frontend/src/store/reducers/CPTJobsReducer.js deleted file mode 100644 index f6eab899..00000000 --- a/frontend/src/store/reducers/CPTJobsReducer.js +++ /dev/null @@ -1,58 +0,0 @@ -import {createSlice, original} from "@reduxjs/toolkit"; -import {CPT_INITIAL_DATA} from "./InitialData"; -import { getCPTUpdatedData, getCPTSummary } from './Utils'; - - -const jobsSlice = createSlice({ - initialState: { - ...CPT_INITIAL_DATA, - }, - name: 'cptES', - reducers: { - getCPTJobsData: (state, action) => { - state.initialState = false - state.copyData = action.payload.data - state.data = action.payload.data - state.testNames = ["All", ...action.payload.testNames] - state.products = ["All", ...action.payload.products] - state.ciSystems = ["All", ...action.payload.ciSystems] - state.jobStatuses = ["All", ...action.payload.jobStatuses] - state.releaseStreams = ["All", ...action.payload.releaseStreams] - state.waitForUpdate = action.payload.waitForUpdate - state.updatedTime = action.payload.updatedTime - state.error = null - Object.assign(state, getCPTSummary(state.data)) - state.startDate = action.payload.startDate - state.endDate = action.payload.endDate - }, - updateCPTDataFilter: (state, action) => { - const {ciSystem, testName, product, jobStatus, releaseStream} = action.payload - state.selectedTestName = testName - state.selectedProduct = product - state.selectedCiSystem = ciSystem - state.selectedJobStatus = jobStatus - state.selectedReleaseStream = releaseStream - state.data = getCPTUpdatedData(original(state.copyData), testName, product, ciSystem, jobStatus, releaseStream) - Object.assign(state, getCPTSummary(state.data)) - }, - updateCPTMetaData: (state, action) => { - state.data = getCPTUpdatedData(action.payload.data, state.selectedTestName, state.selectedProduct, - state.selectedCiSystem, state.selectedJobStatus, state.selectedReleaseStream) - Object.assign(state, getCPTSummary(state.data)) - }, - setWaitForCPTUpdate: (state, action) => { - state.waitForUpdate = action.payload.waitForUpdate - }, - errorCPTCall: (state, action) => { - state.error = action.payload.error - } - } -}) -export const { - getCPTJobsData, - updateCPTDataFilter, - updateCPTMetaData, - setWaitForCPTUpdate, - errorCPTCall, -} = jobsSlice.actions -export default jobsSlice.reducer diff --git a/frontend/src/store/reducers/GraphReducer.js b/frontend/src/store/reducers/GraphReducer.js deleted file mode 100644 index 6110dd03..00000000 --- a/frontend/src/store/reducers/GraphReducer.js +++ /dev/null @@ -1,24 +0,0 @@ -import {createSlice} from "@reduxjs/toolkit"; -import {GRAPH_INITIAL_DATA} from "./InitialData"; - - -const graphReducer = createSlice({ - initialState: { - ...GRAPH_INITIAL_DATA, - }, - name: 'Graph', - reducers: { - getUuidResults: (state, action) => { - Object.assign(state.uuid_results, action.payload) - }, - setGraphError: (state, action) => { - Object.assign(state.graphError, action.payload.error) - } - } -}) - -export const { - getUuidResults, - setGraphError -} = graphReducer.actions -export default graphReducer.reducer diff --git a/frontend/src/store/reducers/InitialData.js b/frontend/src/store/reducers/InitialData.js deleted file mode 100644 index f70aa1bc..00000000 --- a/frontend/src/store/reducers/InitialData.js +++ /dev/null @@ -1,136 +0,0 @@ - -export const OCP_INITIAL_DATA = { - initialState: true, - success: 0, - failure: 0, - total: 0, - others: 0, - duration:0, - benchmarks: ["All"], - versions: ["All"], - workers: ["All"], - ciSystems: ["All"], - networkTypes: ["All"], - jobTypes: ["All"], - rehearses: ["All"], - allIpsec: ["All"], - allFips: ["All"], - allEncrypted: ["All"], - encryptionTypes: ["All"], - allPublish: ["All"], - computeArchs: ["All"], - controlPlaneArchs: ["All"], - selectedBenchmark: "All", - selectedVersion: "All", - selectedPlatform: "All", - selectedWorkerCount: "All", - selectedNetworkType: "All", - selectedCiSystem: "All", - selectedJobType: "All", - selectedRehearse: "All", - selectedIpsec: "All", - selectedFips: "All", - selectedEncrypted: "All", - selectedEncryptionType: "All", - selectedPublish: "All", - selectedComputeArch: "All", - selectedControlPlaneArch: "All", - waitForUpdate: false, - platforms: ["All"], - copyData: [], - data: [], - updatedTime: 'Loading', - error: null, - startDate: '', - endDate: '', - tableData : [{ name: "Benchmark", value: "benchmark" }, - {name:"Release Stream", value: "releaseStream"}, - {name:"Build", value: "build"}, - {name: "Worker Count", value: "workerNodesCount"}, - {name: "Start Date", value: "startDate"}, - {name: "End Date", value: "endDate"}, - {name: "Status", value: "jobStatus"}], -} - -export const QUAY_INITIAL_DATA = { - initialState: true, - success: 0, - failure: 0, - total: 0, - others: 0, - duration:0, - ciSystems: ["All"], - platforms: ["All"], - benchmarks: ["All"], - releaseStreams: ["All"], - workers: ["All"], - hitSizes: ["All"], - concurrencies: ["All"], - imagePushPulls: ["All"], - selectedCiSystem: "All", - selectedPlatform: "All", - selectedBenchmark: "All", - selectedReleaseStream: "All", - selectedWorkerCount: "All", - selectedHitSize: "All", - selectedConcurrency: "All", - selectedImagePushPulls: "All", - waitForUpdate: false, - copyData: [], - data: [], - updatedTime: 'Loading', - error: null, - startDate: '', - endDate: '', - tableData : [{ name: "Benchmark", value: "benchmark" }, - {name:"Release Stream", value: "releaseStream"}, - {name:"Platform", value: "platform"}, - {name: "Worker Count", value: "workerNodesCount"}, - {name: "Start Date", value: "startDate"}, - {name: "End Date", value: "endDate"}, - {name: "Status", value: "jobStatus"}], -} - -export const CPT_INITIAL_DATA = { - initialState: true, - success: 0, - failure: 0, - total: 0, - others: 0, - testNames: ["All"], - products: ["All"], - ciSystems: ["All"], - statuses: ["All"], - releaseStreams: ["All"], - selectedCiSystem: "All", - selectedProduct: "All", - selectedTestName: "All", - selectedJobStatus: "All", - selectedReleaseStream: "All", - waitForUpdate: false, - copyData: [], - data: [], - updatedTime: 'Loading', - error: null, - startDate: '', - endDate: '', - tableData : [{name:"Product", value: "product"}, - { name: "CI System", value: "ciSystem" }, - {name: "Test Name", value: "testName"}, - {name: "Version", value: "version"}, - {name: "Release Stream", value: "releaseStream"}, - {name: "Start Date", value: "startDate"}, - {name: "End Date", value: "endDate"}, - {name: "Build URL", value: "buildUrl"}, - {name: "Status", value: "jobStatus"},], -} - -export const GRAPH_INITIAL_DATA = { - uuid_results: {}, - graphError: false, -} - -export const QUAY_GRAPH_INITIAL_DATA = { - uuid_results: {}, - graphError: false, -} \ No newline at end of file diff --git a/frontend/src/store/reducers/OCPJobsReducer.js b/frontend/src/store/reducers/OCPJobsReducer.js deleted file mode 100644 index 1673e6e6..00000000 --- a/frontend/src/store/reducers/OCPJobsReducer.js +++ /dev/null @@ -1,82 +0,0 @@ -import {createSlice, original} from "@reduxjs/toolkit"; -import {OCP_INITIAL_DATA} from "./InitialData"; -import { getOCPUpdatedData, getOCPSummary } from './Utils'; - - -const jobsSlice = createSlice({ - initialState: { - ...OCP_INITIAL_DATA, - }, - name: 'ocpES', - reducers: { - getOCPJobsData: (state, action) => { - state.initialState = false - state.copyData = action.payload.data - state.data = action.payload.data - state.benchmarks = ["All", ...action.payload.benchmarks] - state.versions = ["All", ...action.payload.versions] - state.waitForUpdate = action.payload.waitForUpdate - state.platforms = ["All", ...action.payload.platforms] - state.workers = ["All", ...action.payload.workers] - state.networkTypes = ["All", ...action.payload.networkTypes] - state.ciSystems = ["All", ...action.payload.ciSystems] - state.jobTypes = ["All", ...action.payload.jobTypes] - state.rehearses = ["All", ...action.payload.rehearses] - state.allIpsec = ["All", ...action.payload.allIpsec] - state.allFips = ["All", ...action.payload.allFips] - state.allEncrypted = ["All", ...action.payload.allEncrypted] - state.encryptionTypes = ["All", ...action.payload.encryptionTypes] - state.allPublish = ["All", ...action.payload.allPublish] - state.computeArchs = ["All", ...action.payload.computeArchs] - state.controlPlaneArchs = ["All", ...action.payload.controlPlaneArchs] - state.updatedTime = action.payload.updatedTime - state.error = null - Object.assign(state, getOCPSummary(state.data)) - state.startDate = action.payload.startDate - state.endDate = action.payload.endDate - }, - updateOCPDataFilter: (state, action) => { - const {ciSystem, platform, benchmark, version, workerCount, networkType, jobType, isRehearse, - ipsec, fips, encrypted, encryptionType, publish, computeArch, controlPlaneArch} = action.payload - state.selectedBenchmark = benchmark - state.selectedVersion = version - state.selectedPlatform = platform - state.selectedNetworkType = networkType - state.selectedWorkerCount = workerCount - state.selectedCiSystem = ciSystem - state.selectedJobType = jobType - state.selectedRehearse = isRehearse - state.selectedIpsec = ipsec - state.selectedFips = fips - state.selectedEncrypted = encrypted - state.selectedEncryptionType = encryptionType - state.selectedPublish = publish - state.selectedComputeArch = computeArch - state.selectedControlPlaneArch = controlPlaneArch - state.data = getOCPUpdatedData(original(state.copyData), platform, benchmark, version, workerCount, networkType, ciSystem, jobType, isRehearse, - ipsec, fips, encrypted, encryptionType, publish, computeArch, controlPlaneArch) - Object.assign(state, getOCPSummary(state.data)) - }, - updateOCPMetaData: (state, action) => { - state.data = getOCPUpdatedData(action.payload.data, state.selectedPlatform, state.selectedBenchmark, - state.selectedVersion, state.selectedWorkerCount, state.selectedNetworkType, state.selectedCiSystem, - state.selectedJobType, state.selectedRehearse, state.selectedIpsec, state.selectedFips, state.selectedEncrypted, - state.selectedEncryptionType, state.selectedPublish, state.selectedComputeArch, state.selectedControlPlaneArch) - Object.assign(state, getOCPSummary(state.data)) - }, - setWaitForOCPUpdate: (state, action) => { - state.waitForUpdate = action.payload.waitForUpdate - }, - errorOCPCall: (state, action) => { - state.error = action.payload.error - } - } -}) -export const { - getOCPJobsData, - updateOCPDataFilter, - updateOCPMetaData, - setWaitForOCPUpdate, - errorOCPCall, -} = jobsSlice.actions -export default jobsSlice.reducer diff --git a/frontend/src/store/reducers/QuayGraphReducer.js b/frontend/src/store/reducers/QuayGraphReducer.js deleted file mode 100644 index 63956603..00000000 --- a/frontend/src/store/reducers/QuayGraphReducer.js +++ /dev/null @@ -1,24 +0,0 @@ -import {createSlice} from "@reduxjs/toolkit"; -import {QUAY_GRAPH_INITIAL_DATA} from "./InitialData"; - - -const quayGraphReducer = createSlice({ - initialState: { - ...QUAY_GRAPH_INITIAL_DATA, - }, - name: 'quayGraph', - reducers: { - getQuayUuidResults: (state, action) => { - Object.assign(state.uuid_results, action.payload) - }, - setQuayGraphError: (state, action) => { - Object.assign(state.graphError, action.payload.error) - } - } -}) - -export const { - getQuayUuidResults, - setQuayGraphError -} = quayGraphReducer.actions -export default quayGraphReducer.reducer diff --git a/frontend/src/store/reducers/QuayJobsReducer.js b/frontend/src/store/reducers/QuayJobsReducer.js deleted file mode 100644 index ead7647e..00000000 --- a/frontend/src/store/reducers/QuayJobsReducer.js +++ /dev/null @@ -1,64 +0,0 @@ -import {createSlice, original} from "@reduxjs/toolkit"; -import {QUAY_INITIAL_DATA} from "./InitialData"; -import { getQuayUpdatedData, getQuaySummary } from './Utils'; - -const jobsSlice = createSlice({ - initialState: { - ...QUAY_INITIAL_DATA, - }, - name: 'quayES', - reducers: { - getQuayJobsData: (state, action) => { - state.initialState = false - state.copyData = action.payload.data - state.data = action.payload.data - state.benchmarks = ["All", ...action.payload.benchmarks] - state.releaseStreams = ["All", ...action.payload.releaseStreams] - state.waitForUpdate = action.payload.waitForUpdate - state.platforms = ["All", ...action.payload.platforms] - state.workers = ["All", ...action.payload.workers] - state.hitSizes = ["All", ...action.payload.hitSizes] - state.concurrencies = ["All", ...action.payload.concurrencies] - state.imagePushPulls = ["All", ...action.payload.imagePushPulls] - state.ciSystems = ["All", ...action.payload.ciSystems] - state.updatedTime = action.payload.updatedTime - state.error = null - Object.assign(state, getQuaySummary(state.data)) - state.startDate = action.payload.startDate - state.endDate = action.payload.endDate - }, - updateQuayDataFilter: (state, action) => { - const {ciSystem, platform, benchmark, releaseStream, workerCount, hitSize, concurrency, imagePushPulls} = action.payload - state.selectedBenchmark = benchmark - state.selectedReleaseStream = releaseStream - state.selectedPlatform = platform - state.selectedWorkerCount = workerCount - state.selectedCiSystem = ciSystem - state.selectedHitSize = hitSize - state.selectedConcurrency = concurrency - state.selectedImagePushPulls = imagePushPulls - state.data = getQuayUpdatedData(original(state.copyData), platform, benchmark, releaseStream, workerCount, ciSystem, hitSize, concurrency, imagePushPulls) - Object.assign(state, getQuaySummary(state.data)) - }, - updateQuayMetaData: (state, action) => { - state.data = getQuayUpdatedData(action.payload.data, state.selectedPlatform, state.selectedBenchmark, - state.selectedReleaseStream, state.selectedWorkerCount, state.selectedCiSystem, state.selectedHitSize, - state.selectedConcurrency, state.selectedImagePushPulls) - Object.assign(state, getQuaySummary(state.data)) - }, - setWaitForQuayUpdate: (state, action) => { - state.waitForUpdate = action.payload.waitForUpdate - }, - errorQuayCall: (state, action) => { - state.error = action.payload.error - } - } -}) -export const { - getQuayJobsData, - updateQuayDataFilter, - updateQuayMetaData, - setWaitForQuayUpdate, - errorQuayCall, -} = jobsSlice.actions -export default jobsSlice.reducer diff --git a/frontend/src/store/reducers/index.js b/frontend/src/store/reducers/index.js deleted file mode 100644 index a057312d..00000000 --- a/frontend/src/store/reducers/index.js +++ /dev/null @@ -1,14 +0,0 @@ -import ocpJobsReducer from "./OCPJobsReducer"; -import cptJobsReducer from "./CPTJobsReducer"; -import quayJobsReducer from "./QuayJobsReducer"; -import graphReducer from "./GraphReducer"; -import quayGraphReducer from "./QuayGraphReducer"; - - -export const rootReducer = { - 'ocpJobs': ocpJobsReducer, - 'cptJobs': cptJobsReducer, - 'quayJobs': quayJobsReducer, - 'graph': graphReducer, - 'quayGraph': quayGraphReducer, -} diff --git a/frontend/src/store/store.js b/frontend/src/store/store.js index 739300ed..afa9133a 100644 --- a/frontend/src/store/store.js +++ b/frontend/src/store/store.js @@ -1,12 +1,5 @@ -import {configureStore} from "@reduxjs/toolkit"; -import {rootReducer} from "./reducers"; -import {logger} from "redux-logger/src"; +import { configureStore } from "@reduxjs/toolkit"; - -const store = configureStore({ - reducer:rootReducer, - middleware: (getDefaultMiddleware) => - window.location.hostname === "localhost" ? getDefaultMiddleware().concat(logger): getDefaultMiddleware() +export default configureStore({ + reducer: {}, }); - -export default store; diff --git a/frontend/src/utils/routeConstants.js b/frontend/src/utils/routeConstants.js new file mode 100644 index 00000000..878f8b90 --- /dev/null +++ b/frontend/src/utils/routeConstants.js @@ -0,0 +1,2 @@ +export const HOME = "Home"; +export const QUAY = "QUAY"; diff --git a/frontend/vite.config.js b/frontend/vite.config.js new file mode 100644 index 00000000..fdd212e8 --- /dev/null +++ b/frontend/vite.config.js @@ -0,0 +1,19 @@ +import { defineConfig } from "vite"; +import path from "path"; +import react from "@vitejs/plugin-react"; +// https://vitejs.dev/config/ +export default defineConfig({ + plugins: [react()], + esbuild: { + jsxFactory: "React.createElement", + jsxFragment: "React.Fragment", + }, + resolve: { + alias: [{ find: "@", replacement: path.resolve(__dirname, "src") }], + extensions: [".js", ".json", ".jsx", ".mjs"], + }, + server: { + port: 3000, + open: true, + }, +}); From cdc0208e3314b0a37f8d09bdc564dfb2ca356741 Mon Sep 17 00:00:00 2001 From: MVarshini Date: Mon, 1 Apr 2024 18:55:58 +0530 Subject: [PATCH 02/14] Layout and Store --- frontend/package-lock.json | 181 +++++++++++++++++- frontend/package.json | 4 +- frontend/src/App.jsx | 8 + frontend/src/actions/sideMenuActions.js | 13 ++ frontend/src/actions/toastActions.js | 43 +++++ frontend/src/actions/types.js | 11 ++ .../src/assets/constants/SidemenuConstants.js | 3 + .../src/assets/constants/toastConstants.js | 4 + .../molecules/SideMenuOptions/index.jsx | 34 +++- .../src/components/organisms/Header/index.jsx | 11 +- .../organisms/LoadingComponent/index.jsx | 9 +- .../components/organisms/SideMenu/index.jsx | 3 +- .../organisms/ToastComponent/index.jsx | 42 +++- frontend/src/containers/MainLayout/index.jsx | 5 +- frontend/src/main.jsx | 5 +- frontend/src/reducers/index.js | 10 + frontend/src/reducers/loadingReducer.js | 19 ++ frontend/src/reducers/sideMenuReducer.js | 24 +++ frontend/src/reducers/toastReducer.js | 25 +++ frontend/src/store/store.js | 12 +- frontend/src/utils/apiConstants.js | 16 ++ frontend/src/utils/axiosInstance.js | 9 + frontend/src/utils/helper.js | 6 + frontend/src/utils/routeConstants.js | 1 + 24 files changed, 470 insertions(+), 28 deletions(-) create mode 100644 frontend/src/actions/sideMenuActions.js create mode 100644 frontend/src/actions/toastActions.js create mode 100644 frontend/src/actions/types.js create mode 100644 frontend/src/assets/constants/SidemenuConstants.js create mode 100644 frontend/src/assets/constants/toastConstants.js create mode 100644 frontend/src/reducers/index.js create mode 100644 frontend/src/reducers/loadingReducer.js create mode 100644 frontend/src/reducers/sideMenuReducer.js create mode 100644 frontend/src/reducers/toastReducer.js create mode 100644 frontend/src/utils/apiConstants.js create mode 100644 frontend/src/utils/axiosInstance.js create mode 100644 frontend/src/utils/helper.js diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 10007d03..197848ab 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -10,10 +10,12 @@ "dependencies": { "@patternfly/react-core": "^5.2.3", "@reduxjs/toolkit": "^2.2.3", + "axios": "^1.6.8", "react": "^18.2.0", "react-dom": "^18.2.0", "react-redux": "^9.1.0", - "react-router-dom": "^6.22.3" + "react-router-dom": "^6.22.3", + "redux-logger": "^3.0.6" }, "devDependencies": { "@types/react": "^18.2.66", @@ -1499,6 +1501,11 @@ "url": "https://github.com/sponsors/ljharb" } }, + "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/attr-accept": { "version": "2.2.2", "resolved": "https://registry.npmjs.org/attr-accept/-/attr-accept-2.2.2.tgz", @@ -1522,6 +1529,16 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/axios": { + "version": "1.6.8", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.8.tgz", + "integrity": "sha512-v/ZHtJDU39mDpyBoFVkETcd/uNdxrWRrg3bKpOKzXFA6Bvqopts6ALSMU3y6ijYxbw2B+wPrIv46egTzJXCLGQ==", + "dependencies": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, "node_modules/balanced-match": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", @@ -1647,6 +1664,17 @@ "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", "dev": true }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, "node_modules/concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", @@ -1759,6 +1787,11 @@ } } }, + "node_modules/deep-diff": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/deep-diff/-/deep-diff-0.3.8.tgz", + "integrity": "sha512-yVn6RZmHiGnxRKR9sJb3iVV2XTF1Ghh2DiWRZ3dMnGc43yUdWWF/kX6lQyk3+P84iprfWKU/8zFTrlkvtFm1ug==" + }, "node_modules/deep-is": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", @@ -1799,6 +1832,14 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/doctrine": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", @@ -2442,6 +2483,25 @@ "tabbable": "^6.2.0" } }, + "node_modules/follow-redirects": { + "version": "1.15.6", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.6.tgz", + "integrity": "sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, "node_modules/for-each": { "version": "0.3.3", "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", @@ -2451,6 +2511,19 @@ "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==", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -3381,6 +3454,25 @@ "node": ">=4" } }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, "node_modules/minimatch": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", @@ -3738,6 +3830,11 @@ "react-is": "^16.13.1" } }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, "node_modules/prr": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", @@ -3888,6 +3985,14 @@ "resolved": "https://registry.npmjs.org/redux/-/redux-5.0.1.tgz", "integrity": "sha512-M9/ELqF6fy8FwmkpnF0S3YKOqMyoWJ4+CS5Efg2ct3oY9daQvd/Pc71FpGZsVsbl3Cpb+IIcjBDUnnyBdQbq4w==" }, + "node_modules/redux-logger": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/redux-logger/-/redux-logger-3.0.6.tgz", + "integrity": "sha512-JoCIok7bg/XpqA1JqCqXFypuqBbQzGQySrhFzewB7ThcnysTO30l4VCst86AuB9T9tuT03MAA56Jw2PNhRSNCg==", + "dependencies": { + "deep-diff": "^0.3.5" + } + }, "node_modules/redux-thunk": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/redux-thunk/-/redux-thunk-3.1.0.tgz", @@ -5635,6 +5740,11 @@ "is-shared-array-buffer": "^1.0.2" } }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" + }, "attr-accept": { "version": "2.2.2", "resolved": "https://registry.npmjs.org/attr-accept/-/attr-accept-2.2.2.tgz", @@ -5649,6 +5759,16 @@ "possible-typed-array-names": "^1.0.0" } }, + "axios": { + "version": "1.6.8", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.8.tgz", + "integrity": "sha512-v/ZHtJDU39mDpyBoFVkETcd/uNdxrWRrg3bKpOKzXFA6Bvqopts6ALSMU3y6ijYxbw2B+wPrIv46egTzJXCLGQ==", + "requires": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" + } + }, "balanced-match": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", @@ -5728,6 +5848,14 @@ "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", "dev": true }, + "combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "requires": { + "delayed-stream": "~1.0.0" + } + }, "concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", @@ -5808,6 +5936,11 @@ "ms": "2.1.2" } }, + "deep-diff": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/deep-diff/-/deep-diff-0.3.8.tgz", + "integrity": "sha512-yVn6RZmHiGnxRKR9sJb3iVV2XTF1Ghh2DiWRZ3dMnGc43yUdWWF/kX6lQyk3+P84iprfWKU/8zFTrlkvtFm1ug==" + }, "deep-is": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", @@ -5836,6 +5969,11 @@ "object-keys": "^1.1.1" } }, + "delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==" + }, "doctrine": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", @@ -6334,6 +6472,11 @@ "tabbable": "^6.2.0" } }, + "follow-redirects": { + "version": "1.15.6", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.6.tgz", + "integrity": "sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==" + }, "for-each": { "version": "0.3.3", "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", @@ -6343,6 +6486,16 @@ "is-callable": "^1.1.3" } }, + "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==", + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + } + }, "fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -6983,6 +7136,19 @@ "dev": true, "optional": true }, + "mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==" + }, + "mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "requires": { + "mime-db": "1.52.0" + } + }, "minimatch": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", @@ -7227,6 +7393,11 @@ "react-is": "^16.13.1" } }, + "proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, "prr": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", @@ -7315,6 +7486,14 @@ "resolved": "https://registry.npmjs.org/redux/-/redux-5.0.1.tgz", "integrity": "sha512-M9/ELqF6fy8FwmkpnF0S3YKOqMyoWJ4+CS5Efg2ct3oY9daQvd/Pc71FpGZsVsbl3Cpb+IIcjBDUnnyBdQbq4w==" }, + "redux-logger": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/redux-logger/-/redux-logger-3.0.6.tgz", + "integrity": "sha512-JoCIok7bg/XpqA1JqCqXFypuqBbQzGQySrhFzewB7ThcnysTO30l4VCst86AuB9T9tuT03MAA56Jw2PNhRSNCg==", + "requires": { + "deep-diff": "^0.3.5" + } + }, "redux-thunk": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/redux-thunk/-/redux-thunk-3.1.0.tgz", diff --git a/frontend/package.json b/frontend/package.json index 0161e7f6..9ea35a84 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -12,10 +12,12 @@ "dependencies": { "@patternfly/react-core": "^5.2.3", "@reduxjs/toolkit": "^2.2.3", + "axios": "^1.6.8", "react": "^18.2.0", "react-dom": "^18.2.0", "react-redux": "^9.1.0", - "react-router-dom": "^6.22.3" + "react-router-dom": "^6.22.3", + "redux-logger": "^3.0.6" }, "devDependencies": { "@types/react": "^18.2.66", diff --git a/frontend/src/App.jsx b/frontend/src/App.jsx index 3d11ca8c..5d6ce6e3 100644 --- a/frontend/src/App.jsx +++ b/frontend/src/App.jsx @@ -6,14 +6,22 @@ import { BrowserRouter, Route, Routes } from "react-router-dom"; import Home from "./components/templates/Home"; import MainLayout from "./containers/MainLayout"; +import { useDispatch } from "react-redux"; +import { useEffect } from "react"; function App() { + const dispatch = useDispatch(); + + useEffect(() => {}, [dispatch]); return (
}> } /> + } /> + } /> + } /> diff --git a/frontend/src/actions/sideMenuActions.js b/frontend/src/actions/sideMenuActions.js new file mode 100644 index 00000000..3e4a2950 --- /dev/null +++ b/frontend/src/actions/sideMenuActions.js @@ -0,0 +1,13 @@ +import * as TYPES from "./types"; + +export const setActiveItem = (item) => { + return { + type: TYPES.SET_ACTIVE_MENU_ITEM, + payload: item, + }; +}; + +export const toggleSideMenu = (isOpen) => ({ + type: TYPES.TOGGLE_SIDE_MENU, + payload: isOpen, +}); diff --git a/frontend/src/actions/toastActions.js b/frontend/src/actions/toastActions.js new file mode 100644 index 00000000..574487f1 --- /dev/null +++ b/frontend/src/actions/toastActions.js @@ -0,0 +1,43 @@ +import * as TYPES from "./types"; + +import { uid } from "@/utils/helper"; + +export const showFailureToast = () => async (dispatch) => { + const toast = { + variant: "danger", + title: "Something went wrong", + message: "Please try again later", + }; + dispatch(showToast(toast.variant, toast.title, toast.message)); +}; + +export const showToast = + (variant, title, message = "") => + (dispatch, getState) => { + const obj = { + variant: variant, + title: title, + message: message, + key: uid(), + }; + const alerts = [...getState().toast.alerts, obj]; + + dispatch({ + type: TYPES.SHOW_TOAST, + payload: alerts, + }); + }; + +export const hideToast = (key) => (dispatch, getState) => { + const alerts = getState().toast.alerts; + const activeAlert = alerts.filter((item) => item.key !== key); + + dispatch({ + type: TYPES.SHOW_TOAST, + payload: activeAlert, + }); +}; + +export const clearToast = () => (dispatch) => { + dispatch({ type: TYPES.CLEAR_TOAST }); +}; diff --git a/frontend/src/actions/types.js b/frontend/src/actions/types.js new file mode 100644 index 00000000..2db9927a --- /dev/null +++ b/frontend/src/actions/types.js @@ -0,0 +1,11 @@ +/* Loading */ +export const LOADING = "LOADING"; +export const COMPLETED = "COMPLETED"; + +/* Toast */ +export const SHOW_TOAST = "SHOW_TOAST"; +export const CLEAR_TOAST = "CLEAR_TOAST"; + +/* Sidemenu */ +export const SET_ACTIVE_MENU_ITEM = "SET_ACTIVE_MENU_ITEM"; +export const TOGGLE_SIDE_MENU = "TOGGLE_SIDE_MENU"; diff --git a/frontend/src/assets/constants/SidemenuConstants.js b/frontend/src/assets/constants/SidemenuConstants.js new file mode 100644 index 00000000..b4c1f631 --- /dev/null +++ b/frontend/src/assets/constants/SidemenuConstants.js @@ -0,0 +1,3 @@ +export const HOME_NAV = "home"; +export const QUAY_NAV = "quay"; +export const OCP_NAV = "ocp"; diff --git a/frontend/src/assets/constants/toastConstants.js b/frontend/src/assets/constants/toastConstants.js new file mode 100644 index 00000000..54534e69 --- /dev/null +++ b/frontend/src/assets/constants/toastConstants.js @@ -0,0 +1,4 @@ +export const DANGER = "danger"; +export const ERROR_MSG = "Something went wrong!"; +export const SUCCESS = "success"; +export const WARNING = "warning"; diff --git a/frontend/src/components/molecules/SideMenuOptions/index.jsx b/frontend/src/components/molecules/SideMenuOptions/index.jsx index ac6693f4..754919da 100644 --- a/frontend/src/components/molecules/SideMenuOptions/index.jsx +++ b/frontend/src/components/molecules/SideMenuOptions/index.jsx @@ -1,31 +1,46 @@ +import * as CONSTANTS from "@/assets/constants/SidemenuConstants"; + import { Nav, NavItem, NavList } from "@patternfly/react-core"; +import { useDispatch, useSelector } from "react-redux"; +import { useLocation, useNavigate } from "react-router-dom"; -import React from "react"; +import { setActiveItem } from "@/actions/sideMenuActions"; +import { useEffect } from "react"; const sideMenuOptions = [ { - id: 0, + id: CONSTANTS.HOME_NAV, key: "home", displayName: "Home", }, { - id: 1, + id: CONSTANTS.OCP_NAV, key: "ocp", displayName: "OCP", }, { - id: 2, + id: CONSTANTS.QUAY_NAV, key: "quay", displayName: "Quay", }, ]; const MenuOptions = () => { - const [activeItem, setActiveItem] = React.useState(0); - const onSelect = (_event, itemId) => { - const item = itemId; - setActiveItem(item); + const dispatch = useDispatch(); + const { pathname } = useLocation(); + const activeMenuItem = useSelector((state) => state.sidemenu.activeMenuItem); + + const onSelect = (_event, item) => { + dispatch(setActiveItem(item.itemId)); }; + const navigate = useNavigate(); + useEffect(() => { + if (pathname) { + const currPath = pathname.replace(/^.*[/]([^/]+)[/]*$/, "$1"); + + dispatch(setActiveItem(currPath)); + } + }, [dispatch, pathname]); return ( <> @@ -36,7 +51,8 @@ const MenuOptions = () => { navigate(option.key)} > {option.displayName} diff --git a/frontend/src/components/organisms/Header/index.jsx b/frontend/src/components/organisms/Header/index.jsx index 3474c352..4090b29a 100644 --- a/frontend/src/components/organisms/Header/index.jsx +++ b/frontend/src/components/organisms/Header/index.jsx @@ -13,15 +13,16 @@ import { useDispatch, useSelector } from "react-redux"; import { BarsIcon } from "@patternfly/react-icons"; import logo from "@/assets//logo.png"; - -// import { toggleSideMenu } from "../../../reducer/HeaderReducer"; +import { showToast } from "@/actions/toastActions"; +import { toggleSideMenu } from "@/actions/sideMenuActions"; const Header = () => { - const isSideMenuOpen = true; + const isSideMenuOpen = useSelector((state) => state.sidemenu.isSidebarOpen); - // const dispatch = useDispatch(); + const dispatch = useDispatch(); const onSidebarToggle = () => { - // dispatch(toggleSideMenu()); + dispatch(toggleSideMenu(false)); + dispatch(showToast("success", "Hello")); }; return ( diff --git a/frontend/src/components/organisms/LoadingComponent/index.jsx b/frontend/src/components/organisms/LoadingComponent/index.jsx index 388361f7..66f57392 100644 --- a/frontend/src/components/organisms/LoadingComponent/index.jsx +++ b/frontend/src/components/organisms/LoadingComponent/index.jsx @@ -1,17 +1,18 @@ import "./index.less"; import { Spinner } from "@patternfly/react-core"; +import { useSelector } from "react-redux"; const LoadingComponent = ({ children }) => { - // const parentClass = isLoading ? "main-with-spinner" : ""; - const isLoading = false; - const parentClass = ""; + const isLoading = useSelector((state) => state.loading.isLoading); + const parentClass = isLoading ? "main-with-spinner" : ""; + return (
{children} {isLoading && (
- +
)}
diff --git a/frontend/src/components/organisms/SideMenu/index.jsx b/frontend/src/components/organisms/SideMenu/index.jsx index 2857205a..cae80271 100644 --- a/frontend/src/components/organisms/SideMenu/index.jsx +++ b/frontend/src/components/organisms/SideMenu/index.jsx @@ -4,7 +4,8 @@ import MenuOptions from "@/components/molecules/SideMenuOptions/index"; import { useSelector } from "react-redux"; const SideMenu = () => { - const isSideMenuOpen = true; + const isSideMenuOpen = useSelector((state) => state.sidemenu.isSidebarOpen); + return ( diff --git a/frontend/src/components/organisms/ToastComponent/index.jsx b/frontend/src/components/organisms/ToastComponent/index.jsx index a894dc24..3f7ff834 100644 --- a/frontend/src/components/organisms/ToastComponent/index.jsx +++ b/frontend/src/components/organisms/ToastComponent/index.jsx @@ -1,5 +1,45 @@ +import { + Alert, + AlertActionCloseButton, + AlertGroup, + AlertVariant, +} from "@patternfly/react-core"; +import { useDispatch, useSelector } from "react-redux"; + +import { hideToast } from "@/actions/toastActions"; + const ToastComponent = () => { - return <>hello; + const { alerts } = useSelector((state) => state.toast); + const dispatch = useDispatch(); + + const removeToast = (key) => { + dispatch(hideToast(key)); + }; + return ( + + {alerts.map((item) => ( + removeToast(item.key)} + actionClose={ + removeToast(item.key)} + /> + } + > + {item?.message && + item?.message.split("\n").map((i) => { + return

{i}

; + })} +
+ ))} +
+ ); }; export default ToastComponent; diff --git a/frontend/src/containers/MainLayout/index.jsx b/frontend/src/containers/MainLayout/index.jsx index 7beaadad..44bca647 100644 --- a/frontend/src/containers/MainLayout/index.jsx +++ b/frontend/src/containers/MainLayout/index.jsx @@ -6,12 +6,15 @@ import { Page, PageSection, PageSectionVariants } from "@patternfly/react-core"; import Header from "@/components/organisms/Header"; import LoadingComponent from "@/components/organisms/LoadingComponent"; import SideMenu from "@/components/organisms/SideMenu"; +import ToastComponent from "@/components/organisms/ToastComponent"; +import { useSelector } from "react-redux"; const MainLayout = () => { const navigate = useNavigate(); - + const { alerts } = useSelector((state) => state.toast); return ( <> + {alerts && alerts.length > 0 && } } sidebar={}> diff --git a/frontend/src/main.jsx b/frontend/src/main.jsx index 4226ca31..dfd24a0a 100644 --- a/frontend/src/main.jsx +++ b/frontend/src/main.jsx @@ -3,14 +3,11 @@ import "./index.css"; import App from "./App.jsx"; import { Provider } from "react-redux"; -import React from "react"; import ReactDOM from "react-dom/client"; import store from "./store/store"; ReactDOM.createRoot(document.getElementById("root")).render( - - - + ); diff --git a/frontend/src/reducers/index.js b/frontend/src/reducers/index.js new file mode 100644 index 00000000..7b225c6c --- /dev/null +++ b/frontend/src/reducers/index.js @@ -0,0 +1,10 @@ +import LoadingReducer from "./loadingReducer"; +import SideMenuReducer from "./sideMenuReducer"; +import ToastReducer from "./toastReducer"; +import { combineReducers } from "redux"; + +export default combineReducers({ + loading: LoadingReducer, + toast: ToastReducer, + sidemenu: SideMenuReducer, +}); diff --git a/frontend/src/reducers/loadingReducer.js b/frontend/src/reducers/loadingReducer.js new file mode 100644 index 00000000..2389a430 --- /dev/null +++ b/frontend/src/reducers/loadingReducer.js @@ -0,0 +1,19 @@ +import { COMPLETED, LOADING } from "@/actions/types"; + +const initialState = { + isLoading: false, +}; + +const LoadingReducer = (state = initialState, action = {}) => { + const { type } = action; + switch (type) { + case LOADING: + return { ...state, isLoading: true }; + case COMPLETED: + return { ...state, isLoading: false }; + default: + return state; + } +}; + +export default LoadingReducer; diff --git a/frontend/src/reducers/sideMenuReducer.js b/frontend/src/reducers/sideMenuReducer.js new file mode 100644 index 00000000..5470a9b6 --- /dev/null +++ b/frontend/src/reducers/sideMenuReducer.js @@ -0,0 +1,24 @@ +import * as CONSTANTS from "@/assets/constants/SidemenuConstants"; + +import { SET_ACTIVE_MENU_ITEM, TOGGLE_SIDE_MENU } from "../actions/types"; + +const initialState = { + activeMenuItem: CONSTANTS.HOME_NAV, + isSideMenuOpen: true, +}; +const SideMenuReducer = (state = initialState, action = {}) => { + const { type, payload } = action; + switch (type) { + case SET_ACTIVE_MENU_ITEM: + return { + ...state, + activeMenuItem: payload, + }; + case TOGGLE_SIDE_MENU: + return { ...state, isSideMenuOpen: payload }; + default: + return state; + } +}; + +export default SideMenuReducer; diff --git a/frontend/src/reducers/toastReducer.js b/frontend/src/reducers/toastReducer.js new file mode 100644 index 00000000..b571804c --- /dev/null +++ b/frontend/src/reducers/toastReducer.js @@ -0,0 +1,25 @@ +import { CLEAR_TOAST, SHOW_TOAST } from "../actions/types"; + +const initialState = { + alerts: [], +}; + +const ToastReducer = (state = initialState, action = {}) => { + const { type, payload } = action; + switch (type) { + case SHOW_TOAST: + return { + ...state, + alerts: [...payload], + }; + case CLEAR_TOAST: + return { + ...state, + ...initialState, + }; + default: + return state; + } +}; + +export default ToastReducer; diff --git a/frontend/src/store/store.js b/frontend/src/store/store.js index afa9133a..8a906b4b 100644 --- a/frontend/src/store/store.js +++ b/frontend/src/store/store.js @@ -1,5 +1,15 @@ import { configureStore } from "@reduxjs/toolkit"; +import { createLogger } from "redux-logger"; +import rootReducer from "../reducers/index.js"; + +const middleware = []; +middleware.push(createLogger()); +const enhancers = [...middleware]; export default configureStore({ - reducer: {}, + reducer: rootReducer, + middleware: (getDefaultMiddleware) => + window.location.hostname === "localhost" + ? getDefaultMiddleware().concat(enhancers) + : getDefaultMiddleware(), }); diff --git a/frontend/src/utils/apiConstants.js b/frontend/src/utils/apiConstants.js new file mode 100644 index 00000000..e6d68406 --- /dev/null +++ b/frontend/src/utils/apiConstants.js @@ -0,0 +1,16 @@ +export const getUrl = () => { + const { hostname, protocol } = window.location; + return hostname === "localhost" + ? "http://localhost:8000" + : `${protocol}//${hostname}`; +}; + +export const BASE_URL = getUrl(); + +export const OCP_JOBS_API_V1 = "/api/v1/ocp/jobs"; +export const OCP_GRAPH_API_V1 = "/api/v1/ocp/graph"; + +export const CPT_JOBS_API_V1 = "/api/v1/cpt/jobs"; + +export const QUAY_JOBS_API_V1 = "/api/v1/quay/jobs"; +export const QUAY_GRAPH_API_V1 = "/api/v1/quay/graph"; diff --git a/frontend/src/utils/axiosInstance.js b/frontend/src/utils/axiosInstance.js new file mode 100644 index 00000000..02eb48e2 --- /dev/null +++ b/frontend/src/utils/axiosInstance.js @@ -0,0 +1,9 @@ +import { BASE_URL } from "./apiConstants"; +import axios from "axios"; + +const axiosInstance = axios.create({ + baseURL: BASE_URL, + responseType: "json", +}); + +export default axiosInstance; diff --git a/frontend/src/utils/helper.js b/frontend/src/utils/helper.js new file mode 100644 index 00000000..7d72ef8b --- /dev/null +++ b/frontend/src/utils/helper.js @@ -0,0 +1,6 @@ +export const uid = () => { + const head = Date.now().toString(36); + const tail = Math.random().toString(36).substring(2); + + return head + tail; +}; diff --git a/frontend/src/utils/routeConstants.js b/frontend/src/utils/routeConstants.js index 878f8b90..ac95bff1 100644 --- a/frontend/src/utils/routeConstants.js +++ b/frontend/src/utils/routeConstants.js @@ -1,2 +1,3 @@ export const HOME = "Home"; +export const OCP = "OCP"; export const QUAY = "QUAY"; From d3ea14418eaa1fcae0703ffffee42027c63e73c0 Mon Sep 17 00:00:00 2001 From: MVarshini Date: Tue, 2 Apr 2024 19:35:49 +0530 Subject: [PATCH 03/14] OCP Table Laout --- frontend/package-lock.json | 41 ++++++++++++++ frontend/package.json | 1 + frontend/src/actions/homeActions.js | 28 ++++++++++ frontend/src/actions/types.js | 3 ++ .../src/components/atoms/TableCell/index.jsx | 54 +++++++++++++++++++ .../molecules/SideMenuOptions/index.jsx | 5 +- .../components/molecules/TableRows/index.jsx | 52 ++++++++++++++++++ .../src/components/organisms/Header/index.jsx | 4 +- .../organisms/TableLayout/index.jsx | 25 +++++++++ .../src/components/templates/Home/index.jsx | 23 +++++++- frontend/src/reducers/homeReducer.js | 36 +++++++++++++ frontend/src/reducers/index.js | 2 + frontend/src/reducers/sideMenuReducer.js | 2 +- frontend/src/utils/helper.js | 23 ++++++++ 14 files changed, 293 insertions(+), 6 deletions(-) create mode 100644 frontend/src/actions/homeActions.js create mode 100644 frontend/src/components/atoms/TableCell/index.jsx create mode 100644 frontend/src/components/molecules/TableRows/index.jsx create mode 100644 frontend/src/components/organisms/TableLayout/index.jsx create mode 100644 frontend/src/reducers/homeReducer.js diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 197848ab..3502c2f8 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -9,6 +9,7 @@ "version": "0.0.0", "dependencies": { "@patternfly/react-core": "^5.2.3", + "@patternfly/react-table": "^5.2.4", "@reduxjs/toolkit": "^2.2.3", "axios": "^1.6.8", "react": "^18.2.0", @@ -965,6 +966,23 @@ "resolved": "https://registry.npmjs.org/@patternfly/react-styles/-/react-styles-5.2.1.tgz", "integrity": "sha512-GT96hzI1QenBhq6Pfc51kxnj9aVLjL1zSLukKZXcYVe0HPOy0BFm90bT1Fo4e/z7V9cDYw4SqSX1XLc3O4jsTw==" }, + "node_modules/@patternfly/react-table": { + "version": "5.2.4", + "resolved": "https://registry.npmjs.org/@patternfly/react-table/-/react-table-5.2.4.tgz", + "integrity": "sha512-WCt4I6XYKRHXcasDqcOX70ctkgPVBAvlOv67KhaZsedxUU+B2NT1kiI7Jr7tD4SrR+jQs0MCF/bbRk1QM+rBqg==", + "dependencies": { + "@patternfly/react-core": "^5.2.3", + "@patternfly/react-icons": "^5.2.1", + "@patternfly/react-styles": "^5.2.1", + "@patternfly/react-tokens": "^5.2.1", + "lodash": "^4.17.19", + "tslib": "^2.5.0" + }, + "peerDependencies": { + "react": "^17 || ^18", + "react-dom": "^17 || ^18" + } + }, "node_modules/@patternfly/react-tokens": { "version": "5.2.1", "resolved": "https://registry.npmjs.org/@patternfly/react-tokens/-/react-tokens-5.2.1.tgz", @@ -3391,6 +3409,11 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" + }, "node_modules/lodash.merge": { "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", @@ -5369,6 +5392,19 @@ "resolved": "https://registry.npmjs.org/@patternfly/react-styles/-/react-styles-5.2.1.tgz", "integrity": "sha512-GT96hzI1QenBhq6Pfc51kxnj9aVLjL1zSLukKZXcYVe0HPOy0BFm90bT1Fo4e/z7V9cDYw4SqSX1XLc3O4jsTw==" }, + "@patternfly/react-table": { + "version": "5.2.4", + "resolved": "https://registry.npmjs.org/@patternfly/react-table/-/react-table-5.2.4.tgz", + "integrity": "sha512-WCt4I6XYKRHXcasDqcOX70ctkgPVBAvlOv67KhaZsedxUU+B2NT1kiI7Jr7tD4SrR+jQs0MCF/bbRk1QM+rBqg==", + "requires": { + "@patternfly/react-core": "^5.2.3", + "@patternfly/react-icons": "^5.2.1", + "@patternfly/react-styles": "^5.2.1", + "@patternfly/react-tokens": "^5.2.1", + "lodash": "^4.17.19", + "tslib": "^2.5.0" + } + }, "@patternfly/react-tokens": { "version": "5.2.1", "resolved": "https://registry.npmjs.org/@patternfly/react-tokens/-/react-tokens-5.2.1.tgz", @@ -7086,6 +7122,11 @@ "p-locate": "^5.0.0" } }, + "lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" + }, "lodash.merge": { "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", diff --git a/frontend/package.json b/frontend/package.json index 9ea35a84..5b7905a8 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -11,6 +11,7 @@ }, "dependencies": { "@patternfly/react-core": "^5.2.3", + "@patternfly/react-table": "^5.2.4", "@reduxjs/toolkit": "^2.2.3", "axios": "^1.6.8", "react": "^18.2.0", diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js new file mode 100644 index 00000000..37ee3233 --- /dev/null +++ b/frontend/src/actions/homeActions.js @@ -0,0 +1,28 @@ +import * as API_ROUTES from "@/utils/apiConstants"; +import * as TYPES from "@/actions/types.js"; + +import API from "@/utils/axiosInstance"; +import { showFailureToast } from "@/actions/toastActions"; + +export const fetchOCPJobsData = () => async (dispatch, getState) => { + try { + dispatch({ type: TYPES.LOADING }); + const { start_date, end_date } = getState().cpt; + const response = await API.get(API_ROUTES.CPT_JOBS_API_V1, { + params: { + pretty: true, + ...(start_date && start_date), + ...(end_date && end_date), + }, + }); + if (response?.data?.results?.length > 0) { + dispatch({ + type: TYPES.SET_OCP_JOBS_DATA, + payload: response.data, + }); + } + } catch (error) { + dispatch(showFailureToast()); + } + dispatch({ type: TYPES.COMPLETED }); +}; diff --git a/frontend/src/actions/types.js b/frontend/src/actions/types.js index 2db9927a..0bbda640 100644 --- a/frontend/src/actions/types.js +++ b/frontend/src/actions/types.js @@ -9,3 +9,6 @@ export const CLEAR_TOAST = "CLEAR_TOAST"; /* Sidemenu */ export const SET_ACTIVE_MENU_ITEM = "SET_ACTIVE_MENU_ITEM"; export const TOGGLE_SIDE_MENU = "TOGGLE_SIDE_MENU"; + +/* OCP Jobs */ +export const SET_OCP_JOBS_DATA = "SET_OCP_JOBS_DATA"; diff --git a/frontend/src/components/atoms/TableCell/index.jsx b/frontend/src/components/atoms/TableCell/index.jsx new file mode 100644 index 00000000..c80b6198 --- /dev/null +++ b/frontend/src/components/atoms/TableCell/index.jsx @@ -0,0 +1,54 @@ +import { Button, Label } from "@patternfly/react-core"; +import { + CheckCircleIcon, + ExclamationCircleIcon, + ExternalLinkSquareAltIcon, +} from "@patternfly/react-icons"; +import { formatDateTime, uid } from "@/utils/helper.js"; + +import { Td } from "@patternfly/react-table"; + +const TableCell = (props) => { + const { col, item } = props; + return ( + + {col.value === "jobStatus" ? ( + + ) : col.value === "startDate" || col.value === "endDate" ? ( + formatDateTime(item[col.value]) + ) : col.value === "buildUrl" ? ( + + ) : ( + item[col.value] + )} + + ); +}; + +const StatusCell = (props) => { + const { item, col } = props; + return item[col.value] === "success" ? ( + + ) : ( + + ); +}; + +const BuildURLCell = (props) => ( + +); + +export default TableCell; diff --git a/frontend/src/components/molecules/SideMenuOptions/index.jsx b/frontend/src/components/molecules/SideMenuOptions/index.jsx index 754919da..9117345b 100644 --- a/frontend/src/components/molecules/SideMenuOptions/index.jsx +++ b/frontend/src/components/molecules/SideMenuOptions/index.jsx @@ -27,15 +27,16 @@ const sideMenuOptions = [ const MenuOptions = () => { const dispatch = useDispatch(); + const navigate = useNavigate(); const { pathname } = useLocation(); const activeMenuItem = useSelector((state) => state.sidemenu.activeMenuItem); const onSelect = (_event, item) => { dispatch(setActiveItem(item.itemId)); }; - const navigate = useNavigate(); + useEffect(() => { - if (pathname) { + if (pathname !== "/") { const currPath = pathname.replace(/^.*[/]([^/]+)[/]*$/, "$1"); dispatch(setActiveItem(currPath)); diff --git a/frontend/src/components/molecules/TableRows/index.jsx b/frontend/src/components/molecules/TableRows/index.jsx new file mode 100644 index 00000000..ea1cd707 --- /dev/null +++ b/frontend/src/components/molecules/TableRows/index.jsx @@ -0,0 +1,52 @@ +import TableCell from "../../atoms/TableCell"; +import { Tr } from "@patternfly/react-table"; +import { uid } from "@/utils/helper.js"; + +const TableRows = (props) => { + const { rows, columns } = props; + return ( + rows?.length > 0 && + rows.map((item) => ( + + {columns.map((col) => ( + + ))} + + )) + ); +}; + +// const TableCell = (props) => { +// const { col, item } = props; +// return ( +// +// {col.name === "Status" ? ( +// item[col.value] === "success" ? ( +// +// ) : ( +// +// ) +// ) : col.name === "Start Date" || col.name === "End Date" ? ( +// formatDateTime(item[col.value]) +// ) : col.name === "Build URL" ? ( +// +// ) : ( +// item[col.value] +// )} +// +// ); +// }; +export default TableRows; diff --git a/frontend/src/components/organisms/Header/index.jsx b/frontend/src/components/organisms/Header/index.jsx index 4090b29a..eefeb44c 100644 --- a/frontend/src/components/organisms/Header/index.jsx +++ b/frontend/src/components/organisms/Header/index.jsx @@ -32,14 +32,14 @@ const Header = () => { aria-label="Global navigation" isSidebarOpen={isSideMenuOpen} onSidebarToggle={onSidebarToggle} - id="fill-nav-toggle" + id="nav-toggle" > - + diff --git a/frontend/src/components/organisms/TableLayout/index.jsx b/frontend/src/components/organisms/TableLayout/index.jsx new file mode 100644 index 00000000..acd602d8 --- /dev/null +++ b/frontend/src/components/organisms/TableLayout/index.jsx @@ -0,0 +1,25 @@ +import { Table, Tbody, Th, Thead, Tr } from "@patternfly/react-table"; + +import TableRows from "../../molecules/TableRows"; +import { uid } from "@/utils/helper.js"; + +const TableLayout = (props) => { + const { tableData, tableColumns } = props; + + const columns = tableColumns?.map((item) => item.value); + return ( + + + + {tableColumns?.length > 0 && + tableColumns.map((col) => )} + + + + + +
{col.name}
+ ); +}; + +export default TableLayout; diff --git a/frontend/src/components/templates/Home/index.jsx b/frontend/src/components/templates/Home/index.jsx index 0b0a9e6a..e502ef2e 100644 --- a/frontend/src/components/templates/Home/index.jsx +++ b/frontend/src/components/templates/Home/index.jsx @@ -1,5 +1,26 @@ +import { useDispatch, useSelector } from "react-redux"; + +import TableLayout from "@/components/organisms/TableLayout"; +import { fetchOCPJobsData } from "@/actions/homeActions.js"; +import { useEffect } from "react"; + const Home = () => { - return <>Test; + const dispatch = useDispatch(); + + const { results, tableColumns } = useSelector((state) => state.cpt); + + useEffect(() => { + dispatch(fetchOCPJobsData()); + }, [dispatch]); + + // const columnNames = {}; + // tableColumns.map((item) => (columnNames[item.value] = item.name)); + + return ( + <> + + + ); }; export default Home; diff --git a/frontend/src/reducers/homeReducer.js b/frontend/src/reducers/homeReducer.js new file mode 100644 index 00000000..a5d26be6 --- /dev/null +++ b/frontend/src/reducers/homeReducer.js @@ -0,0 +1,36 @@ +import * as TYPES from "@/actions/types"; + +const initialState = { + results: [], + start_date: "", + end_date: "", + tableColumns: [ + { name: "Product", value: "product" }, + { name: "CI System", value: "ciSystem" }, + { name: "Test Name", value: "testName" }, + { name: "Version", value: "version" }, + { name: "Release Stream", value: "releaseStream" }, + { name: "Build URL", value: "buildUrl" }, + { name: "Start Date", value: "startDate" }, + { name: "End Date", value: "endDate" }, + { name: "Status", value: "jobStatus" }, + ], +}; + +const HomeReducer = (state = initialState, action = {}) => { + const { type, payload } = action; + switch (type) { + case TYPES.SET_OCP_JOBS_DATA: + return { + ...state, + results: payload.results, + start_date: payload.startDate, + end_date: payload.endDate, + }; + + default: + return state; + } +}; + +export default HomeReducer; diff --git a/frontend/src/reducers/index.js b/frontend/src/reducers/index.js index 7b225c6c..13de5189 100644 --- a/frontend/src/reducers/index.js +++ b/frontend/src/reducers/index.js @@ -1,3 +1,4 @@ +import HomeReducer from "./homeReducer"; import LoadingReducer from "./loadingReducer"; import SideMenuReducer from "./sideMenuReducer"; import ToastReducer from "./toastReducer"; @@ -7,4 +8,5 @@ export default combineReducers({ loading: LoadingReducer, toast: ToastReducer, sidemenu: SideMenuReducer, + cpt: HomeReducer, }); diff --git a/frontend/src/reducers/sideMenuReducer.js b/frontend/src/reducers/sideMenuReducer.js index 5470a9b6..e8bc2c4a 100644 --- a/frontend/src/reducers/sideMenuReducer.js +++ b/frontend/src/reducers/sideMenuReducer.js @@ -15,7 +15,7 @@ const SideMenuReducer = (state = initialState, action = {}) => { activeMenuItem: payload, }; case TOGGLE_SIDE_MENU: - return { ...state, isSideMenuOpen: payload }; + return { ...state, isSideMenuOpen: false }; default: return state; } diff --git a/frontend/src/utils/helper.js b/frontend/src/utils/helper.js index 7d72ef8b..d6f4b1b2 100644 --- a/frontend/src/utils/helper.js +++ b/frontend/src/utils/helper.js @@ -4,3 +4,26 @@ export const uid = () => { return head + tail; }; + +export const isEmptyObject = (obj) => { + return ( + obj && + Object.keys(obj).length === 0 && + Object.getPrototypeOf(obj) === Object.prototype + ); +}; + +/** + * Convert a date string into Locale Date String with seconds and milli seconds removed * + * @function + * @param {string} dateTimeStamp - Date in string format + * @return {string} - Locale Date string + */ + +export const formatDateTime = (dateTimeStamp) => { + const dateObj = new Date(dateTimeStamp); + return new Intl.DateTimeFormat("en-US", { + dateStyle: "medium", + timeStyle: "short", + }).format(dateObj); +}; From 4ecd5b4e50ea475893a5d5aeaae059023affa1c8 Mon Sep 17 00:00:00 2001 From: MVarshini Date: Wed, 3 Apr 2024 14:55:04 +0530 Subject: [PATCH 04/14] sort --- frontend/src/actions/homeActions.js | 53 ++++++++++++++++++- frontend/src/actions/types.js | 5 +- .../organisms/TableLayout/index.jsx | 29 ++++++++-- .../src/components/templates/Home/index.jsx | 27 ++++++++-- frontend/src/reducers/homeReducer.js | 15 ++++-- 5 files changed, 117 insertions(+), 12 deletions(-) diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js index 37ee3233..89aa482b 100644 --- a/frontend/src/actions/homeActions.js +++ b/frontend/src/actions/homeActions.js @@ -17,8 +17,15 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { }); if (response?.data?.results?.length > 0) { dispatch({ - type: TYPES.SET_OCP_JOBS_DATA, - payload: response.data, + type: TYPES.SET_CPT_JOBS_DATA, + payload: response.data.results, + }); + dispatch({ + type: TYPES.SET_CPT_DATE_FILTER, + payload: { + start_date: response.data.startDate, + end_date: response.data.endDate, + }, }); } } catch (error) { @@ -26,3 +33,45 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { } dispatch({ type: TYPES.COMPLETED }); }; + +const getSortableRowValues = (result, tableColumns) => { + const tableKeys = tableColumns.map((item) => item.value); + return tableKeys.map((key) => result[key]); +}; + +export const sortTable = () => (dispatch, getState) => { + const { results, activeSortDir, activeSortIndex, tableColumns } = + getState().cpt; + + if (activeSortIndex) { + const sortedResults = results.sort((a, b) => { + const aValue = getSortableRowValues(a, tableColumns)[activeSortIndex]; + const bValue = getSortableRowValues(b, tableColumns)[activeSortIndex]; + if (typeof aValue === "number") { + if (activeSortDir === "asc") { + return aValue - bValue; + } + return bValue - aValue; + } else { + if (activeSortDir === "asc") { + return aValue.localeCompare(bValue); + } + return bValue.localeCompare(aValue); + } + }); + dispatch({ + type: TYPES.SET_CPT_JOBS_DATA, + payload: sortedResults, + }); + } +}; + +export const setCPTSortIndex = (index) => ({ + type: TYPES.SET_CPT_SORT_INDEX, + payload: index, +}); + +export const setCPTSortDir = (direction) => ({ + type: TYPES.SET_CPT_SORT_DIR, + payload: direction, +}); diff --git a/frontend/src/actions/types.js b/frontend/src/actions/types.js index 0bbda640..16c6063c 100644 --- a/frontend/src/actions/types.js +++ b/frontend/src/actions/types.js @@ -11,4 +11,7 @@ export const SET_ACTIVE_MENU_ITEM = "SET_ACTIVE_MENU_ITEM"; export const TOGGLE_SIDE_MENU = "TOGGLE_SIDE_MENU"; /* OCP Jobs */ -export const SET_OCP_JOBS_DATA = "SET_OCP_JOBS_DATA"; +export const SET_CPT_JOBS_DATA = "SET_CPT_JOBS_DATA"; +export const SET_CPT_DATE_FILTER = "SET_CPT_DATE_FILTER"; +export const SET_CPT_SORT_DIR = "SET_CPT_SORT_DIR"; +export const SET_CPT_SORT_INDEX = "SET_CPT_SORT_INDEX"; diff --git a/frontend/src/components/organisms/TableLayout/index.jsx b/frontend/src/components/organisms/TableLayout/index.jsx index acd602d8..4c216851 100644 --- a/frontend/src/components/organisms/TableLayout/index.jsx +++ b/frontend/src/components/organisms/TableLayout/index.jsx @@ -4,15 +4,38 @@ import TableRows from "../../molecules/TableRows"; import { uid } from "@/utils/helper.js"; const TableLayout = (props) => { - const { tableData, tableColumns } = props; + const { + tableData, + tableColumns, + activeSortIndex, + activeSortDir, + setActiveSortIndex, + setActiveSortDir, + } = props; + + const getSortParams = (columnIndex) => ({ + sortBy: { + index: activeSortIndex, + direction: activeSortDir, + defaultDirection: "asc", + }, + onSort: (_event, index, direction) => { + setActiveSortIndex(index); + setActiveSortDir(direction); + }, + columnIndex, + }); - const columns = tableColumns?.map((item) => item.value); return ( {tableColumns?.length > 0 && - tableColumns.map((col) => )} + tableColumns.map((col, idx) => ( + + ))} diff --git a/frontend/src/components/templates/Home/index.jsx b/frontend/src/components/templates/Home/index.jsx index e502ef2e..2abd4276 100644 --- a/frontend/src/components/templates/Home/index.jsx +++ b/frontend/src/components/templates/Home/index.jsx @@ -1,13 +1,19 @@ +import { + fetchOCPJobsData, + setCPTSortDir, + setCPTSortIndex, +} from "@/actions/homeActions.js"; import { useDispatch, useSelector } from "react-redux"; import TableLayout from "@/components/organisms/TableLayout"; -import { fetchOCPJobsData } from "@/actions/homeActions.js"; import { useEffect } from "react"; const Home = () => { const dispatch = useDispatch(); - const { results, tableColumns } = useSelector((state) => state.cpt); + const { results, tableColumns, activeSortDir, activeSortIndex } = useSelector( + (state) => state.cpt + ); useEffect(() => { dispatch(fetchOCPJobsData()); @@ -16,9 +22,24 @@ const Home = () => { // const columnNames = {}; // tableColumns.map((item) => (columnNames[item.value] = item.name)); + const setActiveSortDir = (dir) => { + dispatch(setCPTSortDir(dir)); + }; + + const setActiveSortIndex = (index) => { + dispatch(setCPTSortIndex(index)); + }; + return ( <> - + ); }; diff --git a/frontend/src/reducers/homeReducer.js b/frontend/src/reducers/homeReducer.js index a5d26be6..fa20047e 100644 --- a/frontend/src/reducers/homeReducer.js +++ b/frontend/src/reducers/homeReducer.js @@ -15,19 +15,28 @@ const initialState = { { name: "End Date", value: "endDate" }, { name: "Status", value: "jobStatus" }, ], + activeSortDir: null, + activeSortIndex: null, }; const HomeReducer = (state = initialState, action = {}) => { const { type, payload } = action; switch (type) { - case TYPES.SET_OCP_JOBS_DATA: + case TYPES.SET_CPT_JOBS_DATA: + return { + ...state, + results: payload, + }; + case TYPES.SET_CPT_DATE_FILTER: return { ...state, - results: payload.results, start_date: payload.startDate, end_date: payload.endDate, }; - + case TYPES.SET_CPT_SORT_INDEX: + return { ...state, activeSortIndex: payload }; + case TYPES.SET_CPT_SORT_DIR: + return { ...state, activeSortDir: payload }; default: return state; } From 6b755a499f6f405e9ad55bbd397bc92415ebc7ad Mon Sep 17 00:00:00 2001 From: MVarshini Date: Wed, 10 Apr 2024 14:06:54 +0530 Subject: [PATCH 05/14] Pagination --- frontend/jsconfig.json | 11 +++ frontend/src/actions/homeActions.js | 61 +++++++++++------ frontend/src/actions/types.js | 3 +- .../assets/constants/paginationConstants.js | 2 + .../src/components/atoms/TableCell/index.jsx | 13 ++++ .../components/organisms/Pagination/index.jsx | 8 ++- .../organisms/TableLayout/index.jsx | 68 ++++++++++++++----- .../src/components/templates/Home/index.jsx | 50 +++++++++++--- frontend/src/reducers/homeReducer.js | 3 + frontend/vite.config.js | 1 + 10 files changed, 171 insertions(+), 49 deletions(-) create mode 100644 frontend/jsconfig.json create mode 100644 frontend/src/assets/constants/paginationConstants.js diff --git a/frontend/jsconfig.json b/frontend/jsconfig.json new file mode 100644 index 00000000..123053e3 --- /dev/null +++ b/frontend/jsconfig.json @@ -0,0 +1,11 @@ +{ + "compilerOptions": { + "baseUrl": "src", + "paths": { + "*": [ + "src/*" + ] + } + } + } + \ No newline at end of file diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js index 89aa482b..b474b5ec 100644 --- a/frontend/src/actions/homeActions.js +++ b/frontend/src/actions/homeActions.js @@ -2,6 +2,7 @@ import * as API_ROUTES from "@/utils/apiConstants"; import * as TYPES from "@/actions/types.js"; import API from "@/utils/axiosInstance"; +import { DEFAULT_PER_PAGE } from "@/assets/constants/paginationConstants"; import { showFailureToast } from "@/actions/toastActions"; export const fetchOCPJobsData = () => async (dispatch, getState) => { @@ -27,6 +28,8 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { end_date: response.data.endDate, }, }); + dispatch(sortTable()); + dispatch(sliceTableRows(0, DEFAULT_PER_PAGE)); } } catch (error) { dispatch(showFailureToast()); @@ -40,29 +43,34 @@ const getSortableRowValues = (result, tableColumns) => { }; export const sortTable = () => (dispatch, getState) => { - const { results, activeSortDir, activeSortIndex, tableColumns } = - getState().cpt; - - if (activeSortIndex) { - const sortedResults = results.sort((a, b) => { - const aValue = getSortableRowValues(a, tableColumns)[activeSortIndex]; - const bValue = getSortableRowValues(b, tableColumns)[activeSortIndex]; - if (typeof aValue === "number") { - if (activeSortDir === "asc") { - return aValue - bValue; - } - return bValue - aValue; - } else { - if (activeSortDir === "asc") { - return aValue.localeCompare(bValue); + const { activeSortDir, activeSortIndex, tableColumns } = getState().cpt; + const results = [...getState().cpt.results]; + try { + if (activeSortIndex) { + const sortedResults = results.sort((a, b) => { + const aValue = getSortableRowValues(a, tableColumns)[activeSortIndex]; + const bValue = getSortableRowValues(b, tableColumns)[activeSortIndex]; + if (typeof aValue === "number") { + if (activeSortDir === "asc") { + return aValue - bValue; + } + return bValue - aValue; + } else { + if (activeSortDir === "asc") { + return aValue.localeCompare(bValue); + } + return bValue.localeCompare(aValue); } - return bValue.localeCompare(aValue); - } - }); - dispatch({ - type: TYPES.SET_CPT_JOBS_DATA, - payload: sortedResults, - }); + }); + dispatch({ + type: TYPES.SET_CPT_JOBS_DATA, + payload: sortedResults, + }); + dispatch(sliceTableRows(0, DEFAULT_PER_PAGE)); + } + } catch (error) { + console.log(error); + dispatch(showFailureToast()); } }; @@ -75,3 +83,12 @@ export const setCPTSortDir = (direction) => ({ type: TYPES.SET_CPT_SORT_DIR, payload: direction, }); + +export const sliceTableRows = (startIdx, endIdx) => (dispatch, getState) => { + const results = [...getState().cpt.results]; + + dispatch({ + type: TYPES.SET_CPT_INIT_JOBS, + payload: results.slice(startIdx, endIdx), + }); +}; diff --git a/frontend/src/actions/types.js b/frontend/src/actions/types.js index 16c6063c..66e8f5b1 100644 --- a/frontend/src/actions/types.js +++ b/frontend/src/actions/types.js @@ -10,8 +10,9 @@ export const CLEAR_TOAST = "CLEAR_TOAST"; export const SET_ACTIVE_MENU_ITEM = "SET_ACTIVE_MENU_ITEM"; export const TOGGLE_SIDE_MENU = "TOGGLE_SIDE_MENU"; -/* OCP Jobs */ +/* CPT Jobs */ export const SET_CPT_JOBS_DATA = "SET_CPT_JOBS_DATA"; export const SET_CPT_DATE_FILTER = "SET_CPT_DATE_FILTER"; export const SET_CPT_SORT_DIR = "SET_CPT_SORT_DIR"; export const SET_CPT_SORT_INDEX = "SET_CPT_SORT_INDEX"; +export const SET_CPT_INIT_JOBS = "SET_CPT_INIT_JOBS"; diff --git a/frontend/src/assets/constants/paginationConstants.js b/frontend/src/assets/constants/paginationConstants.js new file mode 100644 index 00000000..90460014 --- /dev/null +++ b/frontend/src/assets/constants/paginationConstants.js @@ -0,0 +1,2 @@ +export const DEFAULT_PER_PAGE = 25; +export const START_PAGE = 1; diff --git a/frontend/src/components/atoms/TableCell/index.jsx b/frontend/src/components/atoms/TableCell/index.jsx index c80b6198..fec0b390 100644 --- a/frontend/src/components/atoms/TableCell/index.jsx +++ b/frontend/src/components/atoms/TableCell/index.jsx @@ -6,6 +6,7 @@ import { } from "@patternfly/react-icons"; import { formatDateTime, uid } from "@/utils/helper.js"; +import PropTypes from "prop-types"; import { Td } from "@patternfly/react-table"; const TableCell = (props) => { @@ -51,4 +52,16 @@ const BuildURLCell = (props) => ( ); +StatusCell.propTypes = { + item: PropTypes.object, + col: PropTypes.object, +}; +TableCell.propTypes = { + item: PropTypes.object, + col: PropTypes.object, +}; +BuildURLCell.propTypes = { + item: PropTypes.object, + col: PropTypes.object, +}; export default TableCell; diff --git a/frontend/src/components/organisms/Pagination/index.jsx b/frontend/src/components/organisms/Pagination/index.jsx index 0f34dd3b..490a801d 100644 --- a/frontend/src/components/organisms/Pagination/index.jsx +++ b/frontend/src/components/organisms/Pagination/index.jsx @@ -1,6 +1,12 @@ import { Pagination, PaginationVariant } from "@patternfly/react-core"; const RenderPagination = (props) => { + const perPageOptions = [ + { title: "25", value: 25 }, + { title: "50", value: 50 }, + { title: "100", value: 100 }, + ]; + return ( { perPage={props.perPage} page={props.page} variant={PaginationVariant.bottom} - perPageOptions={props.perPageOptions} + perPageOptions={perPageOptions} onSetPage={props.onSetPage} onPerPageSelect={props.onPerPageSelect} /> diff --git a/frontend/src/components/organisms/TableLayout/index.jsx b/frontend/src/components/organisms/TableLayout/index.jsx index 4c216851..64cd0dac 100644 --- a/frontend/src/components/organisms/TableLayout/index.jsx +++ b/frontend/src/components/organisms/TableLayout/index.jsx @@ -1,6 +1,8 @@ import { Table, Tbody, Th, Thead, Tr } from "@patternfly/react-table"; -import TableRows from "../../molecules/TableRows"; +import PropTypes from "prop-types"; +import RenderPagination from "@/components/organisms/Pagination"; +import TableRows from "@/components/molecules/TableRows"; import { uid } from "@/utils/helper.js"; const TableLayout = (props) => { @@ -11,6 +13,14 @@ const TableLayout = (props) => { activeSortDir, setActiveSortIndex, setActiveSortDir, + handleOnSort, + page, + perPage, + setPage, + setPerPage, + onSetPage, + onPerPageSelect, + totalItems, } = props; const getSortParams = (columnIndex) => ({ @@ -22,27 +32,53 @@ const TableLayout = (props) => { onSort: (_event, index, direction) => { setActiveSortIndex(index); setActiveSortDir(direction); + handleOnSort(); }, columnIndex, }); return ( -
{col.name} + {col.name} +
- - - {tableColumns?.length > 0 && - tableColumns.map((col, idx) => ( - - ))} - - - - - -
- {col.name} -
+ <> + + + + {tableColumns?.length > 0 && + tableColumns.map((col, idx) => ( + + ))} + + + + + +
+ {col.name} +
+ + ); }; +TableLayout.propTypes = { + tableData: PropTypes.array, + tableColumns: PropTypes.array, + activeSortIndex: PropTypes.number || PropTypes.object, + activeSortDir: PropTypes.string || PropTypes.object, + setActiveSortIndex: PropTypes.func, + setActiveSortDir: PropTypes.func, + handleOnSort: PropTypes.func, + totalItems: PropTypes.number, + page: PropTypes.number, + perPage: PropTypes.number, + onPerPageSelect: PropTypes.func, + onSetPage: PropTypes.func, +}; export default TableLayout; diff --git a/frontend/src/components/templates/Home/index.jsx b/frontend/src/components/templates/Home/index.jsx index 2abd4276..1360a487 100644 --- a/frontend/src/components/templates/Home/index.jsx +++ b/frontend/src/components/templates/Home/index.jsx @@ -1,44 +1,76 @@ +import { + DEFAULT_PER_PAGE, + START_PAGE, +} from "@/assets/constants/paginationConstants"; import { fetchOCPJobsData, setCPTSortDir, setCPTSortIndex, + sliceTableRows, + sortTable, } from "@/actions/homeActions.js"; +import { useCallback, useEffect, useState } from "react"; import { useDispatch, useSelector } from "react-redux"; import TableLayout from "@/components/organisms/TableLayout"; -import { useEffect } from "react"; const Home = () => { const dispatch = useDispatch(); - const { results, tableColumns, activeSortDir, activeSortIndex } = useSelector( - (state) => state.cpt - ); + const { results, tableColumns, activeSortDir, activeSortIndex, tableData } = + useSelector((state) => state.cpt); useEffect(() => { dispatch(fetchOCPJobsData()); }, [dispatch]); - // const columnNames = {}; - // tableColumns.map((item) => (columnNames[item.value] = item.name)); - + //Sorting const setActiveSortDir = (dir) => { dispatch(setCPTSortDir(dir)); }; - const setActiveSortIndex = (index) => { dispatch(setCPTSortIndex(index)); }; + const handleOnSort = () => { + dispatch(sortTable()); + }; + // Sorting + // Pagination Helper + const [perPage, setPerPage] = useState(DEFAULT_PER_PAGE); + const [page, setPage] = useState(START_PAGE); + + const onSetPage = useCallback( + (_evt, newPage, _perPage, startIdx, endIdx) => { + setPage(newPage); + dispatch(sliceTableRows(startIdx, endIdx)); + }, + [dispatch] + ); + const onPerPageSelect = useCallback( + (_evt, newPerPage, newPage, startIdx, endIdx) => { + setPerPage(newPerPage); + setPage(newPage); + dispatch(sliceTableRows(startIdx, endIdx)); + }, + [dispatch] + ); + // Pagination helper return ( <> ); diff --git a/frontend/src/reducers/homeReducer.js b/frontend/src/reducers/homeReducer.js index fa20047e..f2d9cc76 100644 --- a/frontend/src/reducers/homeReducer.js +++ b/frontend/src/reducers/homeReducer.js @@ -17,6 +17,7 @@ const initialState = { ], activeSortDir: null, activeSortIndex: null, + tableData: [], }; const HomeReducer = (state = initialState, action = {}) => { @@ -37,6 +38,8 @@ const HomeReducer = (state = initialState, action = {}) => { return { ...state, activeSortIndex: payload }; case TYPES.SET_CPT_SORT_DIR: return { ...state, activeSortDir: payload }; + case TYPES.SET_CPT_INIT_JOBS: + return { ...state, tableData: payload }; default: return state; } diff --git a/frontend/vite.config.js b/frontend/vite.config.js index fdd212e8..76f9be17 100644 --- a/frontend/vite.config.js +++ b/frontend/vite.config.js @@ -2,6 +2,7 @@ import { defineConfig } from "vite"; import path from "path"; import react from "@vitejs/plugin-react"; // https://vitejs.dev/config/ + export default defineConfig({ plugins: [react()], esbuild: { From 7f78e17a166112714158c5529150d7655b4266ab Mon Sep 17 00:00:00 2001 From: MVarshini Date: Wed, 10 Apr 2024 14:46:34 +0530 Subject: [PATCH 06/14] Readme file update --- frontend/README.md | 71 +++++++++++++++++-- .../components/molecules/TableRows/index.jsx | 33 --------- 2 files changed, 66 insertions(+), 38 deletions(-) diff --git a/frontend/README.md b/frontend/README.md index f768e33f..36c15bac 100644 --- a/frontend/README.md +++ b/frontend/README.md @@ -1,8 +1,69 @@ -# React + Vite -This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules. -Currently, two official plugins are available: -- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react/README.md) uses [Babel](https://babeljs.io/) for Fast Refresh -- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh +# Openshift Performance Dashbaord + +## Dashboard directory structure + +### [`src`](src/) + +The CPT Dashboard Javascript source plus additional CSS/LESS and artifacts. + +#### [`src/assets`](src/assets/) + +Assets placed in the `src/assets/images` directory are only referenced within component or layout definitions and are packaged in the generated `***.js` file during the build process. + +#### [`src/modules`](src/modules/) + +`modules` directory has all containers (patent layouts) and components (react components). + +#### [`src/utils`](src/utils/) + +The `utils` directory has all helper/utility scripts. + +#### [`src/reducers`](src/reducers) + +Contains functions that manage store via actions + +## Cloning and Running the Application Locally + +- Install [Node.js](https://nodejs.org) +- Clone the [CPT Dashboard code](https://github.com/cloud-bulldozer/cpt-dashboard) to a local file system +- Install all the npm packages + +Type the following command to install all npm packages + +```bash +$ npm install +``` + +In order to run the application use the following command + +```bash +$ npm run dev +``` + +The application runs on http://localhost:3000 in the default browser. + +## Build + +To build the application run the following command + +```bash +$ npm run build +``` +This will generate the `build` folder in the root directory, which contains packaged files such as `***.js`, `***.css`, and `index.html`. + +Then, copy the `build` folder to the proper place on the server for deployment. + +## Template + +This application is based on v5 of PatternFly which is a production-ready UI solution for admin interfaces. For more information regarding the foundation and template of the application, please visit [PatternFly](https://www.patternfly.org/get-started/develop) + +## Resources + +- [Vite](https://vitejs.dev/guide/) + +- [ReactJS](https://reactjs.org/) + +- [React-Redux](https://github.com/reduxjs/react-redux) \ No newline at end of file diff --git a/frontend/src/components/molecules/TableRows/index.jsx b/frontend/src/components/molecules/TableRows/index.jsx index ea1cd707..15d6683a 100644 --- a/frontend/src/components/molecules/TableRows/index.jsx +++ b/frontend/src/components/molecules/TableRows/index.jsx @@ -16,37 +16,4 @@ const TableRows = (props) => { ); }; -// const TableCell = (props) => { -// const { col, item } = props; -// return ( -// -// {col.name === "Status" ? ( -// item[col.value] === "success" ? ( -// -// ) : ( -// -// ) -// ) : col.name === "Start Date" || col.name === "End Date" ? ( -// formatDateTime(item[col.value]) -// ) : col.name === "Build URL" ? ( -// -// ) : ( -// item[col.value] -// )} -// -// ); -// }; export default TableRows; From c6004ddebfbe30ff30451523a5ccaa1e43f2a9f8 Mon Sep 17 00:00:00 2001 From: MVarshini Date: Fri, 12 Apr 2024 22:02:40 +0530 Subject: [PATCH 07/14] Dockerfile update --- frontend/frontend.containerfile | 31 ++++++++++++------------------- frontend/vite.config.js | 2 +- local-compose.sh | 2 +- 3 files changed, 14 insertions(+), 21 deletions(-) diff --git a/frontend/frontend.containerfile b/frontend/frontend.containerfile index e40cf576..360b53e7 100644 --- a/frontend/frontend.containerfile +++ b/frontend/frontend.containerfile @@ -1,20 +1,13 @@ -# pull official base image -FROM docker.io/library/node:16.20.2-alpine3.18 - -RUN mkdir /app - - -COPY . /app -WORKDIR /app - -RUN yarn install -RUN yarn add react-scripts -ENV PATH="${PATH}:/app/node_modules/.bin/react-scripts" - +FROM node:18 +# Create app directory +WORKDIR /usr/src/cpt-dashboard +# Install app dependencies +# A wildcard is used to ensure both package.json AND package-lock.json are copied +# where available (npm@5+) +COPY package*.json ./ +RUN npm install +#To bundle your app’s source code inside the Docker image, use the COPY instruction: +COPY . . +#Your app binds to port 3000 so you’ll use the EXPOSE instruction to have it mapped by the docker daemon: EXPOSE 3000 - -RUN chown -R node /app -USER node - -# start app -CMD ["yarn", "run", "start"] +CMD ["npm", "run", "dev"] \ No newline at end of file diff --git a/frontend/vite.config.js b/frontend/vite.config.js index 76f9be17..317c052e 100644 --- a/frontend/vite.config.js +++ b/frontend/vite.config.js @@ -15,6 +15,6 @@ export default defineConfig({ }, server: { port: 3000, - open: true, + open: false, }, }); diff --git a/local-compose.sh b/local-compose.sh index e18ee8f8..8f7801b6 100755 --- a/local-compose.sh +++ b/local-compose.sh @@ -12,4 +12,4 @@ podman build -f frontend/frontend.containerfile --tag frontend podman run -d --name=back -p 8000:8000 -v "$PWD/backend/ocpperf.toml:/backend/ocpperf.toml:Z" localhost/backend -podman run -d --name=front -p 3000:3000 localhost/frontend +podman run -d --name=front --net=host -p 3000:3000 localhost/frontend From 84f16d7b4e3486b0714ae84a52eba6e3b0bc6d47 Mon Sep 17 00:00:00 2001 From: MVarshini Date: Wed, 24 Apr 2024 15:06:51 +0530 Subject: [PATCH 08/14] PANDA-364 Table Filters with Category Menu --- frontend/src/actions/homeActions.js | 44 ++++++++++++- frontend/src/actions/types.js | 3 + .../components/molecules/SelectBox/index.jsx | 61 +++++++++++++++++++ .../organisms/TableFilters/index.jsx | 43 +++++++++++++ .../src/components/templates/Home/index.jsx | 13 +++- frontend/src/reducers/homeReducer.js | 21 ++++++- 6 files changed, 179 insertions(+), 6 deletions(-) create mode 100644 frontend/src/components/molecules/SelectBox/index.jsx create mode 100644 frontend/src/components/organisms/TableFilters/index.jsx diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js index b474b5ec..4df76019 100644 --- a/frontend/src/actions/homeActions.js +++ b/frontend/src/actions/homeActions.js @@ -12,8 +12,10 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { const response = await API.get(API_ROUTES.CPT_JOBS_API_V1, { params: { pretty: true, - ...(start_date && start_date), - ...(end_date && end_date), + start_date: "2024-04-21", + end_date: "2024-04-22", + // ...(start_date && start_date), + // ...(end_date && end_date), }, }); if (response?.data?.results?.length > 0) { @@ -30,6 +32,7 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { }); dispatch(sortTable()); dispatch(sliceTableRows(0, DEFAULT_PER_PAGE)); + dispatch(buildFilterData()); } } catch (error) { dispatch(showFailureToast()); @@ -92,3 +95,40 @@ export const sliceTableRows = (startIdx, endIdx) => (dispatch, getState) => { payload: results.slice(startIdx, endIdx), }); }; + +export const buildFilterData = () => (dispatch, getState) => { + const results = [...getState().cpt.results]; + + const tableFilters = [...getState().cpt.tableFilters]; + + const filterData = []; + for (const filter of tableFilters) { + const key = filter.value; + let obj = { + name: filter.name, + key, + value: [...new Set(results.map((item) => item[key])), "ALL"], + }; + filterData.push(obj); + } + dispatch({ + type: TYPES.SET_CPT_FILTER_DATA, + payload: filterData, + }); + dispatch(setCatFilters(tableFilters[0].name)); +}; + +export const setCatFilters = (category) => (dispatch, getState) => { + const filterData = [...getState().cpt.filterData]; + const options = filterData.filter((item) => item.name === category)[0].value; + const list = options.map((item) => ({ name: item, value: item })); + + dispatch({ + type: TYPES.SET_CATEGORY_FILTER, + payload: category, + }); + dispatch({ + type: TYPES.SET_FILTER_OPTIONS, + payload: list, + }); +}; diff --git a/frontend/src/actions/types.js b/frontend/src/actions/types.js index 66e8f5b1..421fcbfe 100644 --- a/frontend/src/actions/types.js +++ b/frontend/src/actions/types.js @@ -16,3 +16,6 @@ export const SET_CPT_DATE_FILTER = "SET_CPT_DATE_FILTER"; export const SET_CPT_SORT_DIR = "SET_CPT_SORT_DIR"; export const SET_CPT_SORT_INDEX = "SET_CPT_SORT_INDEX"; export const SET_CPT_INIT_JOBS = "SET_CPT_INIT_JOBS"; +export const SET_CPT_FILTER_DATA = "SET_CPT_FILTER_DATA"; +export const SET_CATEGORY_FILTER = "SET_CATEGORY_FILTER"; +export const SET_FILTER_OPTIONS = "SET_FILTER_OPTIONS"; diff --git a/frontend/src/components/molecules/SelectBox/index.jsx b/frontend/src/components/molecules/SelectBox/index.jsx new file mode 100644 index 00000000..12d0d147 --- /dev/null +++ b/frontend/src/components/molecules/SelectBox/index.jsx @@ -0,0 +1,61 @@ +import { + MenuToggle, + Select, + SelectList, + SelectOption, +} from "@patternfly/react-core"; + +import PropTypes from "prop-types"; +import { useState } from "react"; + +const SelectBox = (props) => { + const [isOpen, setIsOpen] = useState(false); + + const onToggleClick = () => { + setIsOpen(!isOpen); + }; + const onSelect = (_event, value) => { + setIsOpen(false); + props.onChange(_event, value); + }; + const toggle = (toggleRef) => ( + + {props.selected} + + ); + return ( + <> + + + ); +}; + +SelectBox.propTypes = { + options: PropTypes.array, + onChange: PropTypes.func, + selected: PropTypes.string, +}; +export default SelectBox; diff --git a/frontend/src/components/organisms/TableFilters/index.jsx b/frontend/src/components/organisms/TableFilters/index.jsx new file mode 100644 index 00000000..568d9041 --- /dev/null +++ b/frontend/src/components/organisms/TableFilters/index.jsx @@ -0,0 +1,43 @@ +import { Toolbar, ToolbarContent, ToolbarItem } from "@patternfly/react-core"; +import { useDispatch, useSelector } from "react-redux"; + +import SelectBox from "@/components/molecules/SelectBox"; +import { setCatFilters } from "@/actions/homeActions"; + +const TableFilter = () => { + const dispatch = useDispatch(); + const { tableFilters, categoryFilterValue, filterOptions } = useSelector( + (state) => state.cpt + ); + + const onCategoryChange = (_event, value) => { + dispatch(setCatFilters(value)); + }; + const onOptionsChange = (_event, value) => { + console.log(value); + }; + return ( + <> + + + + + + + + + + + + ); +}; + +export default TableFilter; diff --git a/frontend/src/components/templates/Home/index.jsx b/frontend/src/components/templates/Home/index.jsx index 1360a487..5367275e 100644 --- a/frontend/src/components/templates/Home/index.jsx +++ b/frontend/src/components/templates/Home/index.jsx @@ -12,13 +12,20 @@ import { import { useCallback, useEffect, useState } from "react"; import { useDispatch, useSelector } from "react-redux"; +import TableFilter from "@/components/organisms/TableFilters"; import TableLayout from "@/components/organisms/TableLayout"; const Home = () => { const dispatch = useDispatch(); - const { results, tableColumns, activeSortDir, activeSortIndex, tableData } = - useSelector((state) => state.cpt); + const { + results, + tableColumns, + activeSortDir, + activeSortIndex, + tableData, + filterOptions, + } = useSelector((state) => state.cpt); useEffect(() => { dispatch(fetchOCPJobsData()); @@ -58,6 +65,8 @@ const Home = () => { return ( <> + {filterOptions?.length > 0 && } + { @@ -31,8 +42,8 @@ const HomeReducer = (state = initialState, action = {}) => { case TYPES.SET_CPT_DATE_FILTER: return { ...state, - start_date: payload.startDate, - end_date: payload.endDate, + start_date: payload.start_date, + end_date: payload.end_date, }; case TYPES.SET_CPT_SORT_INDEX: return { ...state, activeSortIndex: payload }; @@ -40,6 +51,12 @@ const HomeReducer = (state = initialState, action = {}) => { return { ...state, activeSortDir: payload }; case TYPES.SET_CPT_INIT_JOBS: return { ...state, tableData: payload }; + case TYPES.SET_CPT_FILTER_DATA: + return { ...state, filterData: payload }; + case TYPES.SET_CATEGORY_FILTER: + return { ...state, categoryFilterValue: payload }; + case TYPES.SET_FILTER_OPTIONS: + return { ...state, filterOptions: payload }; default: return state; } From d406fbcf9a1494de922e8d485a5db0b98badec66 Mon Sep 17 00:00:00 2001 From: MVarshini Date: Wed, 24 Apr 2024 23:08:57 +0530 Subject: [PATCH 09/14] url update fro filters --- frontend/src/actions/homeActions.js | 82 ++++++++++++++++++- frontend/src/actions/types.js | 2 + .../src/components/atoms/TableCell/index.jsx | 2 +- .../organisms/TableFilters/index.jsx | 45 ++++++++-- .../src/components/templates/Home/index.jsx | 11 ++- frontend/src/reducers/homeReducer.js | 7 +- 6 files changed, 135 insertions(+), 14 deletions(-) diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js index 4df76019..aa6f9c28 100644 --- a/frontend/src/actions/homeActions.js +++ b/frontend/src/actions/homeActions.js @@ -30,6 +30,7 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { end_date: response.data.endDate, }, }); + dispatch(applyFilters()); dispatch(sortTable()); dispatch(sliceTableRows(0, DEFAULT_PER_PAGE)); dispatch(buildFilterData()); @@ -47,7 +48,7 @@ const getSortableRowValues = (result, tableColumns) => { export const sortTable = () => (dispatch, getState) => { const { activeSortDir, activeSortIndex, tableColumns } = getState().cpt; - const results = [...getState().cpt.results]; + const results = [...getState().cpt.filteredResults]; try { if (activeSortIndex) { const sortedResults = results.sort((a, b) => { @@ -88,7 +89,7 @@ export const setCPTSortDir = (direction) => ({ }); export const sliceTableRows = (startIdx, endIdx) => (dispatch, getState) => { - const results = [...getState().cpt.results]; + const results = [...getState().cpt.filteredResults]; dispatch({ type: TYPES.SET_CPT_INIT_JOBS, @@ -132,3 +133,80 @@ export const setCatFilters = (category) => (dispatch, getState) => { payload: list, }); }; + +export const setAppliedFilters = + (selectedOption, navigate) => (dispatch, getState) => { + const { categoryFilterValue, filterData } = getState().cpt; + const appliedFilters = { ...getState().cpt.appliedFilters }; + + const category = filterData.filter( + (item) => item.name === categoryFilterValue + )[0].key; + appliedFilters[category] = selectedOption; + + dispatch({ + type: TYPES.SET_APPLIED_FILTERS, + payload: appliedFilters, + }); + buildQueryString(appliedFilters, navigate); + dispatch(applyFilters()); + }; + +export const removeAppliedFilters = + (filterKey, navigate) => (dispatch, getState) => { + const appliedFilters = { ...getState().cpt.appliedFilters }; + delete appliedFilters[filterKey]; + dispatch({ + type: TYPES.SET_APPLIED_FILTERS, + payload: appliedFilters, + }); + buildQueryString(appliedFilters, navigate); + dispatch(applyFilters()); + }; + +export const buildQueryString = (appliedFilters, navigate) => { + const queryString = new URLSearchParams(appliedFilters).toString(); + navigate({ + pathname: window.location.pathname, + search: `?${queryString}`, + }); + + // navigate({ + // pathname: "listing", + // search: createSearchParams({ + // foo: "bar" + // }).toString() + // }); +}; +export const applyFilters = () => (dispatch, getState) => { + const { appliedFilters } = getState().cpt; + + const results = [...getState().cpt.results]; + + const isFilterApplied = + Object.keys(appliedFilters).length > 0 && + !Object.values(appliedFilters).includes(""); + + let filtered = []; + if (isFilterApplied) { + filtered = results.filter((el) => { + for (const key in appliedFilters) { + if (el[key] !== appliedFilters[key]) { + return false; + } + } + return true; + }); + } + + dispatch({ + type: TYPES.SET_FILTERED_DATA, + payload: isFilterApplied ? filtered : results, + }); + dispatch(sliceTableRows(0, DEFAULT_PER_PAGE)); +}; + +export const setFilterFromURL = (searchParams) => ({ + type: TYPES.SET_APPLIED_FILTERS, + payload: Object.fromEntries(searchParams), +}); diff --git a/frontend/src/actions/types.js b/frontend/src/actions/types.js index 421fcbfe..3539ac38 100644 --- a/frontend/src/actions/types.js +++ b/frontend/src/actions/types.js @@ -19,3 +19,5 @@ export const SET_CPT_INIT_JOBS = "SET_CPT_INIT_JOBS"; export const SET_CPT_FILTER_DATA = "SET_CPT_FILTER_DATA"; export const SET_CATEGORY_FILTER = "SET_CATEGORY_FILTER"; export const SET_FILTER_OPTIONS = "SET_FILTER_OPTIONS"; +export const SET_FILTERED_DATA = "SET_FILTERED_DATA"; +export const SET_APPLIED_FILTERS = "SET_APPLIED_FILTERS"; diff --git a/frontend/src/components/atoms/TableCell/index.jsx b/frontend/src/components/atoms/TableCell/index.jsx index fec0b390..1bdfbe3d 100644 --- a/frontend/src/components/atoms/TableCell/index.jsx +++ b/frontend/src/components/atoms/TableCell/index.jsx @@ -28,7 +28,7 @@ const TableCell = (props) => { const StatusCell = (props) => { const { item, col } = props; - return item[col.value] === "success" ? ( + return item[col.value]?.toLowerCase() === "success" ? ( diff --git a/frontend/src/components/organisms/TableFilters/index.jsx b/frontend/src/components/organisms/TableFilters/index.jsx index 568d9041..991555e8 100644 --- a/frontend/src/components/organisms/TableFilters/index.jsx +++ b/frontend/src/components/organisms/TableFilters/index.jsx @@ -1,20 +1,46 @@ -import { Toolbar, ToolbarContent, ToolbarItem } from "@patternfly/react-core"; +import { + Chip, + Toolbar, + ToolbarContent, + ToolbarItem, +} from "@patternfly/react-core"; +import { + removeAppliedFilters, + setAppliedFilters, + setCatFilters, +} from "@/actions/homeActions"; import { useDispatch, useSelector } from "react-redux"; import SelectBox from "@/components/molecules/SelectBox"; -import { setCatFilters } from "@/actions/homeActions"; +import { useNavigate } from "react-router-dom"; const TableFilter = () => { const dispatch = useDispatch(); - const { tableFilters, categoryFilterValue, filterOptions } = useSelector( - (state) => state.cpt - ); - + const navigate = useNavigate(); + const { + tableFilters, + categoryFilterValue, + filterOptions, + filterData, + appliedFilters, + } = useSelector((state) => state.cpt); + const category = filterData.filter( + (item) => item.name === categoryFilterValue + )[0].key; const onCategoryChange = (_event, value) => { dispatch(setCatFilters(value)); }; const onOptionsChange = (_event, value) => { console.log(value); + dispatch(setAppliedFilters(value, navigate)); + }; + const deleteItem = (key) => { + console.log("hey"); + dispatch(removeAppliedFilters(key, navigate)); + }; + const getFilterName = (key) => { + const filter = tableFilters.find((item) => item.value === key); + return filter.name; }; return ( <> @@ -31,11 +57,16 @@ const TableFilter = () => { + {Object.keys(appliedFilters).map((key) => ( + deleteItem(key)}> + {getFilterName(key)} : {appliedFilters[key]} + + ))} ); }; diff --git a/frontend/src/components/templates/Home/index.jsx b/frontend/src/components/templates/Home/index.jsx index 5367275e..a40dbc9b 100644 --- a/frontend/src/components/templates/Home/index.jsx +++ b/frontend/src/components/templates/Home/index.jsx @@ -6,6 +6,7 @@ import { fetchOCPJobsData, setCPTSortDir, setCPTSortIndex, + setFilterFromURL, sliceTableRows, sortTable, } from "@/actions/homeActions.js"; @@ -14,12 +15,13 @@ import { useDispatch, useSelector } from "react-redux"; import TableFilter from "@/components/organisms/TableFilters"; import TableLayout from "@/components/organisms/TableLayout"; +import { useSearchParams } from "react-router-dom"; const Home = () => { const dispatch = useDispatch(); - + const [searchParams] = useSearchParams(); const { - results, + filteredResults, tableColumns, activeSortDir, activeSortIndex, @@ -31,6 +33,9 @@ const Home = () => { dispatch(fetchOCPJobsData()); }, [dispatch]); + useEffect(() => { + dispatch(setFilterFromURL(searchParams)); + }, []); //Sorting const setActiveSortDir = (dir) => { dispatch(setCPTSortDir(dir)); @@ -79,7 +84,7 @@ const Home = () => { onSetPage={onSetPage} page={page} perPage={perPage} - totalItems={results.length} + totalItems={filteredResults.length} /> ); diff --git a/frontend/src/reducers/homeReducer.js b/frontend/src/reducers/homeReducer.js index bc49ad32..a3766326 100644 --- a/frontend/src/reducers/homeReducer.js +++ b/frontend/src/reducers/homeReducer.js @@ -28,7 +28,8 @@ const initialState = { tableData: [], categoryFilterValue: "", filterOptions: [], - appliedFilters: [], + appliedFilters: {}, + filteredResults: [], }; const HomeReducer = (state = initialState, action = {}) => { @@ -57,6 +58,10 @@ const HomeReducer = (state = initialState, action = {}) => { return { ...state, categoryFilterValue: payload }; case TYPES.SET_FILTER_OPTIONS: return { ...state, filterOptions: payload }; + case TYPES.SET_APPLIED_FILTERS: + return { ...state, appliedFilters: payload }; + case TYPES.SET_FILTERED_DATA: + return { ...state, filteredResults: payload }; default: return state; } From 261670f059615b993d42e6fb2389947b7cb893af Mon Sep 17 00:00:00 2001 From: MVarshini Date: Thu, 25 Apr 2024 22:26:25 +0530 Subject: [PATCH 10/14] Date filter --- frontend/package-lock.json | 301 +++++++++++++++++- frontend/package.json | 1 + frontend/src/actions/homeActions.js | 34 +- .../components/molecules/SelectBox/index.jsx | 5 +- .../organisms/TableFilters/index.jsx | 50 ++- .../organisms/TableFilters/index.less | 12 + frontend/src/utils/helper.js | 27 ++ 7 files changed, 404 insertions(+), 26 deletions(-) create mode 100644 frontend/src/components/organisms/TableFilters/index.less diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 3502c2f8..4987e95f 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -13,6 +13,7 @@ "@reduxjs/toolkit": "^2.2.3", "axios": "^1.6.8", "react": "^18.2.0", + "react-date-picker": "^10.6.0", "react-dom": "^18.2.0", "react-redux": "^9.1.0", "react-router-dom": "^6.22.3", @@ -1281,7 +1282,7 @@ "version": "18.2.23", "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.2.23.tgz", "integrity": "sha512-ZQ71wgGOTmDYpnav2knkjr3qXdAFu0vsk8Ci5w3pGAIdj7/kKAyn+VsQDhXsmzzzepAiI9leWMmubXz690AI/A==", - "dev": true, + "devOptional": true, "dependencies": { "@types/react": "*" } @@ -1316,6 +1317,14 @@ "vite": "^4.2.0 || ^5.0.0" } }, + "node_modules/@wojtekmaj/date-utils": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/@wojtekmaj/date-utils/-/date-utils-1.5.1.tgz", + "integrity": "sha512-+i7+JmNiE/3c9FKxzWFi2IjRJ+KzZl1QPu6QNrsgaa2MuBgXvUy4gA1TVzf/JMdIIloB76xSKikTWuyYAIVLww==", + "funding": { + "url": "https://github.com/wojtekmaj/date-utils?sponsor=1" + } + }, "node_modules/acorn": { "version": "8.11.3", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.3.tgz", @@ -1667,6 +1676,14 @@ "node": ">=4" } }, + "node_modules/clsx": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", + "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==", + "engines": { + "node": ">=6" + } + }, "node_modules/color-convert": { "version": "1.9.3", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", @@ -1858,6 +1875,14 @@ "node": ">=0.4.0" } }, + "node_modules/detect-element-overflow": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/detect-element-overflow/-/detect-element-overflow-1.4.2.tgz", + "integrity": "sha512-4m6cVOtvm/GJLjo7WFkPfwXoEIIbM7GQwIh4WEa4g7IsNi1YzwUsGL5ApNLrrHL29bHeNeQ+/iZhw+YHqgE2Fw==", + "funding": { + "url": "https://github.com/wojtekmaj/detect-element-overflow?sponsor=1" + } + }, "node_modules/doctrine": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", @@ -2643,6 +2668,17 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/get-user-locale": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/get-user-locale/-/get-user-locale-2.3.2.tgz", + "integrity": "sha512-O2GWvQkhnbDoWFUJfaBlDIKUEdND8ATpBXD6KXcbhxlfktyD/d8w6mkzM/IlQEqGZAMz/PW6j6Hv53BiigKLUQ==", + "dependencies": { + "mem": "^8.0.0" + }, + "funding": { + "url": "https://github.com/wojtekmaj/get-user-locale?sponsor=1" + } + }, "node_modules/glob": { "version": "7.2.3", "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", @@ -3464,6 +3500,40 @@ "semver": "bin/semver" } }, + "node_modules/make-event-props": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/make-event-props/-/make-event-props-1.6.2.tgz", + "integrity": "sha512-iDwf7mA03WPiR8QxvcVHmVWEPfMY1RZXerDVNCRYW7dUr2ppH3J58Rwb39/WG39yTZdRSxr3x+2v22tvI0VEvA==", + "funding": { + "url": "https://github.com/wojtekmaj/make-event-props?sponsor=1" + } + }, + "node_modules/map-age-cleaner": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/map-age-cleaner/-/map-age-cleaner-0.1.3.tgz", + "integrity": "sha512-bJzx6nMoP6PDLPBFmg7+xRKeFZvFboMrGlxmNj9ClvX53KrmvM5bXFXEWjbz4cz1AFn+jWJ9z/DJSz7hrs0w3w==", + "dependencies": { + "p-defer": "^1.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/mem": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/mem/-/mem-8.1.1.tgz", + "integrity": "sha512-qFCFUDs7U3b8mBDPyz5EToEKoAkgCzqquIgi9nkkR9bixxOVOre+09lbuH7+9Kn2NFpm56M3GUWVbU2hQgdACA==", + "dependencies": { + "map-age-cleaner": "^0.1.3", + "mimic-fn": "^3.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/mem?sponsor=1" + } + }, "node_modules/mime": { "version": "1.6.0", "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", @@ -3496,6 +3566,14 @@ "node": ">= 0.6" } }, + "node_modules/mimic-fn": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-3.1.0.tgz", + "integrity": "sha512-Ysbi9uYW9hFyfrThdDEQuykN4Ey6BuwPD2kpI5ES/nFTDn/98yxYNLZJcgUAKPT/mcrLLKaGzJR9YVxJrIdASQ==", + "engines": { + "node": ">=8" + } + }, "node_modules/minimatch": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", @@ -3697,6 +3775,14 @@ "node": ">= 0.8.0" } }, + "node_modules/p-defer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-defer/-/p-defer-1.0.0.tgz", + "integrity": "sha512-wB3wfAxZpk2AzOfUMJNL+d36xothRSyj8EXOa4f6GMqYDN9BJaaSISbsk+wS9abmnebVw95C2Kb5t85UmpCxuw==", + "engines": { + "node": ">=4" + } + }, "node_modules/p-limit": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", @@ -3905,6 +3991,59 @@ "node": ">=0.10.0" } }, + "node_modules/react-calendar": { + "version": "4.8.0", + "resolved": "https://registry.npmjs.org/react-calendar/-/react-calendar-4.8.0.tgz", + "integrity": "sha512-qFgwo+p58sgv1QYMI1oGNaop90eJVKuHTZ3ZgBfrrpUb+9cAexxsKat0sAszgsizPMVo7vOXedV7Lqa0GQGMvA==", + "dependencies": { + "@wojtekmaj/date-utils": "^1.1.3", + "clsx": "^2.0.0", + "get-user-locale": "^2.2.1", + "prop-types": "^15.6.0", + "warning": "^4.0.0" + }, + "funding": { + "url": "https://github.com/wojtekmaj/react-calendar?sponsor=1" + }, + "peerDependencies": { + "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0", + "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, + "node_modules/react-date-picker": { + "version": "10.6.0", + "resolved": "https://registry.npmjs.org/react-date-picker/-/react-date-picker-10.6.0.tgz", + "integrity": "sha512-db5lcmU/52X8ur8SU1QU3PYBiaDG5SbzZDlqWk3YruPx5Ti9w6UpqCRsd1TXycVla9Ut2I3Qb4BUe27jxSwHeg==", + "dependencies": { + "@wojtekmaj/date-utils": "^1.1.3", + "clsx": "^2.0.0", + "get-user-locale": "^2.2.1", + "make-event-props": "^1.6.0", + "prop-types": "^15.6.0", + "react-calendar": "^4.6.0", + "react-fit": "^1.7.0", + "update-input-width": "^1.4.0" + }, + "funding": { + "url": "https://github.com/wojtekmaj/react-date-picker?sponsor=1" + }, + "peerDependencies": { + "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0", + "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + } + } + }, "node_modules/react-dom": { "version": "18.2.0", "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.2.0.tgz", @@ -3933,6 +4072,33 @@ "react": ">= 16.8 || 18.0.0" } }, + "node_modules/react-fit": { + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/react-fit/-/react-fit-1.7.1.tgz", + "integrity": "sha512-y/TYovCCBzfIwRJsbLj0rH4Es40wPQhU5GPPq9GlbdF09b0OdzTdMSkBza0QixSlgFzTm6dkM7oTFzaVvaBx+w==", + "dependencies": { + "detect-element-overflow": "^1.4.0", + "prop-types": "^15.6.0", + "tiny-warning": "^1.0.0" + }, + "funding": { + "url": "https://github.com/wojtekmaj/react-fit?sponsor=1" + }, + "peerDependencies": { + "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0", + "@types/react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0", + "react": "^16.8.0 || ^17.0.0 || ^18.0.0", + "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0" + }, + "peerDependenciesMeta": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { + "optional": true + } + } + }, "node_modules/react-is": { "version": "16.13.1", "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", @@ -4466,6 +4632,11 @@ "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", "dev": true }, + "node_modules/tiny-warning": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/tiny-warning/-/tiny-warning-1.0.3.tgz", + "integrity": "sha512-lBN9zLN/oAf68o3zNXYrdCt1kP8WsiGW8Oo2ka41b2IM5JL/S1CTyX1rW0mb/zSuJun0ZUrDxx4sqvYS2FWzPA==" + }, "node_modules/to-fast-properties": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", @@ -4622,6 +4793,14 @@ "browserslist": ">= 4.21.0" } }, + "node_modules/update-input-width": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/update-input-width/-/update-input-width-1.4.2.tgz", + "integrity": "sha512-/p0XLhrQQQ4bMWD7bL9duYObwYCO1qGr8R19xcMmoMSmXuQ7/1//veUnCObQ7/iW6E2pGS6rFkS4TfH4ur7e/g==", + "funding": { + "url": "https://github.com/wojtekmaj/update-input-width?sponsor=1" + } + }, "node_modules/uri-js": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", @@ -4694,6 +4873,14 @@ } } }, + "node_modules/warning": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/warning/-/warning-4.0.3.tgz", + "integrity": "sha512-rpJyN222KWIvHJ/F53XSZv0Zl/accqHR8et1kpaMTD/fLCRxtV8iX8czMzY7sVZupTI3zcUTg8eycS2kNF9l6w==", + "dependencies": { + "loose-envify": "^1.0.0" + } + }, "node_modules/which": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", @@ -5598,7 +5785,7 @@ "version": "18.2.23", "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.2.23.tgz", "integrity": "sha512-ZQ71wgGOTmDYpnav2knkjr3qXdAFu0vsk8Ci5w3pGAIdj7/kKAyn+VsQDhXsmzzzepAiI9leWMmubXz690AI/A==", - "dev": true, + "devOptional": true, "requires": { "@types/react": "*" } @@ -5627,6 +5814,11 @@ "react-refresh": "^0.14.0" } }, + "@wojtekmaj/date-utils": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/@wojtekmaj/date-utils/-/date-utils-1.5.1.tgz", + "integrity": "sha512-+i7+JmNiE/3c9FKxzWFi2IjRJ+KzZl1QPu6QNrsgaa2MuBgXvUy4gA1TVzf/JMdIIloB76xSKikTWuyYAIVLww==" + }, "acorn": { "version": "8.11.3", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.3.tgz", @@ -5869,6 +6061,11 @@ "supports-color": "^5.3.0" } }, + "clsx": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz", + "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==" + }, "color-convert": { "version": "1.9.3", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", @@ -6010,6 +6207,11 @@ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==" }, + "detect-element-overflow": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/detect-element-overflow/-/detect-element-overflow-1.4.2.tgz", + "integrity": "sha512-4m6cVOtvm/GJLjo7WFkPfwXoEIIbM7GQwIh4WEa4g7IsNi1YzwUsGL5ApNLrrHL29bHeNeQ+/iZhw+YHqgE2Fw==" + }, "doctrine": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", @@ -6599,6 +6801,14 @@ "get-intrinsic": "^1.2.4" } }, + "get-user-locale": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/get-user-locale/-/get-user-locale-2.3.2.tgz", + "integrity": "sha512-O2GWvQkhnbDoWFUJfaBlDIKUEdND8ATpBXD6KXcbhxlfktyD/d8w6mkzM/IlQEqGZAMz/PW6j6Hv53BiigKLUQ==", + "requires": { + "mem": "^8.0.0" + } + }, "glob": { "version": "7.2.3", "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", @@ -7170,6 +7380,28 @@ } } }, + "make-event-props": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/make-event-props/-/make-event-props-1.6.2.tgz", + "integrity": "sha512-iDwf7mA03WPiR8QxvcVHmVWEPfMY1RZXerDVNCRYW7dUr2ppH3J58Rwb39/WG39yTZdRSxr3x+2v22tvI0VEvA==" + }, + "map-age-cleaner": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/map-age-cleaner/-/map-age-cleaner-0.1.3.tgz", + "integrity": "sha512-bJzx6nMoP6PDLPBFmg7+xRKeFZvFboMrGlxmNj9ClvX53KrmvM5bXFXEWjbz4cz1AFn+jWJ9z/DJSz7hrs0w3w==", + "requires": { + "p-defer": "^1.0.0" + } + }, + "mem": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/mem/-/mem-8.1.1.tgz", + "integrity": "sha512-qFCFUDs7U3b8mBDPyz5EToEKoAkgCzqquIgi9nkkR9bixxOVOre+09lbuH7+9Kn2NFpm56M3GUWVbU2hQgdACA==", + "requires": { + "map-age-cleaner": "^0.1.3", + "mimic-fn": "^3.1.0" + } + }, "mime": { "version": "1.6.0", "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", @@ -7190,6 +7422,11 @@ "mime-db": "1.52.0" } }, + "mimic-fn": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-3.1.0.tgz", + "integrity": "sha512-Ysbi9uYW9hFyfrThdDEQuykN4Ey6BuwPD2kpI5ES/nFTDn/98yxYNLZJcgUAKPT/mcrLLKaGzJR9YVxJrIdASQ==" + }, "minimatch": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", @@ -7331,6 +7568,11 @@ "type-check": "^0.4.0" } }, + "p-defer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-defer/-/p-defer-1.0.0.tgz", + "integrity": "sha512-wB3wfAxZpk2AzOfUMJNL+d36xothRSyj8EXOa4f6GMqYDN9BJaaSISbsk+wS9abmnebVw95C2Kb5t85UmpCxuw==" + }, "p-limit": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", @@ -7466,6 +7708,33 @@ "loose-envify": "^1.1.0" } }, + "react-calendar": { + "version": "4.8.0", + "resolved": "https://registry.npmjs.org/react-calendar/-/react-calendar-4.8.0.tgz", + "integrity": "sha512-qFgwo+p58sgv1QYMI1oGNaop90eJVKuHTZ3ZgBfrrpUb+9cAexxsKat0sAszgsizPMVo7vOXedV7Lqa0GQGMvA==", + "requires": { + "@wojtekmaj/date-utils": "^1.1.3", + "clsx": "^2.0.0", + "get-user-locale": "^2.2.1", + "prop-types": "^15.6.0", + "warning": "^4.0.0" + } + }, + "react-date-picker": { + "version": "10.6.0", + "resolved": "https://registry.npmjs.org/react-date-picker/-/react-date-picker-10.6.0.tgz", + "integrity": "sha512-db5lcmU/52X8ur8SU1QU3PYBiaDG5SbzZDlqWk3YruPx5Ti9w6UpqCRsd1TXycVla9Ut2I3Qb4BUe27jxSwHeg==", + "requires": { + "@wojtekmaj/date-utils": "^1.1.3", + "clsx": "^2.0.0", + "get-user-locale": "^2.2.1", + "make-event-props": "^1.6.0", + "prop-types": "^15.6.0", + "react-calendar": "^4.6.0", + "react-fit": "^1.7.0", + "update-input-width": "^1.4.0" + } + }, "react-dom": { "version": "18.2.0", "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.2.0.tgz", @@ -7485,6 +7754,16 @@ "prop-types": "^15.8.1" } }, + "react-fit": { + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/react-fit/-/react-fit-1.7.1.tgz", + "integrity": "sha512-y/TYovCCBzfIwRJsbLj0rH4Es40wPQhU5GPPq9GlbdF09b0OdzTdMSkBza0QixSlgFzTm6dkM7oTFzaVvaBx+w==", + "requires": { + "detect-element-overflow": "^1.4.0", + "prop-types": "^15.6.0", + "tiny-warning": "^1.0.0" + } + }, "react-is": { "version": "16.13.1", "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", @@ -7851,6 +8130,11 @@ "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", "dev": true }, + "tiny-warning": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/tiny-warning/-/tiny-warning-1.0.3.tgz", + "integrity": "sha512-lBN9zLN/oAf68o3zNXYrdCt1kP8WsiGW8Oo2ka41b2IM5JL/S1CTyX1rW0mb/zSuJun0ZUrDxx4sqvYS2FWzPA==" + }, "to-fast-properties": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", @@ -7951,6 +8235,11 @@ "picocolors": "^1.0.0" } }, + "update-input-width": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/update-input-width/-/update-input-width-1.4.2.tgz", + "integrity": "sha512-/p0XLhrQQQ4bMWD7bL9duYObwYCO1qGr8R19xcMmoMSmXuQ7/1//veUnCObQ7/iW6E2pGS6rFkS4TfH4ur7e/g==" + }, "uri-js": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", @@ -7978,6 +8267,14 @@ "rollup": "^4.13.0" } }, + "warning": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/warning/-/warning-4.0.3.tgz", + "integrity": "sha512-rpJyN222KWIvHJ/F53XSZv0Zl/accqHR8et1kpaMTD/fLCRxtV8iX8czMzY7sVZupTI3zcUTg8eycS2kNF9l6w==", + "requires": { + "loose-envify": "^1.0.0" + } + }, "which": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", diff --git a/frontend/package.json b/frontend/package.json index 5b7905a8..edb48595 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -15,6 +15,7 @@ "@reduxjs/toolkit": "^2.2.3", "axios": "^1.6.8", "react": "^18.2.0", + "react-date-picker": "^10.6.0", "react-dom": "^18.2.0", "react-redux": "^9.1.0", "react-router-dom": "^6.22.3", diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js index aa6f9c28..f9b08772 100644 --- a/frontend/src/actions/homeActions.js +++ b/frontend/src/actions/homeActions.js @@ -3,6 +3,7 @@ import * as TYPES from "@/actions/types.js"; import API from "@/utils/axiosInstance"; import { DEFAULT_PER_PAGE } from "@/assets/constants/paginationConstants"; +import { appendQueryString } from "@/utils/helper"; import { showFailureToast } from "@/actions/toastActions"; export const fetchOCPJobsData = () => async (dispatch, getState) => { @@ -12,10 +13,10 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { const response = await API.get(API_ROUTES.CPT_JOBS_API_V1, { params: { pretty: true, + // ...(start_date && { start_date }), + // ...(end_date && { end_date }), start_date: "2024-04-21", end_date: "2024-04-22", - // ...(start_date && start_date), - // ...(end_date && end_date), }, }); if (response?.data?.results?.length > 0) { @@ -148,7 +149,7 @@ export const setAppliedFilters = type: TYPES.SET_APPLIED_FILTERS, payload: appliedFilters, }); - buildQueryString(appliedFilters, navigate); + appendQueryString(appliedFilters, navigate); dispatch(applyFilters()); }; @@ -160,24 +161,10 @@ export const removeAppliedFilters = type: TYPES.SET_APPLIED_FILTERS, payload: appliedFilters, }); - buildQueryString(appliedFilters, navigate); + appendQueryString(appliedFilters, navigate); dispatch(applyFilters()); }; -export const buildQueryString = (appliedFilters, navigate) => { - const queryString = new URLSearchParams(appliedFilters).toString(); - navigate({ - pathname: window.location.pathname, - search: `?${queryString}`, - }); - - // navigate({ - // pathname: "listing", - // search: createSearchParams({ - // foo: "bar" - // }).toString() - // }); -}; export const applyFilters = () => (dispatch, getState) => { const { appliedFilters } = getState().cpt; @@ -210,3 +197,14 @@ export const setFilterFromURL = (searchParams) => ({ type: TYPES.SET_APPLIED_FILTERS, payload: Object.fromEntries(searchParams), }); + +export const setDateFilter = (start_date, end_date) => (dispatch) => { + dispatch({ + type: TYPES.SET_CPT_DATE_FILTER, + payload: { + start_date, + end_date, + }, + }); + dispatch(fetchOCPJobsData()); +}; diff --git a/frontend/src/components/molecules/SelectBox/index.jsx b/frontend/src/components/molecules/SelectBox/index.jsx index 12d0d147..6ee3d5ca 100644 --- a/frontend/src/components/molecules/SelectBox/index.jsx +++ b/frontend/src/components/molecules/SelectBox/index.jsx @@ -21,10 +21,11 @@ const SelectBox = (props) => { const toggle = (toggleRef) => ( {props.selected} @@ -57,5 +58,7 @@ SelectBox.propTypes = { options: PropTypes.array, onChange: PropTypes.func, selected: PropTypes.string, + width: PropTypes.string, + icon: PropTypes.any, }; export default SelectBox; diff --git a/frontend/src/components/organisms/TableFilters/index.jsx b/frontend/src/components/organisms/TableFilters/index.jsx index 991555e8..84574adb 100644 --- a/frontend/src/components/organisms/TableFilters/index.jsx +++ b/frontend/src/components/organisms/TableFilters/index.jsx @@ -1,3 +1,7 @@ +import "react-date-picker/dist/DatePicker.css"; +import "react-calendar/dist/Calendar.css"; +import "./index.less"; + import { Chip, Toolbar, @@ -8,10 +12,14 @@ import { removeAppliedFilters, setAppliedFilters, setCatFilters, + setDateFilter, } from "@/actions/homeActions"; import { useDispatch, useSelector } from "react-redux"; +import DatePicker from "react-date-picker"; +import { FilterIcon } from "@patternfly/react-icons"; import SelectBox from "@/components/molecules/SelectBox"; +import { formatDate } from "@/utils/helper"; import { useNavigate } from "react-router-dom"; const TableFilter = () => { @@ -23,6 +31,8 @@ const TableFilter = () => { filterOptions, filterData, appliedFilters, + start_date, + end_date, } = useSelector((state) => state.cpt); const category = filterData.filter( (item) => item.name === categoryFilterValue @@ -31,33 +41,63 @@ const TableFilter = () => { dispatch(setCatFilters(value)); }; const onOptionsChange = (_event, value) => { - console.log(value); dispatch(setAppliedFilters(value, navigate)); }; const deleteItem = (key) => { - console.log("hey"); dispatch(removeAppliedFilters(key, navigate)); }; const getFilterName = (key) => { const filter = tableFilters.find((item) => item.value === key); return filter.name; }; + + const startDateChangeHandler = (date, key) => { + dispatch(setDateFilter(date, key)); + }; + const endDateChangeHandler = (date, key) => { + dispatch(setDateFilter(key, date)); + }; return ( <> - - + + } + width={"200px"} /> + + + + + + startDateChangeHandler(formatDate(date), end_date) + } + clearIcon={null} + value={start_date} + /> + + to + + + endDateChangeHandler(formatDate(date), start_date) + } + minDate={new Date(start_date)} + clearIcon={null} + value={end_date} /> diff --git a/frontend/src/components/organisms/TableFilters/index.less b/frontend/src/components/organisms/TableFilters/index.less new file mode 100644 index 00000000..0b388805 --- /dev/null +++ b/frontend/src/components/organisms/TableFilters/index.less @@ -0,0 +1,12 @@ +#filter-toolbar { + display: flex; + .react-date-picker { + height: 4vh; + .react-date-picker__wrapper { + padding: 5px; + width: 10vw; + border: 1px solid #f0f0f0; + border-bottom-color: #8a8d90; + } + } +} \ No newline at end of file diff --git a/frontend/src/utils/helper.js b/frontend/src/utils/helper.js index d6f4b1b2..58c6ba78 100644 --- a/frontend/src/utils/helper.js +++ b/frontend/src/utils/helper.js @@ -27,3 +27,30 @@ export const formatDateTime = (dateTimeStamp) => { timeStyle: "short", }).format(dateObj); }; + +/** + * Build the url with query params* + * @function + * @param {Object} queryObj - Query + * @return {string} - update the url with the query string + */ + +export const appendQueryString = (queryObj, navigate) => { + const queryString = new URLSearchParams(queryObj).toString(); + navigate({ + pathname: window.location.pathname, + search: `?${queryString}`, + }); +}; + +export const formatDate = (date) => { + let d = new Date(date), + month = "" + (d.getMonth() + 1), + day = "" + d.getDate(), + year = d.getFullYear(); + + if (month.length < 2) month = "0" + month; + if (day.length < 2) day = "0" + day; + + return [year, month, day].join("-"); +}; From a5278a4f6909f63798b3d2db4a0ba91cf4b269d2 Mon Sep 17 00:00:00 2001 From: MVarshini Date: Fri, 26 Apr 2024 13:26:52 +0530 Subject: [PATCH 11/14] Date filter in url --- frontend/src/actions/homeActions.js | 54 ++++++++++++------- .../organisms/TableFilters/index.jsx | 17 +++--- .../organisms/TableFilters/index.less | 3 +- .../src/components/templates/Home/index.jsx | 16 ++++-- 4 files changed, 59 insertions(+), 31 deletions(-) diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js index f9b08772..6a2c8f60 100644 --- a/frontend/src/actions/homeActions.js +++ b/frontend/src/actions/homeActions.js @@ -20,6 +20,14 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { }, }); if (response?.data?.results?.length > 0) { + const startDate = response.data.startDate, + endDate = response.data.endDate; + + window.history.pushState( + {}, + "", + `?start_date=${startDate}&end_date=${endDate}` + ); dispatch({ type: TYPES.SET_CPT_JOBS_DATA, payload: response.data.results, @@ -27,8 +35,8 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { dispatch({ type: TYPES.SET_CPT_DATE_FILTER, payload: { - start_date: response.data.startDate, - end_date: response.data.endDate, + start_date: startDate, + end_date: endDate, }, }); dispatch(applyFilters()); @@ -109,7 +117,7 @@ export const buildFilterData = () => (dispatch, getState) => { let obj = { name: filter.name, key, - value: [...new Set(results.map((item) => item[key])), "ALL"], + value: [...new Set(results.map((item) => item[key]))], }; filterData.push(obj); } @@ -137,7 +145,8 @@ export const setCatFilters = (category) => (dispatch, getState) => { export const setAppliedFilters = (selectedOption, navigate) => (dispatch, getState) => { - const { categoryFilterValue, filterData } = getState().cpt; + const { categoryFilterValue, filterData, start_date, end_date } = + getState().cpt; const appliedFilters = { ...getState().cpt.appliedFilters }; const category = filterData.filter( @@ -149,19 +158,22 @@ export const setAppliedFilters = type: TYPES.SET_APPLIED_FILTERS, payload: appliedFilters, }); - appendQueryString(appliedFilters, navigate); + appendQueryString({ ...appliedFilters, start_date, end_date }, navigate); dispatch(applyFilters()); }; export const removeAppliedFilters = (filterKey, navigate) => (dispatch, getState) => { const appliedFilters = { ...getState().cpt.appliedFilters }; - delete appliedFilters[filterKey]; + const { start_date, end_date } = getState().cpt; + + delete appliedFilters[filterKey]; //find to remove key + //let { [filterKey], ...newObject } = { ...params }; dispatch({ type: TYPES.SET_APPLIED_FILTERS, payload: appliedFilters, }); - appendQueryString(appliedFilters, navigate); + appendQueryString({ ...appliedFilters, start_date, end_date }, navigate); dispatch(applyFilters()); }; @@ -195,16 +207,22 @@ export const applyFilters = () => (dispatch, getState) => { export const setFilterFromURL = (searchParams) => ({ type: TYPES.SET_APPLIED_FILTERS, - payload: Object.fromEntries(searchParams), + payload: searchParams, }); -export const setDateFilter = (start_date, end_date) => (dispatch) => { - dispatch({ - type: TYPES.SET_CPT_DATE_FILTER, - payload: { - start_date, - end_date, - }, - }); - dispatch(fetchOCPJobsData()); -}; +export const setDateFilter = + (start_date, end_date, navigate) => (dispatch, getState) => { + const appliedFilters = getState().cpt.appliedFilters; + + dispatch({ + type: TYPES.SET_CPT_DATE_FILTER, + payload: { + start_date, + end_date, + }, + }); + + appendQueryString({ ...appliedFilters, start_date, end_date }, navigate); + + dispatch(fetchOCPJobsData()); + }; diff --git a/frontend/src/components/organisms/TableFilters/index.jsx b/frontend/src/components/organisms/TableFilters/index.jsx index 84574adb..42c44fad 100644 --- a/frontend/src/components/organisms/TableFilters/index.jsx +++ b/frontend/src/components/organisms/TableFilters/index.jsx @@ -25,6 +25,7 @@ import { useNavigate } from "react-router-dom"; const TableFilter = () => { const dispatch = useDispatch(); const navigate = useNavigate(); + const { tableFilters, categoryFilterValue, @@ -34,6 +35,7 @@ const TableFilter = () => { start_date, end_date, } = useSelector((state) => state.cpt); + const category = filterData.filter( (item) => item.name === categoryFilterValue )[0].key; @@ -52,10 +54,10 @@ const TableFilter = () => { }; const startDateChangeHandler = (date, key) => { - dispatch(setDateFilter(date, key)); + dispatch(setDateFilter(date, key, navigate)); }; const endDateChangeHandler = (date, key) => { - dispatch(setDateFilter(key, date)); + dispatch(setDateFilter(key, date, navigate)); }; return ( <> @@ -102,11 +104,12 @@ const TableFilter = () => { - {Object.keys(appliedFilters).map((key) => ( - deleteItem(key)}> - {getFilterName(key)} : {appliedFilters[key]} - - ))} + {Object.keys(appliedFilters).length > 0 && + Object.keys(appliedFilters).map((key) => ( + deleteItem(key)}> + {getFilterName(key)} : {appliedFilters[key]} + + ))} ); }; diff --git a/frontend/src/components/organisms/TableFilters/index.less b/frontend/src/components/organisms/TableFilters/index.less index 0b388805..f990b26a 100644 --- a/frontend/src/components/organisms/TableFilters/index.less +++ b/frontend/src/components/organisms/TableFilters/index.less @@ -1,9 +1,8 @@ #filter-toolbar { display: flex; .react-date-picker { - height: 4vh; .react-date-picker__wrapper { - padding: 5px; + padding: 4px; width: 10vw; border: 1px solid #f0f0f0; border-bottom-color: #8a8d90; diff --git a/frontend/src/components/templates/Home/index.jsx b/frontend/src/components/templates/Home/index.jsx index a40dbc9b..cc4c2225 100644 --- a/frontend/src/components/templates/Home/index.jsx +++ b/frontend/src/components/templates/Home/index.jsx @@ -20,6 +20,7 @@ import { useSearchParams } from "react-router-dom"; const Home = () => { const dispatch = useDispatch(); const [searchParams] = useSearchParams(); + const { filteredResults, tableColumns, @@ -30,12 +31,19 @@ const Home = () => { } = useSelector((state) => state.cpt); useEffect(() => { - dispatch(fetchOCPJobsData()); - }, [dispatch]); + if (searchParams.size > 0) { + // date filter is set in fetchOCPJobsData() + searchParams.delete("start_date"); + searchParams.delete("end_date"); - useEffect(() => { - dispatch(setFilterFromURL(searchParams)); + const params = Object.fromEntries(searchParams); + dispatch(setFilterFromURL(params)); + } }, []); + + useEffect(() => { + dispatch(fetchOCPJobsData()); + }, [dispatch]); //Sorting const setActiveSortDir = (dir) => { dispatch(setCPTSortDir(dir)); From 5f257766d183ace085b57bde472408c8ebda243c Mon Sep 17 00:00:00 2001 From: MVarshini Date: Fri, 26 Apr 2024 14:53:46 +0530 Subject: [PATCH 12/14] Update the filter component to be common --- frontend/src/actions/homeActions.js | 21 ++++---- .../organisms/TableFilters/index.jsx | 53 +++++++++---------- .../organisms/TableFilters/index.less | 3 ++ .../src/components/templates/Home/index.jsx | 48 +++++++++++++++-- 4 files changed, 84 insertions(+), 41 deletions(-) diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js index 6a2c8f60..42a47e67 100644 --- a/frontend/src/actions/homeActions.js +++ b/frontend/src/actions/homeActions.js @@ -22,12 +22,14 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { if (response?.data?.results?.length > 0) { const startDate = response.data.startDate, endDate = response.data.endDate; + //on initial load startDate and endDate are empty, so from response append to url + const searchParams = new URLSearchParams(window.location.search); + if (!searchParams.has("start_date") || !searchParams.has("end_date")) { + searchParams.set("start_date", startDate); + searchParams.set("end_date", endDate); + window.history.pushState({}, "", `?${searchParams.toString()}`); + } - window.history.pushState( - {}, - "", - `?start_date=${startDate}&end_date=${endDate}` - ); dispatch({ type: TYPES.SET_CPT_JOBS_DATA, payload: response.data.results, @@ -166,14 +168,15 @@ export const removeAppliedFilters = (filterKey, navigate) => (dispatch, getState) => { const appliedFilters = { ...getState().cpt.appliedFilters }; const { start_date, end_date } = getState().cpt; + const name = filterKey; + // eslint-disable-next-line no-unused-vars + const { [name]: removedProperty, ...remainingObject } = appliedFilters; - delete appliedFilters[filterKey]; //find to remove key - //let { [filterKey], ...newObject } = { ...params }; dispatch({ type: TYPES.SET_APPLIED_FILTERS, - payload: appliedFilters, + payload: remainingObject, }); - appendQueryString({ ...appliedFilters, start_date, end_date }, navigate); + appendQueryString({ ...remainingObject, start_date, end_date }, navigate); dispatch(applyFilters()); }; diff --git a/frontend/src/components/organisms/TableFilters/index.jsx b/frontend/src/components/organisms/TableFilters/index.jsx index 42c44fad..36c8dc89 100644 --- a/frontend/src/components/organisms/TableFilters/index.jsx +++ b/frontend/src/components/organisms/TableFilters/index.jsx @@ -8,24 +8,14 @@ import { ToolbarContent, ToolbarItem, } from "@patternfly/react-core"; -import { - removeAppliedFilters, - setAppliedFilters, - setCatFilters, - setDateFilter, -} from "@/actions/homeActions"; -import { useDispatch, useSelector } from "react-redux"; import DatePicker from "react-date-picker"; import { FilterIcon } from "@patternfly/react-icons"; +import PropTypes from "prop-types"; import SelectBox from "@/components/molecules/SelectBox"; import { formatDate } from "@/utils/helper"; -import { useNavigate } from "react-router-dom"; - -const TableFilter = () => { - const dispatch = useDispatch(); - const navigate = useNavigate(); +const TableFilter = (props) => { const { tableFilters, categoryFilterValue, @@ -34,31 +24,22 @@ const TableFilter = () => { appliedFilters, start_date, end_date, - } = useSelector((state) => state.cpt); + onCategoryChange, + onOptionsChange, + deleteItem, + startDateChangeHandler, + endDateChangeHandler, + } = props; const category = filterData.filter( (item) => item.name === categoryFilterValue )[0].key; - const onCategoryChange = (_event, value) => { - dispatch(setCatFilters(value)); - }; - const onOptionsChange = (_event, value) => { - dispatch(setAppliedFilters(value, navigate)); - }; - const deleteItem = (key) => { - dispatch(removeAppliedFilters(key, navigate)); - }; + const getFilterName = (key) => { const filter = tableFilters.find((item) => item.value === key); return filter.name; }; - const startDateChangeHandler = (date, key) => { - dispatch(setDateFilter(date, key, navigate)); - }; - const endDateChangeHandler = (date, key) => { - dispatch(setDateFilter(key, date, navigate)); - }; return ( <> @@ -91,7 +72,7 @@ const TableFilter = () => { value={start_date} /> - to + to @@ -114,4 +95,18 @@ const TableFilter = () => { ); }; +TableFilter.propTypes = { + tableFilters: PropTypes.array, + categoryFilterValue: PropTypes.string, + filterOptions: PropTypes.array, + filterData: PropTypes.array, + appliedFilters: PropTypes.object, + start_date: PropTypes.string, + end_date: PropTypes.string, + onCategoryChange: PropTypes.func, + onOptionsChange: PropTypes.func, + deleteItem: PropTypes.func, + startDateChangeHandler: PropTypes.func, + endDateChangeHandler: PropTypes.func, +}; export default TableFilter; diff --git a/frontend/src/components/organisms/TableFilters/index.less b/frontend/src/components/organisms/TableFilters/index.less index f990b26a..b100a012 100644 --- a/frontend/src/components/organisms/TableFilters/index.less +++ b/frontend/src/components/organisms/TableFilters/index.less @@ -8,4 +8,7 @@ border-bottom-color: #8a8d90; } } + .to-text { + padding: 5px 0; + } } \ No newline at end of file diff --git a/frontend/src/components/templates/Home/index.jsx b/frontend/src/components/templates/Home/index.jsx index cc4c2225..0c78c9ae 100644 --- a/frontend/src/components/templates/Home/index.jsx +++ b/frontend/src/components/templates/Home/index.jsx @@ -4,22 +4,27 @@ import { } from "@/assets/constants/paginationConstants"; import { fetchOCPJobsData, + removeAppliedFilters, + setAppliedFilters, setCPTSortDir, setCPTSortIndex, + setCatFilters, + setDateFilter, setFilterFromURL, sliceTableRows, sortTable, } from "@/actions/homeActions.js"; import { useCallback, useEffect, useState } from "react"; import { useDispatch, useSelector } from "react-redux"; +import { useNavigate, useSearchParams } from "react-router-dom"; import TableFilter from "@/components/organisms/TableFilters"; import TableLayout from "@/components/organisms/TableLayout"; -import { useSearchParams } from "react-router-dom"; const Home = () => { const dispatch = useDispatch(); const [searchParams] = useSearchParams(); + const navigate = useNavigate(); const { filteredResults, @@ -28,6 +33,12 @@ const Home = () => { activeSortIndex, tableData, filterOptions, + tableFilters, + categoryFilterValue, + filterData, + appliedFilters, + start_date, + end_date, } = useSelector((state) => state.cpt); useEffect(() => { @@ -75,10 +86,41 @@ const Home = () => { [dispatch] ); // Pagination helper - + // Filter Helper + const onCategoryChange = (_event, value) => { + dispatch(setCatFilters(value)); + }; + const onOptionsChange = (_event, value) => { + dispatch(setAppliedFilters(value, navigate)); + }; + const deleteItem = (key) => { + dispatch(removeAppliedFilters(key, navigate)); + }; + const startDateChangeHandler = (date, key) => { + dispatch(setDateFilter(date, key, navigate)); + }; + const endDateChangeHandler = (date, key) => { + dispatch(setDateFilter(key, date, navigate)); + }; + // Filter Helper return ( <> - {filterOptions?.length > 0 && } + {tableFilters?.length > 0 && filterOptions?.length > 0 && ( + + )} Date: Fri, 26 Apr 2024 15:25:35 +0530 Subject: [PATCH 13/14] date update --- frontend/src/actions/homeActions.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/frontend/src/actions/homeActions.js b/frontend/src/actions/homeActions.js index 42a47e67..f5efdb05 100644 --- a/frontend/src/actions/homeActions.js +++ b/frontend/src/actions/homeActions.js @@ -13,10 +13,10 @@ export const fetchOCPJobsData = () => async (dispatch, getState) => { const response = await API.get(API_ROUTES.CPT_JOBS_API_V1, { params: { pretty: true, - // ...(start_date && { start_date }), - // ...(end_date && { end_date }), - start_date: "2024-04-21", - end_date: "2024-04-22", + ...(start_date && { start_date }), + ...(end_date && { end_date }), + // start_date: "2024-04-21", + // end_date: "2024-04-22", }, }); if (response?.data?.results?.length > 0) { From 4d8d3851d297815dce64732d618d7c771360c5d5 Mon Sep 17 00:00:00 2001 From: vvijayra Date: Tue, 7 May 2024 06:11:06 +0530 Subject: [PATCH 14/14] Initial UI test code/setup commit --- frontend/tests/UI/.gitignore | 176 +++++++++++++++++++++++++++++ frontend/tests/UI/README.md | 24 ++++ frontend/tests/UI/requirements.txt | 5 + frontend/tests/UI/test_home.py | 5 + 4 files changed, 210 insertions(+) create mode 100644 frontend/tests/UI/.gitignore create mode 100644 frontend/tests/UI/README.md create mode 100644 frontend/tests/UI/requirements.txt create mode 100644 frontend/tests/UI/test_home.py diff --git a/frontend/tests/UI/.gitignore b/frontend/tests/UI/.gitignore new file mode 100644 index 00000000..ad4a1f17 --- /dev/null +++ b/frontend/tests/UI/.gitignore @@ -0,0 +1,176 @@ +# Created by https://www.toptal.com/developers/gitignore/api/python +# Edit at https://www.toptal.com/developers/gitignore?templates=python + +### Python ### +# Byte-compiled / optimized / DLL files +__pycache__/ +*.py[cod] +*$py.class + +# C extensions +*.so + +# Distribution / packaging +.Python +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +share/python-wheels/ +*.egg-info/ +.installed.cfg +*.egg +MANIFEST + +# PyInstaller +# Usually these files are written by a python script from a template +# before PyInstaller builds the exe, so as to inject date/other infos into it. +*.manifest +*.spec + +# Installer logs +pip-log.txt +pip-delete-this-directory.txt + +# Unit test / coverage reports +htmlcov/ +.tox/ +.nox/ +.coverage +.coverage.* +.cache +nosetests.xml +coverage.xml +*.cover +*.py,cover +.hypothesis/ +.pytest_cache/ +cover/ + +# Translations +*.mo +*.pot + +# Django stuff: +*.log +local_settings.py +db.sqlite3 +db.sqlite3-journal + +# Flask stuff: +instance/ +.webassets-cache + +# Scrapy stuff: +.scrapy + +# Sphinx documentation +docs/_build/ + +# PyBuilder +.pybuilder/ +target/ + +# Jupyter Notebook +.ipynb_checkpoints + +# IPython +profile_default/ +ipython_config.py + +# pyenv +# For a library or package, you might want to ignore these files since the code is +# intended to run in multiple environments; otherwise, check them in: +# .python-version + +# pipenv +# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. +# However, in case of collaboration, if having platform-specific dependencies or dependencies +# having no cross-platform support, pipenv may install dependencies that don't work, or not +# install all needed dependencies. +#Pipfile.lock + +# poetry +# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control. +# This is especially recommended for binary packages to ensure reproducibility, and is more +# commonly ignored for libraries. +# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control +#poetry.lock + +# pdm +# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control. +#pdm.lock +# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it +# in version control. +# https://pdm.fming.dev/#use-with-ide +.pdm.toml + +# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm +__pypackages__/ + +# Celery stuff +celerybeat-schedule +celerybeat.pid + +# SageMath parsed files +*.sage.py + +# Environments +.env +.venv +env/ +venv/ +ENV/ +env.bak/ +venv.bak/ + +# Spyder project settings +.spyderproject +.spyproject + +# Rope project settings +.ropeproject + +# mkdocs documentation +/site + +# mypy +.mypy_cache/ +.dmypy.json +dmypy.json + +# Pyre type checker +.pyre/ + +# pytype static type analyzer +.pytype/ + +# Cython debug symbols +cython_debug/ + +# PyCharm +# JetBrains specific template is maintained in a separate JetBrains.gitignore that can +# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore +# and can be added to the global gitignore or merged into this file. For a more nuclear +# option (not recommended) you can uncomment the following to ignore the entire idea folder. +#.idea/ + +### Python Patch ### +# Poetry local configuration file - https://python-poetry.org/docs/configuration/#local-configuration +poetry.toml + +# ruff +.ruff_cache/ + +# LSP config files +pyrightconfig.json + +# End of https://www.toptal.com/developers/gitignore/api/python diff --git a/frontend/tests/UI/README.md b/frontend/tests/UI/README.md new file mode 100644 index 00000000..24b2c99e --- /dev/null +++ b/frontend/tests/UI/README.md @@ -0,0 +1,24 @@ +[Playwright Framework Docs](https://playwright.dev/python/docs/intro) + +# Setup +- Installing dependencies + ```console + $ pip install -r requirements.txt + ``` +- Download browser drivers for playwright + ```console + $ playwright install + ``` + +# Run Test +- Run test + ```console + $ pytest -vs --headed --base-url + ``` +- Generate allure report + ```console + $ pytest --alluredir allure-results + $ allure serve allure-results + ``` + > Note: Make sure you have allure installed before you run these commands. [Installation Docs](https://allurereport.org/docs/gettingstarted-installation) + \ No newline at end of file diff --git a/frontend/tests/UI/requirements.txt b/frontend/tests/UI/requirements.txt new file mode 100644 index 00000000..83d61ba8 --- /dev/null +++ b/frontend/tests/UI/requirements.txt @@ -0,0 +1,5 @@ +pytest +playwright +pytest-playwright +allure-pytest +# pytest-reporter-html1 diff --git a/frontend/tests/UI/test_home.py b/frontend/tests/UI/test_home.py new file mode 100644 index 00000000..c5f0f1d5 --- /dev/null +++ b/frontend/tests/UI/test_home.py @@ -0,0 +1,5 @@ +from playwright.sync_api import expect + +def test_homepage(page): + page.goto("/home") + expect(page.get_by_text("CPT Dashboard")).to_be_visible()