diff --git a/client/.firebase/hosting.ZGlzdA.cache b/client/.firebase/hosting.ZGlzdA.cache
new file mode 100644
index 0000000..938168c
--- /dev/null
+++ b/client/.firebase/hosting.ZGlzdA.cache
@@ -0,0 +1,5 @@
+vite.svg,1742260251222,699a02e0e68a579f687d364bbbe7633161244f35af068220aee37b1b33dfb3c7
+index.html,1743020052734,7d2eb91387179f1d05ad06419df078478c94c53aeae2dadc2a98a59085597d02
+assets/index-BDAipEXg.css,1743020052734,3128af03d137a6590aef6fa8bd00ccaf2e1f2d385c1c864281afad3acfa51df3
+assets/hero-Cj08RxkT.svg,1743020052734,7ddfec926291860a7098cf85085b547922c8d995a9d669d9777c0b1d720cc4c9
+assets/index-aFz7XF--.js,1743020052734,071409052e4fc1d15e11f291b0a6731fcf632e5e9537f81f58675e96088559d2
diff --git a/client/.firebaserc b/client/.firebaserc
new file mode 100644
index 0000000..1d385cc
--- /dev/null
+++ b/client/.firebaserc
@@ -0,0 +1,5 @@
+{
+ "projects": {
+ "default": "garuda-nihon-web-client"
+ }
+}
diff --git a/client/.gitignore b/client/.gitignore
new file mode 100644
index 0000000..1cac559
--- /dev/null
+++ b/client/.gitignore
@@ -0,0 +1,25 @@
+# Logs
+logs
+*.log
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+pnpm-debug.log*
+lerna-debug.log*
+
+node_modules
+dist
+dist-ssr
+*.local
+
+# Editor directories and files
+.vscode/*
+!.vscode/extensions.json
+.idea
+.DS_Store
+*.suo
+*.ntvs*
+*.njsproj
+*.sln
+*.sw?
+.env
\ No newline at end of file
diff --git a/client/README.md b/client/README.md
new file mode 100644
index 0000000..fd3b758
--- /dev/null
+++ b/client/README.md
@@ -0,0 +1,12 @@
+# 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
+
+## Expanding the ESLint configuration
+
+If you are developing a production application, we recommend using TypeScript and enable type-aware lint rules. Check out the [TS template](https://github.com/vitejs/vite/tree/main/packages/create-vite/template-react-ts) to integrate TypeScript and [`typescript-eslint`](https://typescript-eslint.io) in your project.
diff --git a/client/eslint.config.js b/client/eslint.config.js
new file mode 100644
index 0000000..ec2b712
--- /dev/null
+++ b/client/eslint.config.js
@@ -0,0 +1,33 @@
+import js from '@eslint/js'
+import globals from 'globals'
+import reactHooks from 'eslint-plugin-react-hooks'
+import reactRefresh from 'eslint-plugin-react-refresh'
+
+export default [
+ { ignores: ['dist'] },
+ {
+ files: ['**/*.{js,jsx}'],
+ languageOptions: {
+ ecmaVersion: 2020,
+ globals: globals.browser,
+ parserOptions: {
+ ecmaVersion: 'latest',
+ ecmaFeatures: { jsx: true },
+ sourceType: 'module',
+ },
+ },
+ plugins: {
+ 'react-hooks': reactHooks,
+ 'react-refresh': reactRefresh,
+ },
+ rules: {
+ ...js.configs.recommended.rules,
+ ...reactHooks.configs.recommended.rules,
+ 'no-unused-vars': ['error', { varsIgnorePattern: '^[A-Z_]' }],
+ 'react-refresh/only-export-components': [
+ 'warn',
+ { allowConstantExport: true },
+ ],
+ },
+ },
+]
diff --git a/client/firebase.json b/client/firebase.json
new file mode 100644
index 0000000..2c33c29
--- /dev/null
+++ b/client/firebase.json
@@ -0,0 +1,16 @@
+{
+ "hosting": {
+ "public": "dist",
+ "ignore": [
+ "firebase.json",
+ "**/.*",
+ "**/node_modules/**"
+ ],
+ "rewrites": [
+ {
+ "source": "**",
+ "destination": "/index.html"
+ }
+ ]
+ }
+}
diff --git a/client/index.html b/client/index.html
new file mode 100644
index 0000000..285d422
--- /dev/null
+++ b/client/index.html
@@ -0,0 +1,18 @@
+
+
+
+
+
+
+
+
+ Garuda Nihon
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/client/package-lock.json b/client/package-lock.json
new file mode 100644
index 0000000..59d6e94
--- /dev/null
+++ b/client/package-lock.json
@@ -0,0 +1,4616 @@
+{
+ "name": "client",
+ "version": "0.0.0",
+ "lockfileVersion": 3,
+ "requires": true,
+ "packages": {
+ "": {
+ "name": "client",
+ "version": "0.0.0",
+ "dependencies": {
+ "@reduxjs/toolkit": "^2.6.1",
+ "axios": "^1.8.4",
+ "dotenv": "^16.4.7",
+ "openai": "^4.89.0",
+ "react": "^19.0.0",
+ "react-dom": "^19.0.0",
+ "react-redux": "^9.2.0",
+ "react-router": "^7.4.0",
+ "sweetalert2": "^11.17.2"
+ },
+ "devDependencies": {
+ "@eslint/js": "^9.21.0",
+ "@types/react": "^19.0.10",
+ "@types/react-dom": "^19.0.4",
+ "@vitejs/plugin-react": "^4.3.4",
+ "autoprefixer": "^10.4.21",
+ "eslint": "^9.21.0",
+ "eslint-plugin-react-hooks": "^5.1.0",
+ "eslint-plugin-react-refresh": "^0.4.19",
+ "globals": "^15.15.0",
+ "postcss": "^8.5.3",
+ "tailwindcss": "^3.4.17",
+ "vite": "^6.2.0"
+ }
+ },
+ "node_modules/@alloc/quick-lru": {
+ "version": "5.2.0",
+ "resolved": "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz",
+ "integrity": "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "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,
+ "license": "Apache-2.0",
+ "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.26.2",
+ "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.26.2.tgz",
+ "integrity": "sha512-RJlIHRueQgwWitWgF8OdFYGZX328Ax5BCemNGlqHfplnRT9ESi8JkFlvaVYbS+UubVY6dpv87Fs2u5M29iNFVQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-validator-identifier": "^7.25.9",
+ "js-tokens": "^4.0.0",
+ "picocolors": "^1.0.0"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/compat-data": {
+ "version": "7.26.8",
+ "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.26.8.tgz",
+ "integrity": "sha512-oH5UPLMWR3L2wEFLnFJ1TZXqHufiTKAiLfqw5zkhS4dKXLJ10yVztfil/twG8EDTA4F/tvVNw9nOl4ZMslB8rQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/core": {
+ "version": "7.26.10",
+ "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.26.10.tgz",
+ "integrity": "sha512-vMqyb7XCDMPvJFFOaT9kxtiRh42GwlZEg1/uIgtZshS5a/8OaduUfCi7kynKgc3Tw/6Uo2D+db9qBttghhmxwQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@ampproject/remapping": "^2.2.0",
+ "@babel/code-frame": "^7.26.2",
+ "@babel/generator": "^7.26.10",
+ "@babel/helper-compilation-targets": "^7.26.5",
+ "@babel/helper-module-transforms": "^7.26.0",
+ "@babel/helpers": "^7.26.10",
+ "@babel/parser": "^7.26.10",
+ "@babel/template": "^7.26.9",
+ "@babel/traverse": "^7.26.10",
+ "@babel/types": "^7.26.10",
+ "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.27.0",
+ "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.27.0.tgz",
+ "integrity": "sha512-VybsKvpiN1gU1sdMZIp7FcqphVVKEwcuj02x73uvcHE0PTihx1nlBcowYWhDwjpoAXRv43+gDzyggGnn1XZhVw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/parser": "^7.27.0",
+ "@babel/types": "^7.27.0",
+ "@jridgewell/gen-mapping": "^0.3.5",
+ "@jridgewell/trace-mapping": "^0.3.25",
+ "jsesc": "^3.0.2"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-compilation-targets": {
+ "version": "7.27.0",
+ "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.0.tgz",
+ "integrity": "sha512-LVk7fbXml0H2xH34dFzKQ7TDZ2G4/rVTOrq9V+icbbadjbVxxeFeDsNHv2SrZeWoA+6ZiTyWYWtScEIW07EAcA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/compat-data": "^7.26.8",
+ "@babel/helper-validator-option": "^7.25.9",
+ "browserslist": "^4.24.0",
+ "lru-cache": "^5.1.1",
+ "semver": "^6.3.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-module-imports": {
+ "version": "7.25.9",
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.25.9.tgz",
+ "integrity": "sha512-tnUA4RsrmflIM6W6RFTLFSXITtl0wKjgpnLgXyowocVPrbYrLUXSBXDgTs8BlbmIzIdlBySRQjINYs2BAkiLtw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/traverse": "^7.25.9",
+ "@babel/types": "^7.25.9"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-module-transforms": {
+ "version": "7.26.0",
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.26.0.tgz",
+ "integrity": "sha512-xO+xu6B5K2czEnQye6BHA7DolFFmS3LB7stHZFaOLb1pAwO1HWLS8fXA+eh0A2yIvltPVmx3eNNDBJA2SLHXFw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-module-imports": "^7.25.9",
+ "@babel/helper-validator-identifier": "^7.25.9",
+ "@babel/traverse": "^7.25.9"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0"
+ }
+ },
+ "node_modules/@babel/helper-plugin-utils": {
+ "version": "7.26.5",
+ "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.26.5.tgz",
+ "integrity": "sha512-RS+jZcRdZdRFzMyr+wcsaqOmld1/EqTghfaBGQQd/WnRdzdlvSZ//kF7U8VQTxf1ynZ4cjUcYgjVGx13ewNPMg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-string-parser": {
+ "version": "7.25.9",
+ "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.25.9.tgz",
+ "integrity": "sha512-4A/SCr/2KLd5jrtOMFzaKjVtAei3+2r/NChoBNoZ3EyP/+GlhoaEGoWOZUmFmoITP7zOJyHIMm+DYRd8o3PvHA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-validator-identifier": {
+ "version": "7.25.9",
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.25.9.tgz",
+ "integrity": "sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-validator-option": {
+ "version": "7.25.9",
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.25.9.tgz",
+ "integrity": "sha512-e/zv1co8pp55dNdEcCynfj9X7nyUKUXoUEwfXqaZt0omVOmDe9oOTdKStH4GmAw6zxMFs50ZayuMfHDKlO7Tfw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helpers": {
+ "version": "7.27.0",
+ "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.27.0.tgz",
+ "integrity": "sha512-U5eyP/CTFPuNE3qk+WZMxFkp/4zUzdceQlfzf7DdGdhp+Fezd7HD+i8Y24ZuTMKX3wQBld449jijbGq6OdGNQg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/template": "^7.27.0",
+ "@babel/types": "^7.27.0"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/parser": {
+ "version": "7.27.0",
+ "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.27.0.tgz",
+ "integrity": "sha512-iaepho73/2Pz7w2eMS0Q5f83+0RKI7i4xmiYeBmDzfRVbQtTOG7Ts0S4HzJVsTMGI9keU8rNfuZr8DKfSt7Yyg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/types": "^7.27.0"
+ },
+ "bin": {
+ "parser": "bin/babel-parser.js"
+ },
+ "engines": {
+ "node": ">=6.0.0"
+ }
+ },
+ "node_modules/@babel/plugin-transform-react-jsx-self": {
+ "version": "7.25.9",
+ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.25.9.tgz",
+ "integrity": "sha512-y8quW6p0WHkEhmErnfe58r7x0A70uKphQm8Sp8cV7tjNQwK56sNVK0M73LK3WuYmsuyrftut4xAkjjgU0twaMg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-plugin-utils": "^7.25.9"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0-0"
+ }
+ },
+ "node_modules/@babel/plugin-transform-react-jsx-source": {
+ "version": "7.25.9",
+ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.25.9.tgz",
+ "integrity": "sha512-+iqjT8xmXhhYv4/uiYd8FNQsraMFZIfxVSqxxVSZP0WbbSAWvBXAul0m/zu+7Vv4O/3WtApy9pmaTMiumEZgfg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-plugin-utils": "^7.25.9"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0-0"
+ }
+ },
+ "node_modules/@babel/template": {
+ "version": "7.27.0",
+ "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.0.tgz",
+ "integrity": "sha512-2ncevenBqXI6qRMukPlXwHKHchC7RyMuu4xv5JBXRfOGVcTy1mXCD12qrp7Jsoxll1EV3+9sE4GugBVRjT2jFA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/code-frame": "^7.26.2",
+ "@babel/parser": "^7.27.0",
+ "@babel/types": "^7.27.0"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/traverse": {
+ "version": "7.27.0",
+ "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.27.0.tgz",
+ "integrity": "sha512-19lYZFzYVQkkHkl4Cy4WrAVcqBkgvV2YM2TU3xG6DIwO7O3ecbDPfW3yM3bjAGcqcQHi+CCtjMR3dIEHxsd6bA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/code-frame": "^7.26.2",
+ "@babel/generator": "^7.27.0",
+ "@babel/parser": "^7.27.0",
+ "@babel/template": "^7.27.0",
+ "@babel/types": "^7.27.0",
+ "debug": "^4.3.1",
+ "globals": "^11.1.0"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/traverse/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,
+ "license": "MIT",
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/@babel/types": {
+ "version": "7.27.0",
+ "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.27.0.tgz",
+ "integrity": "sha512-H45s8fVLYjbhFH62dIJ3WtmJ6RSPt/3DRO0ZcT2SUiYiQyz3BLVb9ADEnLl91m74aQPS3AzzeajZHYOalWe3bg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/helper-string-parser": "^7.25.9",
+ "@babel/helper-validator-identifier": "^7.25.9"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@esbuild/aix-ppc64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.1.tgz",
+ "integrity": "sha512-kfYGy8IdzTGy+z0vFGvExZtxkFlA4zAxgKEahG9KE1ScBjpQnFsNOX8KTU5ojNru5ed5CVoJYXFtoxaq5nFbjQ==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "aix"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/android-arm": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.1.tgz",
+ "integrity": "sha512-dp+MshLYux6j/JjdqVLnMglQlFu+MuVeNrmT5nk6q07wNhCdSnB7QZj+7G8VMUGh1q+vj2Bq8kRsuyA00I/k+Q==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/android-arm64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.1.tgz",
+ "integrity": "sha512-50tM0zCJW5kGqgG7fQ7IHvQOcAn9TKiVRuQ/lN0xR+T2lzEFvAi1ZcS8DiksFcEpf1t/GYOeOfCAgDHFpkiSmA==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/android-x64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.1.tgz",
+ "integrity": "sha512-GCj6WfUtNldqUzYkN/ITtlhwQqGWu9S45vUXs7EIYf+7rCiiqH9bCloatO9VhxsL0Pji+PF4Lz2XXCES+Q8hDw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/darwin-arm64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.1.tgz",
+ "integrity": "sha512-5hEZKPf+nQjYoSr/elb62U19/l1mZDdqidGfmFutVUjjUZrOazAtwK+Kr+3y0C/oeJfLlxo9fXb1w7L+P7E4FQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/darwin-x64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.1.tgz",
+ "integrity": "sha512-hxVnwL2Dqs3fM1IWq8Iezh0cX7ZGdVhbTfnOy5uURtao5OIVCEyj9xIzemDi7sRvKsuSdtCAhMKarxqtlyVyfA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/freebsd-arm64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.1.tgz",
+ "integrity": "sha512-1MrCZs0fZa2g8E+FUo2ipw6jw5qqQiH+tERoS5fAfKnRx6NXH31tXBKI3VpmLijLH6yriMZsxJtaXUyFt/8Y4A==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/freebsd-x64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.1.tgz",
+ "integrity": "sha512-0IZWLiTyz7nm0xuIs0q1Y3QWJC52R8aSXxe40VUxm6BB1RNmkODtW6LHvWRrGiICulcX7ZvyH6h5fqdLu4gkww==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-arm": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.1.tgz",
+ "integrity": "sha512-NdKOhS4u7JhDKw9G3cY6sWqFcnLITn6SqivVArbzIaf3cemShqfLGHYMx8Xlm/lBit3/5d7kXvriTUGa5YViuQ==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-arm64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.1.tgz",
+ "integrity": "sha512-jaN3dHi0/DDPelk0nLcXRm1q7DNJpjXy7yWaWvbfkPvI+7XNSc/lDOnCLN7gzsyzgu6qSAmgSvP9oXAhP973uQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-ia32": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.1.tgz",
+ "integrity": "sha512-OJykPaF4v8JidKNGz8c/q1lBO44sQNUQtq1KktJXdBLn1hPod5rE/Hko5ugKKZd+D2+o1a9MFGUEIUwO2YfgkQ==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-loong64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.1.tgz",
+ "integrity": "sha512-nGfornQj4dzcq5Vp835oM/o21UMlXzn79KobKlcs3Wz9smwiifknLy4xDCLUU0BWp7b/houtdrgUz7nOGnfIYg==",
+ "cpu": [
+ "loong64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-mips64el": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.1.tgz",
+ "integrity": "sha512-1osBbPEFYwIE5IVB/0g2X6i1qInZa1aIoj1TdL4AaAb55xIIgbg8Doq6a5BzYWgr+tEcDzYH67XVnTmUzL+nXg==",
+ "cpu": [
+ "mips64el"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-ppc64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.1.tgz",
+ "integrity": "sha512-/6VBJOwUf3TdTvJZ82qF3tbLuWsscd7/1w+D9LH0W/SqUgM5/JJD0lrJ1fVIfZsqB6RFmLCe0Xz3fmZc3WtyVg==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-riscv64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.1.tgz",
+ "integrity": "sha512-nSut/Mx5gnilhcq2yIMLMe3Wl4FK5wx/o0QuuCLMtmJn+WeWYoEGDN1ipcN72g1WHsnIbxGXd4i/MF0gTcuAjQ==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-s390x": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.1.tgz",
+ "integrity": "sha512-cEECeLlJNfT8kZHqLarDBQso9a27o2Zd2AQ8USAEoGtejOrCYHNtKP8XQhMDJMtthdF4GBmjR2au3x1udADQQQ==",
+ "cpu": [
+ "s390x"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/linux-x64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.1.tgz",
+ "integrity": "sha512-xbfUhu/gnvSEg+EGovRc+kjBAkrvtk38RlerAzQxvMzlB4fXpCFCeUAYzJvrnhFtdeyVCDANSjJvOvGYoeKzFA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/netbsd-arm64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.1.tgz",
+ "integrity": "sha512-O96poM2XGhLtpTh+s4+nP7YCCAfb4tJNRVZHfIE7dgmax+yMP2WgMd2OecBuaATHKTHsLWHQeuaxMRnCsH8+5g==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "netbsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/netbsd-x64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.1.tgz",
+ "integrity": "sha512-X53z6uXip6KFXBQ+Krbx25XHV/NCbzryM6ehOAeAil7X7oa4XIq+394PWGnwaSQ2WRA0KI6PUO6hTO5zeF5ijA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "netbsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/openbsd-arm64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.1.tgz",
+ "integrity": "sha512-Na9T3szbXezdzM/Kfs3GcRQNjHzM6GzFBeU1/6IV/npKP5ORtp9zbQjvkDJ47s6BCgaAZnnnu/cY1x342+MvZg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "openbsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/openbsd-x64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.1.tgz",
+ "integrity": "sha512-T3H78X2h1tszfRSf+txbt5aOp/e7TAz3ptVKu9Oyir3IAOFPGV6O9c2naym5TOriy1l0nNf6a4X5UXRZSGX/dw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "openbsd"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/sunos-x64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.1.tgz",
+ "integrity": "sha512-2H3RUvcmULO7dIE5EWJH8eubZAI4xw54H1ilJnRNZdeo8dTADEZ21w6J22XBkXqGJbe0+wnNJtw3UXRoLJnFEg==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "sunos"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/win32-arm64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.1.tgz",
+ "integrity": "sha512-GE7XvrdOzrb+yVKB9KsRMq+7a2U/K5Cf/8grVFRAGJmfADr/e/ODQ134RK2/eeHqYV5eQRFxb1hY7Nr15fv1NQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/win32-ia32": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.1.tgz",
+ "integrity": "sha512-uOxSJCIcavSiT6UnBhBzE8wy3n0hOkJsBOzy7HDAuTDE++1DJMRRVCPGisULScHL+a/ZwdXPpXD3IyFKjA7K8A==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@esbuild/win32-x64": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.1.tgz",
+ "integrity": "sha512-Y1EQdcfwMSeQN/ujR5VayLOJ1BHaK+ssyk0AEzPjC+t1lITgsnccPqFjb6V+LsTp/9Iov4ysfjxLaGJ9RPtkVg==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@eslint-community/eslint-utils": {
+ "version": "4.5.1",
+ "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.5.1.tgz",
+ "integrity": "sha512-soEIOALTfTK6EjmKMMoLugwaP0rzkad90iIWd1hMO9ARkSAyjfMfkRRhLvD5qH7vvM0Cg72pieUfR6yh6XxC4w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "eslint-visitor-keys": "^3.4.3"
+ },
+ "engines": {
+ "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0"
+ }
+ },
+ "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": {
+ "version": "3.4.3",
+ "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz",
+ "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/@eslint-community/regexpp": {
+ "version": "4.12.1",
+ "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz",
+ "integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^12.0.0 || ^14.0.0 || >=16.0.0"
+ }
+ },
+ "node_modules/@eslint/config-array": {
+ "version": "0.19.2",
+ "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.19.2.tgz",
+ "integrity": "sha512-GNKqxfHG2ySmJOBSHg7LxeUx4xpuCoFjacmlCoYWEbaPXLwvfIjixRI12xCQZeULksQb23uiA8F40w5TojpV7w==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@eslint/object-schema": "^2.1.6",
+ "debug": "^4.3.1",
+ "minimatch": "^3.1.2"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/config-helpers": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.2.0.tgz",
+ "integrity": "sha512-yJLLmLexii32mGrhW29qvU3QBVTu0GUmEf/J4XsBtVhp4JkIUFN/BjWqTF63yRvGApIDpZm5fa97LtYtINmfeQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/core": {
+ "version": "0.12.0",
+ "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.12.0.tgz",
+ "integrity": "sha512-cmrR6pytBuSMTaBweKoGMwu3EiHiEC+DoyupPmlZ0HxBJBtIxwe+j/E4XPIKNx+Q74c8lXKPwYawBf5glsTkHg==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@types/json-schema": "^7.0.15"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/eslintrc": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.1.tgz",
+ "integrity": "sha512-gtF186CXhIl1p4pJNGZw8Yc6RlshoePRvE0X91oPGb3vZ8pM3qOS9W9NGPat9LziaBV7XrJWGylNQXkGcnM3IQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ajv": "^6.12.4",
+ "debug": "^4.3.2",
+ "espree": "^10.0.1",
+ "globals": "^14.0.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": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/@eslint/eslintrc/node_modules/globals": {
+ "version": "14.0.0",
+ "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz",
+ "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/@eslint/js": {
+ "version": "9.23.0",
+ "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.23.0.tgz",
+ "integrity": "sha512-35MJ8vCPU0ZMxo7zfev2pypqTwWTofFZO6m4KAtdoFhRpLJUpHTZZ+KB3C7Hb1d7bULYwO4lJXGCi5Se+8OMbw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/object-schema": {
+ "version": "2.1.6",
+ "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.6.tgz",
+ "integrity": "sha512-RBMg5FRL0I0gs51M/guSAj5/e14VQ4tpZnQNWwuDT66P14I43ItmPfIZRhO9fUVIPOAQXU47atlywZ/czoqFPA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@eslint/plugin-kit": {
+ "version": "0.2.7",
+ "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.7.tgz",
+ "integrity": "sha512-JubJ5B2pJ4k4yGxaNLdbjrnk9d/iDz6/q8wOilpIowd6PJPgaxCuHBnBszq7Ce2TyMrywm5r4PnKm6V3iiZF+g==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@eslint/core": "^0.12.0",
+ "levn": "^0.4.1"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ }
+ },
+ "node_modules/@humanfs/core": {
+ "version": "0.19.1",
+ "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.1.tgz",
+ "integrity": "sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=18.18.0"
+ }
+ },
+ "node_modules/@humanfs/node": {
+ "version": "0.16.6",
+ "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.6.tgz",
+ "integrity": "sha512-YuI2ZHQL78Q5HbhDiBA1X4LmYdXCKCMQIfw0pw7piHJwyREFebJUvrQN4cMssyES6x+vfUbx1CIpaQUKYdQZOw==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@humanfs/core": "^0.19.1",
+ "@humanwhocodes/retry": "^0.3.0"
+ },
+ "engines": {
+ "node": ">=18.18.0"
+ }
+ },
+ "node_modules/@humanfs/node/node_modules/@humanwhocodes/retry": {
+ "version": "0.3.1",
+ "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.1.tgz",
+ "integrity": "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=18.18"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/nzakas"
+ }
+ },
+ "node_modules/@humanwhocodes/module-importer": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz",
+ "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=12.22"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/nzakas"
+ }
+ },
+ "node_modules/@humanwhocodes/retry": {
+ "version": "0.4.2",
+ "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.2.tgz",
+ "integrity": "sha512-xeO57FpIu4p1Ri3Jq/EXq4ClRm86dVF2z/+kvFnyqVYRavTZmaFaUBbWCOuuTh0o/g7DSsk6kc2vrS4Vl5oPOQ==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=18.18"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/nzakas"
+ }
+ },
+ "node_modules/@isaacs/cliui": {
+ "version": "8.0.2",
+ "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz",
+ "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "string-width": "^5.1.2",
+ "string-width-cjs": "npm:string-width@^4.2.0",
+ "strip-ansi": "^7.0.1",
+ "strip-ansi-cjs": "npm:strip-ansi@^6.0.1",
+ "wrap-ansi": "^8.1.0",
+ "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@jridgewell/gen-mapping": {
+ "version": "0.3.8",
+ "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.8.tgz",
+ "integrity": "sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA==",
+ "dev": true,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.0.0"
+ }
+ },
+ "node_modules/@jridgewell/sourcemap-codec": {
+ "version": "1.5.0",
+ "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz",
+ "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "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,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "dependencies": {
+ "@nodelib/fs.scandir": "2.1.5",
+ "fastq": "^1.6.0"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/@pkgjs/parseargs": {
+ "version": "0.11.0",
+ "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz",
+ "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==",
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "engines": {
+ "node": ">=14"
+ }
+ },
+ "node_modules/@reduxjs/toolkit": {
+ "version": "2.6.1",
+ "resolved": "https://registry.npmjs.org/@reduxjs/toolkit/-/toolkit-2.6.1.tgz",
+ "integrity": "sha512-SSlIqZNYhqm/oMkXbtofwZSt9lrncblzo6YcZ9zoX+zLngRBrCOjK4lNLdkNucJF58RHOWrD9txT3bT3piH7Zw==",
+ "license": "MIT",
+ "dependencies": {
+ "immer": "^10.0.3",
+ "redux": "^5.0.1",
+ "redux-thunk": "^3.1.0",
+ "reselect": "^5.1.0"
+ },
+ "peerDependencies": {
+ "react": "^16.9.0 || ^17.0.0 || ^18 || ^19",
+ "react-redux": "^7.2.1 || ^8.1.3 || ^9.0.0"
+ },
+ "peerDependenciesMeta": {
+ "react": {
+ "optional": true
+ },
+ "react-redux": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/@rollup/rollup-android-arm-eabi": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.37.0.tgz",
+ "integrity": "sha512-l7StVw6WAa8l3vA1ov80jyetOAEo1FtHvZDbzXDO/02Sq/QVvqlHkYoFwDJPIMj0GKiistsBudfx5tGFnwYWDQ==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ]
+ },
+ "node_modules/@rollup/rollup-android-arm64": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.37.0.tgz",
+ "integrity": "sha512-6U3SlVyMxezt8Y+/iEBcbp945uZjJwjZimu76xoG7tO1av9VO691z8PkhzQ85ith2I8R2RddEPeSfcbyPfD4hA==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ]
+ },
+ "node_modules/@rollup/rollup-darwin-arm64": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.37.0.tgz",
+ "integrity": "sha512-+iTQ5YHuGmPt10NTzEyMPbayiNTcOZDWsbxZYR1ZnmLnZxG17ivrPSWFO9j6GalY0+gV3Jtwrrs12DBscxnlYA==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ]
+ },
+ "node_modules/@rollup/rollup-darwin-x64": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.37.0.tgz",
+ "integrity": "sha512-m8W2UbxLDcmRKVjgl5J/k4B8d7qX2EcJve3Sut7YGrQoPtCIQGPH5AMzuFvYRWZi0FVS0zEY4c8uttPfX6bwYQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ]
+ },
+ "node_modules/@rollup/rollup-freebsd-arm64": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.37.0.tgz",
+ "integrity": "sha512-FOMXGmH15OmtQWEt174v9P1JqqhlgYge/bUjIbiVD1nI1NeJ30HYT9SJlZMqdo1uQFyt9cz748F1BHghWaDnVA==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ]
+ },
+ "node_modules/@rollup/rollup-freebsd-x64": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.37.0.tgz",
+ "integrity": "sha512-SZMxNttjPKvV14Hjck5t70xS3l63sbVwl98g3FlVVx2YIDmfUIy29jQrsw06ewEYQ8lQSuY9mpAPlmgRD2iSsA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm-gnueabihf": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.37.0.tgz",
+ "integrity": "sha512-hhAALKJPidCwZcj+g+iN+38SIOkhK2a9bqtJR+EtyxrKKSt1ynCBeqrQy31z0oWU6thRZzdx53hVgEbRkuI19w==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm-musleabihf": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.37.0.tgz",
+ "integrity": "sha512-jUb/kmn/Gd8epbHKEqkRAxq5c2EwRt0DqhSGWjPFxLeFvldFdHQs/n8lQ9x85oAeVb6bHcS8irhTJX2FCOd8Ag==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm64-gnu": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.37.0.tgz",
+ "integrity": "sha512-oNrJxcQT9IcbcmKlkF+Yz2tmOxZgG9D9GRq+1OE6XCQwCVwxixYAa38Z8qqPzQvzt1FCfmrHX03E0pWoXm1DqA==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm64-musl": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.37.0.tgz",
+ "integrity": "sha512-pfxLBMls+28Ey2enpX3JvjEjaJMBX5XlPCZNGxj4kdJyHduPBXtxYeb8alo0a7bqOoWZW2uKynhHxF/MWoHaGQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-loongarch64-gnu": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.37.0.tgz",
+ "integrity": "sha512-yCE0NnutTC/7IGUq/PUHmoeZbIwq3KRh02e9SfFh7Vmc1Z7atuJRYWhRME5fKgT8aS20mwi1RyChA23qSyRGpA==",
+ "cpu": [
+ "loong64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-powerpc64le-gnu": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.37.0.tgz",
+ "integrity": "sha512-NxcICptHk06E2Lh3a4Pu+2PEdZ6ahNHuK7o6Np9zcWkrBMuv21j10SQDJW3C9Yf/A/P7cutWoC/DptNLVsZ0VQ==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-riscv64-gnu": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.37.0.tgz",
+ "integrity": "sha512-PpWwHMPCVpFZLTfLq7EWJWvrmEuLdGn1GMYcm5MV7PaRgwCEYJAwiN94uBuZev0/J/hFIIJCsYw4nLmXA9J7Pw==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-riscv64-musl": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.37.0.tgz",
+ "integrity": "sha512-DTNwl6a3CfhGTAOYZ4KtYbdS8b+275LSLqJVJIrPa5/JuIufWWZ/QFvkxp52gpmguN95eujrM68ZG+zVxa8zHA==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-s390x-gnu": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.37.0.tgz",
+ "integrity": "sha512-hZDDU5fgWvDdHFuExN1gBOhCuzo/8TMpidfOR+1cPZJflcEzXdCy1LjnklQdW8/Et9sryOPJAKAQRw8Jq7Tg+A==",
+ "cpu": [
+ "s390x"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-x64-gnu": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.37.0.tgz",
+ "integrity": "sha512-pKivGpgJM5g8dwj0ywBwe/HeVAUSuVVJhUTa/URXjxvoyTT/AxsLTAbkHkDHG7qQxLoW2s3apEIl26uUe08LVQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-x64-musl": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.37.0.tgz",
+ "integrity": "sha512-E2lPrLKE8sQbY/2bEkVTGDEk4/49UYRVWgj90MY8yPjpnGBQ+Xi1Qnr7b7UIWw1NOggdFQFOLZ8+5CzCiz143w==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-arm64-msvc": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.37.0.tgz",
+ "integrity": "sha512-Jm7biMazjNzTU4PrQtr7VS8ibeys9Pn29/1bm4ph7CP2kf21950LgN+BaE2mJ1QujnvOc6p54eWWiVvn05SOBg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-ia32-msvc": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.37.0.tgz",
+ "integrity": "sha512-e3/1SFm1OjefWICB2Ucstg2dxYDkDTZGDYgwufcbsxTHyqQps1UQf33dFEChBNmeSsTOyrjw2JJq0zbG5GF6RA==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-x64-msvc": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.37.0.tgz",
+ "integrity": "sha512-LWbXUBwn/bcLx2sSsqy7pK5o+Nr+VCoRoAohfJ5C/aBio9nfJmGQqHAhU6pwxV/RmyTk5AqdySma7uwWGlmeuA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/parser": "^7.20.7",
+ "@babel/types": "^7.20.7",
+ "@types/babel__generator": "*",
+ "@types/babel__template": "*",
+ "@types/babel__traverse": "*"
+ }
+ },
+ "node_modules/@types/babel__generator": {
+ "version": "7.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,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/types": "^7.0.0"
+ }
+ },
+ "node_modules/@types/babel__template": {
+ "version": "7.4.4",
+ "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz",
+ "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/parser": "^7.1.0",
+ "@babel/types": "^7.0.0"
+ }
+ },
+ "node_modules/@types/babel__traverse": {
+ "version": "7.20.6",
+ "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.20.6.tgz",
+ "integrity": "sha512-r1bzfrm0tomOI8g1SzvCaQHo6Lcv6zu0EA+W2kHrt8dyrHQxGzBBL4kdkzIS+jBMV+EYcMAEAqXqYaLJq5rOZg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/types": "^7.20.7"
+ }
+ },
+ "node_modules/@types/cookie": {
+ "version": "0.6.0",
+ "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.6.0.tgz",
+ "integrity": "sha512-4Kh9a6B2bQciAhf7FSuMRRkUWecJgJu9nPnx3yzpsfXX/c50REIqpHY4C82bXP90qrLtXtkDxTZosYO3UpOwlA==",
+ "license": "MIT"
+ },
+ "node_modules/@types/estree": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.7.tgz",
+ "integrity": "sha512-w28IoSUCJpidD/TGviZwwMJckNESJZXFu7NBZ5YJ4mEUnNraUn9Pm8HSZm/jDF1pDWYKspWE7oVphigUPRakIQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@types/json-schema": {
+ "version": "7.0.15",
+ "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz",
+ "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@types/node": {
+ "version": "18.19.83",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-18.19.83.tgz",
+ "integrity": "sha512-D69JeR5SfFS5H6FLbUaS0vE4r1dGhmMBbG4Ed6BNS4wkDK8GZjsdCShT5LCN59vOHEUHnFCY9J4aclXlIphMkA==",
+ "license": "MIT",
+ "dependencies": {
+ "undici-types": "~5.26.4"
+ }
+ },
+ "node_modules/@types/node-fetch": {
+ "version": "2.6.12",
+ "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.6.12.tgz",
+ "integrity": "sha512-8nneRWKCg3rMtF69nLQJnOYUcbafYeFSjqkw3jCRLsqkWFlHaoQrr5mXmofFGOx3DKn7UfmBMyov8ySvLRVldA==",
+ "license": "MIT",
+ "dependencies": {
+ "@types/node": "*",
+ "form-data": "^4.0.0"
+ }
+ },
+ "node_modules/@types/react": {
+ "version": "19.0.12",
+ "resolved": "https://registry.npmjs.org/@types/react/-/react-19.0.12.tgz",
+ "integrity": "sha512-V6Ar115dBDrjbtXSrS+/Oruobc+qVbbUxDFC1RSbRqLt5SYvxxyIDrSC85RWml54g+jfNeEMZhEj7wW07ONQhA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "csstype": "^3.0.2"
+ }
+ },
+ "node_modules/@types/react-dom": {
+ "version": "19.0.4",
+ "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.0.4.tgz",
+ "integrity": "sha512-4fSQ8vWFkg+TGhePfUzVmat3eC14TXYSsiiDSLI0dVLsrm9gZFABjPy/Qu6TKgl1tq1Bu1yDsuQgY3A3DOjCcg==",
+ "dev": true,
+ "license": "MIT",
+ "peerDependencies": {
+ "@types/react": "^19.0.0"
+ }
+ },
+ "node_modules/@types/use-sync-external-store": {
+ "version": "0.0.6",
+ "resolved": "https://registry.npmjs.org/@types/use-sync-external-store/-/use-sync-external-store-0.0.6.tgz",
+ "integrity": "sha512-zFDAD+tlpf2r4asuHEj0XH6pY6i0g5NeAHPn+15wk3BV6JA69eERFXC1gyGThDkVa1zCyKr5jox1+2LbV/AMLg==",
+ "license": "MIT"
+ },
+ "node_modules/@vitejs/plugin-react": {
+ "version": "4.3.4",
+ "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.3.4.tgz",
+ "integrity": "sha512-SCCPBJtYLdE8PX/7ZQAs1QAZ8Jqwih+0VBLum1EGqmCCQal+MIUqLCzj3ZUy8ufbC0cAM4LRlSTm7IQJwWT4ug==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@babel/core": "^7.26.0",
+ "@babel/plugin-transform-react-jsx-self": "^7.25.9",
+ "@babel/plugin-transform-react-jsx-source": "^7.25.9",
+ "@types/babel__core": "^7.20.5",
+ "react-refresh": "^0.14.2"
+ },
+ "engines": {
+ "node": "^14.18.0 || >=16.0.0"
+ },
+ "peerDependencies": {
+ "vite": "^4.2.0 || ^5.0.0 || ^6.0.0"
+ }
+ },
+ "node_modules/abort-controller": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz",
+ "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==",
+ "license": "MIT",
+ "dependencies": {
+ "event-target-shim": "^5.0.0"
+ },
+ "engines": {
+ "node": ">=6.5"
+ }
+ },
+ "node_modules/acorn": {
+ "version": "8.14.1",
+ "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.1.tgz",
+ "integrity": "sha512-OvQ/2pUDKmgfCg++xsTX1wGxfTaszcHVcTctW4UJB4hibJx2HXxxO5UmVgyjMa+ZDsiaf5wWLXYpRWMmBI0QHg==",
+ "dev": true,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "peerDependencies": {
+ "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
+ }
+ },
+ "node_modules/agentkeepalive": {
+ "version": "4.6.0",
+ "resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-4.6.0.tgz",
+ "integrity": "sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ==",
+ "license": "MIT",
+ "dependencies": {
+ "humanize-ms": "^1.2.1"
+ },
+ "engines": {
+ "node": ">= 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,
+ "license": "MIT",
+ "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": "6.1.0",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz",
+ "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/ansi-regex?sponsor=1"
+ }
+ },
+ "node_modules/ansi-styles": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
+ "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "color-convert": "^2.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/ansi-styles?sponsor=1"
+ }
+ },
+ "node_modules/any-promise": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz",
+ "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/anymatch": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz",
+ "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "normalize-path": "^3.0.0",
+ "picomatch": "^2.0.4"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/arg": {
+ "version": "5.0.2",
+ "resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz",
+ "integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "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,
+ "license": "Python-2.0"
+ },
+ "node_modules/asynckit": {
+ "version": "0.4.0",
+ "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz",
+ "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==",
+ "license": "MIT"
+ },
+ "node_modules/autoprefixer": {
+ "version": "10.4.21",
+ "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.21.tgz",
+ "integrity": "sha512-O+A6LWV5LDHSJD3LjHYoNi4VLsj/Whi7k6zG12xTYaU4cQ8oxQGckXNX8cRHK5yOZ/ppVHe0ZBXGzSV9jXdVbQ==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/autoprefixer"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "browserslist": "^4.24.4",
+ "caniuse-lite": "^1.0.30001702",
+ "fraction.js": "^4.3.7",
+ "normalize-range": "^0.1.2",
+ "picocolors": "^1.1.1",
+ "postcss-value-parser": "^4.2.0"
+ },
+ "bin": {
+ "autoprefixer": "bin/autoprefixer"
+ },
+ "engines": {
+ "node": "^10 || ^12 || >=14"
+ },
+ "peerDependencies": {
+ "postcss": "^8.1.0"
+ }
+ },
+ "node_modules/axios": {
+ "version": "1.8.4",
+ "resolved": "https://registry.npmjs.org/axios/-/axios-1.8.4.tgz",
+ "integrity": "sha512-eBSYY4Y68NNlHbHBMdeDmKNtDgXWhQsJcGqzO3iLUM0GraQFSS9cVgPX5I9b3lbdFKyYoAEGAZF1DwhTaljNAw==",
+ "license": "MIT",
+ "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",
+ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/binary-extensions": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz",
+ "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "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,
+ "license": "MIT",
+ "dependencies": {
+ "balanced-match": "^1.0.0",
+ "concat-map": "0.0.1"
+ }
+ },
+ "node_modules/braces": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz",
+ "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "fill-range": "^7.1.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/browserslist": {
+ "version": "4.24.4",
+ "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.24.4.tgz",
+ "integrity": "sha512-KDi1Ny1gSePi1vm0q4oxSF8b4DR44GF4BbmS2YdhPLOEqd8pDviZOGH/GsmRwoWJ2+5Lr085X7naowMwKHDG1A==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "caniuse-lite": "^1.0.30001688",
+ "electron-to-chromium": "^1.5.73",
+ "node-releases": "^2.0.19",
+ "update-browserslist-db": "^1.1.1"
+ },
+ "bin": {
+ "browserslist": "cli.js"
+ },
+ "engines": {
+ "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
+ }
+ },
+ "node_modules/call-bind-apply-helpers": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
+ "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/callsites": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz",
+ "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/camelcase-css": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz",
+ "integrity": "sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/caniuse-lite": {
+ "version": "1.0.30001707",
+ "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001707.tgz",
+ "integrity": "sha512-3qtRjw/HQSMlDWf+X79N206fepf4SOOU6SQLMaq/0KkZLmSjPxAkBOQQ+FxbHKfHmYLZFfdWsO3KA90ceHPSnw==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/caniuse-lite"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "CC-BY-4.0"
+ },
+ "node_modules/chalk": {
+ "version": "4.1.2",
+ "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
+ "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-styles": "^4.1.0",
+ "supports-color": "^7.1.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/chalk?sponsor=1"
+ }
+ },
+ "node_modules/chokidar": {
+ "version": "3.6.0",
+ "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz",
+ "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "anymatch": "~3.1.2",
+ "braces": "~3.0.2",
+ "glob-parent": "~5.1.2",
+ "is-binary-path": "~2.1.0",
+ "is-glob": "~4.0.1",
+ "normalize-path": "~3.0.0",
+ "readdirp": "~3.6.0"
+ },
+ "engines": {
+ "node": ">= 8.10.0"
+ },
+ "funding": {
+ "url": "https://paulmillr.com/funding/"
+ },
+ "optionalDependencies": {
+ "fsevents": "~2.3.2"
+ }
+ },
+ "node_modules/chokidar/node_modules/glob-parent": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
+ "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "is-glob": "^4.0.1"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/color-convert": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
+ "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "color-name": "~1.1.4"
+ },
+ "engines": {
+ "node": ">=7.0.0"
+ }
+ },
+ "node_modules/color-name": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
+ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/combined-stream": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
+ "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
+ "license": "MIT",
+ "dependencies": {
+ "delayed-stream": "~1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/commander": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz",
+ "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/concat-map": {
+ "version": "0.0.1",
+ "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+ "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/convert-source-map": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz",
+ "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/cookie": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/cookie/-/cookie-1.0.2.tgz",
+ "integrity": "sha512-9Kr/j4O16ISv8zBBhJoi4bXOYNTkFLOqSL3UDB0njXxCXNezjeyVrJyGOWtgfs/q2km1gwBcfH8q1yEGoMYunA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/cross-spawn": {
+ "version": "7.0.6",
+ "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz",
+ "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "path-key": "^3.1.0",
+ "shebang-command": "^2.0.0",
+ "which": "^2.0.1"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/cssesc": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz",
+ "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "cssesc": "bin/cssesc"
+ },
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/csstype": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz",
+ "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/debug": {
+ "version": "4.4.0",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz",
+ "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ms": "^2.1.3"
+ },
+ "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,
+ "license": "MIT"
+ },
+ "node_modules/delayed-stream": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
+ "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.4.0"
+ }
+ },
+ "node_modules/didyoumean": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.2.tgz",
+ "integrity": "sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw==",
+ "dev": true,
+ "license": "Apache-2.0"
+ },
+ "node_modules/dlv": {
+ "version": "1.1.3",
+ "resolved": "https://registry.npmjs.org/dlv/-/dlv-1.1.3.tgz",
+ "integrity": "sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/dotenv": {
+ "version": "16.4.7",
+ "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz",
+ "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==",
+ "license": "BSD-2-Clause",
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://dotenvx.com"
+ }
+ },
+ "node_modules/dunder-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
+ "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
+ "license": "MIT",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.1",
+ "es-errors": "^1.3.0",
+ "gopd": "^1.2.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/eastasianwidth": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz",
+ "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/electron-to-chromium": {
+ "version": "1.5.123",
+ "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.123.tgz",
+ "integrity": "sha512-refir3NlutEZqlKaBLK0tzlVLe5P2wDKS7UQt/3SpibizgsRAPOsqQC3ffw1nlv3ze5gjRQZYHoPymgVZkplFA==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/emoji-regex": {
+ "version": "9.2.2",
+ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz",
+ "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/es-define-property": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
+ "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-errors": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
+ "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-object-atoms": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz",
+ "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-set-tostringtag": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz",
+ "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==",
+ "license": "MIT",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "get-intrinsic": "^1.2.6",
+ "has-tostringtag": "^1.0.2",
+ "hasown": "^2.0.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/esbuild": {
+ "version": "0.25.1",
+ "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.1.tgz",
+ "integrity": "sha512-BGO5LtrGC7vxnqucAe/rmvKdJllfGaYWdyABvyMoXQlfYMb2bbRuReWR5tEGE//4LcNJj9XrkovTqNYRFZHAMQ==",
+ "dev": true,
+ "hasInstallScript": true,
+ "license": "MIT",
+ "bin": {
+ "esbuild": "bin/esbuild"
+ },
+ "engines": {
+ "node": ">=18"
+ },
+ "optionalDependencies": {
+ "@esbuild/aix-ppc64": "0.25.1",
+ "@esbuild/android-arm": "0.25.1",
+ "@esbuild/android-arm64": "0.25.1",
+ "@esbuild/android-x64": "0.25.1",
+ "@esbuild/darwin-arm64": "0.25.1",
+ "@esbuild/darwin-x64": "0.25.1",
+ "@esbuild/freebsd-arm64": "0.25.1",
+ "@esbuild/freebsd-x64": "0.25.1",
+ "@esbuild/linux-arm": "0.25.1",
+ "@esbuild/linux-arm64": "0.25.1",
+ "@esbuild/linux-ia32": "0.25.1",
+ "@esbuild/linux-loong64": "0.25.1",
+ "@esbuild/linux-mips64el": "0.25.1",
+ "@esbuild/linux-ppc64": "0.25.1",
+ "@esbuild/linux-riscv64": "0.25.1",
+ "@esbuild/linux-s390x": "0.25.1",
+ "@esbuild/linux-x64": "0.25.1",
+ "@esbuild/netbsd-arm64": "0.25.1",
+ "@esbuild/netbsd-x64": "0.25.1",
+ "@esbuild/openbsd-arm64": "0.25.1",
+ "@esbuild/openbsd-x64": "0.25.1",
+ "@esbuild/sunos-x64": "0.25.1",
+ "@esbuild/win32-arm64": "0.25.1",
+ "@esbuild/win32-ia32": "0.25.1",
+ "@esbuild/win32-x64": "0.25.1"
+ }
+ },
+ "node_modules/escalade": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
+ "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/escape-string-regexp": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
+ "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/eslint": {
+ "version": "9.23.0",
+ "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.23.0.tgz",
+ "integrity": "sha512-jV7AbNoFPAY1EkFYpLq5bslU9NLNO8xnEeQXwErNibVryjk67wHVmddTBilc5srIttJDBrB0eMHKZBFbSIABCw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@eslint-community/eslint-utils": "^4.2.0",
+ "@eslint-community/regexpp": "^4.12.1",
+ "@eslint/config-array": "^0.19.2",
+ "@eslint/config-helpers": "^0.2.0",
+ "@eslint/core": "^0.12.0",
+ "@eslint/eslintrc": "^3.3.1",
+ "@eslint/js": "9.23.0",
+ "@eslint/plugin-kit": "^0.2.7",
+ "@humanfs/node": "^0.16.6",
+ "@humanwhocodes/module-importer": "^1.0.1",
+ "@humanwhocodes/retry": "^0.4.2",
+ "@types/estree": "^1.0.6",
+ "@types/json-schema": "^7.0.15",
+ "ajv": "^6.12.4",
+ "chalk": "^4.0.0",
+ "cross-spawn": "^7.0.6",
+ "debug": "^4.3.2",
+ "escape-string-regexp": "^4.0.0",
+ "eslint-scope": "^8.3.0",
+ "eslint-visitor-keys": "^4.2.0",
+ "espree": "^10.3.0",
+ "esquery": "^1.5.0",
+ "esutils": "^2.0.2",
+ "fast-deep-equal": "^3.1.3",
+ "file-entry-cache": "^8.0.0",
+ "find-up": "^5.0.0",
+ "glob-parent": "^6.0.2",
+ "ignore": "^5.2.0",
+ "imurmurhash": "^0.1.4",
+ "is-glob": "^4.0.0",
+ "json-stable-stringify-without-jsonify": "^1.0.1",
+ "lodash.merge": "^4.6.2",
+ "minimatch": "^3.1.2",
+ "natural-compare": "^1.4.0",
+ "optionator": "^0.9.3"
+ },
+ "bin": {
+ "eslint": "bin/eslint.js"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://eslint.org/donate"
+ },
+ "peerDependencies": {
+ "jiti": "*"
+ },
+ "peerDependenciesMeta": {
+ "jiti": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/eslint-plugin-react-hooks": {
+ "version": "5.2.0",
+ "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-5.2.0.tgz",
+ "integrity": "sha512-+f15FfK64YQwZdJNELETdn5ibXEUQmW1DZL6KXhNnc2heoy/sg9VJJeT7n8TlMWouzWqSWavFkIhHyIbIAEapg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=10"
+ },
+ "peerDependencies": {
+ "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0"
+ }
+ },
+ "node_modules/eslint-plugin-react-refresh": {
+ "version": "0.4.19",
+ "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.4.19.tgz",
+ "integrity": "sha512-eyy8pcr/YxSYjBoqIFSrlbn9i/xvxUFa8CjzAYo9cFjgGXqq1hyjihcpZvxRLalpaWmueWR81xn7vuKmAFijDQ==",
+ "dev": true,
+ "license": "MIT",
+ "peerDependencies": {
+ "eslint": ">=8.40"
+ }
+ },
+ "node_modules/eslint-scope": {
+ "version": "8.3.0",
+ "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.3.0.tgz",
+ "integrity": "sha512-pUNxi75F8MJ/GdeKtVLSbYg4ZI34J6C0C7sbL4YOp2exGwen7ZsuBqKzUhXd0qMQ362yET3z+uPwKeg/0C2XCQ==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "esrecurse": "^4.3.0",
+ "estraverse": "^5.2.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/eslint-visitor-keys": {
+ "version": "4.2.0",
+ "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz",
+ "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/espree": {
+ "version": "10.3.0",
+ "resolved": "https://registry.npmjs.org/espree/-/espree-10.3.0.tgz",
+ "integrity": "sha512-0QYC8b24HWY8zjRnDTL6RiHfDbAWn63qb4LMj1Z4b076A4une81+z03Kg7l7mn/48PUTqoLptSXez8oknU8Clg==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "acorn": "^8.14.0",
+ "acorn-jsx": "^5.3.2",
+ "eslint-visitor-keys": "^4.2.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/eslint"
+ }
+ },
+ "node_modules/esquery": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz",
+ "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "dependencies": {
+ "estraverse": "^5.1.0"
+ },
+ "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,
+ "license": "BSD-2-Clause",
+ "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,
+ "license": "BSD-2-Clause",
+ "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,
+ "license": "BSD-2-Clause",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/event-target-shim": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz",
+ "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "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,
+ "license": "MIT"
+ },
+ "node_modules/fast-glob": {
+ "version": "3.3.3",
+ "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz",
+ "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@nodelib/fs.stat": "^2.0.2",
+ "@nodelib/fs.walk": "^1.2.3",
+ "glob-parent": "^5.1.2",
+ "merge2": "^1.3.0",
+ "micromatch": "^4.0.8"
+ },
+ "engines": {
+ "node": ">=8.6.0"
+ }
+ },
+ "node_modules/fast-glob/node_modules/glob-parent": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
+ "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "is-glob": "^4.0.1"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/fast-json-stable-stringify": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
+ "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "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,
+ "license": "MIT"
+ },
+ "node_modules/fastq": {
+ "version": "1.19.1",
+ "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz",
+ "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "reusify": "^1.0.4"
+ }
+ },
+ "node_modules/file-entry-cache": {
+ "version": "8.0.0",
+ "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz",
+ "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "flat-cache": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=16.0.0"
+ }
+ },
+ "node_modules/fill-range": {
+ "version": "7.1.1",
+ "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz",
+ "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "to-regex-range": "^5.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/find-up": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+ "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+ "dev": true,
+ "license": "MIT",
+ "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": "4.0.1",
+ "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz",
+ "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "flatted": "^3.2.9",
+ "keyv": "^4.5.4"
+ },
+ "engines": {
+ "node": ">=16"
+ }
+ },
+ "node_modules/flatted": {
+ "version": "3.3.3",
+ "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz",
+ "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/follow-redirects": {
+ "version": "1.15.9",
+ "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz",
+ "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==",
+ "funding": [
+ {
+ "type": "individual",
+ "url": "https://github.com/sponsors/RubenVerborgh"
+ }
+ ],
+ "license": "MIT",
+ "engines": {
+ "node": ">=4.0"
+ },
+ "peerDependenciesMeta": {
+ "debug": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/foreground-child": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.1.tgz",
+ "integrity": "sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "cross-spawn": "^7.0.6",
+ "signal-exit": "^4.0.1"
+ },
+ "engines": {
+ "node": ">=14"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
+ "node_modules/form-data": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.2.tgz",
+ "integrity": "sha512-hGfm/slu0ZabnNt4oaRZ6uREyfCj6P4fT/n6A1rGV+Z0VdGXjfOhVUpkn6qVQONHGIFwmveGXyDs75+nr6FM8w==",
+ "license": "MIT",
+ "dependencies": {
+ "asynckit": "^0.4.0",
+ "combined-stream": "^1.0.8",
+ "es-set-tostringtag": "^2.1.0",
+ "mime-types": "^2.1.12"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/form-data-encoder": {
+ "version": "1.7.2",
+ "resolved": "https://registry.npmjs.org/form-data-encoder/-/form-data-encoder-1.7.2.tgz",
+ "integrity": "sha512-qfqtYan3rxrnCk1VYaA4H+Ms9xdpPqvLZa6xmMgFvhO32x7/3J/ExcTd6qpxM0vH2GdMI+poehyBZvqfMTto8A==",
+ "license": "MIT"
+ },
+ "node_modules/formdata-node": {
+ "version": "4.4.1",
+ "resolved": "https://registry.npmjs.org/formdata-node/-/formdata-node-4.4.1.tgz",
+ "integrity": "sha512-0iirZp3uVDjVGt9p49aTaqjk84TrglENEDuqfdlZQ1roC9CWlPk6Avf8EEnZNcAqPonwkG35x4n3ww/1THYAeQ==",
+ "license": "MIT",
+ "dependencies": {
+ "node-domexception": "1.0.0",
+ "web-streams-polyfill": "4.0.0-beta.3"
+ },
+ "engines": {
+ "node": ">= 12.20"
+ }
+ },
+ "node_modules/fraction.js": {
+ "version": "4.3.7",
+ "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.3.7.tgz",
+ "integrity": "sha512-ZsDfxO51wGAXREY55a7la9LScWpwv9RxIrYABrlvOFBlH/ShPnrtsXeuUIfXKKOVicNxQ+o8JTbJvjS4M89yew==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "*"
+ },
+ "funding": {
+ "type": "patreon",
+ "url": "https://github.com/sponsors/rawify"
+ }
+ },
+ "node_modules/fsevents": {
+ "version": "2.3.3",
+ "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
+ "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
+ "dev": true,
+ "hasInstallScript": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
+ }
+ },
+ "node_modules/function-bind": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
+ "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
+ "license": "MIT",
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/gensync": {
+ "version": "1.0.0-beta.2",
+ "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz",
+ "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/get-intrinsic": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
+ "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
+ "license": "MIT",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.2",
+ "es-define-property": "^1.0.1",
+ "es-errors": "^1.3.0",
+ "es-object-atoms": "^1.1.1",
+ "function-bind": "^1.1.2",
+ "get-proto": "^1.0.1",
+ "gopd": "^1.2.0",
+ "has-symbols": "^1.1.0",
+ "hasown": "^2.0.2",
+ "math-intrinsics": "^1.1.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/get-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
+ "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
+ "license": "MIT",
+ "dependencies": {
+ "dunder-proto": "^1.0.1",
+ "es-object-atoms": "^1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/glob": {
+ "version": "10.4.5",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz",
+ "integrity": "sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "foreground-child": "^3.1.0",
+ "jackspeak": "^3.1.2",
+ "minimatch": "^9.0.4",
+ "minipass": "^7.1.2",
+ "package-json-from-dist": "^1.0.0",
+ "path-scurry": "^1.11.1"
+ },
+ "bin": {
+ "glob": "dist/esm/bin.mjs"
+ },
+ "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,
+ "license": "ISC",
+ "dependencies": {
+ "is-glob": "^4.0.3"
+ },
+ "engines": {
+ "node": ">=10.13.0"
+ }
+ },
+ "node_modules/glob/node_modules/brace-expansion": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
+ "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "balanced-match": "^1.0.0"
+ }
+ },
+ "node_modules/glob/node_modules/minimatch": {
+ "version": "9.0.5",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz",
+ "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "brace-expansion": "^2.0.1"
+ },
+ "engines": {
+ "node": ">=16 || 14 >=14.17"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
+ "node_modules/globals": {
+ "version": "15.15.0",
+ "resolved": "https://registry.npmjs.org/globals/-/globals-15.15.0.tgz",
+ "integrity": "sha512-7ACyT3wmyp3I61S4fG682L0VA2RGD9otkqGJIwNUMF1SWUombIIk+af1unuDYgMm082aHYwD+mzJvv9Iu8dsgg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/gopd": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
+ "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-flag": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
+ "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/has-symbols": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
+ "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-tostringtag": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz",
+ "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==",
+ "license": "MIT",
+ "dependencies": {
+ "has-symbols": "^1.0.3"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/hasown": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
+ "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
+ "license": "MIT",
+ "dependencies": {
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/humanize-ms": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/humanize-ms/-/humanize-ms-1.2.1.tgz",
+ "integrity": "sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==",
+ "license": "MIT",
+ "dependencies": {
+ "ms": "^2.0.0"
+ }
+ },
+ "node_modules/ignore": {
+ "version": "5.3.2",
+ "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz",
+ "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 4"
+ }
+ },
+ "node_modules/immer": {
+ "version": "10.1.1",
+ "resolved": "https://registry.npmjs.org/immer/-/immer-10.1.1.tgz",
+ "integrity": "sha512-s2MPrmjovJcoMaHtx6K11Ra7oD05NT97w1IC5zpMkT6Atjr7H8LjaDd81iIxUYpMKSRRNMJE703M1Fhr/TctHw==",
+ "license": "MIT",
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/immer"
+ }
+ },
+ "node_modules/import-fresh": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz",
+ "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==",
+ "dev": true,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.8.19"
+ }
+ },
+ "node_modules/is-binary-path": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
+ "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "binary-extensions": "^2.0.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/is-core-module": {
+ "version": "2.16.1",
+ "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz",
+ "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "hasown": "^2.0.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/is-extglob": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
+ "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/is-fullwidth-code-point": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz",
+ "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/is-glob": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
+ "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "is-extglob": "^2.1.1"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/is-number": {
+ "version": "7.0.0",
+ "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
+ "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.12.0"
+ }
+ },
+ "node_modules/isexe": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
+ "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/jackspeak": {
+ "version": "3.4.3",
+ "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz",
+ "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==",
+ "dev": true,
+ "license": "BlueOak-1.0.0",
+ "dependencies": {
+ "@isaacs/cliui": "^8.0.2"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ },
+ "optionalDependencies": {
+ "@pkgjs/parseargs": "^0.11.0"
+ }
+ },
+ "node_modules/jiti": {
+ "version": "1.21.7",
+ "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.7.tgz",
+ "integrity": "sha512-/imKNG4EbWNrVjoNC/1H5/9GFy+tqjGBHCaSsN+P2RnPqjsLmv6UD3Ej+Kj8nBWaRAwyk7kK5ZUc+OEatnTR3A==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "jiti": "bin/jiti.js"
+ }
+ },
+ "node_modules/js-tokens": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
+ "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/js-yaml": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz",
+ "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "argparse": "^2.0.1"
+ },
+ "bin": {
+ "js-yaml": "bin/js-yaml.js"
+ }
+ },
+ "node_modules/jsesc": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz",
+ "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "jsesc": "bin/jsesc"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/json-buffer": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz",
+ "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "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,
+ "license": "MIT"
+ },
+ "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,
+ "license": "MIT"
+ },
+ "node_modules/json5": {
+ "version": "2.2.3",
+ "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz",
+ "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "json5": "lib/cli.js"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/keyv": {
+ "version": "4.5.4",
+ "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
+ "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "json-buffer": "3.0.1"
+ }
+ },
+ "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,
+ "license": "MIT",
+ "dependencies": {
+ "prelude-ls": "^1.2.1",
+ "type-check": "~0.4.0"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/lilconfig": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-3.1.3.tgz",
+ "integrity": "sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=14"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/antonk52"
+ }
+ },
+ "node_modules/lines-and-columns": {
+ "version": "1.2.4",
+ "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz",
+ "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/locate-path": {
+ "version": "6.0.0",
+ "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+ "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+ "dev": true,
+ "license": "MIT",
+ "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,
+ "license": "MIT"
+ },
+ "node_modules/lru-cache": {
+ "version": "5.1.1",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
+ "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "yallist": "^3.0.2"
+ }
+ },
+ "node_modules/math-intrinsics": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
+ "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/merge2": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz",
+ "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/micromatch": {
+ "version": "4.0.8",
+ "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz",
+ "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "braces": "^3.0.3",
+ "picomatch": "^2.3.1"
+ },
+ "engines": {
+ "node": ">=8.6"
+ }
+ },
+ "node_modules/mime-db": {
+ "version": "1.52.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/mime-types": {
+ "version": "2.1.35",
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+ "license": "MIT",
+ "dependencies": {
+ "mime-db": "1.52.0"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/minimatch": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+ "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "brace-expansion": "^1.1.7"
+ },
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/minipass": {
+ "version": "7.1.2",
+ "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz",
+ "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==",
+ "dev": true,
+ "license": "ISC",
+ "engines": {
+ "node": ">=16 || 14 >=14.17"
+ }
+ },
+ "node_modules/ms": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
+ "license": "MIT"
+ },
+ "node_modules/mz": {
+ "version": "2.7.0",
+ "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
+ "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "any-promise": "^1.0.0",
+ "object-assign": "^4.0.1",
+ "thenify-all": "^1.0.0"
+ }
+ },
+ "node_modules/nanoid": {
+ "version": "3.3.11",
+ "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz",
+ "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "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,
+ "license": "MIT"
+ },
+ "node_modules/node-domexception": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz",
+ "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==",
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/jimmywarting"
+ },
+ {
+ "type": "github",
+ "url": "https://paypal.me/jimmywarting"
+ }
+ ],
+ "license": "MIT",
+ "engines": {
+ "node": ">=10.5.0"
+ }
+ },
+ "node_modules/node-fetch": {
+ "version": "2.7.0",
+ "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz",
+ "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==",
+ "license": "MIT",
+ "dependencies": {
+ "whatwg-url": "^5.0.0"
+ },
+ "engines": {
+ "node": "4.x || >=6.0.0"
+ },
+ "peerDependencies": {
+ "encoding": "^0.1.0"
+ },
+ "peerDependenciesMeta": {
+ "encoding": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/node-releases": {
+ "version": "2.0.19",
+ "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.19.tgz",
+ "integrity": "sha512-xxOWJsBKtzAq7DY0J+DTzuz58K8e7sJbdgwkbMWQe8UYB6ekmsQ45q0M/tJDsGaZmbC+l7n57UV8Hl5tHxO9uw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/normalize-path": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
+ "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/normalize-range": {
+ "version": "0.1.2",
+ "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz",
+ "integrity": "sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/object-assign": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/object-hash": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-3.0.0.tgz",
+ "integrity": "sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/openai": {
+ "version": "4.89.0",
+ "resolved": "https://registry.npmjs.org/openai/-/openai-4.89.0.tgz",
+ "integrity": "sha512-XNI0q2l8/Os6jmojxaID5EhyQjxZgzR2gWcpEjYWK5hGKwE7AcifxEY7UNwFDDHJQXqeiosQ0CJwQN+rvnwdjA==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@types/node": "^18.11.18",
+ "@types/node-fetch": "^2.6.4",
+ "abort-controller": "^3.0.0",
+ "agentkeepalive": "^4.2.1",
+ "form-data-encoder": "1.7.2",
+ "formdata-node": "^4.3.2",
+ "node-fetch": "^2.6.7"
+ },
+ "bin": {
+ "openai": "bin/cli"
+ },
+ "peerDependencies": {
+ "ws": "^8.18.0",
+ "zod": "^3.23.8"
+ },
+ "peerDependenciesMeta": {
+ "ws": {
+ "optional": true
+ },
+ "zod": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/optionator": {
+ "version": "0.9.4",
+ "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz",
+ "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "deep-is": "^0.1.3",
+ "fast-levenshtein": "^2.0.6",
+ "levn": "^0.4.1",
+ "prelude-ls": "^1.2.1",
+ "type-check": "^0.4.0",
+ "word-wrap": "^1.2.5"
+ },
+ "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,
+ "license": "MIT",
+ "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,
+ "license": "MIT",
+ "dependencies": {
+ "p-limit": "^3.0.2"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/package-json-from-dist": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz",
+ "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==",
+ "dev": true,
+ "license": "BlueOak-1.0.0"
+ },
+ "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,
+ "license": "MIT",
+ "dependencies": {
+ "callsites": "^3.0.0"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/path-exists": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+ "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/path-key": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
+ "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/path-parse": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
+ "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/path-scurry": {
+ "version": "1.11.1",
+ "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz",
+ "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==",
+ "dev": true,
+ "license": "BlueOak-1.0.0",
+ "dependencies": {
+ "lru-cache": "^10.2.0",
+ "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0"
+ },
+ "engines": {
+ "node": ">=16 || 14 >=14.18"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
+ "node_modules/path-scurry/node_modules/lru-cache": {
+ "version": "10.4.3",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz",
+ "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/picocolors": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
+ "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/picomatch": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz",
+ "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8.6"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/jonschlinkert"
+ }
+ },
+ "node_modules/pify": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz",
+ "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/pirates": {
+ "version": "4.0.6",
+ "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.6.tgz",
+ "integrity": "sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/postcss": {
+ "version": "8.5.3",
+ "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.3.tgz",
+ "integrity": "sha512-dle9A3yYxlBSrt8Fu+IpjGT8SY8hN0mlaA6GY8t0P5PjIOZemULz/E2Bnm/2dcUOena75OTNkHI76uZBNUUq3A==",
+ "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"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "nanoid": "^3.3.8",
+ "picocolors": "^1.1.1",
+ "source-map-js": "^1.2.1"
+ },
+ "engines": {
+ "node": "^10 || ^12 || >=14"
+ }
+ },
+ "node_modules/postcss-import": {
+ "version": "15.1.0",
+ "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz",
+ "integrity": "sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "postcss-value-parser": "^4.0.0",
+ "read-cache": "^1.0.0",
+ "resolve": "^1.1.7"
+ },
+ "engines": {
+ "node": ">=14.0.0"
+ },
+ "peerDependencies": {
+ "postcss": "^8.0.0"
+ }
+ },
+ "node_modules/postcss-js": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/postcss-js/-/postcss-js-4.0.1.tgz",
+ "integrity": "sha512-dDLF8pEO191hJMtlHFPRa8xsizHaM82MLfNkUHdUtVEV3tgTp5oj+8qbEqYM57SLfc74KSbw//4SeJma2LRVIw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "camelcase-css": "^2.0.1"
+ },
+ "engines": {
+ "node": "^12 || ^14 || >= 16"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ "peerDependencies": {
+ "postcss": "^8.4.21"
+ }
+ },
+ "node_modules/postcss-load-config": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-4.0.2.tgz",
+ "integrity": "sha512-bSVhyJGL00wMVoPUzAVAnbEoWyqRxkjv64tUl427SKnPrENtq6hJwUojroMz2VB+Q1edmi4IfrAPpami5VVgMQ==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "lilconfig": "^3.0.0",
+ "yaml": "^2.3.4"
+ },
+ "engines": {
+ "node": ">= 14"
+ },
+ "peerDependencies": {
+ "postcss": ">=8.0.9",
+ "ts-node": ">=9.0.0"
+ },
+ "peerDependenciesMeta": {
+ "postcss": {
+ "optional": true
+ },
+ "ts-node": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/postcss-nested": {
+ "version": "6.2.0",
+ "resolved": "https://registry.npmjs.org/postcss-nested/-/postcss-nested-6.2.0.tgz",
+ "integrity": "sha512-HQbt28KulC5AJzG+cZtj9kvKB93CFCdLvog1WFLf1D+xmMvPGlBstkpTEZfK5+AN9hfJocyBFCNiqyS48bpgzQ==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "postcss-selector-parser": "^6.1.1"
+ },
+ "engines": {
+ "node": ">=12.0"
+ },
+ "peerDependencies": {
+ "postcss": "^8.2.14"
+ }
+ },
+ "node_modules/postcss-selector-parser": {
+ "version": "6.1.2",
+ "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz",
+ "integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "cssesc": "^3.0.0",
+ "util-deprecate": "^1.0.2"
+ },
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/postcss-value-parser": {
+ "version": "4.2.0",
+ "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz",
+ "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "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,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/proxy-from-env": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
+ "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==",
+ "license": "MIT"
+ },
+ "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,
+ "license": "MIT",
+ "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"
+ }
+ ],
+ "license": "MIT"
+ },
+ "node_modules/react": {
+ "version": "19.0.0",
+ "resolved": "https://registry.npmjs.org/react/-/react-19.0.0.tgz",
+ "integrity": "sha512-V8AVnmPIICiWpGfm6GLzCR/W5FXLchHop40W4nXBmdlEceh16rCN8O8LNWm5bh5XUX91fh7KpA+W0TgMKmgTpQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/react-dom": {
+ "version": "19.0.0",
+ "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.0.0.tgz",
+ "integrity": "sha512-4GV5sHFG0e/0AD4X+ySy6UJd3jVl1iNsNHdpad0qhABJ11twS3TTBnseqsKurKcsNqCEFeGL3uLpVChpIO3QfQ==",
+ "license": "MIT",
+ "dependencies": {
+ "scheduler": "^0.25.0"
+ },
+ "peerDependencies": {
+ "react": "^19.0.0"
+ }
+ },
+ "node_modules/react-redux": {
+ "version": "9.2.0",
+ "resolved": "https://registry.npmjs.org/react-redux/-/react-redux-9.2.0.tgz",
+ "integrity": "sha512-ROY9fvHhwOD9ySfrF0wmvu//bKCQ6AeZZq1nJNtbDC+kk5DuSuNX/n6YWYF/SYy7bSba4D4FSz8DJeKY/S/r+g==",
+ "license": "MIT",
+ "dependencies": {
+ "@types/use-sync-external-store": "^0.0.6",
+ "use-sync-external-store": "^1.4.0"
+ },
+ "peerDependencies": {
+ "@types/react": "^18.2.25 || ^19",
+ "react": "^18.0 || ^19",
+ "redux": "^5.0.0"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "redux": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/react-refresh": {
+ "version": "0.14.2",
+ "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.14.2.tgz",
+ "integrity": "sha512-jCvmsr+1IUSMUyzOkRcvnVbX3ZYC6g9TDrDbFuFmRDq7PD4yaGbLKNQL6k2jnArV8hjYxh7hVhAZB6s9HDGpZA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/react-router": {
+ "version": "7.4.0",
+ "resolved": "https://registry.npmjs.org/react-router/-/react-router-7.4.0.tgz",
+ "integrity": "sha512-Y2g5ObjkvX3VFeVt+0CIPuYd9PpgqCslG7ASSIdN73LwA1nNWzcMLaoMRJfP3prZFI92svxFwbn7XkLJ+UPQ6A==",
+ "license": "MIT",
+ "dependencies": {
+ "@types/cookie": "^0.6.0",
+ "cookie": "^1.0.1",
+ "set-cookie-parser": "^2.6.0",
+ "turbo-stream": "2.4.0"
+ },
+ "engines": {
+ "node": ">=20.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=18",
+ "react-dom": ">=18"
+ },
+ "peerDependenciesMeta": {
+ "react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/read-cache": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
+ "integrity": "sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "pify": "^2.3.0"
+ }
+ },
+ "node_modules/readdirp": {
+ "version": "3.6.0",
+ "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz",
+ "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "picomatch": "^2.2.1"
+ },
+ "engines": {
+ "node": ">=8.10.0"
+ }
+ },
+ "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==",
+ "license": "MIT"
+ },
+ "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==",
+ "license": "MIT",
+ "peerDependencies": {
+ "redux": "^5.0.0"
+ }
+ },
+ "node_modules/reselect": {
+ "version": "5.1.1",
+ "resolved": "https://registry.npmjs.org/reselect/-/reselect-5.1.1.tgz",
+ "integrity": "sha512-K/BG6eIky/SBpzfHZv/dd+9JBFiS4SWV7FIujVyJRux6e45+73RaUHXLmIR1f7WOMaQ0U1km6qwklRQxpJJY0w==",
+ "license": "MIT"
+ },
+ "node_modules/resolve": {
+ "version": "1.22.10",
+ "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz",
+ "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "is-core-module": "^2.16.0",
+ "path-parse": "^1.0.7",
+ "supports-preserve-symlinks-flag": "^1.0.0"
+ },
+ "bin": {
+ "resolve": "bin/resolve"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/resolve-from": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz",
+ "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/reusify": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz",
+ "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "iojs": ">=1.0.0",
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/rollup": {
+ "version": "4.37.0",
+ "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.37.0.tgz",
+ "integrity": "sha512-iAtQy/L4QFU+rTJ1YUjXqJOJzuwEghqWzCEYD2FEghT7Gsy1VdABntrO4CLopA5IkflTyqNiLNwPcOJ3S7UKLg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/estree": "1.0.6"
+ },
+ "bin": {
+ "rollup": "dist/bin/rollup"
+ },
+ "engines": {
+ "node": ">=18.0.0",
+ "npm": ">=8.0.0"
+ },
+ "optionalDependencies": {
+ "@rollup/rollup-android-arm-eabi": "4.37.0",
+ "@rollup/rollup-android-arm64": "4.37.0",
+ "@rollup/rollup-darwin-arm64": "4.37.0",
+ "@rollup/rollup-darwin-x64": "4.37.0",
+ "@rollup/rollup-freebsd-arm64": "4.37.0",
+ "@rollup/rollup-freebsd-x64": "4.37.0",
+ "@rollup/rollup-linux-arm-gnueabihf": "4.37.0",
+ "@rollup/rollup-linux-arm-musleabihf": "4.37.0",
+ "@rollup/rollup-linux-arm64-gnu": "4.37.0",
+ "@rollup/rollup-linux-arm64-musl": "4.37.0",
+ "@rollup/rollup-linux-loongarch64-gnu": "4.37.0",
+ "@rollup/rollup-linux-powerpc64le-gnu": "4.37.0",
+ "@rollup/rollup-linux-riscv64-gnu": "4.37.0",
+ "@rollup/rollup-linux-riscv64-musl": "4.37.0",
+ "@rollup/rollup-linux-s390x-gnu": "4.37.0",
+ "@rollup/rollup-linux-x64-gnu": "4.37.0",
+ "@rollup/rollup-linux-x64-musl": "4.37.0",
+ "@rollup/rollup-win32-arm64-msvc": "4.37.0",
+ "@rollup/rollup-win32-ia32-msvc": "4.37.0",
+ "@rollup/rollup-win32-x64-msvc": "4.37.0",
+ "fsevents": "~2.3.2"
+ }
+ },
+ "node_modules/rollup/node_modules/@types/estree": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.6.tgz",
+ "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "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"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "queue-microtask": "^1.2.2"
+ }
+ },
+ "node_modules/scheduler": {
+ "version": "0.25.0",
+ "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.25.0.tgz",
+ "integrity": "sha512-xFVuu11jh+xcO7JOAGJNOXld8/TcEHK/4CituBUeUb5hqxJLj9YuemAEuvm9gQ/+pgXYfbQuqAkiYu+u7YEsNA==",
+ "license": "MIT"
+ },
+ "node_modules/semver": {
+ "version": "6.3.1",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
+ "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==",
+ "dev": true,
+ "license": "ISC",
+ "bin": {
+ "semver": "bin/semver.js"
+ }
+ },
+ "node_modules/set-cookie-parser": {
+ "version": "2.7.1",
+ "resolved": "https://registry.npmjs.org/set-cookie-parser/-/set-cookie-parser-2.7.1.tgz",
+ "integrity": "sha512-IOc8uWeOZgnb3ptbCURJWNjWUPcO3ZnTTdzsurqERrP6nPyv+paC55vJM0LpOlT2ne+Ix+9+CRG1MNLlyZ4GjQ==",
+ "license": "MIT"
+ },
+ "node_modules/shebang-command": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
+ "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "shebang-regex": "^3.0.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/shebang-regex": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
+ "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/signal-exit": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz",
+ "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==",
+ "dev": true,
+ "license": "ISC",
+ "engines": {
+ "node": ">=14"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
+ "node_modules/source-map-js": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
+ "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
+ "dev": true,
+ "license": "BSD-3-Clause",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/string-width": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz",
+ "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "eastasianwidth": "^0.2.0",
+ "emoji-regex": "^9.2.2",
+ "strip-ansi": "^7.0.1"
+ },
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/string-width-cjs": {
+ "name": "string-width",
+ "version": "4.2.3",
+ "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
+ "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "emoji-regex": "^8.0.0",
+ "is-fullwidth-code-point": "^3.0.0",
+ "strip-ansi": "^6.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/string-width-cjs/node_modules/ansi-regex": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
+ "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/string-width-cjs/node_modules/emoji-regex": {
+ "version": "8.0.0",
+ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
+ "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/string-width-cjs/node_modules/strip-ansi": {
+ "version": "6.0.1",
+ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
+ "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-regex": "^5.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/strip-ansi": {
+ "version": "7.1.0",
+ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz",
+ "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-regex": "^6.0.1"
+ },
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/strip-ansi?sponsor=1"
+ }
+ },
+ "node_modules/strip-ansi-cjs": {
+ "name": "strip-ansi",
+ "version": "6.0.1",
+ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
+ "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-regex": "^5.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/strip-ansi-cjs/node_modules/ansi-regex": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
+ "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/strip-json-comments": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz",
+ "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/sucrase": {
+ "version": "3.35.0",
+ "resolved": "https://registry.npmjs.org/sucrase/-/sucrase-3.35.0.tgz",
+ "integrity": "sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@jridgewell/gen-mapping": "^0.3.2",
+ "commander": "^4.0.0",
+ "glob": "^10.3.10",
+ "lines-and-columns": "^1.1.6",
+ "mz": "^2.7.0",
+ "pirates": "^4.0.1",
+ "ts-interface-checker": "^0.1.9"
+ },
+ "bin": {
+ "sucrase": "bin/sucrase",
+ "sucrase-node": "bin/sucrase-node"
+ },
+ "engines": {
+ "node": ">=16 || 14 >=14.17"
+ }
+ },
+ "node_modules/supports-color": {
+ "version": "7.2.0",
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
+ "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "has-flag": "^4.0.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/supports-preserve-symlinks-flag": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz",
+ "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/sweetalert2": {
+ "version": "11.17.2",
+ "resolved": "https://registry.npmjs.org/sweetalert2/-/sweetalert2-11.17.2.tgz",
+ "integrity": "sha512-HKxDr1IyV3Lxr3W6sb61qm/p2epFIEdr5EKwteRFHnIg6f8nHFl2kX++DBVz16Mac+fFiU3hMpjq1L6yE2Ge5w==",
+ "license": "MIT",
+ "funding": {
+ "type": "individual",
+ "url": "https://github.com/sponsors/limonte"
+ }
+ },
+ "node_modules/tailwindcss": {
+ "version": "3.4.17",
+ "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.17.tgz",
+ "integrity": "sha512-w33E2aCvSDP0tW9RZuNXadXlkHXqFzSkQew/aIa2i/Sj8fThxwovwlXHSPXTbAHwEIhBFXAedUhP2tueAKP8Og==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@alloc/quick-lru": "^5.2.0",
+ "arg": "^5.0.2",
+ "chokidar": "^3.6.0",
+ "didyoumean": "^1.2.2",
+ "dlv": "^1.1.3",
+ "fast-glob": "^3.3.2",
+ "glob-parent": "^6.0.2",
+ "is-glob": "^4.0.3",
+ "jiti": "^1.21.6",
+ "lilconfig": "^3.1.3",
+ "micromatch": "^4.0.8",
+ "normalize-path": "^3.0.0",
+ "object-hash": "^3.0.0",
+ "picocolors": "^1.1.1",
+ "postcss": "^8.4.47",
+ "postcss-import": "^15.1.0",
+ "postcss-js": "^4.0.1",
+ "postcss-load-config": "^4.0.2",
+ "postcss-nested": "^6.2.0",
+ "postcss-selector-parser": "^6.1.2",
+ "resolve": "^1.22.8",
+ "sucrase": "^3.35.0"
+ },
+ "bin": {
+ "tailwind": "lib/cli.js",
+ "tailwindcss": "lib/cli.js"
+ },
+ "engines": {
+ "node": ">=14.0.0"
+ }
+ },
+ "node_modules/thenify": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz",
+ "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "any-promise": "^1.0.0"
+ }
+ },
+ "node_modules/thenify-all": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz",
+ "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "thenify": ">= 3.1.0 < 4"
+ },
+ "engines": {
+ "node": ">=0.8"
+ }
+ },
+ "node_modules/to-regex-range": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
+ "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "is-number": "^7.0.0"
+ },
+ "engines": {
+ "node": ">=8.0"
+ }
+ },
+ "node_modules/tr46": {
+ "version": "0.0.3",
+ "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz",
+ "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==",
+ "license": "MIT"
+ },
+ "node_modules/ts-interface-checker": {
+ "version": "0.1.13",
+ "resolved": "https://registry.npmjs.org/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz",
+ "integrity": "sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==",
+ "dev": true,
+ "license": "Apache-2.0"
+ },
+ "node_modules/turbo-stream": {
+ "version": "2.4.0",
+ "resolved": "https://registry.npmjs.org/turbo-stream/-/turbo-stream-2.4.0.tgz",
+ "integrity": "sha512-FHncC10WpBd2eOmGwpmQsWLDoK4cqsA/UT/GqNoaKOQnT8uzhtCbg3EoUDMvqpOSAI0S26mr0rkjzbOO6S3v1g==",
+ "license": "ISC"
+ },
+ "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,
+ "license": "MIT",
+ "dependencies": {
+ "prelude-ls": "^1.2.1"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/undici-types": {
+ "version": "5.26.5",
+ "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz",
+ "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==",
+ "license": "MIT"
+ },
+ "node_modules/update-browserslist-db": {
+ "version": "1.1.3",
+ "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.3.tgz",
+ "integrity": "sha512-UxhIZQ+QInVdunkDAaiazvvT/+fXL5Osr0JZlJulepYu6Jd7qJtDZjlur0emRlT71EN3ScPoE7gvsuIKKNavKw==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "escalade": "^3.2.0",
+ "picocolors": "^1.1.1"
+ },
+ "bin": {
+ "update-browserslist-db": "cli.js"
+ },
+ "peerDependencies": {
+ "browserslist": ">= 4.21.0"
+ }
+ },
+ "node_modules/uri-js": {
+ "version": "4.4.1",
+ "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
+ "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "punycode": "^2.1.0"
+ }
+ },
+ "node_modules/use-sync-external-store": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.4.0.tgz",
+ "integrity": "sha512-9WXSPC5fMv61vaupRkCKCxsPxBocVnwakBEkMIHHpkTTg6icbJtg6jzgtLDm4bl3cSHAca52rYWih0k4K3PfHw==",
+ "license": "MIT",
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/util-deprecate": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/vite": {
+ "version": "6.2.3",
+ "resolved": "https://registry.npmjs.org/vite/-/vite-6.2.3.tgz",
+ "integrity": "sha512-IzwM54g4y9JA/xAeBPNaDXiBF8Jsgl3VBQ2YQ/wOY6fyW3xMdSoltIV3Bo59DErdqdE6RxUfv8W69DvUorE4Eg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "esbuild": "^0.25.0",
+ "postcss": "^8.5.3",
+ "rollup": "^4.30.1"
+ },
+ "bin": {
+ "vite": "bin/vite.js"
+ },
+ "engines": {
+ "node": "^18.0.0 || ^20.0.0 || >=22.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 || >=22.0.0",
+ "jiti": ">=1.21.0",
+ "less": "*",
+ "lightningcss": "^1.21.0",
+ "sass": "*",
+ "sass-embedded": "*",
+ "stylus": "*",
+ "sugarss": "*",
+ "terser": "^5.16.0",
+ "tsx": "^4.8.1",
+ "yaml": "^2.4.2"
+ },
+ "peerDependenciesMeta": {
+ "@types/node": {
+ "optional": true
+ },
+ "jiti": {
+ "optional": true
+ },
+ "less": {
+ "optional": true
+ },
+ "lightningcss": {
+ "optional": true
+ },
+ "sass": {
+ "optional": true
+ },
+ "sass-embedded": {
+ "optional": true
+ },
+ "stylus": {
+ "optional": true
+ },
+ "sugarss": {
+ "optional": true
+ },
+ "terser": {
+ "optional": true
+ },
+ "tsx": {
+ "optional": true
+ },
+ "yaml": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/web-streams-polyfill": {
+ "version": "4.0.0-beta.3",
+ "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-4.0.0-beta.3.tgz",
+ "integrity": "sha512-QW95TCTaHmsYfHDybGMwO5IJIM93I/6vTRk+daHTWFPhwh+C8Cg7j7XyKrwrj8Ib6vYXe0ocYNrmzY4xAAN6ug==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 14"
+ }
+ },
+ "node_modules/webidl-conversions": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz",
+ "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==",
+ "license": "BSD-2-Clause"
+ },
+ "node_modules/whatwg-url": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz",
+ "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==",
+ "license": "MIT",
+ "dependencies": {
+ "tr46": "~0.0.3",
+ "webidl-conversions": "^3.0.0"
+ }
+ },
+ "node_modules/which": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
+ "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
+ "dev": true,
+ "license": "ISC",
+ "dependencies": {
+ "isexe": "^2.0.0"
+ },
+ "bin": {
+ "node-which": "bin/node-which"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/word-wrap": {
+ "version": "1.2.5",
+ "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz",
+ "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/wrap-ansi": {
+ "version": "8.1.0",
+ "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz",
+ "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-styles": "^6.1.0",
+ "string-width": "^5.0.1",
+ "strip-ansi": "^7.0.1"
+ },
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/wrap-ansi?sponsor=1"
+ }
+ },
+ "node_modules/wrap-ansi-cjs": {
+ "name": "wrap-ansi",
+ "version": "7.0.0",
+ "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz",
+ "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-styles": "^4.0.0",
+ "string-width": "^4.1.0",
+ "strip-ansi": "^6.0.0"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/wrap-ansi?sponsor=1"
+ }
+ },
+ "node_modules/wrap-ansi-cjs/node_modules/ansi-regex": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
+ "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": {
+ "version": "8.0.0",
+ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz",
+ "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/wrap-ansi-cjs/node_modules/string-width": {
+ "version": "4.2.3",
+ "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz",
+ "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "emoji-regex": "^8.0.0",
+ "is-fullwidth-code-point": "^3.0.0",
+ "strip-ansi": "^6.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": {
+ "version": "6.0.1",
+ "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz",
+ "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "ansi-regex": "^5.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/wrap-ansi/node_modules/ansi-styles": {
+ "version": "6.2.1",
+ "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz",
+ "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/chalk/ansi-styles?sponsor=1"
+ }
+ },
+ "node_modules/yallist": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+ "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+ "dev": true,
+ "license": "ISC"
+ },
+ "node_modules/yaml": {
+ "version": "2.7.0",
+ "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.7.0.tgz",
+ "integrity": "sha512-+hSoy/QHluxmC9kCIJyL/uyFmLmc+e5CFR5Wa+bpIhIj85LVb9ZH2nVnqrHoSvKogwODv0ClqZkmiSSaIH5LTA==",
+ "dev": true,
+ "license": "ISC",
+ "bin": {
+ "yaml": "bin.mjs"
+ },
+ "engines": {
+ "node": ">= 14"
+ }
+ },
+ "node_modules/yocto-queue": {
+ "version": "0.1.0",
+ "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
+ "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ }
+ }
+}
diff --git a/client/package.json b/client/package.json
new file mode 100644
index 0000000..d1d05c1
--- /dev/null
+++ b/client/package.json
@@ -0,0 +1,37 @@
+{
+ "name": "client",
+ "private": true,
+ "version": "0.0.0",
+ "type": "module",
+ "scripts": {
+ "dev": "vite",
+ "build": "vite build",
+ "lint": "eslint .",
+ "preview": "vite preview"
+ },
+ "dependencies": {
+ "@reduxjs/toolkit": "^2.6.1",
+ "axios": "^1.8.4",
+ "dotenv": "^16.4.7",
+ "openai": "^4.89.0",
+ "react": "^19.0.0",
+ "react-dom": "^19.0.0",
+ "react-redux": "^9.2.0",
+ "react-router": "^7.4.0",
+ "sweetalert2": "^11.17.2"
+ },
+ "devDependencies": {
+ "@eslint/js": "^9.21.0",
+ "@types/react": "^19.0.10",
+ "@types/react-dom": "^19.0.4",
+ "@vitejs/plugin-react": "^4.3.4",
+ "autoprefixer": "^10.4.21",
+ "eslint": "^9.21.0",
+ "eslint-plugin-react-hooks": "^5.1.0",
+ "eslint-plugin-react-refresh": "^0.4.19",
+ "globals": "^15.15.0",
+ "postcss": "^8.5.3",
+ "tailwindcss": "^3.4.17",
+ "vite": "^6.2.0"
+ }
+}
diff --git a/client/postcss.config.js b/client/postcss.config.js
new file mode 100644
index 0000000..2e7af2b
--- /dev/null
+++ b/client/postcss.config.js
@@ -0,0 +1,6 @@
+export default {
+ plugins: {
+ tailwindcss: {},
+ autoprefixer: {},
+ },
+}
diff --git a/client/public/vite.svg b/client/public/vite.svg
new file mode 100644
index 0000000..e7b8dfb
--- /dev/null
+++ b/client/public/vite.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/client/redux.js b/client/redux.js
new file mode 100644
index 0000000..924fa30
--- /dev/null
+++ b/client/redux.js
@@ -0,0 +1,25 @@
+const { configureStore } = require("@reduxjs/toolkit");
+
+
+const counterSlice = createSlice({
+ name: 'counter',
+
+ // tempat untuk menyimpan state (diisi dengan nilai awal)
+ initialState: {
+ value: 0 // hitungan count nya
+ },
+ reducers: {
+ // kumpulan function yang bisa merubah state
+ incremented: (state, action) => {
+ state.value += 1
+ },
+ decremented: (state, action) => {
+ state.value -= 1
+ }
+ }
+})
+
+const store = configureStore({
+ // reducer -> pure function untuk mutate state
+ reducer: counterSlice.reducer
+})
\ No newline at end of file
diff --git a/client/src/App.css b/client/src/App.css
new file mode 100644
index 0000000..e69de29
diff --git a/client/src/App.jsx b/client/src/App.jsx
new file mode 100644
index 0000000..a252af2
--- /dev/null
+++ b/client/src/App.jsx
@@ -0,0 +1,52 @@
+import { useEffect, useState } from 'react'
+import './App.css'
+import { BrowserRouter, Navigate, Outlet, Route, Routes, useNavigate } from 'react-router'
+import LoginPage from './pages/LoginPage'
+import LandingPage from './pages/LandingPage'
+import RegisterPage from './pages/RegisterPage'
+import CoursePage from './pages/CoursesPage'
+import Navbar from './components/navbar'
+import DetailCourse from './pages/DetailCourse'
+import KanjiPage from './pages/KanjiPage'
+
+
+function IndexLayout() {
+ const navigate = useNavigate()
+ useEffect(() => {
+ if (!localStorage.getItem("access_token")) {
+ navigate('/login')
+ }
+ }, [])
+
+ return (
+
+
+
+
+ )
+}
+
+function App() {
+
+ return (
+
+
+ } />
+ } />
+ } />
+
+ }>
+
+ {/* Redirect root path '/' to '/courses' */}
+ } />
+
+ } />
+ } />
+ } />
+
+
+
+ )
+}
+
+export default App
diff --git a/client/src/assets/dot-circle.svg b/client/src/assets/dot-circle.svg
new file mode 100644
index 0000000..79194d8
--- /dev/null
+++ b/client/src/assets/dot-circle.svg
@@ -0,0 +1,2 @@
+
+
diff --git a/client/src/assets/hero.svg b/client/src/assets/hero.svg
new file mode 100644
index 0000000..45f948b
--- /dev/null
+++ b/client/src/assets/hero.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/client/src/assets/react.svg b/client/src/assets/react.svg
new file mode 100644
index 0000000..6c87de9
--- /dev/null
+++ b/client/src/assets/react.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/client/src/components/CourseCard.jsx b/client/src/components/CourseCard.jsx
new file mode 100644
index 0000000..4ff092a
--- /dev/null
+++ b/client/src/components/CourseCard.jsx
@@ -0,0 +1,26 @@
+import { NavLink } from "react-router";
+
+function CourseCard(props) {
+ const { title, sensei, desc, imageUrl, id } = props
+ return (
+
+

+
+
{title}
+
+
{sensei}
+
+
{desc}
+
+ Lihat Kursus
+
+
+
+
+
+ )
+}
+
+export default CourseCard;
\ No newline at end of file
diff --git a/client/src/components/Navbar.jsx b/client/src/components/Navbar.jsx
new file mode 100644
index 0000000..210cb5f
--- /dev/null
+++ b/client/src/components/Navbar.jsx
@@ -0,0 +1,108 @@
+import { NavLink, useNavigate } from "react-router";
+import Swal from "sweetalert2";
+
+function Navbar() {
+ const navigate = useNavigate()
+ return (
+
+
+
+ )
+}
+
+export default Navbar;
\ No newline at end of file
diff --git a/client/src/components/ReduxTest.jsx b/client/src/components/ReduxTest.jsx
new file mode 100644
index 0000000..b41e1ad
--- /dev/null
+++ b/client/src/components/ReduxTest.jsx
@@ -0,0 +1,8 @@
+import { useSelector } from "react-redux"
+
+export default function ReduxTest() {
+ const counter = useSelector((state) => state.counter)
+ return (
+ {counter.value}
+ )
+}
diff --git a/client/src/helpers/axiosInstance.js b/client/src/helpers/axiosInstance.js
new file mode 100644
index 0000000..5d4875c
--- /dev/null
+++ b/client/src/helpers/axiosInstance.js
@@ -0,0 +1,7 @@
+import axios from "axios"
+
+const axiosInstance = axios.create({
+ baseURL: 'https://api.garudanihon.my.id'
+})
+
+export default axiosInstance
\ No newline at end of file
diff --git a/client/src/index.css b/client/src/index.css
new file mode 100644
index 0000000..bd6213e
--- /dev/null
+++ b/client/src/index.css
@@ -0,0 +1,3 @@
+@tailwind base;
+@tailwind components;
+@tailwind utilities;
\ No newline at end of file
diff --git a/client/src/main.jsx b/client/src/main.jsx
new file mode 100644
index 0000000..1223bda
--- /dev/null
+++ b/client/src/main.jsx
@@ -0,0 +1,15 @@
+import { StrictMode } from 'react'
+import { createRoot } from 'react-dom/client'
+import './index.css'
+import App from './App.jsx'
+import { Provider } from 'react-redux'
+import { store } from './store/index.js'
+
+createRoot(document.getElementById('root')).render(
+
+ {/* bungkus sama redux store */}
+
+
+
+ ,
+)
diff --git a/client/src/pages/CoursesPage.jsx b/client/src/pages/CoursesPage.jsx
new file mode 100644
index 0000000..0f0ab76
--- /dev/null
+++ b/client/src/pages/CoursesPage.jsx
@@ -0,0 +1,30 @@
+import CourseCard from "../components/CourseCard"
+import { useDispatch, useSelector } from "react-redux"
+import { useEffect } from "react"
+import { fetchCourses } from "../store/coursesSlice"
+
+function CoursesPage() {
+ const dispatch = useDispatch()
+ const courseReduxState = useSelector(function (state) {
+ return state.courses
+ })
+ const courses = courseReduxState.data
+
+ useEffect(() => {
+ dispatch(fetchCourses())
+ }, [])
+
+ return (
+
+
+ {courses.map((item, i) => {
+ return
+ })}
+
+
+
+
+ )
+}
+
+export default CoursesPage
\ No newline at end of file
diff --git a/client/src/pages/DetailCourse.jsx b/client/src/pages/DetailCourse.jsx
new file mode 100644
index 0000000..f652337
--- /dev/null
+++ b/client/src/pages/DetailCourse.jsx
@@ -0,0 +1,144 @@
+import { useDispatch, useSelector } from "react-redux";
+import { useParams } from "react-router";
+import { useEffect } from "react";
+import { show } from "../store/coursesSlice";
+import { index } from "../store/materialSlice";
+import { index as indexUserCourse, toggleUserCourse, toogleFavoriteUpdate } from "../store/userCourseSlice";
+import dotcircle from "../assets/dot-circle.svg"
+import Swal from "sweetalert2";
+
+function DetailCourse() {
+ // redux store -> reducer -> state
+ const dispatch = useDispatch()
+ const courseReduxState = useSelector(function (state) {
+ return state.courses
+ })
+
+ const userId = localStorage.getItem("user_id")
+ const materialState = useSelector(function (state) {
+ return state.materialReducer
+ })
+
+ const userCourseState = useSelector(function (state) {
+ return state.userCourseReducer
+ })
+
+ //console.log(JSON.stringify(userCourseState.isEnrolled))
+
+ const { courseId } = useParams()
+
+ const course = courseReduxState.course
+ const materials = materialState.data
+
+ useEffect(() => {
+ dispatch(show(courseId))
+ dispatch(index(courseId))
+ dispatch(indexUserCourse({ courseId, userId }))
+ }, [])
+
+ const handleToggleEnrollment = () => {
+ dispatch(toggleUserCourse({ courseId, userId }))
+ }
+
+ const handleToogleFavorite = () => {
+ dispatch(toogleFavoriteUpdate({ courseId, userId }))
+ }
+
+ const handleVideoClick = (e, videoUrl) => {
+ if (!userCourseState.isEnrolled) {
+ e.preventDefault()
+ Swal.fire("Gagal", "Anda harus mengambil kursus terlebih dahulu", "error")
+ } else {
+ window.open(videoUrl, "_blank")
+ }
+ }
+
+ return (
+
+
+
+
+
{course.title}
+
{course.desc}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+

+
+
+
+
+
+
Materi
+
+ {materials.map((item, i) => {
+ return
+
+
+

+
+
+
+
+ })}
+
+
+
+
+
+ )
+}
+
+export default DetailCourse
\ No newline at end of file
diff --git a/client/src/pages/KanjiPage.jsx b/client/src/pages/KanjiPage.jsx
new file mode 100644
index 0000000..4574839
--- /dev/null
+++ b/client/src/pages/KanjiPage.jsx
@@ -0,0 +1,170 @@
+import { useEffect, useState } from "react"
+import { useDispatch, useSelector } from "react-redux"
+import { index } from "../store/kanjiSlice"
+import axiosInstance from "../helpers/axiosInstance"
+import Swal from "sweetalert2"
+
+function KanjiPage() {
+ const [romaji, setRomaji] = useState("")
+ const [translateTextInput, setTranslateTextInput] = useState("")
+
+ const [transliteration, setTransliteration] = useState("")
+ const [translation, setTranslation] = useState("")
+
+ const [transliterator, setTransliterator] = useState("romaji-kanji")
+ const [translatorMode, setTranslatorMode] = useState("indonesia-japan")
+
+ const onSubmitTransliteration = async () => {
+ try {
+
+ if (transliterator === "romaji-kanji") {
+ const result = await axiosInstance({
+ method: "POST",
+ url: "/romaji-transliterator",
+ data: {
+ romaji: romaji
+ },
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ }
+ })
+ const responseBody = result.data
+ console.log(responseBody)
+ setTransliteration(responseBody.transliteration)
+ } else {
+ const result = await axiosInstance({
+ method: "POST",
+ url: "/kanji-transliterator",
+ data: {
+ kanji: romaji
+ },
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ }
+ })
+ const responseBody = result.data
+ setTransliteration(responseBody.transliteration)
+ }
+ } catch (error) {
+ Swal.fire("Error", error.message, 'error')
+ }
+ }
+
+ const onSubmitTranslation = async () => {
+ try {
+ if (translatorMode === "indonesia-japan") {
+ const result = await axiosInstance({
+ method: "POST",
+ url: "/translate-to-japanese",
+ data: {
+ text: translateTextInput
+ },
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ }
+ })
+ const responseBody = result.data
+ setTranslation(responseBody.translation)
+ } else {
+ const result = await axiosInstance({
+ method: "POST",
+ url: "/translate-to-indonesia",
+ data: {
+ text: translateTextInput
+ },
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ }
+ })
+ const responseBody = result.data
+ console.log(responseBody)
+ setTranslation(responseBody.translation)
+ }
+ } catch (error) {
+ Swal.fire("Error", error.message, 'error')
+ }
+ }
+
+ const dispatch = useDispatch()
+ const kanjiState = useSelector(function (state) {
+ return state.kanjiReducer
+ })
+
+ const kanjis = kanjiState.data
+ console.log(kanjis)
+
+ useEffect(() => {
+ dispatch(index())
+ }, [])
+
+ return (
+
+
+
+
+
+
{(transliterator === "romaji-kanji") ? "Romaji to Kanji" : "Kanji to Romaji"}
+
+
+
+ { setRomaji(e.target.value) }} className="px-2 py-1 border-2 rounded-md" placeholder="Masukkan romaji.." />
+
+
+
+
+
+
+
+
+
+
+
+
+
{(translatorMode === "indonesia-japan") ? "Terjemahan (IND - JPN)" : "Terjemahan (JPN - IND)"}
+
+
+
+ {
+ setTranslateTextInput(e.target.value)
+ }} className="px-2 py-1 border-2 rounded-md" placeholder="Masukkan teks.." />
+
+
+
+
+
+
+
+
+
+
+ {kanjis.map((item) => {
+ return
+ {item}
+
+ })}
+
+
+
+
+ )
+}
+
+export default KanjiPage
\ No newline at end of file
diff --git a/client/src/pages/LandingPage.jsx b/client/src/pages/LandingPage.jsx
new file mode 100644
index 0000000..4ffa62d
--- /dev/null
+++ b/client/src/pages/LandingPage.jsx
@@ -0,0 +1,36 @@
+import hero from "../assets/hero.svg";
+import Navbar from "../components/navbar";
+
+function LandingPage() {
+ return (
+
+
+
+
+
+
Platform Belajar Bahasa Jepang Terbaik
+
Garuda nihon telah membimbing lebih dari 10.000 siswa untuk lulus ujian JLPT dan JFT. Sekarang adalah giliranmu untuk bergabung.
+
+
+
+
+
+
+

+
+
+
+
+
+ )
+}
+
+export default LandingPage;
\ No newline at end of file
diff --git a/client/src/pages/LoginPage.jsx b/client/src/pages/LoginPage.jsx
new file mode 100644
index 0000000..5eb32ac
--- /dev/null
+++ b/client/src/pages/LoginPage.jsx
@@ -0,0 +1,214 @@
+import { useEffect, useState } from "react";
+import { useDispatch, useSelector } from "react-redux";
+import axiosInstance from "../helpers/axiosInstance";
+import { useNavigate, NavLink } from "react-router"
+import Swal from "sweetalert2"
+
+function LoginPage() {
+ const navigate = useNavigate()
+
+ async function handleCredentialResponse(response) {
+ console.log("Encoded JWT ID token: " + response.credential);
+ try {
+ const result = await axiosInstance({
+ method: "POST",
+ url: "/google-login",
+ data: {
+ googleToken: response.credential
+ }
+ })
+ const responseBody = result.data
+
+ // save tokennya ke localStorage
+ localStorage.setItem("access_token", responseBody.access_token)
+ localStorage.setItem("user_id", responseBody.data.id)
+
+ // pindah halaman
+ navigate('/')
+ } catch (error) {
+ if (error.response && error.response.data) {
+ Swal.fire({
+ icon: 'error',
+ title: 'Oops...',
+ text: error.response.data.message,
+ })
+ } else {
+ Swal.fire({
+ icon: 'error',
+ title: 'Oops...',
+ text: 'Terjadi kesalahan. Silakan coba lagi.',
+ })
+ }
+ }
+ }
+
+ const [email, setEmail] = useState('')
+ const [password, setPassword] = useState('')
+
+ const onClick = async () => {
+ try {
+ const result = await axiosInstance({
+ method: "POST",
+ url: "/login",
+ data: {
+ email: email,
+ password: password
+ }
+ })
+ const response = result.data
+
+ // save tokennya ke localStorage
+ localStorage.setItem("access_token", response.access_token)
+ localStorage.setItem("user_id", response.data.id)
+
+ Swal.fire({
+ icon: 'success',
+ title: 'Success',
+ text: 'Login Berhasil'
+ })
+
+ // pindah halaman
+ navigate('/')
+
+ } catch (error) {
+ if (error.response && error.response.data) {
+ Swal.fire({
+ icon: 'error',
+ title: 'Oops...',
+ text: error.response.data.message,
+ })
+ } else {
+ Swal.fire({
+ icon: 'error',
+ title: 'Oops...',
+ text: 'Terjadi kesalahan. Silakan coba lagi.',
+ })
+ }
+ }
+ }
+
+ useEffect(() => {
+ if (localStorage.getItem("access_token")) {
+ navigate('/')
+ }
+
+ google.accounts.id.initialize({
+ client_id: process.env.GOOGLE_CLIENT_ID,
+ callback: handleCredentialResponse
+ });
+ google.accounts.id.renderButton(
+ document.getElementById("btnGoogleLogin"),
+ { theme: "outline", size: "large" } // customization attributes
+ );
+ }, [])
+
+ const dispatch = useDispatch() // <-- untuk invoke function pengubah
+
+ const counter = useSelector((state) => state.counter)
+ return (
+ <>
+
+
+
+
+
+
+
+
+ LOGIN
+
+
+
+
+
+
+
+
+
+
+ Or continue with
+
+
+
+ {/* login with google */}
+
+ {/*
*/}
+
Loading...
+
+
+
+
+
+ Belum memiliki akun?
+
+ {" "}Daftar Sekarang
+
+
+
+
+
+
+ >
+ );
+}
+
+export default LoginPage;
\ No newline at end of file
diff --git a/client/src/pages/RegisterPage.jsx b/client/src/pages/RegisterPage.jsx
new file mode 100644
index 0000000..c9885fb
--- /dev/null
+++ b/client/src/pages/RegisterPage.jsx
@@ -0,0 +1,168 @@
+import { useEffect, useState } from "react";
+import Swal from "sweetalert2"
+import axiosInstance from "../helpers/axiosInstance"
+import { useNavigate, NavLink } from "react-router"
+
+function RegisterPage() {
+ const navigate = useNavigate()
+
+ const [email, setEmail] = useState('')
+ const [password, setPassword] = useState('')
+ const [address, setAddress] = useState('')
+ const [phone, setPhone] = useState('')
+
+ useEffect(() => {
+ if (localStorage.getItem("access_token")) {
+ navigate('/')
+ }
+ }, [])
+
+ const onClick = async () => {
+ try {
+ const result = await axiosInstance({
+ method: "POST",
+ url: "/register",
+ data: {
+ email: email,
+ password: password,
+ address: address,
+ phone: phone
+ }
+ })
+ const response = result.data
+ Swal.fire({
+ icon: 'success',
+ title: 'Success',
+ text: 'Berhasil membuat akun'
+ })
+
+ // pindah halaman
+ navigate('/login')
+
+ } catch (error) {
+ if (error.response && error.response.data) {
+ Swal.fire({
+ icon: 'error',
+ title: 'Oops...',
+ text: error.response.data.message,
+ })
+ } else {
+ Swal.fire({
+ icon: 'error',
+ title: 'Oops...',
+ text: `Terjadi kesalahan. Silakan coba lagi ${error}`,
+ })
+ }
+ }
+ }
+
+ return (
+ <>
+
+
+
+
+
+
+
+
+ REGISTER
+
+
+
+
+
+
+
+ Sudah memiliki akun?{' '}
+
+ {" "}Login Sekarang
+
+
+
+
+
+ >
+ );
+}
+
+export default RegisterPage
\ No newline at end of file
diff --git a/client/src/store/coursesSlice.js b/client/src/store/coursesSlice.js
new file mode 100644
index 0000000..5f8c7a2
--- /dev/null
+++ b/client/src/store/coursesSlice.js
@@ -0,0 +1,73 @@
+import { createAsyncThunk, createSlice } from '@reduxjs/toolkit'
+import axiosInstance from '../helpers/axiosInstance'
+import Swal from 'sweetalert2'
+
+// membuaat sebuah slice/toolkit
+const coursesSlice = createSlice({
+ name: 'courses',
+ initialState: {
+ responseBody: {
+ message: "",
+ data: []
+ },
+ data: [], // diisi di action index
+ course: {} // diisi di action show
+ },
+ reducers: { // <-- reducers : bertugas menghandle sekumpulan actions
+ // seluruh actions yang kita miliki ada disini
+
+ responseBodyAction(state, action) {
+ state.responseBody = action.payload
+ },
+ indexAction(state, action) {
+ // state data akan diisi pada saat thunk dijalankan di view melalui payload
+ state.data = action.payload
+ },
+ showAction(state, action) {
+ state.course = action.payload
+ }
+ }
+})
+
+// destructure actions kita untuk di export
+export const { responseBodyAction, indexAction, showAction } = coursesSlice.actions
+
+// export reducer kita
+export const coursesReducer = coursesSlice.reducer
+
+
+// ini akan diinvoke di views
+export const fetchCourses = createAsyncThunk("courses/fetchCourses", async (_, { dispatch }) => {
+ try {
+ const result = await axiosInstance({
+ method: 'GET',
+ url: '/courses',
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ }
+ })
+ const responseBody = result.data
+ dispatch(responseBodyAction(responseBody)) // mau coba-coba ambil seluruh body aja
+ dispatch(indexAction(responseBody.data))
+ } catch (error) {
+ console.log(error)
+ Swal.fire("Error", error, 'error')
+ }
+})
+
+export const show = createAsyncThunk("courses/show", async (id, { dispatch }) => {
+ try {
+ const result = await axiosInstance({
+ method: 'GET',
+ url: `/courses/${id}`,
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ }
+ })
+ const responseBody = result.data
+ dispatch(showAction(responseBody.data))
+ } catch (error) {
+ console.log(error)
+ Swal.fire("Error", error, 'error')
+ }
+})
diff --git a/client/src/store/index.js b/client/src/store/index.js
new file mode 100644
index 0000000..cd2ec66
--- /dev/null
+++ b/client/src/store/index.js
@@ -0,0 +1,16 @@
+import { configureStore } from "@reduxjs/toolkit";
+import { coursesReducer } from "./coursesSlice";
+import { materialReducer } from "./materialSlice";
+import { userCourseReducer } from "./userCourseSlice";
+import { kanjiReducer } from "./kanjiSlice";
+
+export const store = configureStore({
+ reducer: {
+ courses: coursesReducer, // todo naming convention belum ajib
+ materialReducer: materialReducer,
+ userCourseReducer: userCourseReducer,
+ kanjiReducer: kanjiReducer
+ }
+})
+
+
diff --git a/client/src/store/kanjiSlice.js b/client/src/store/kanjiSlice.js
new file mode 100644
index 0000000..038d86d
--- /dev/null
+++ b/client/src/store/kanjiSlice.js
@@ -0,0 +1,28 @@
+import { createAsyncThunk, createSlice } from "@reduxjs/toolkit";
+import axios from "axios";
+import Swal from "sweetalert2";
+
+const kanjiSlice = createSlice({
+ name: "kanjiSlice",
+ initialState: {
+ data: []
+ },
+ reducers: {
+ indexAction(state, action) {
+ state.data = action.payload
+ }
+ }
+})
+
+export const { indexAction } = kanjiSlice.actions
+export const kanjiReducer = kanjiSlice.reducer
+
+// ini akan diinvoke di views
+export const index = createAsyncThunk("kanjiSlice/index", async (_, { dispatch }) => {
+ try {
+ const response = await axios.get('https://kanjiapi.dev/v1/kanji/joyo');
+ dispatch(indexAction(response.data)) // array of string
+ } catch (error) {
+ Swal.fire("Error", "Terjadi kesalahan", 'error')
+ }
+})
\ No newline at end of file
diff --git a/client/src/store/materialSlice.js b/client/src/store/materialSlice.js
new file mode 100644
index 0000000..64b5ac1
--- /dev/null
+++ b/client/src/store/materialSlice.js
@@ -0,0 +1,39 @@
+import { createAsyncThunk, createSlice } from '@reduxjs/toolkit'
+import axiosInstance from '../helpers/axiosInstance'
+import Swal from 'sweetalert2'
+
+// buat slicenya
+const materialSlice = createSlice({
+ name: "materialSlice",
+ initialState: {
+ data: []
+ },
+ reducers: {
+ indexAction(state, action) {
+ state.data = action.payload
+ }
+ }
+})
+
+// destructure dan export actions dan reducernya
+export const { indexAction } = materialSlice.actions
+export const materialReducer = materialSlice.reducer
+
+// buat async thunk untuk menggenerate function yang mereturn sebuah action
+// ini akan diapakai untuk didispatch di view agar state bisa diperbaruhi
+export const index = createAsyncThunk("materialSlice/index", async (courseId, { dispatch }) => {
+ try {
+ const result = await axiosInstance({
+ method: 'GET',
+ url: `/courses/${courseId}/materials`,
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ }
+ })
+ const responseBody = result.data
+ dispatch(indexAction(responseBody.data))
+ } catch (error) {
+ console.log(error)
+ Swal.fire("Error", error, 'error')
+ }
+})
diff --git a/client/src/store/userCourseSlice.js b/client/src/store/userCourseSlice.js
new file mode 100644
index 0000000..6658bc8
--- /dev/null
+++ b/client/src/store/userCourseSlice.js
@@ -0,0 +1,171 @@
+import { createAsyncThunk, createSlice } from '@reduxjs/toolkit'
+import axiosInstance from '../helpers/axiosInstance'
+import Swal from 'sweetalert2'
+
+// buat slicenya
+const userCourseSlice = createSlice({
+ name: "userCourseSlice",
+ initialState: {
+ data: [],
+ isEnrolled: false,
+ isFavorite: false
+ },
+ reducers: {
+ indexAction(state, action) {
+ state.data = action.payload
+ },
+ isEnrolledAction(state, action) {
+ state.isEnrolled = action.payload
+ },
+ setEnrollmentStatus(state, action) {
+ state.isEnrolled = action.payload
+ },
+ isFavoriteAction(state, action) {
+ state.isFavorite = action.payload
+ },
+ setFavoriteStatus(state, action) {
+ state.isFavorite = action.payload
+ }
+ }
+})
+
+// destructure dan export actions dan reducernya
+export const { indexAction, isEnrolledAction, setEnrollmentStatus, isFavoriteAction, setFavoriteStatus } = userCourseSlice.actions
+export const userCourseReducer = userCourseSlice.reducer
+
+// buat async thunk untuk menggenerate function yang mereturn sebuah action
+// ini akan diapakai untuk didispatch di view agar state bisa diperbaruhi
+export const index = createAsyncThunk("userCourseSlice/index", async ({ courseId, userId }, { dispatch }) => {
+ try {
+ const result = await axiosInstance({
+ method: 'GET',
+ url: `/user-courses`,
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ }
+ })
+ const responseBody = result.data
+
+ // check if user is enrolled
+ let isEnrolledCurrently = false
+ for (let i = 0; i < responseBody.length; i++) {
+ if (responseBody[i].UserCourse.CourseId === Number(courseId) && responseBody[i].UserCourse.UserId === Number(userId)) {
+ isEnrolledCurrently = true
+ break
+ }
+ // console.log(responseBody[i].UserCourse.CourseId, courseId, responseBody[i].UserCourse.UserId, userId)
+ }
+
+
+ // check if user has favorited the course
+ let isFavoriteCurrently = false
+ for (let i = 0; i < responseBody.length; i++) {
+ if (responseBody[i].UserCourse.CourseId === Number(courseId) && responseBody[i].UserCourse.UserId === Number(userId)) {
+ isFavoriteCurrently = responseBody[i].UserCourse.favorite || false
+ console.log(isFavoriteCurrently)
+ break
+ }
+ }
+
+
+ dispatch(isEnrolledAction(isEnrolledCurrently))
+ dispatch(isFavoriteAction(isFavoriteCurrently))
+ dispatch(indexAction(responseBody))
+ } catch (error) {
+ console.log(error)
+ Swal.fire("Error", error, 'error')
+ }
+})
+
+
+// Create async thunk to toggle user course enrollment
+export const toggleUserCourse = createAsyncThunk("userCourseSlice/toggleUserCourse", async ({ courseId, userId }, { dispatch, getState }) => {
+ try {
+ // Check if user is enrolled
+ const state = getState()
+ const isEnrolled = state.userCourseReducer.isEnrolled
+
+ if (isEnrolled) {
+ // User is enrolled, call DELETE endpoint
+ await axiosInstance({
+ method: 'DELETE',
+ url: `/user-courses`,
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ },
+ data: {
+ courseId,
+ userId
+ }
+ })
+ Swal.fire("Success", "You have been unenrolled from the course", 'success')
+ dispatch(setEnrollmentStatus(false))
+ dispatch(setFavoriteStatus(false))
+ } else {
+ // User is not enrolled, call POST endpoint
+ await axiosInstance({
+ method: 'POST',
+ url: `/user-courses`,
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ },
+ data: {
+ courseId,
+ userId
+ }
+ })
+ Swal.fire("Success", "You have been enrolled in the course", 'success')
+ dispatch(setEnrollmentStatus(true))
+ }
+
+ // // Re-fetch user courses to update the state
+ // dispatch(indexUserCourse({ courseId, userId }))
+ // ^^ di views aja
+ } catch (error) {
+ console.log(error)
+ Swal.fire("Error", error, 'error')
+ }
+})
+
+export const toogleFavoriteUpdate = createAsyncThunk("userCourseSlice/toogleFavoriteUpdate", async ({ courseId, userId }, { dispatch, getState }) => {
+ try {
+ // Check if user is enrolled
+ const state = getState()
+ const isFavorite = state.userCourseReducer.isFavorite
+
+ if (isFavorite) {
+ // User is enrolled, call DELETE endpoint
+ await axiosInstance({
+ method: 'PATCH',
+ url: `/user-courses`,
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ },
+ data: {
+ courseId,
+ userId
+ }
+ })
+ Swal.fire("Success", "Berhasil dihapus dari favorit", 'success')
+ dispatch(setFavoriteStatus(false))
+ } else {
+ // User is not enrolled, call POST endpoint
+ await axiosInstance({
+ method: 'PATCH',
+ url: `/user-courses`,
+ headers: {
+ Authorization: `Bearer ${localStorage.getItem("access_token")}`
+ },
+ data: {
+ courseId,
+ userId
+ }
+ })
+ Swal.fire("Success", "Berhasil ditandai sebagai favorit", 'success')
+ dispatch(setFavoriteStatus(true))
+ }
+ } catch (error) {
+ console.log(error)
+ Swal.fire("Error", error, 'error')
+ }
+})
\ No newline at end of file
diff --git a/client/tailwind.config.js b/client/tailwind.config.js
new file mode 100644
index 0000000..d37737f
--- /dev/null
+++ b/client/tailwind.config.js
@@ -0,0 +1,12 @@
+/** @type {import('tailwindcss').Config} */
+export default {
+ content: [
+ "./index.html",
+ "./src/**/*.{js,ts,jsx,tsx}",
+ ],
+ theme: {
+ extend: {},
+ },
+ plugins: [],
+}
+
diff --git a/client/vite.config.js b/client/vite.config.js
new file mode 100644
index 0000000..3ba048b
--- /dev/null
+++ b/client/vite.config.js
@@ -0,0 +1,14 @@
+import { defineConfig } from 'vite'
+import react from '@vitejs/plugin-react'
+import dotenv from 'dotenv';
+
+// Load environment variables
+dotenv.config();
+
+// https://vite.dev/config/
+export default defineConfig({
+ plugins: [react()],
+ define: {
+ 'process.env.GOOGLE_CLIENT_ID': JSON.stringify(process.env.GOOGLE_CLIENT_ID)
+ }
+})
diff --git a/server/.gitignore b/server/.gitignore
new file mode 100644
index 0000000..f6b6a55
--- /dev/null
+++ b/server/.gitignore
@@ -0,0 +1,4 @@
+my-server-notes.md
+node_modules
+.env
+test_env.js
\ No newline at end of file
diff --git a/server/API_DOCS.md b/server/API_DOCS.md
new file mode 100644
index 0000000..16e1363
--- /dev/null
+++ b/server/API_DOCS.md
@@ -0,0 +1,549 @@
+API Documentation
+# Endpoints :
+List of available endpoints:
+
+- GET /
+
+## AUTHENTICATION
+
+- POST /login
+- POST /google-login
+
+- POST /register
+- GET /verify-email
+
+## COURSES
+- GET /courses
+- GET /courses/:id
+- GET /courses/:courseId/materials
+
+## USER COURSES
+- POST /user-courses
+- DELETE /user-courses
+- GET /user-courses
+- PATCH /user-courses
+
+## KANJI
+- POST /romaji-transliterator
+- POST /kanji-transliterator
+- POST /translate-to-japanese
+- POST /translate-to-indonesia
+
+---
+
+### POST /login
+#### Request:
+- Body:
+```json
+{
+ "email": "string",
+ "password": "string"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "message": "login success",
+ "access_token": "string",
+ "data": {
+ "id": "integer"
+ }
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "email is required"
+}
+```
+or
+```json
+{
+ "message": "password is required"
+}
+```
+- 401 Unauthorized:
+```json
+{
+ "message": "email or password is invalid"
+}
+```
+or
+```json
+{
+ "message": "Please verify your email before logging in"
+}
+```
+
+---
+
+### POST /google-login
+#### Request:
+- Body:
+```json
+{
+ "googleToken": "string"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "message": "login success",
+ "access_token": "string",
+ "data": {
+ "id": "integer"
+ }
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "googleToken is required"
+}
+```
+
+---
+
+### POST /register
+#### Request:
+- Body:
+```json
+{
+ "email": "string",
+ "password": "string"
+}
+```
+
+#### Response:
+- 201 Created:
+```json
+{
+ "message": "register success",
+ "data": {
+ "email": "string"
+ }
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "email is required"
+}
+```
+or
+```json
+{
+ "message": "password is required"
+}
+```
+
+---
+
+### GET /verify-email
+#### Request:
+- Query Parameters:
+```json
+{
+ "token": "string"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "message": "Email verification successful"
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "Token is required"
+}
+```
+- 401 Unauthorized:
+```json
+{
+ "message": "Invalid token"
+}
+```
+- 404 Not Found:
+```json
+{
+ "message": "User not found"
+}
+```
+
+## COURSES
+
+### GET /courses
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+
+#### Response:
+- 200 OK:
+```json
+{
+ "data": [
+ {
+ "id": "integer",
+ "title": "string",
+ "description": "string",
+ "price": "number",
+ "createdAt": "string",
+ "updatedAt": "string"
+ }
+ ]
+}
+```
+- 401 Unauthorized:
+```json
+{
+ "message": "Unauthorized"
+}
+```
+
+---
+
+### GET /courses/:id
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+
+#### Response:
+- 200 OK:
+```json
+{
+ "data": {
+ "id": "integer",
+ "title": "string",
+ "description": "string",
+ "price": "number",
+ "createdAt": "string",
+ "updatedAt": "string"
+ }
+}
+```
+- 401 Unauthorized:
+```json
+{
+ "message": "Unauthorized"
+}
+```
+- 404 Not Found:
+```json
+{
+ "message": "Course not found"
+}
+```
+
+---
+### GET /courses/:courseId/materials
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+
+#### Response:
+- 200 OK:
+```json
+{
+ "message": "materials retrieved successfully",
+ "data": [
+ {
+ "id": 1,
+ "title": "string",
+ "imageUrl": "string",
+ "videoUrl": "string",
+ "duration": "00:39:23",
+ "desc": "string",
+ "CourseId": 1,
+ "createdAt": "string",
+ "updatedAt": "string"
+ }
+ ]
+}
+```
+- 401 Unauthorized:
+```json
+{
+ "message": "Unauthorized"
+}
+```
+## USER COURSES
+
+### POST /user-courses
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+- Body:
+```json
+{
+ "userId": "integer",
+ "courseId": "integer"
+}
+```
+
+#### Response:
+- 201 Created:
+```json
+{
+ "UserId": "integer",
+ "CourseId": "integer",
+ "updatedAt": "string",
+ "createdAt": "string",
+ "favorite": "boolean"
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "courseId is required"
+}
+```
+- 404 Not Found:
+```json
+{
+ "message": "error not found"
+}
+```
+- 401 Unauthorized:
+```json
+{
+ "message": "You need to verify your email first"
+}
+```
+
+---
+
+### DELETE /user-courses
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+- Body:
+```json
+{
+ "userId": "integer",
+ "courseId": "integer"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "message": "Course unenrolled successfully"
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "courseId is required"
+}
+```
+- 404 Not Found:
+```json
+{
+ "message": "error not found"
+}
+```
+- 401 Unauthorized:
+```json
+{
+ "message": "You are not authorized to unenroll from this course"
+}
+```
+
+---
+
+### GET /user-courses
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+
+#### Response:
+- 200 OK:
+```json
+[
+ {
+ "id": 1,
+ "title": "string",
+ "desc": "string",
+ "sensei": "string",
+ "imageUrl": "string",
+ "createdAt": "string",
+ "updatedAt": "string",
+ "UserCourse": {
+ "UserId": "integer",
+ "CourseId": "integer",
+ "favorite": "boolean",
+ "createdAt": "string",
+ "updatedAt": "string"
+ }
+ }
+]
+```
+- 401 Unauthorized:
+```json
+{
+ "message": "Unauthorized"
+}
+```
+
+---
+
+### PATCH /user-courses
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+- Body:
+```json
+{
+ "userId": "integer",
+ "courseId": "integer"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "message": "Course added to favorite"
+}
+```
+or
+```json
+{
+ "message": "Course removed from favorite"
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "courseId is required"
+}
+```
+- 404 Not Found:
+```json
+{
+ "message": "error not found"
+}
+```
+
+## KANJI
+
+### POST /romaji-transliterator
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+- Body:
+```json
+{
+ "romaji": "string"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "transliteration": "string"
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "Inputan tidak boleh kosong"
+}
+```
+
+---
+
+### POST /kanji-transliterator
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+- Body:
+```json
+{
+ "kanji": "string"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "transliteration": "string"
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "Inputan tidak boleh kosong"
+}
+```
+or
+```json
+{
+ "message": "Inputan bukan kanji"
+}
+```
+
+---
+
+### POST /translate-to-japanese
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+- Body:
+```json
+{
+ "text": "string"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "translation": "string"
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "Inputan tidak boleh kosong"
+}
+```
+
+---
+
+### POST /translate-to-indonesia
+#### Request:
+- Headers:
+ - Authorization: Bearer ``
+- Body:
+```json
+{
+ "text": "string"
+}
+```
+
+#### Response:
+- 200 OK:
+```json
+{
+ "translation": "string"
+}
+```
+- 400 Bad Request:
+```json
+{
+ "message": "Inputan tidak boleh kosong"
+}
+```
\ No newline at end of file
diff --git a/server/__test__/AiController.test.js b/server/__test__/AiController.test.js
new file mode 100644
index 0000000..14f7e79
--- /dev/null
+++ b/server/__test__/AiController.test.js
@@ -0,0 +1,105 @@
+const request = require('supertest')
+const { expect, test, afterAll, describe } = require('@jest/globals')
+const app = require('../app')
+const { User } = require('../models/index')
+const { signToken } = require('../helpers/jwt')
+const { hashingPassword } = require('../helpers/bcrypt')
+
+beforeAll(async () => {
+ const user = await User.create({
+ email: 'testuser@example.com',
+ password: await hashingPassword('password123'),
+ emailVerifiedAt: new Date()
+ })
+ accessToken = signToken({ id: user.id, email: user.email })
+})
+
+afterAll(async () => {
+ await User.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+})
+
+describe('POST /romaji-transliterator', () => {
+ const requestBody = {
+ userId: 1
+ };
+ test('should return kanji', async () => {
+ const response = await request(app)
+ .post('/romaji-transliterator')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({
+ romaji: 'jinsei'
+ })
+ .expect(200)
+ expect(response.body.transliteration).toBe('人生')
+ })
+
+ test('should return 400 when no romaji provided', async () => {
+ const response = await request(app)
+ .post('/romaji-transliterator')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({
+
+ })
+ .expect(400)
+ expect(response.body.message).toBe('Inputan tidak boleh kosong')
+ })
+
+ test('should return romaji', async () => {
+ const response = await request(app)
+ .post('/kanji-transliterator')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({
+ kanji: '人生'
+ })
+ .expect(200)
+ expect(response.body.transliteration).toBe('jinsei')
+ })
+
+ test('should return 400 when no kanji provided', async () => {
+ const response = await request(app)
+ .post('/kanji-transliterator')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({
+
+ })
+ .expect(400)
+ expect(response.body.message).toBe('Inputan tidak boleh kosong')
+ })
+
+ test('should return 400 when input non kanji', async () => {
+ const response = await request(app)
+ .post('/kanji-transliterator')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({
+ kanji: 'romaji'
+ })
+ .expect(400)
+ expect(response.body.message).toBe('Inputan bukan kanji')
+ })
+
+ test('should return japanese translation', async () => {
+ const response = await request(app)
+ .post('/translate-to-japanese')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({
+ text: 'bahasa jepang'
+ })
+ .expect(200)
+ expect(response.body.translation).toBe('日本語')
+ })
+
+ test('should return indonesia translation', async () => {
+ const response = await request(app)
+ .post('/translate-to-indonesia')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({
+ text: '日本語'
+ })
+ .expect(200)
+ expect(response.body.translation).toBe('Bahasa Jepang')
+ })
+})
\ No newline at end of file
diff --git a/server/__test__/UserController.test.js b/server/__test__/UserController.test.js
new file mode 100644
index 0000000..196583c
--- /dev/null
+++ b/server/__test__/UserController.test.js
@@ -0,0 +1,186 @@
+const request = require("supertest")
+const app = require('../app')
+const { expect, describe, beforeAll, afterAll } = require("@jest/globals")
+const { User } = require('../models/index')
+const { hashingPassword } = require("../helpers/bcrypt")
+const { signToken } = require("../helpers/jwt")
+const MOCK_GOOGLE_TOKEN = require('./test_env')
+
+
+beforeAll(async () => {
+ await User.bulkCreate([
+ {
+ email: 'usertest@example.com',
+ password: await hashingPassword('12345678'),
+ role: 'User',
+ emailVerifiedAt: new Date(),
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ },
+ ])
+})
+
+afterAll(async () => {
+ await User.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+})
+
+describe('POST /login', () => {
+ const email = "usertest@example.com"
+ const password = "12345678"
+ it('should success and send access_token', async () => {
+ const reqBody = {
+ email: email,
+ password: password
+ }
+ const response = await request(app).post("/login").send(reqBody)
+ expect(response.status).toBe(200)
+ expect(response.body).toHaveProperty("access_token")
+ })
+
+ it('should fail when email is not provided', async () => {
+ const reqBody = {
+ password: password
+ };
+
+ const response = await request(app).post("/login").send(reqBody)
+
+ expect(response.status).toBe(400)
+ expect(response.body).toHaveProperty("message")
+ expect(response.body.message).toMatch('email is required')
+ })
+
+ it('should fail when password is not provided', async () => {
+ const reqBody = {
+ email: email
+ };
+
+ const response = await request(app).post("/login").send(reqBody)
+
+ expect(response.status).toBe(400);
+ expect(response.body).toHaveProperty("message");
+ expect(response.body.message).toMatch('password is required')
+ })
+
+ it('should invalid when email not registered', async () => {
+ const reqBody = {
+ email: "unregistered@example.com",
+ password: "12345678"
+ };
+
+ const response = await request(app).post("/login").send(reqBody)
+
+ expect(response.status).toBe(401);
+ expect(response.body).toHaveProperty("message");
+ expect(response.body.message).toMatch("email or password is invalid");
+ })
+
+ it('should invalid when password not registered', async () => {
+ const reqBody = {
+ email: "admin@example.com",
+ password: "123456789"
+ };
+
+ const response = await request(app).post("/login").send(reqBody)
+
+ expect(response.status).toBe(401);
+ expect(response.body).toHaveProperty("message");
+ expect(response.body.message).toMatch('email or password is invalid');
+ })
+})
+
+describe('POST /register', () => {
+ it('should successfully register a new user and send verification email', async () => {
+ const reqBody = {
+ email: "newuser@example.com",
+ password: "12345678"
+ };
+
+ const response = await request(app).post("/register").send(reqBody);
+
+ expect(response.status).toBe(201);
+ expect(response.body).toHaveProperty("message", "register success");
+ expect(response.body.data).toHaveProperty("email", reqBody.email);
+ expect(response.body.data).not.toHaveProperty("password");
+ });
+
+ it('should fail when email is not provided', async () => {
+ const reqBody = {
+ password: "12345678"
+ };
+
+ const response = await request(app).post("/register").send(reqBody);
+
+ expect(response.status).toBe(400);
+ expect(response.body).toHaveProperty("message", "email is required");
+ });
+
+ it('should fail when password is not provided', async () => {
+ const reqBody = {
+ email: "newuser@example.com"
+ };
+
+ const response = await request(app).post("/register").send(reqBody);
+
+ expect(response.status).toBe(400);
+ expect(response.body).toHaveProperty("message", "password is required");
+ });
+});
+
+describe('GET /verify-email', () => {
+ let verificationToken;
+
+ beforeAll(async () => {
+ const user = await User.create({
+ email: "verifyuser@example.com",
+ password: await hashingPassword("12345678"),
+ emailVerifiedAt: null,
+ createdAt: new Date(),
+ updatedAt: new Date()
+ });
+
+ verificationToken = signToken({ id: user.id }, '1h');
+ });
+
+ it('should successfully verify email', async () => {
+ const response = await request(app).get(`/verify-email?token=${verificationToken}`);
+
+ expect(response.status).toBe(200);
+ expect(response.body).toHaveProperty("message", "Email verification successful");
+ });
+
+ it('should fail when token is not provided', async () => {
+ const response = await request(app).get("/verify-email");
+
+ expect(response.status).toBe(400);
+ expect(response.body).toHaveProperty("message", "Token is required");
+ });
+
+ it('should fail when token is invalid', async () => {
+ const response = await request(app).get("/verify-email?token=invalidtoken");
+
+ expect(response.status).toBe(401);
+ expect(response.body).toHaveProperty("message");
+ });
+});
+
+describe('POST /google-login', () => {
+ it('should successfully login or register a user with Google', async () => {
+ const mockGoogleToken = MOCK_GOOGLE_TOKEN; // Replace with a valid mock token if needed
+ const response = await request(app).post("/google-login").send({ googleToken: mockGoogleToken });
+ expect(response.status).toBe(200);
+ expect(response.body).toHaveProperty("message", "login success");
+ expect(response.body).toHaveProperty("access_token");
+ expect(response.body.data).toHaveProperty("id");
+ });
+
+ it('should fail when googleToken is not provided', async () => {
+ const response = await request(app).post("/google-login").send({});
+ expect(response.status).toBe(400);
+ expect(response.body).toHaveProperty("message");
+ });
+});
+
diff --git a/server/__test__/UserCourseController.test.js b/server/__test__/UserCourseController.test.js
new file mode 100644
index 0000000..f778945
--- /dev/null
+++ b/server/__test__/UserCourseController.test.js
@@ -0,0 +1,401 @@
+const request = require('supertest')
+const { expect, test, afterAll, describe } = require('@jest/globals')
+const app = require('../app')
+const { Course, User, Material, UserCourse } = require('../models/index')
+const { signToken } = require('../helpers/jwt')
+const { hashingPassword } = require('../helpers/bcrypt')
+
+beforeAll(async () => {
+ const user = await User.create({
+ email: 'testuser@example.com',
+ password: await hashingPassword('password123'),
+ emailVerifiedAt: new Date()
+ })
+
+ const course = await Course.create(
+ {
+ title: 'Hiragana bersama Ryu Sensei',
+ desc: 'Bergabunglah dengan kursus komprehensif kami untuk menguasai Hiragana, dasar dari sistem penulisan Jepang, dengan bimbingan ahli dari Ryu Sensei.',
+ sensei: 'Ryu Sensei',
+ imageUrl: 'https://i.ytimg.com/vi/nrv_iMuMLFk/hqdefault.jpg?sqp=-oaymwEXCOADEI4CSFryq4qpAwkIARUAAIhCGAE=&rs=AOn4CLBFRz9nFqgGVX23V3mvxtb_PwzVMg',
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ }
+ )
+
+ await UserCourse.create({
+ UserId: user.id,
+ CourseId: course.id
+ })
+
+ accessToken = signToken({ id: user.id, email: user.email })
+})
+
+afterAll(async () => {
+ await User.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+ await Course.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+
+ await UserCourse.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+})
+
+describe('GET /user-courses', () => {
+ test('should return a list of user courses', async () => {
+ const response = await request(app)
+ .get('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .expect(200)
+ expect(response.body).toBeInstanceOf(Array)
+ })
+})
+
+describe('POST /user-courses', () => {
+ const requestBody = {
+ userId: 1
+ };
+ test('should return 400 if courseId is not provided', async () => {
+ const response = await request(app)
+ .post('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send(requestBody)
+ .expect(400)
+ expect(response.body.message).toBe('courseId is required')
+ })
+})
+
+
+describe('DELETE /user-courses', () => {
+ test('should return 400 if courseId is not provided', async () => {
+ const response = await request(app)
+ .delete('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .expect(400)
+ expect(response.body.message).toBe('courseId is required')
+ })
+})
+
+describe('POST /user-courses (store)', () => {
+ test('should enroll a user in a course successfully', async () => {
+ const newCourse = await Course.create({
+ title: 'Katakana bersama Ryu Sensei',
+ desc: 'Pelajari Katakana dengan mudah bersama Ryu Sensei.',
+ sensei: 'Ryu Sensei',
+ imageUrl: 'https://example.com/katakana.jpg',
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ });
+
+ const requestBody = {
+ userId: 1, // Assuming the user created in beforeAll has ID 1
+ courseId: newCourse.id,
+ };
+
+ const response = await request(app)
+ .post('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send(requestBody)
+ .expect(200);
+
+ expect(response.body).toHaveProperty('UserId', requestBody.userId);
+ expect(response.body).toHaveProperty('CourseId', requestBody.courseId);
+ });
+
+ test('should return 404 if user is not found', async () => {
+ const newCourse = await Course.create({
+ title: 'Katakana bersama Ryu Sensei',
+ desc: 'Pelajari Katakana dengan mudah bersama Ryu Sensei.',
+ sensei: 'Ryu Sensei',
+ imageUrl: 'https://example.com/katakana.jpg',
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ });
+
+ const requestBody = {
+ userId: 9999, // Assuming the user created in beforeAll has ID 1
+ courseId: newCourse.id,
+ };
+
+ const response = await request(app)
+ .post('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send(requestBody)
+ .expect(404);
+
+ expect(response.body.message).toBe("error not found");
+ });
+
+ test('should return 400 if courseId is not provided', async () => {
+ const requestBody = {
+ userId: 1
+ };
+
+ const response = await request(app)
+ .post('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send(requestBody)
+ .expect(400);
+
+ expect(response.body.message).toBe("courseId is required");
+ });
+
+ test('should return 401 if user email is not verified', async () => {
+ const unverifiedUser = await User.create({
+ email: 'unverified@example.com',
+ password: await hashingPassword('password123'),
+ emailVerifiedAt: null, // Email not verified
+ });
+
+ const newCourse = await Course.create({
+ title: 'Katakana bersama Ryu Sensei',
+ desc: 'Pelajari Katakana dengan mudah bersama Ryu Sensei.',
+ sensei: 'Ryu Sensei',
+ imageUrl: 'https://example.com/katakana.jpg',
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ });
+
+ const response = await request(app)
+ .post('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({
+ userId: unverifiedUser.id,
+ courseId: newCourse.id,
+ })
+ .expect(401);
+
+ expect(response.body.message).toBe("You need to verify your email first");
+ });
+
+ test('should return 404 if course is not found', async () => {
+ const newCourse = await Course.create({
+ title: 'Katakana bersama Ryu Sensei',
+ desc: 'Pelajari Katakana dengan mudah bersama Ryu Sensei.',
+ sensei: 'Ryu Sensei',
+ imageUrl: 'https://example.com/katakana.jpg',
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ });
+
+ const requestBody = {
+ userId: 1, // Assuming the user created in beforeAll has ID 1
+ courseId: 999,
+ };
+
+ const response = await request(app)
+ .post('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send(requestBody)
+ .expect(404);
+
+ expect(response.body.message).toBe("error not found");
+ });
+
+ describe('POST /user-courses (store)', () => {
+ test('should return 400 if user is already enrolled in the course', async () => {
+ const existingCourse = await Course.create({
+ title: 'Kanji bersama Ryu Sensei',
+ desc: 'Pelajari Kanji dengan mudah bersama Ryu Sensei.',
+ sensei: 'Ryu Sensei',
+ imageUrl: 'https://example.com/kanji.jpg',
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ });
+
+ const existingUserCourse = await UserCourse.create({
+ UserId: 1, // Assuming the user created in beforeAll has ID 1
+ CourseId: existingCourse.id,
+ });
+
+ const requestBody = {
+ userId: existingUserCourse.UserId,
+ courseId: existingUserCourse.CourseId,
+ };
+
+ const response = await request(app)
+ .post('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send(requestBody)
+ .expect(400);
+
+ expect(response.body.message).toBe("You already enrolled in this course");
+ });
+ });
+});
+
+describe('DELETE /user-courses', () => {
+ test('should unenroll a user from a course successfully', async () => {
+ UserCourse.create({
+ UserId: 1,
+ CourseId: 1
+ })
+
+ const response = await request(app)
+ .delete('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: 1, courseId: 1 })
+ .expect(200)
+ expect(response.body.message).toBe('Course unenrolled successfully')
+ })
+
+ test('should return 400 if courseId is not provided', async () => {
+ UserCourse.create({
+ UserId: 1,
+ CourseId: 1
+ })
+
+ const response = await request(app)
+ .delete('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: 1 })
+ .expect(400);
+
+ expect(response.body.message).toBe('courseId is required');
+ });
+
+ test('should return 404 if user is not found', async () => {
+ UserCourse.create({
+ UserId: 1,
+ CourseId: 1
+ })
+ const response = await request(app)
+ .delete('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: 9999, courseId: 1 })
+ .expect(404);
+
+ expect(response.body.message).toBe('error not found');
+ });
+
+ test('should return 404 if course is not found', async () => {
+ UserCourse.create({
+ UserId: 1,
+ CourseId: 1
+ })
+ const response = await request(app)
+ .delete('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: 1, courseId: 9999 })
+ .expect(404);
+
+ expect(response.body.message).toBe('error not found');
+ });
+
+ test('should return 400 if user is not enrolled in the course', async () => {
+ const user = await User.create({
+ email: 'notenrolleduser@example.com',
+ password: await hashingPassword('password123'),
+ emailVerifiedAt: new Date(),
+ });
+
+ UserCourse.create({
+ UserId: 1,
+ CourseId: 1
+ })
+
+ const response = await request(app)
+ .delete('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: user.id, courseId: 1 })
+ .expect(400);
+
+ expect(response.body.message).toBe('You are not enrolled in this course');
+ });
+
+ test('should return 401 if unenrolled another user course', async () => {
+ const currentUser = await User.create({
+ email: 'newUseragain@example.com',
+ password: await hashingPassword('password123'),
+ emailVerifiedAt: new Date(),
+ });
+ const currentUserAccessToken = signToken({ id: currentUser.id, email: currentUser.email })
+
+ const anotherUser = await User.create({
+ email: 'anotheruser@example.com',
+ password: await hashingPassword('password123'),
+ emailVerifiedAt: new Date(),
+ });
+ UserCourse.create({
+ UserId: anotherUser.id,
+ CourseId: 1
+ })
+
+
+ const response = await request(app)
+ .delete('/user-courses')
+ .set('Authorization', `Bearer ${currentUserAccessToken}`)
+ .send({ userId: anotherUser.id, courseId: 1 })
+ .expect(401);
+
+ expect(response.body.message).toBe('You are not authorized to unenroll from this course');
+ });
+})
+
+describe('PATCH /user-courses', () => {
+ test('should add a course to favorites successfully', async () => {
+ const userCourse = UserCourse.create({
+ UserId: 1,
+ CourseId: 1
+ })
+ const response = await request(app)
+ .patch('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: 1, courseId: 1 })
+ .expect(200);
+ expect(response.body.message).toBe('Course added to favorite');
+ });
+
+ test('should remove a course from favorites successfully', async () => {
+ UserCourse.create({
+ UserId: 1,
+ CourseId: 1,
+ favorite: true
+ })
+ const response = await request(app)
+ .patch('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: 1, courseId: 1 })
+ .expect(200);
+ expect(response.body.message).toBe('Course removed from favorite');
+ });
+
+ test('should return 400 if userId is not provided', async () => {
+ const response = await request(app)
+ .patch('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ courseId: 1 })
+ .expect(400);
+
+ expect(response.body.message).toBe('userId is required');
+ });
+
+ test('should return 400 if courseId is not provided', async () => {
+ const response = await request(app)
+ .patch('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: 1 })
+ .expect(400);
+
+ expect(response.body.message).toBe('courseId is required');
+ });
+
+ test('should return 404 if userCourse is not found', async () => {
+ const response = await request(app)
+ .patch('/user-courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .send({ userId: 9999, courseId: 9999 })
+ .expect(404);
+
+ expect(response.body.message).toBe('error not found');
+ });
+});
\ No newline at end of file
diff --git a/server/__test__/authenticaton.test.js b/server/__test__/authenticaton.test.js
new file mode 100644
index 0000000..beb384a
--- /dev/null
+++ b/server/__test__/authenticaton.test.js
@@ -0,0 +1,62 @@
+const authentication = require('../middleware/authentication');
+const { verifyToken } = require('../helpers/jwt');
+
+// server/middleware/authentication.test.js
+
+jest.mock('../helpers/jwt');
+
+describe('Authentication Middleware', () => {
+ let req, res, next;
+
+ beforeEach(() => {
+ req = { headers: {} };
+ res = {};
+ next = jest.fn();
+ });
+
+ it('should call next with an error if no authorization header is provided', () => {
+ authentication(req, res, next);
+ expect(next).toHaveBeenCalledWith(expect.objectContaining({
+ name: "Unauthorized",
+ message: "unauthenticated : no headers with key authorization"
+ }));
+ });
+
+ it('should call next with an error if authorization header does not start with "Bearer"', () => {
+ req.headers.authorization = "Token abc123";
+ authentication(req, res, next);
+ expect(next).toHaveBeenCalledWith(expect.objectContaining({
+ name: "Unauthorized",
+ message: "unauthenticated : no bearer token"
+ }));
+ });
+
+ it('should call next with an error if authorization header is missing the token', () => {
+ req.headers.authorization = "Bearer";
+ authentication(req, res, next);
+ expect(next).toHaveBeenCalledWith(expect.objectContaining({
+ name: "Unauthorized",
+ message: "unauthenticated : no bearer token"
+ }));
+ });
+
+ it('should call next with an error if verifyToken throws an error', () => {
+ req.headers.authorization = "Bearer invalidToken";
+ verifyToken.mockImplementation(() => {
+ throw new Error("Invalid token");
+ });
+ authentication(req, res, next);
+ expect(next).toHaveBeenCalledWith(expect.any(Error));
+ });
+
+ it('should set req.user and call next if token is valid', () => {
+ const mockPayload = { id: 1, email: "test@example.com" };
+ req.headers.authorization = "Bearer validToken";
+ verifyToken.mockReturnValue(mockPayload);
+
+ authentication(req, res, next);
+
+ expect(req.user).toEqual(mockPayload);
+ expect(next).toHaveBeenCalledWith();
+ });
+});
\ No newline at end of file
diff --git a/server/__test__/authorization.test.js b/server/__test__/authorization.test.js
new file mode 100644
index 0000000..09bdf18
--- /dev/null
+++ b/server/__test__/authorization.test.js
@@ -0,0 +1,88 @@
+const authorization = require('../middleware/authorization');
+const { Post } = require('../models/index');
+
+jest.mock('../models/index', () => ({
+ Post: {
+ findByPk: jest.fn(),
+ },
+}));
+
+describe('authorization middleware', () => {
+ let mockReq, mockRes, mockNext;
+
+ beforeEach(() => {
+ mockReq = {
+ user: {},
+ params: {},
+ };
+ mockRes = {};
+ mockNext = jest.fn();
+ });
+
+ afterEach(() => {
+ jest.clearAllMocks();
+ });
+
+ test('should call next if user role is authorized', async () => {
+ mockReq.user = { role: 'Admin' };
+ const middleware = authorization(['Admin', 'Staff']);
+
+ await middleware(mockReq, mockRes, mockNext);
+
+ expect(mockNext).toHaveBeenCalled();
+ });
+
+ test('should throw Forbidden error if user role is not authorized', async () => {
+ mockReq.user = { role: 'User' };
+ const middleware = authorization(['Admin', 'Staff']);
+
+ await middleware(mockReq, mockRes, mockNext);
+
+ expect(mockNext).toHaveBeenCalledWith({
+ name: 'Forbidden',
+ message: 'unauthorized : role must be Admin,Staff',
+ });
+ });
+
+ test('should throw NotFound error if post is not found for Staff role', async () => {
+ mockReq.user = { role: 'Staff', id: 1 };
+ mockReq.params = { id: 1 };
+ Post.findByPk.mockResolvedValue(null);
+ const middleware = authorization(['Admin', 'Staff']);
+
+ await middleware(mockReq, mockRes, mockNext);
+
+ expect(Post.findByPk).toHaveBeenCalledWith(1);
+ expect(mockNext).toHaveBeenCalledWith({
+ name: 'NotFound',
+ message: 'error not found',
+ });
+ });
+
+ test('should throw Forbidden error if Staff tries to modify another user\'s post', async () => {
+ mockReq.user = { role: 'Staff', id: 1 };
+ mockReq.params = { id: 1 };
+ Post.findByPk.mockResolvedValue({ authorId: 2 });
+ const middleware = authorization(['Admin', 'Staff']);
+
+ await middleware(mockReq, mockRes, mockNext);
+
+ expect(Post.findByPk).toHaveBeenCalledWith(1);
+ expect(mockNext).toHaveBeenCalledWith({
+ name: 'Forbidden',
+ message: 'unauthorized : you only can modify your own post',
+ });
+ });
+
+ test('should call next if Staff modifies their own post', async () => {
+ mockReq.user = { role: 'Staff', id: 1 };
+ mockReq.params = { id: 1 };
+ Post.findByPk.mockResolvedValue({ authorId: 1 });
+ const middleware = authorization(['Admin', 'Staff']);
+
+ await middleware(mockReq, mockRes, mockNext);
+
+ expect(Post.findByPk).toHaveBeenCalledWith(1);
+ expect(mockNext).toHaveBeenCalled();
+ });
+});
\ No newline at end of file
diff --git a/server/__test__/courses.test.js b/server/__test__/courses.test.js
new file mode 100644
index 0000000..a800393
--- /dev/null
+++ b/server/__test__/courses.test.js
@@ -0,0 +1,79 @@
+const request = require('supertest')
+const { expect, test, afterAll, describe } = require('@jest/globals')
+
+const app = require('../app')
+const { Course, User } = require('../models/index')
+const { signToken } = require('../helpers/jwt')
+const { hashingPassword } = require('../helpers/bcrypt')
+
+
+let accessToken
+
+beforeAll(async () => {
+ const user = await User.create({
+ email: 'testuser@example.com',
+ password: await hashingPassword('password123'),
+ emailVerifiedAt: new Date()
+ })
+
+ await Course.create(
+ {
+ title: 'Hiragana bersama Ryu Sensei',
+ desc: 'Bergabunglah dengan kursus komprehensif kami untuk menguasai Hiragana, dasar dari sistem penulisan Jepang, dengan bimbingan ahli dari Ryu Sensei.',
+ sensei: 'Ryu Sensei',
+ imageUrl: 'https://i.ytimg.com/vi/nrv_iMuMLFk/hqdefault.jpg?sqp=-oaymwEXCOADEI4CSFryq4qpAwkIARUAAIhCGAE=&rs=AOn4CLBFRz9nFqgGVX23V3mvxtb_PwzVMg',
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ }
+ )
+
+ accessToken = signToken({ id: user.id, email: user.email })
+})
+
+afterAll(async () => {
+ await User.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+ await Course.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+})
+
+describe('GET /courses', () => {
+ test('should return a list of courses', async () => {
+ const response = await request(app)
+ .get('/courses')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .expect(200)
+
+ expect(response.body.data).toBeInstanceOf(Array)
+ })
+})
+
+// test for GET /courses/:id
+describe('GET /courses/:id', () => {
+ test('should return a course', async () => {
+ const course = await Course.findByPk(1)
+ const response = await request(app)
+ .get(`/courses/${course.id}`)
+ .set('Authorization', `Bearer ${accessToken}`)
+ .expect(200)
+
+ expect(response.body.data.name).toBe(course.name)
+ expect(response.body.data.description).toBe(course.description)
+ expect(response.body.data.price).toBe(course.price)
+ })
+
+ test('should return 404 if course not found', async () => {
+ const response = await request(app)
+ .get('/courses/999')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .expect(404)
+
+ expect(response.body.message).toBe('error not found')
+ })
+})
\ No newline at end of file
diff --git a/server/__test__/errorHandler.test.js b/server/__test__/errorHandler.test.js
new file mode 100644
index 0000000..5543661
--- /dev/null
+++ b/server/__test__/errorHandler.test.js
@@ -0,0 +1,100 @@
+const request = require('supertest')
+const { expect, test, afterAll, describe } = require('@jest/globals')
+
+const app = require('../app')
+const { Course, User } = require('../models/index')
+const { signToken } = require('../helpers/jwt')
+const { hashingPassword } = require('../helpers/bcrypt')
+const errorHandler = require('../middleware/errorHandler')
+describe('errorHandler middleware', () => {
+ const mockReq = {};
+ const mockRes = {
+ status: jest.fn().mockReturnThis(),
+ json: jest.fn()
+ };
+ const mockNext = jest.fn();
+
+ afterEach(() => {
+ jest.clearAllMocks();
+ });
+
+ test('should handle BadRequest error', () => {
+ const err = { name: "BadRequest", message: "Bad request error" };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(400);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "Bad request error" });
+ });
+
+ test('should handle Unauthorized error', () => {
+ const err = { name: "Unauthorized", message: "Unauthorized access" };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(401);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "Unauthorized access" });
+ });
+
+ test('should handle JsonWebTokenError', () => {
+ const err = { name: "JsonWebTokenError" };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(401);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "token is invalid" });
+ });
+
+ test('should handle Forbidden error', () => {
+ const err = { name: "Forbidden", message: "Access forbidden" };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(403);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "Access forbidden" });
+ });
+
+ test('should handle SequelizeValidationError', () => {
+ const err = { name: "SequelizeValidationError", errors: [{ message: "Validation error" }] };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(400);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "Validation error" });
+ });
+
+ test('should handle SequelizeUniqueConstraintError', () => {
+ const err = { name: "SequelizeUniqueConstraintError", errors: [{ message: "Unique constraint error" }] };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(400);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "Unique constraint error" });
+ });
+
+ test('should handle NotFound error', () => {
+ const err = { name: "NotFound", message: "Resource not found" };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(404);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "Resource not found" });
+ });
+
+ test('should handle SequelizeDatabaseError', () => {
+ const err = { name: "SequelizeDatabaseError", message: "Database error" };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(400);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "Database error" });
+ });
+
+ test('should handle TokenExpiredError', () => {
+ const err = { name: "TokenExpiredError" };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(401);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: "Verification link expired" });
+ });
+
+ test('should handle default case for unknown errors', () => {
+ const err = { name: "UnknownError", message: "Something went wrong" };
+ errorHandler(err, mockReq, mockRes, mockNext);
+
+ expect(mockRes.status).toHaveBeenCalledWith(500);
+ expect(mockRes.json).toHaveBeenCalledWith({ message: `internal server error: ${err}` });
+ });
+});
\ No newline at end of file
diff --git a/server/__test__/material.test.js b/server/__test__/material.test.js
new file mode 100644
index 0000000..d8e04e4
--- /dev/null
+++ b/server/__test__/material.test.js
@@ -0,0 +1,71 @@
+const request = require('supertest')
+const { expect, test, afterAll, describe } = require('@jest/globals')
+const app = require('../app')
+const { Course, User, Material } = require('../models/index')
+const { signToken } = require('../helpers/jwt')
+const { hashingPassword } = require('../helpers/bcrypt')
+
+beforeAll(async () => {
+ const user = await User.create({
+ email: 'testuser@example.com',
+ password: await hashingPassword('password123'),
+ emailVerifiedAt: new Date()
+ })
+
+ await Course.create(
+ {
+ title: 'Hiragana bersama Ryu Sensei',
+ desc: 'Bergabunglah dengan kursus komprehensif kami untuk menguasai Hiragana, dasar dari sistem penulisan Jepang, dengan bimbingan ahli dari Ryu Sensei.',
+ sensei: 'Ryu Sensei',
+ imageUrl: 'https://i.ytimg.com/vi/nrv_iMuMLFk/hqdefault.jpg?sqp=-oaymwEXCOADEI4CSFryq4qpAwkIARUAAIhCGAE=&rs=AOn4CLBFRz9nFqgGVX23V3mvxtb_PwzVMg',
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ }
+ )
+
+ await Material.create(
+ {
+ title: 'Belajar Bahasa Jepang - CARA HAFAL HIRAGANA',
+ desc: 'こんにちは !! Hari ini mari kita belajar membaca Hiragana sampai bisa ya. Buat teman2 yg mau belajar materi lainnya silahkan cek playlist yaa Arigatou Minasan (^^)',
+ imageUrl: 'https://i.ytimg.com/vi/nrv_iMuMLFk/hqdefault.jpg?sqp=-oaymwEXCOADEI4CSFryq4qpAwkIARUAAIhCGAE=&rs=AOn4CLBFRz9nFqgGVX23V3mvxtb_PwzVMg',
+ videoUrl: 'https://www.youtube.com/watch?v=nrv_iMuMLFk&list=PLhQ568NlkEK2O3NhQdNzl9YNLEgXMSW-J&index=1',
+ duration: '00:39:23',
+ CourseId: 1,
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ }
+ )
+
+ accessToken = signToken({ id: user.id, email: user.email })
+})
+
+afterAll(async () => {
+ await User.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+ await Course.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+
+ await Material.destroy({
+ truncate: true, // hapus semua
+ cascade: true, // hapus data yang berelasi
+ restartIdentity: true // restart id dari 1
+ })
+})
+
+// buat testing untuk GET /courses/:courseId/materials
+describe('GET /courses/:courseId/materials', () => {
+ test('should return a list of materials', async () => {
+ const response = await request(app)
+ .get('/courses/1/materials')
+ .set('Authorization', `Bearer ${accessToken}`)
+ .expect(200)
+
+ expect(response.body.data).toBeInstanceOf(Array)
+ })
+})
\ No newline at end of file
diff --git a/server/__test__/randompassword.test.js b/server/__test__/randompassword.test.js
new file mode 100644
index 0000000..0264914
--- /dev/null
+++ b/server/__test__/randompassword.test.js
@@ -0,0 +1,22 @@
+const randomPasswordGenerator = require('../helpers/randomPasswordGenerator');
+
+describe('randomPasswordGenerator', () => {
+ test('should generate a password of default length 12', () => {
+ const password = randomPasswordGenerator();
+ expect(password).toHaveLength(12);
+ });
+
+ test('should generate a password of specified length', () => {
+ const length = 16;
+ const password = randomPasswordGenerator(length);
+ expect(password).toHaveLength(length);
+ });
+
+ test('should generate a password containing only valid characters', () => {
+ const validCharacters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789!@#$%^&*()_+[]{}|;:,.<>?';
+ const password = randomPasswordGenerator();
+ for (let char of password) {
+ expect(validCharacters).toContain(char);
+ }
+ });
+});
diff --git a/server/__test__/test_env_example.js b/server/__test__/test_env_example.js
new file mode 100644
index 0000000..237eac2
--- /dev/null
+++ b/server/__test__/test_env_example.js
@@ -0,0 +1,3 @@
+const MOCK_GOOGLE_TOKEN = ""
+
+module.exports = MOCK_GOOGLE_TOKEN
\ No newline at end of file
diff --git a/server/app.js b/server/app.js
new file mode 100644
index 0000000..f1178aa
--- /dev/null
+++ b/server/app.js
@@ -0,0 +1,22 @@
+if (process.env.NODE_ENV !== "production") {
+ require('dotenv').config()
+}
+
+const errorHandler = require('./middleware/errorHandler')
+const apiRoute = require('./routes/api')
+const express = require('express')
+const cors = require('cors');
+
+const app = express()
+
+// Enable CORS for all domains (bisa juga untuk domain/origin tertentu)
+app.use(cors());
+
+app.use(express.urlencoded({ extended: false }))
+app.use(express.json())
+
+app.use('/', apiRoute)
+
+app.use(errorHandler)
+
+module.exports = app
\ No newline at end of file
diff --git a/server/bin/www b/server/bin/www
new file mode 100644
index 0000000..f4cf6e4
--- /dev/null
+++ b/server/bin/www
@@ -0,0 +1,6 @@
+const app = require('../app')
+
+const port = process.env.PORT || 3000
+app.listen(port, () => {
+ console.log(`Server app listening on port ${port}`)
+})
diff --git a/server/config/config.json b/server/config/config.json
new file mode 100644
index 0000000..e1934a4
--- /dev/null
+++ b/server/config/config.json
@@ -0,0 +1,19 @@
+{
+ "development": {
+ "username": "postgres",
+ "password": "postgres",
+ "database": "ip_rmt59",
+ "host": "localhost",
+ "dialect": "postgres"
+ },
+ "test": {
+ "username": "postgres",
+ "password": "postgres",
+ "database": "ip_rmt59_test",
+ "host": "localhost",
+ "dialect": "postgres"
+ },
+ "production": {
+ "use_env_variable": "DATABASE_URL"
+ }
+}
\ No newline at end of file
diff --git a/server/controllers/AIController.js b/server/controllers/AIController.js
new file mode 100644
index 0000000..b40f10a
--- /dev/null
+++ b/server/controllers/AIController.js
@@ -0,0 +1,108 @@
+const { GoogleGenAI } = require("@google/genai");
+const ai = new GoogleGenAI({ apiKey: process.env.GEMINI_API_KEY });
+
+class AIController {
+ static async romajiToKanji(req, res, next) {
+ try {
+ const { romaji } = req.body
+
+ if (!romaji) {
+ return res.status(400).json({
+ message: "Inputan tidak boleh kosong"
+ })
+ }
+
+ const response = await ai.models.generateContent({
+ model: "gemini-2.0-flash",
+ contents: `berikan transliterasi untuk romaji '${romaji}' ke bentuk kanjinya. berikan response hanya kanjinya saja. jika yang diinputkan bukan romaji maka berikan repsponse huruf itu sendiri`,
+ });
+
+ res.json({
+ transliteration: response.text.replace(/\n/g, '')
+ })
+ console.log(response)
+ } catch (error) {
+ next(error)
+ }
+ }
+
+ static async kanjiToRomaji(req, res, next) {
+ try {
+ const { kanji } = req.body
+
+ if (!kanji) {
+ return res.status(400).json({
+ message: "Inputan tidak boleh kosong"
+ })
+ }
+
+ const response = await ai.models.generateContent({
+ model: "gemini-2.0-flash",
+ contents: `berikan transliterasi untuk kanji '${kanji}' ke bentuk romajinya. berikan response hanya romajinya saja. jika yang diinputkan bukan kanji maka berikan response '400:badrequest'`,
+ });
+
+ if (response.text.replace(/\n/g, '') === "400:badrequest") {
+ return res.status(400).json({
+ message: "Inputan bukan kanji"
+ })
+ }
+
+ res.json({
+ transliteration: response.text.replace(/\n/g, '')
+ })
+ console.log(response)
+ } catch (error) {
+ next(error)
+ }
+ }
+
+ // translate from indonesia to japanese
+ static async translateToJapanese(req, res, next) {
+ try {
+ const { text } = req.body
+
+ if (!text) {
+ return res.status(400).json({
+ message: "Inputan tidak boleh kosong"
+ })
+ }
+
+ const response = await ai.models.generateContent({
+ model: "gemini-2.0-flash",
+ contents: `terjemahkan '${text}' ke bahasa jepang. berikan response hanya terjemahannya saja`,
+ });
+ res.json({
+ translation: response.text.replace(/\n/g, '')
+ })
+ console.log(response)
+ } catch (error) {
+ next(error)
+ }
+ }
+
+ // translate from japanese to indonesia
+ static async translateToIndonesia(req, res, next) {
+ try {
+ const { text } = req.body
+
+ if (!text) {
+ return res.status(400).json({
+ message: "Inputan tidak boleh kosong"
+ })
+ }
+
+ const response = await ai.models.generateContent({
+ model: "gemini-2.0-flash",
+ contents: `terjemahkan '${text}' ke bahasa indonesia. berikan response hanya terjemahannya saja`,
+ });
+ res.json({
+ translation: response.text.replace(/\n/g, '')
+ })
+ console.log(response)
+ } catch (error) {
+ next(error)
+ }
+ }
+}
+
+module.exports = AIController
\ No newline at end of file
diff --git a/server/controllers/CourseController.js b/server/controllers/CourseController.js
new file mode 100644
index 0000000..ba1eb83
--- /dev/null
+++ b/server/controllers/CourseController.js
@@ -0,0 +1,36 @@
+const { Course } = require('../models/index')
+class CourseController {
+ static async show(req, res, next) {
+ try {
+ const course = await Course.findByPk(req.params.id)
+
+ if (!course) {
+ throw {
+ name: "NotFound",
+ message: "error not found"
+ }
+ }
+
+ res.status(200).json({
+ message: "course retrieved successfully",
+ data: course
+ })
+
+ } catch (error) {
+ next(error);
+ }
+ }
+ static async index(req, res, next) {
+ try {
+ const courses = await Course.findAll()
+ res.status(200).json({
+ message: "courses retrieved successfully",
+ data: courses
+ })
+ } catch (error) {
+ next(error);
+ }
+ }
+}
+
+module.exports = CourseController
\ No newline at end of file
diff --git a/server/controllers/MaterialController.js b/server/controllers/MaterialController.js
new file mode 100644
index 0000000..864a5d7
--- /dev/null
+++ b/server/controllers/MaterialController.js
@@ -0,0 +1,25 @@
+const { Material } = require('../models/index');
+class MaterialController {
+ static async index(req, res, next) {
+ {
+ try {
+ // find materials by course id
+ const materials = await Material.findAll({
+ where: {
+ CourseId: req.params.courseId
+ }
+ });
+
+ res.status(200).json({
+ message: "materials retrieved successfully",
+ data: materials
+ });
+
+ } catch (error) {
+ next(error);
+ }
+ }
+ }
+}
+
+module.exports = MaterialController;
\ No newline at end of file
diff --git a/server/controllers/UserController.js b/server/controllers/UserController.js
new file mode 100644
index 0000000..2a9d2b2
--- /dev/null
+++ b/server/controllers/UserController.js
@@ -0,0 +1,196 @@
+const { User } = require('../models/index')
+const { comparePassword, hashingPassword } = require('../helpers/bcrypt')
+const { signToken, verifyToken } = require('../helpers/jwt')
+const { verify } = require('jsonwebtoken')
+const emailHelper = require('../helpers/emailHelper')
+const { OAuth2Client } = require('google-auth-library');
+const randomPasswordGenerator = require('../helpers/randomPasswordGenerator')
+
+class UserController {
+ static async login(req, res, next) {
+ try {
+ const { email, password } = req.body
+ if (!email) {
+ throw {
+ name: "BadRequest",
+ message: "email is required"
+ }
+ }
+
+ if (!password) {
+ throw {
+ name: "BadRequest",
+ message: "password is required"
+ }
+ }
+
+ const user = await User.findOne({
+ where: {
+ email: email
+ }
+ })
+
+ if (!user) {
+ throw {
+ name: "Unauthorized",
+ message: "email or password is invalid"
+ }
+ }
+
+ const isValidPassword = await comparePassword(password, user.password)
+ if (!isValidPassword) {
+ throw {
+ name: "Unauthorized",
+ message: "email, or password is invalid"
+ }
+ }
+
+ if (!user.emailVerifiedAt) {
+ throw {
+ name: "Unauthorized",
+ message: `Please verify your email before logging in`
+ }
+ }
+
+ const { id } = user
+ const bearerToken = await signToken({
+ id: id
+ })
+
+ res.status(200).json({
+ message: `login success`,
+ access_token: bearerToken,
+ data: {
+ id: id
+ }
+ })
+ } catch (error) {
+ next(error)
+ }
+ }
+
+ static async register(req, res, next) {
+ try {
+ const { email, password } = req.body
+ if (!email) {
+ throw {
+ name: "BadRequest",
+ message: "email is required"
+ }
+ }
+
+ if (!password) {
+ throw {
+ name: "BadRequest",
+ message: "password is required"
+ }
+ }
+
+ const newUser = await User.create({
+ email: email,
+ password: await hashingPassword(password),
+ verifiedAt: null
+ })
+
+ // Generate verification token
+ const verificationToken = signToken({ id: newUser.id }, '1h');
+ const verificationLink = `${process.env.BASE_URL}/verify-email?token=${verificationToken}`;
+
+ // Send verification email
+ await emailHelper.transporter.sendMail({
+ from: process.env.EMAIL_USER,
+ to: newUser.email,
+ subject: 'Verify Your Email',
+ html: emailHelper.emailPageTemplate(verificationLink)
+ });
+
+ // remove password from response
+ delete newUser.dataValues.password
+
+ res.status(201).json({
+ message: "register success",
+ data: newUser
+ })
+ } catch (error) {
+ next(error)
+ }
+ }
+
+ static async verifyEmail(req, res, next) {
+ try {
+ const { token } = req.query;
+ if (!token) throw { name: "BadRequest", message: "Token is required" };
+ const decoded = verifyToken(token)
+ const user = await User.findByPk(decoded.id);
+
+ if (!user) throw { name: "NotFound", message: "User not found" };
+ if (user.emailVerifiedAt) return res.status(200).json({ message: "Email already verified" });
+
+ await User.update(
+ { emailVerifiedAt: new Date() },
+ { where: { id: user.id } }
+ );
+ res.status(200).json({ message: "Email verification successful" });
+ } catch (error) {
+ next(error);
+ }
+ }
+
+ static async googleLogin(req, res, next) {
+ try {
+ const { googleToken } = req.body
+
+ if (!googleToken) {
+ throw {
+ name: "BadRequest",
+ message: "googleToken is required"
+ }
+ }
+
+ const client = new OAuth2Client();
+ const ticket = await client.verifyIdToken({
+ idToken: googleToken,
+ audience: process.env.WEB_CLIENT_ID,
+ });
+ const payload = ticket.getPayload();
+
+ const [user, created] = await User.findOrCreate({
+ where: { email: payload.email },
+ defaults: {
+ email: payload.email,
+ password: await hashingPassword(randomPasswordGenerator()),
+ emailVerifiedAt: new Date(),
+ role: "User"
+ },
+ });
+
+ // jika dia sudah membuat akun tetapi email belum terverifikasi
+ // jika dia login dengan google maka otomatis email terverifikasi
+ if (!user.emailVerifiedAt) {
+ await User.update(
+ { emailVerifiedAt: new Date() },
+ { where: { id: user.id } }
+ );
+ }
+
+ const bearerToken = await signToken({
+ id: user.id,
+ email: user.email,
+ role: user.role
+ })
+
+ res.json({
+ message: `login success`,
+ access_token: bearerToken,
+ data: {
+ id: user.id
+ }
+ })
+ } catch (error) {
+ next(error)
+ }
+ }
+
+}
+
+module.exports = UserController
\ No newline at end of file
diff --git a/server/controllers/UserCourseController.js b/server/controllers/UserCourseController.js
new file mode 100644
index 0000000..07805a4
--- /dev/null
+++ b/server/controllers/UserCourseController.js
@@ -0,0 +1,195 @@
+const { Op } = require('sequelize');
+const { User, Course, UserCourse } = require('../models/index');
+
+class UserCourseController {
+ // enroll
+ static async store(req, res, next) {
+ try {
+ const { userId, courseId } = req.body;
+
+ if (!courseId) {
+ throw {
+ name: "BadRequest",
+ message: "courseId is required"
+ }
+ }
+
+ const user = await User.findByPk(userId);
+ if (!user) {
+ throw {
+ name: "NotFound",
+ message: "error not found"
+ }
+ }
+
+ // if emailVerifiedAt is null, then the user is not verified
+ if (!user.emailVerifiedAt) {
+ throw {
+ name: "Unauthorized",
+ message: "You need to verify your email first"
+ }
+ }
+
+ const course = await Course.findByPk(courseId);
+ if (!course) {
+ throw {
+ name: "NotFound",
+ message: "error not found"
+ }
+ }
+
+ // if user already enrolled in the course
+ const userCourses = await UserCourse.findOne({
+ where: {
+ [Op.and]: [{ UserId: userId }, { CourseId: courseId }]
+ }
+ });
+ if (userCourses) {
+ throw {
+ name: "BadRequest",
+ message: "You already enrolled in this course"
+ }
+ }
+
+ const userCourse = await UserCourse.create({
+ "UserId": userId,
+ "CourseId": courseId
+
+ });
+ return res.json(userCourse);
+ } catch (error) {
+ next(error)
+ }
+ }
+
+ // unenroll
+ static async delete(req, res, next) {
+ try {
+ const currentUserId = req.user.id;
+ const { userId, courseId } = req.body;
+
+ if (!courseId) {
+ throw {
+ name: "BadRequest",
+ message: "courseId is required"
+ }
+ }
+
+ const user = await User.findByPk(userId);
+ if (!user) {
+ throw {
+ name: "NotFound",
+ message: "error not found"
+ }
+ }
+
+ const course = await Course.findByPk(courseId);
+ if (!course) {
+ throw {
+ name: "NotFound",
+ message: "error not found"
+ }
+ }
+
+ const userCourses = await UserCourse.findOne({
+ where: {
+ [Op.and]: [{ UserId: userId }, { CourseId: courseId }]
+ }
+ });
+ if (!userCourses) {
+ throw {
+ name: "BadRequest",
+ message: "You are not enrolled in this course"
+ }
+ }
+
+ // validation if user is unenrolling from a course that is not his/her own
+ if (userCourses.UserId !== currentUserId) {
+ throw {
+ name: "Unauthorized",
+ message: "You are not authorized to unenroll from this course"
+ }
+ }
+
+ await UserCourse.destroy({
+ where: {
+ [Op.and]: [{ UserId: userId }, { CourseId: courseId }]
+ }
+ });
+ return res.status(200).json({ message: "Course unenrolled successfully" });
+ } catch (error) {
+ next(error);
+ }
+ }
+
+ static async index(req, res, next) {
+ try {
+ const user = req.user
+ const query = {
+ where: {
+ id: user.id
+ },
+ include: {
+ model: Course
+ }
+ }
+ const userCourses = await User.findOne(
+ query
+ )
+ return res.json(userCourses.Courses);
+ } catch (error) {
+ next(error);
+ }
+ }
+
+ // update favorite
+ static async updateFavorite(req, res, next) {
+ try {
+ const { userId, courseId } = req.body
+
+ if (!userId) {
+ throw {
+ name: "BadRequest",
+ message: "userId is required"
+ }
+ }
+
+ if (!courseId) {
+ throw {
+ name: "BadRequest",
+ message: "courseId is required"
+ }
+ }
+
+ const userCourse = await UserCourse.findOne({
+ where: {
+ [Op.and]: [{ UserId: userId }, { CourseId: courseId }]
+ }
+ });
+
+ if (!userCourse) {
+ throw {
+ name: "NotFound",
+ message: "error not found"
+ }
+ }
+
+ // if userCourse favorite is null or false
+ if (!userCourse.favorite) {
+ await userCourse.update({
+ favorite: true
+ });
+ return res.json({ message: "Course added to favorite" });
+ } else {
+ await userCourse.update({
+ favorite: false
+ });
+ return res.json({ message: "Course removed from favorite" });
+ }
+ } catch (error) {
+ next(error);
+ }
+ }
+}
+
+module.exports = UserCourseController;
\ No newline at end of file
diff --git a/server/coverage/clover.xml b/server/coverage/clover.xml
new file mode 100644
index 0000000..14f5739
--- /dev/null
+++ b/server/coverage/clover.xml
@@ -0,0 +1,411 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/server/coverage/coverage-final.json b/server/coverage/coverage-final.json
new file mode 100644
index 0000000..e0acd37
--- /dev/null
+++ b/server/coverage/coverage-final.json
@@ -0,0 +1,21 @@
+{"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\app.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\app.js","statementMap":{"0":{"start":{"line":1,"column":0},"end":{"line":3,"column":1}},"1":{"start":{"line":2,"column":4},"end":{"line":2,"column":30}},"2":{"start":{"line":5,"column":21},"end":{"line":5,"column":57}},"3":{"start":{"line":6,"column":17},"end":{"line":6,"column":40}},"4":{"start":{"line":7,"column":16},"end":{"line":7,"column":34}},"5":{"start":{"line":8,"column":13},"end":{"line":8,"column":28}},"6":{"start":{"line":10,"column":12},"end":{"line":10,"column":21}},"7":{"start":{"line":13,"column":0},"end":{"line":13,"column":16}},"8":{"start":{"line":15,"column":0},"end":{"line":15,"column":48}},"9":{"start":{"line":16,"column":0},"end":{"line":16,"column":23}},"10":{"start":{"line":18,"column":0},"end":{"line":18,"column":22}},"11":{"start":{"line":20,"column":0},"end":{"line":20,"column":21}},"12":{"start":{"line":22,"column":0},"end":{"line":22,"column":20}}},"fnMap":{},"branchMap":{"0":{"loc":{"start":{"line":1,"column":0},"end":{"line":3,"column":1}},"type":"if","locations":[{"start":{"line":1,"column":0},"end":{"line":3,"column":1}},{"start":{},"end":{}}],"line":1}},"s":{"0":6,"1":6,"2":6,"3":6,"4":6,"5":6,"6":6,"7":6,"8":6,"9":6,"10":6,"11":6,"12":6},"f":{},"b":{"0":[6,0]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"a25ae1348dbdd38e5698428d36d4a83accb648a5"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\__test__\\test_env.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\__test__\\test_env.js","statementMap":{"0":{"start":{"line":1,"column":26},"end":{"line":1,"column":1226}},"1":{"start":{"line":3,"column":0},"end":{"line":3,"column":34}}},"fnMap":{},"branchMap":{},"s":{"0":1,"1":1},"f":{},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"c2a5c0d5ea87e266c1fa8fb466ed24d9e05ee2f6"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\AIController.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\AIController.js","statementMap":{"0":{"start":{"line":1,"column":24},"end":{"line":1,"column":48}},"1":{"start":{"line":2,"column":11},"end":{"line":2,"column":66}},"2":{"start":{"line":6,"column":8},"end":{"line":26,"column":9}},"3":{"start":{"line":7,"column":31},"end":{"line":7,"column":39}},"4":{"start":{"line":9,"column":12},"end":{"line":13,"column":13}},"5":{"start":{"line":10,"column":16},"end":{"line":12,"column":18}},"6":{"start":{"line":15,"column":29},"end":{"line":18,"column":14}},"7":{"start":{"line":20,"column":12},"end":{"line":22,"column":14}},"8":{"start":{"line":23,"column":12},"end":{"line":23,"column":33}},"9":{"start":{"line":25,"column":12},"end":{"line":25,"column":23}},"10":{"start":{"line":30,"column":8},"end":{"line":56,"column":9}},"11":{"start":{"line":31,"column":30},"end":{"line":31,"column":38}},"12":{"start":{"line":33,"column":12},"end":{"line":37,"column":13}},"13":{"start":{"line":34,"column":16},"end":{"line":36,"column":18}},"14":{"start":{"line":39,"column":29},"end":{"line":42,"column":14}},"15":{"start":{"line":44,"column":12},"end":{"line":48,"column":13}},"16":{"start":{"line":45,"column":16},"end":{"line":47,"column":18}},"17":{"start":{"line":50,"column":12},"end":{"line":52,"column":14}},"18":{"start":{"line":53,"column":12},"end":{"line":53,"column":33}},"19":{"start":{"line":55,"column":12},"end":{"line":55,"column":23}},"20":{"start":{"line":61,"column":8},"end":{"line":80,"column":9}},"21":{"start":{"line":62,"column":29},"end":{"line":62,"column":37}},"22":{"start":{"line":64,"column":12},"end":{"line":68,"column":13}},"23":{"start":{"line":65,"column":16},"end":{"line":67,"column":18}},"24":{"start":{"line":70,"column":29},"end":{"line":73,"column":14}},"25":{"start":{"line":74,"column":12},"end":{"line":76,"column":14}},"26":{"start":{"line":77,"column":12},"end":{"line":77,"column":33}},"27":{"start":{"line":79,"column":12},"end":{"line":79,"column":23}},"28":{"start":{"line":85,"column":8},"end":{"line":104,"column":9}},"29":{"start":{"line":86,"column":29},"end":{"line":86,"column":37}},"30":{"start":{"line":88,"column":12},"end":{"line":92,"column":13}},"31":{"start":{"line":89,"column":16},"end":{"line":91,"column":18}},"32":{"start":{"line":94,"column":29},"end":{"line":97,"column":14}},"33":{"start":{"line":98,"column":12},"end":{"line":100,"column":14}},"34":{"start":{"line":101,"column":12},"end":{"line":101,"column":33}},"35":{"start":{"line":103,"column":12},"end":{"line":103,"column":23}},"36":{"start":{"line":108,"column":0},"end":{"line":108,"column":29}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":5,"column":4},"end":{"line":5,"column":5}},"loc":{"start":{"line":5,"column":47},"end":{"line":27,"column":5}},"line":5},"1":{"name":"(anonymous_1)","decl":{"start":{"line":29,"column":4},"end":{"line":29,"column":5}},"loc":{"start":{"line":29,"column":47},"end":{"line":57,"column":5}},"line":29},"2":{"name":"(anonymous_2)","decl":{"start":{"line":60,"column":4},"end":{"line":60,"column":5}},"loc":{"start":{"line":60,"column":53},"end":{"line":81,"column":5}},"line":60},"3":{"name":"(anonymous_3)","decl":{"start":{"line":84,"column":4},"end":{"line":84,"column":5}},"loc":{"start":{"line":84,"column":54},"end":{"line":105,"column":5}},"line":84}},"branchMap":{"0":{"loc":{"start":{"line":9,"column":12},"end":{"line":13,"column":13}},"type":"if","locations":[{"start":{"line":9,"column":12},"end":{"line":13,"column":13}},{"start":{},"end":{}}],"line":9},"1":{"loc":{"start":{"line":33,"column":12},"end":{"line":37,"column":13}},"type":"if","locations":[{"start":{"line":33,"column":12},"end":{"line":37,"column":13}},{"start":{},"end":{}}],"line":33},"2":{"loc":{"start":{"line":44,"column":12},"end":{"line":48,"column":13}},"type":"if","locations":[{"start":{"line":44,"column":12},"end":{"line":48,"column":13}},{"start":{},"end":{}}],"line":44},"3":{"loc":{"start":{"line":64,"column":12},"end":{"line":68,"column":13}},"type":"if","locations":[{"start":{"line":64,"column":12},"end":{"line":68,"column":13}},{"start":{},"end":{}}],"line":64},"4":{"loc":{"start":{"line":88,"column":12},"end":{"line":92,"column":13}},"type":"if","locations":[{"start":{"line":88,"column":12},"end":{"line":92,"column":13}},{"start":{},"end":{}}],"line":88}},"s":{"0":6,"1":6,"2":2,"3":2,"4":2,"5":1,"6":1,"7":1,"8":1,"9":0,"10":3,"11":3,"12":3,"13":1,"14":2,"15":2,"16":1,"17":1,"18":1,"19":0,"20":1,"21":1,"22":1,"23":0,"24":1,"25":1,"26":1,"27":0,"28":1,"29":1,"30":1,"31":0,"32":1,"33":1,"34":1,"35":0,"36":6},"f":{"0":2,"1":3,"2":1,"3":1},"b":{"0":[1,1],"1":[1,2],"2":[1,1],"3":[0,1],"4":[0,1]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"209ef097ea4e3202edba651ee34c66071178a338"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\CourseController.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\CourseController.js","statementMap":{"0":{"start":{"line":1,"column":19},"end":{"line":1,"column":45}},"1":{"start":{"line":4,"column":8},"end":{"line":21,"column":9}},"2":{"start":{"line":5,"column":27},"end":{"line":5,"column":63}},"3":{"start":{"line":7,"column":12},"end":{"line":12,"column":13}},"4":{"start":{"line":8,"column":16},"end":{"line":11,"column":17}},"5":{"start":{"line":14,"column":12},"end":{"line":17,"column":14}},"6":{"start":{"line":20,"column":12},"end":{"line":20,"column":24}},"7":{"start":{"line":24,"column":8},"end":{"line":32,"column":9}},"8":{"start":{"line":25,"column":28},"end":{"line":25,"column":50}},"9":{"start":{"line":26,"column":12},"end":{"line":29,"column":14}},"10":{"start":{"line":31,"column":12},"end":{"line":31,"column":24}},"11":{"start":{"line":36,"column":0},"end":{"line":36,"column":33}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":3,"column":4},"end":{"line":3,"column":5}},"loc":{"start":{"line":3,"column":38},"end":{"line":22,"column":5}},"line":3},"1":{"name":"(anonymous_1)","decl":{"start":{"line":23,"column":4},"end":{"line":23,"column":5}},"loc":{"start":{"line":23,"column":39},"end":{"line":33,"column":5}},"line":23}},"branchMap":{"0":{"loc":{"start":{"line":7,"column":12},"end":{"line":12,"column":13}},"type":"if","locations":[{"start":{"line":7,"column":12},"end":{"line":12,"column":13}},{"start":{},"end":{}}],"line":7}},"s":{"0":6,"1":2,"2":2,"3":2,"4":1,"5":1,"6":1,"7":1,"8":1,"9":1,"10":0,"11":6},"f":{"0":2,"1":1},"b":{"0":[1,1]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"cf62e065453062d8c8e6e388973c8b8573c5752a"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\MaterialController.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\MaterialController.js","statementMap":{"0":{"start":{"line":1,"column":21},"end":{"line":1,"column":47}},"1":{"start":{"line":5,"column":12},"end":{"line":20,"column":13}},"2":{"start":{"line":7,"column":34},"end":{"line":11,"column":18}},"3":{"start":{"line":13,"column":16},"end":{"line":16,"column":19}},"4":{"start":{"line":19,"column":16},"end":{"line":19,"column":28}},"5":{"start":{"line":25,"column":0},"end":{"line":25,"column":36}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":3,"column":4},"end":{"line":3,"column":5}},"loc":{"start":{"line":3,"column":39},"end":{"line":22,"column":5}},"line":3}},"branchMap":{},"s":{"0":6,"1":1,"2":1,"3":1,"4":0,"5":6},"f":{"0":1},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"b52e3a938526ab5a1cc2a1991fb06fe02f1809ba"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\UserController.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\UserController.js","statementMap":{"0":{"start":{"line":1,"column":17},"end":{"line":1,"column":43}},"1":{"start":{"line":2,"column":45},"end":{"line":2,"column":73}},"2":{"start":{"line":3,"column":35},"end":{"line":3,"column":60}},"3":{"start":{"line":4,"column":19},"end":{"line":4,"column":42}},"4":{"start":{"line":5,"column":20},"end":{"line":5,"column":53}},"5":{"start":{"line":6,"column":25},"end":{"line":6,"column":55}},"6":{"start":{"line":7,"column":32},"end":{"line":7,"column":77}},"7":{"start":{"line":11,"column":8},"end":{"line":69,"column":9}},"8":{"start":{"line":12,"column":40},"end":{"line":12,"column":48}},"9":{"start":{"line":13,"column":12},"end":{"line":18,"column":13}},"10":{"start":{"line":14,"column":16},"end":{"line":17,"column":17}},"11":{"start":{"line":20,"column":12},"end":{"line":25,"column":13}},"12":{"start":{"line":21,"column":16},"end":{"line":24,"column":17}},"13":{"start":{"line":27,"column":25},"end":{"line":31,"column":14}},"14":{"start":{"line":33,"column":12},"end":{"line":38,"column":13}},"15":{"start":{"line":34,"column":16},"end":{"line":37,"column":17}},"16":{"start":{"line":40,"column":36},"end":{"line":40,"column":82}},"17":{"start":{"line":41,"column":12},"end":{"line":46,"column":13}},"18":{"start":{"line":42,"column":16},"end":{"line":45,"column":17}},"19":{"start":{"line":48,"column":12},"end":{"line":53,"column":13}},"20":{"start":{"line":49,"column":16},"end":{"line":52,"column":17}},"21":{"start":{"line":55,"column":27},"end":{"line":55,"column":31}},"22":{"start":{"line":56,"column":32},"end":{"line":58,"column":14}},"23":{"start":{"line":60,"column":12},"end":{"line":66,"column":14}},"24":{"start":{"line":68,"column":12},"end":{"line":68,"column":23}},"25":{"start":{"line":73,"column":8},"end":{"line":116,"column":9}},"26":{"start":{"line":74,"column":40},"end":{"line":74,"column":48}},"27":{"start":{"line":75,"column":12},"end":{"line":80,"column":13}},"28":{"start":{"line":76,"column":16},"end":{"line":79,"column":17}},"29":{"start":{"line":82,"column":12},"end":{"line":87,"column":13}},"30":{"start":{"line":83,"column":16},"end":{"line":86,"column":17}},"31":{"start":{"line":89,"column":28},"end":{"line":93,"column":14}},"32":{"start":{"line":96,"column":38},"end":{"line":96,"column":73}},"33":{"start":{"line":97,"column":37},"end":{"line":97,"column":102}},"34":{"start":{"line":100,"column":12},"end":{"line":105,"column":15}},"35":{"start":{"line":108,"column":12},"end":{"line":108,"column":46}},"36":{"start":{"line":110,"column":12},"end":{"line":113,"column":14}},"37":{"start":{"line":115,"column":12},"end":{"line":115,"column":23}},"38":{"start":{"line":120,"column":8},"end":{"line":136,"column":9}},"39":{"start":{"line":121,"column":30},"end":{"line":121,"column":39}},"40":{"start":{"line":122,"column":12},"end":{"line":122,"column":83}},"41":{"start":{"line":122,"column":24},"end":{"line":122,"column":83}},"42":{"start":{"line":123,"column":28},"end":{"line":123,"column":46}},"43":{"start":{"line":124,"column":25},"end":{"line":124,"column":56}},"44":{"start":{"line":126,"column":12},"end":{"line":126,"column":77}},"45":{"start":{"line":126,"column":23},"end":{"line":126,"column":77}},"46":{"start":{"line":127,"column":12},"end":{"line":127,"column":105}},"47":{"start":{"line":127,"column":38},"end":{"line":127,"column":105}},"48":{"start":{"line":129,"column":12},"end":{"line":132,"column":14}},"49":{"start":{"line":133,"column":12},"end":{"line":133,"column":79}},"50":{"start":{"line":135,"column":12},"end":{"line":135,"column":24}},"51":{"start":{"line":140,"column":8},"end":{"line":191,"column":9}},"52":{"start":{"line":141,"column":36},"end":{"line":141,"column":44}},"53":{"start":{"line":143,"column":12},"end":{"line":148,"column":13}},"54":{"start":{"line":144,"column":16},"end":{"line":147,"column":17}},"55":{"start":{"line":150,"column":27},"end":{"line":150,"column":45}},"56":{"start":{"line":151,"column":27},"end":{"line":154,"column":14}},"57":{"start":{"line":155,"column":28},"end":{"line":155,"column":47}},"58":{"start":{"line":157,"column":36},"end":{"line":165,"column":14}},"59":{"start":{"line":169,"column":12},"end":{"line":174,"column":13}},"60":{"start":{"line":170,"column":16},"end":{"line":173,"column":18}},"61":{"start":{"line":176,"column":32},"end":{"line":180,"column":14}},"62":{"start":{"line":182,"column":12},"end":{"line":188,"column":14}},"63":{"start":{"line":190,"column":12},"end":{"line":190,"column":23}},"64":{"start":{"line":196,"column":0},"end":{"line":196,"column":31}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":10,"column":4},"end":{"line":10,"column":5}},"loc":{"start":{"line":10,"column":39},"end":{"line":70,"column":5}},"line":10},"1":{"name":"(anonymous_1)","decl":{"start":{"line":72,"column":4},"end":{"line":72,"column":5}},"loc":{"start":{"line":72,"column":42},"end":{"line":117,"column":5}},"line":72},"2":{"name":"(anonymous_2)","decl":{"start":{"line":119,"column":4},"end":{"line":119,"column":5}},"loc":{"start":{"line":119,"column":45},"end":{"line":137,"column":5}},"line":119},"3":{"name":"(anonymous_3)","decl":{"start":{"line":139,"column":4},"end":{"line":139,"column":5}},"loc":{"start":{"line":139,"column":45},"end":{"line":192,"column":5}},"line":139}},"branchMap":{"0":{"loc":{"start":{"line":13,"column":12},"end":{"line":18,"column":13}},"type":"if","locations":[{"start":{"line":13,"column":12},"end":{"line":18,"column":13}},{"start":{},"end":{}}],"line":13},"1":{"loc":{"start":{"line":20,"column":12},"end":{"line":25,"column":13}},"type":"if","locations":[{"start":{"line":20,"column":12},"end":{"line":25,"column":13}},{"start":{},"end":{}}],"line":20},"2":{"loc":{"start":{"line":33,"column":12},"end":{"line":38,"column":13}},"type":"if","locations":[{"start":{"line":33,"column":12},"end":{"line":38,"column":13}},{"start":{},"end":{}}],"line":33},"3":{"loc":{"start":{"line":41,"column":12},"end":{"line":46,"column":13}},"type":"if","locations":[{"start":{"line":41,"column":12},"end":{"line":46,"column":13}},{"start":{},"end":{}}],"line":41},"4":{"loc":{"start":{"line":48,"column":12},"end":{"line":53,"column":13}},"type":"if","locations":[{"start":{"line":48,"column":12},"end":{"line":53,"column":13}},{"start":{},"end":{}}],"line":48},"5":{"loc":{"start":{"line":75,"column":12},"end":{"line":80,"column":13}},"type":"if","locations":[{"start":{"line":75,"column":12},"end":{"line":80,"column":13}},{"start":{},"end":{}}],"line":75},"6":{"loc":{"start":{"line":82,"column":12},"end":{"line":87,"column":13}},"type":"if","locations":[{"start":{"line":82,"column":12},"end":{"line":87,"column":13}},{"start":{},"end":{}}],"line":82},"7":{"loc":{"start":{"line":122,"column":12},"end":{"line":122,"column":83}},"type":"if","locations":[{"start":{"line":122,"column":12},"end":{"line":122,"column":83}},{"start":{},"end":{}}],"line":122},"8":{"loc":{"start":{"line":126,"column":12},"end":{"line":126,"column":77}},"type":"if","locations":[{"start":{"line":126,"column":12},"end":{"line":126,"column":77}},{"start":{},"end":{}}],"line":126},"9":{"loc":{"start":{"line":127,"column":12},"end":{"line":127,"column":105}},"type":"if","locations":[{"start":{"line":127,"column":12},"end":{"line":127,"column":105}},{"start":{},"end":{}}],"line":127},"10":{"loc":{"start":{"line":143,"column":12},"end":{"line":148,"column":13}},"type":"if","locations":[{"start":{"line":143,"column":12},"end":{"line":148,"column":13}},{"start":{},"end":{}}],"line":143},"11":{"loc":{"start":{"line":169,"column":12},"end":{"line":174,"column":13}},"type":"if","locations":[{"start":{"line":169,"column":12},"end":{"line":174,"column":13}},{"start":{},"end":{}}],"line":169}},"s":{"0":6,"1":6,"2":6,"3":6,"4":6,"5":6,"6":6,"7":5,"8":5,"9":5,"10":1,"11":4,"12":1,"13":3,"14":3,"15":2,"16":1,"17":1,"18":0,"19":1,"20":0,"21":1,"22":1,"23":1,"24":4,"25":3,"26":3,"27":3,"28":1,"29":2,"30":1,"31":1,"32":1,"33":1,"34":1,"35":1,"36":1,"37":2,"38":3,"39":3,"40":3,"41":1,"42":2,"43":1,"44":1,"45":0,"46":1,"47":0,"48":1,"49":1,"50":2,"51":2,"52":2,"53":2,"54":1,"55":1,"56":1,"57":1,"58":1,"59":1,"60":0,"61":1,"62":1,"63":1,"64":6},"f":{"0":5,"1":3,"2":3,"3":2},"b":{"0":[1,4],"1":[1,3],"2":[2,1],"3":[0,1],"4":[0,1],"5":[1,2],"6":[1,1],"7":[1,2],"8":[0,1],"9":[0,1],"10":[1,1],"11":[0,1]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"f90771fcea6af55a7988ec1f0259329418142680"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\UserCourseController.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\controllers\\UserCourseController.js","statementMap":{"0":{"start":{"line":1,"column":15},"end":{"line":1,"column":35}},"1":{"start":{"line":2,"column":37},"end":{"line":2,"column":63}},"2":{"start":{"line":7,"column":8},"end":{"line":62,"column":9}},"3":{"start":{"line":8,"column":41},"end":{"line":8,"column":49}},"4":{"start":{"line":10,"column":12},"end":{"line":15,"column":13}},"5":{"start":{"line":11,"column":16},"end":{"line":14,"column":17}},"6":{"start":{"line":17,"column":25},"end":{"line":17,"column":52}},"7":{"start":{"line":18,"column":12},"end":{"line":23,"column":13}},"8":{"start":{"line":19,"column":16},"end":{"line":22,"column":17}},"9":{"start":{"line":26,"column":12},"end":{"line":31,"column":13}},"10":{"start":{"line":27,"column":16},"end":{"line":30,"column":17}},"11":{"start":{"line":33,"column":27},"end":{"line":33,"column":58}},"12":{"start":{"line":34,"column":12},"end":{"line":39,"column":13}},"13":{"start":{"line":35,"column":16},"end":{"line":38,"column":17}},"14":{"start":{"line":42,"column":32},"end":{"line":46,"column":14}},"15":{"start":{"line":47,"column":12},"end":{"line":52,"column":13}},"16":{"start":{"line":48,"column":16},"end":{"line":51,"column":17}},"17":{"start":{"line":54,"column":31},"end":{"line":58,"column":14}},"18":{"start":{"line":59,"column":12},"end":{"line":59,"column":40}},"19":{"start":{"line":61,"column":12},"end":{"line":61,"column":23}},"20":{"start":{"line":67,"column":8},"end":{"line":122,"column":9}},"21":{"start":{"line":68,"column":34},"end":{"line":68,"column":45}},"22":{"start":{"line":69,"column":41},"end":{"line":69,"column":49}},"23":{"start":{"line":71,"column":12},"end":{"line":76,"column":13}},"24":{"start":{"line":72,"column":16},"end":{"line":75,"column":17}},"25":{"start":{"line":78,"column":25},"end":{"line":78,"column":52}},"26":{"start":{"line":79,"column":12},"end":{"line":84,"column":13}},"27":{"start":{"line":80,"column":16},"end":{"line":83,"column":17}},"28":{"start":{"line":86,"column":27},"end":{"line":86,"column":58}},"29":{"start":{"line":87,"column":12},"end":{"line":92,"column":13}},"30":{"start":{"line":88,"column":16},"end":{"line":91,"column":17}},"31":{"start":{"line":94,"column":32},"end":{"line":98,"column":14}},"32":{"start":{"line":99,"column":12},"end":{"line":104,"column":13}},"33":{"start":{"line":100,"column":16},"end":{"line":103,"column":17}},"34":{"start":{"line":107,"column":12},"end":{"line":112,"column":13}},"35":{"start":{"line":108,"column":16},"end":{"line":111,"column":17}},"36":{"start":{"line":114,"column":12},"end":{"line":118,"column":15}},"37":{"start":{"line":119,"column":12},"end":{"line":119,"column":87}},"38":{"start":{"line":121,"column":12},"end":{"line":121,"column":24}},"39":{"start":{"line":126,"column":8},"end":{"line":142,"column":9}},"40":{"start":{"line":127,"column":25},"end":{"line":127,"column":33}},"41":{"start":{"line":128,"column":26},"end":{"line":135,"column":13}},"42":{"start":{"line":136,"column":32},"end":{"line":138,"column":13}},"43":{"start":{"line":139,"column":12},"end":{"line":139,"column":49}},"44":{"start":{"line":141,"column":12},"end":{"line":141,"column":24}},"45":{"start":{"line":147,"column":8},"end":{"line":191,"column":9}},"46":{"start":{"line":148,"column":41},"end":{"line":148,"column":49}},"47":{"start":{"line":150,"column":12},"end":{"line":155,"column":13}},"48":{"start":{"line":151,"column":16},"end":{"line":154,"column":17}},"49":{"start":{"line":157,"column":12},"end":{"line":162,"column":13}},"50":{"start":{"line":158,"column":16},"end":{"line":161,"column":17}},"51":{"start":{"line":164,"column":31},"end":{"line":168,"column":14}},"52":{"start":{"line":170,"column":12},"end":{"line":175,"column":13}},"53":{"start":{"line":171,"column":16},"end":{"line":174,"column":17}},"54":{"start":{"line":178,"column":12},"end":{"line":188,"column":13}},"55":{"start":{"line":179,"column":16},"end":{"line":181,"column":19}},"56":{"start":{"line":182,"column":16},"end":{"line":182,"column":73}},"57":{"start":{"line":184,"column":16},"end":{"line":186,"column":19}},"58":{"start":{"line":187,"column":16},"end":{"line":187,"column":77}},"59":{"start":{"line":190,"column":12},"end":{"line":190,"column":24}},"60":{"start":{"line":195,"column":0},"end":{"line":195,"column":38}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":6,"column":4},"end":{"line":6,"column":5}},"loc":{"start":{"line":6,"column":39},"end":{"line":63,"column":5}},"line":6},"1":{"name":"(anonymous_1)","decl":{"start":{"line":66,"column":4},"end":{"line":66,"column":5}},"loc":{"start":{"line":66,"column":40},"end":{"line":123,"column":5}},"line":66},"2":{"name":"(anonymous_2)","decl":{"start":{"line":125,"column":4},"end":{"line":125,"column":5}},"loc":{"start":{"line":125,"column":39},"end":{"line":143,"column":5}},"line":125},"3":{"name":"(anonymous_3)","decl":{"start":{"line":146,"column":4},"end":{"line":146,"column":5}},"loc":{"start":{"line":146,"column":48},"end":{"line":192,"column":5}},"line":146}},"branchMap":{"0":{"loc":{"start":{"line":10,"column":12},"end":{"line":15,"column":13}},"type":"if","locations":[{"start":{"line":10,"column":12},"end":{"line":15,"column":13}},{"start":{},"end":{}}],"line":10},"1":{"loc":{"start":{"line":18,"column":12},"end":{"line":23,"column":13}},"type":"if","locations":[{"start":{"line":18,"column":12},"end":{"line":23,"column":13}},{"start":{},"end":{}}],"line":18},"2":{"loc":{"start":{"line":26,"column":12},"end":{"line":31,"column":13}},"type":"if","locations":[{"start":{"line":26,"column":12},"end":{"line":31,"column":13}},{"start":{},"end":{}}],"line":26},"3":{"loc":{"start":{"line":34,"column":12},"end":{"line":39,"column":13}},"type":"if","locations":[{"start":{"line":34,"column":12},"end":{"line":39,"column":13}},{"start":{},"end":{}}],"line":34},"4":{"loc":{"start":{"line":47,"column":12},"end":{"line":52,"column":13}},"type":"if","locations":[{"start":{"line":47,"column":12},"end":{"line":52,"column":13}},{"start":{},"end":{}}],"line":47},"5":{"loc":{"start":{"line":71,"column":12},"end":{"line":76,"column":13}},"type":"if","locations":[{"start":{"line":71,"column":12},"end":{"line":76,"column":13}},{"start":{},"end":{}}],"line":71},"6":{"loc":{"start":{"line":79,"column":12},"end":{"line":84,"column":13}},"type":"if","locations":[{"start":{"line":79,"column":12},"end":{"line":84,"column":13}},{"start":{},"end":{}}],"line":79},"7":{"loc":{"start":{"line":87,"column":12},"end":{"line":92,"column":13}},"type":"if","locations":[{"start":{"line":87,"column":12},"end":{"line":92,"column":13}},{"start":{},"end":{}}],"line":87},"8":{"loc":{"start":{"line":99,"column":12},"end":{"line":104,"column":13}},"type":"if","locations":[{"start":{"line":99,"column":12},"end":{"line":104,"column":13}},{"start":{},"end":{}}],"line":99},"9":{"loc":{"start":{"line":107,"column":12},"end":{"line":112,"column":13}},"type":"if","locations":[{"start":{"line":107,"column":12},"end":{"line":112,"column":13}},{"start":{},"end":{}}],"line":107},"10":{"loc":{"start":{"line":150,"column":12},"end":{"line":155,"column":13}},"type":"if","locations":[{"start":{"line":150,"column":12},"end":{"line":155,"column":13}},{"start":{},"end":{}}],"line":150},"11":{"loc":{"start":{"line":157,"column":12},"end":{"line":162,"column":13}},"type":"if","locations":[{"start":{"line":157,"column":12},"end":{"line":162,"column":13}},{"start":{},"end":{}}],"line":157},"12":{"loc":{"start":{"line":170,"column":12},"end":{"line":175,"column":13}},"type":"if","locations":[{"start":{"line":170,"column":12},"end":{"line":175,"column":13}},{"start":{},"end":{}}],"line":170},"13":{"loc":{"start":{"line":178,"column":12},"end":{"line":188,"column":13}},"type":"if","locations":[{"start":{"line":178,"column":12},"end":{"line":188,"column":13}},{"start":{"line":183,"column":19},"end":{"line":188,"column":13}}],"line":178}},"s":{"0":6,"1":6,"2":7,"3":7,"4":7,"5":2,"6":5,"7":5,"8":1,"9":4,"10":1,"11":3,"12":3,"13":1,"14":2,"15":2,"16":1,"17":1,"18":1,"19":6,"20":7,"21":7,"22":7,"23":7,"24":2,"25":5,"26":5,"27":1,"28":4,"29":4,"30":1,"31":3,"32":3,"33":1,"34":2,"35":1,"36":1,"37":1,"38":6,"39":1,"40":1,"41":1,"42":1,"43":1,"44":0,"45":5,"46":5,"47":5,"48":1,"49":4,"50":1,"51":3,"52":3,"53":1,"54":2,"55":1,"56":1,"57":1,"58":1,"59":3,"60":6},"f":{"0":7,"1":7,"2":1,"3":5},"b":{"0":[2,5],"1":[1,4],"2":[1,3],"3":[1,2],"4":[1,1],"5":[2,5],"6":[1,4],"7":[1,3],"8":[1,2],"9":[1,1],"10":[1,4],"11":[1,3],"12":[1,2],"13":[1,1]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"44e34d7183209654082c3dcf270544240adcdf88"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\helpers\\bcrypt.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\helpers\\bcrypt.js","statementMap":{"0":{"start":{"line":1,"column":15},"end":{"line":1,"column":34}},"1":{"start":{"line":4,"column":17},"end":{"line":4,"column":41}},"2":{"start":{"line":5,"column":17},"end":{"line":5,"column":55}},"3":{"start":{"line":6,"column":4},"end":{"line":6,"column":16}},"4":{"start":{"line":10,"column":20},"end":{"line":10,"column":71}},"5":{"start":{"line":11,"column":4},"end":{"line":11,"column":18}},"6":{"start":{"line":14,"column":0},"end":{"line":17,"column":1}}},"fnMap":{"0":{"name":"hashingPassword","decl":{"start":{"line":3,"column":15},"end":{"line":3,"column":30}},"loc":{"start":{"line":3,"column":46},"end":{"line":7,"column":1}},"line":3},"1":{"name":"comparePassword","decl":{"start":{"line":9,"column":15},"end":{"line":9,"column":30}},"loc":{"start":{"line":9,"column":62},"end":{"line":12,"column":1}},"line":9}},"branchMap":{},"s":{"0":6,"1":12,"2":12,"3":12,"4":1,"5":1,"6":6},"f":{"0":12,"1":1},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"236f1f3ab338d6d9c8918393f902d239044c8943"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\helpers\\emailHelper.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\helpers\\emailHelper.js","statementMap":{"0":{"start":{"line":1,"column":19},"end":{"line":1,"column":40}},"1":{"start":{"line":11,"column":20},"end":{"line":20,"column":2}},"2":{"start":{"line":23,"column":2},"end":{"line":68,"column":1}},"3":{"start":{"line":72,"column":0},"end":{"line":72,"column":51}}},"fnMap":{"0":{"name":"emailPageTemplate","decl":{"start":{"line":22,"column":9},"end":{"line":22,"column":26}},"loc":{"start":{"line":22,"column":45},"end":{"line":69,"column":1}},"line":22}},"branchMap":{},"s":{"0":6,"1":6,"2":1,"3":6},"f":{"0":1},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"8af70e41c676bfba2b8abfb431678a51cae2139b"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\helpers\\jwt.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\helpers\\jwt.js","statementMap":{"0":{"start":{"line":1,"column":12},"end":{"line":1,"column":35}},"1":{"start":{"line":2,"column":15},"end":{"line":2,"column":37}},"2":{"start":{"line":5,"column":4},"end":{"line":9,"column":5}},"3":{"start":{"line":6,"column":8},"end":{"line":6,"column":66}},"4":{"start":{"line":8,"column":8},"end":{"line":8,"column":40}},"5":{"start":{"line":13,"column":4},"end":{"line":13,"column":36}},"6":{"start":{"line":16,"column":0},"end":{"line":19,"column":1}}},"fnMap":{"0":{"name":"signToken","decl":{"start":{"line":4,"column":9},"end":{"line":4,"column":18}},"loc":{"start":{"line":4,"column":39},"end":{"line":10,"column":1}},"line":4},"1":{"name":"verifyToken","decl":{"start":{"line":12,"column":9},"end":{"line":12,"column":20}},"loc":{"start":{"line":12,"column":28},"end":{"line":14,"column":1}},"line":12}},"branchMap":{"0":{"loc":{"start":{"line":5,"column":4},"end":{"line":9,"column":5}},"type":"if","locations":[{"start":{"line":5,"column":4},"end":{"line":9,"column":5}},{"start":{"line":7,"column":11},"end":{"line":9,"column":5}}],"line":5}},"s":{"0":7,"1":7,"2":9,"3":2,"4":7,"5":33,"6":7},"f":{"0":9,"1":33},"b":{"0":[2,7]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"db5df2336864ad5859b8e7b00e61a814c1439238"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\helpers\\randomPasswordGenerator.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\helpers\\randomPasswordGenerator.js","statementMap":{"0":{"start":{"line":2,"column":23},"end":{"line":2,"column":111}},"1":{"start":{"line":3,"column":19},"end":{"line":3,"column":21}},"2":{"start":{"line":4,"column":4},"end":{"line":7,"column":5}},"3":{"start":{"line":4,"column":17},"end":{"line":4,"column":18}},"4":{"start":{"line":5,"column":28},"end":{"line":5,"column":73}},"5":{"start":{"line":6,"column":8},"end":{"line":6,"column":51}},"6":{"start":{"line":8,"column":4},"end":{"line":8,"column":20}},"7":{"start":{"line":11,"column":0},"end":{"line":11,"column":41}}},"fnMap":{"0":{"name":"randomPasswordGenerator","decl":{"start":{"line":1,"column":9},"end":{"line":1,"column":32}},"loc":{"start":{"line":1,"column":46},"end":{"line":9,"column":1}},"line":1}},"branchMap":{"0":{"loc":{"start":{"line":1,"column":33},"end":{"line":1,"column":44}},"type":"default-arg","locations":[{"start":{"line":1,"column":42},"end":{"line":1,"column":44}}],"line":1}},"s":{"0":4,"1":4,"2":4,"3":4,"4":52,"5":52,"6":4,"7":7},"f":{"0":4},"b":{"0":[3]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"46a376a295bb668547a959607c8e6b3642b97fbe"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\middleware\\authentication.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\middleware\\authentication.js","statementMap":{"0":{"start":{"line":1,"column":24},"end":{"line":1,"column":49}},"1":{"start":{"line":3,"column":4},"end":{"line":25,"column":5}},"2":{"start":{"line":4,"column":8},"end":{"line":9,"column":9}},"3":{"start":{"line":5,"column":12},"end":{"line":8,"column":13}},"4":{"start":{"line":11,"column":37},"end":{"line":11,"column":62}},"5":{"start":{"line":12,"column":30},"end":{"line":12,"column":61}},"6":{"start":{"line":14,"column":8},"end":{"line":19,"column":9}},"7":{"start":{"line":15,"column":12},"end":{"line":18,"column":13}},"8":{"start":{"line":20,"column":24},"end":{"line":20,"column":42}},"9":{"start":{"line":21,"column":8},"end":{"line":21,"column":26}},"10":{"start":{"line":22,"column":8},"end":{"line":22,"column":14}},"11":{"start":{"line":24,"column":8},"end":{"line":24,"column":19}},"12":{"start":{"line":28,"column":0},"end":{"line":28,"column":31}}},"fnMap":{"0":{"name":"authentication","decl":{"start":{"line":2,"column":9},"end":{"line":2,"column":23}},"loc":{"start":{"line":2,"column":40},"end":{"line":26,"column":1}},"line":2}},"branchMap":{"0":{"loc":{"start":{"line":4,"column":8},"end":{"line":9,"column":9}},"type":"if","locations":[{"start":{"line":4,"column":8},"end":{"line":9,"column":9}},{"start":{},"end":{}}],"line":4},"1":{"loc":{"start":{"line":14,"column":8},"end":{"line":19,"column":9}},"type":"if","locations":[{"start":{"line":14,"column":8},"end":{"line":19,"column":9}},{"start":{},"end":{}}],"line":14},"2":{"loc":{"start":{"line":14,"column":12},"end":{"line":14,"column":52}},"type":"binary-expr","locations":[{"start":{"line":14,"column":12},"end":{"line":14,"column":29}},{"start":{"line":14,"column":33},"end":{"line":14,"column":52}}],"line":14}},"s":{"0":7,"1":36,"2":36,"3":1,"4":35,"5":35,"6":35,"7":2,"8":33,"9":32,"10":32,"11":4,"12":7},"f":{"0":36},"b":{"0":[1,35],"1":[2,33],"2":[35,34]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"f6633e2368f197d7e950a32659eeaff12a1398d9"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\middleware\\authorization.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\middleware\\authorization.js","statementMap":{"0":{"start":{"line":1,"column":17},"end":{"line":1,"column":43}},"1":{"start":{"line":3,"column":4},"end":{"line":34,"column":5}},"2":{"start":{"line":4,"column":8},"end":{"line":33,"column":9}},"3":{"start":{"line":5,"column":25},"end":{"line":5,"column":33}},"4":{"start":{"line":6,"column":12},"end":{"line":11,"column":13}},"5":{"start":{"line":7,"column":16},"end":{"line":10,"column":17}},"6":{"start":{"line":13,"column":12},"end":{"line":28,"column":13}},"7":{"start":{"line":14,"column":29},"end":{"line":14,"column":63}},"8":{"start":{"line":15,"column":16},"end":{"line":20,"column":17}},"9":{"start":{"line":16,"column":20},"end":{"line":19,"column":21}},"10":{"start":{"line":22,"column":16},"end":{"line":27,"column":17}},"11":{"start":{"line":23,"column":20},"end":{"line":26,"column":21}},"12":{"start":{"line":30,"column":12},"end":{"line":30,"column":18}},"13":{"start":{"line":32,"column":12},"end":{"line":32,"column":23}},"14":{"start":{"line":37,"column":0},"end":{"line":37,"column":30}}},"fnMap":{"0":{"name":"authorization","decl":{"start":{"line":2,"column":9},"end":{"line":2,"column":22}},"loc":{"start":{"line":2,"column":30},"end":{"line":35,"column":1}},"line":2},"1":{"name":"(anonymous_1)","decl":{"start":{"line":3,"column":11},"end":{"line":3,"column":12}},"loc":{"start":{"line":3,"column":37},"end":{"line":34,"column":5}},"line":3}},"branchMap":{"0":{"loc":{"start":{"line":6,"column":12},"end":{"line":11,"column":13}},"type":"if","locations":[{"start":{"line":6,"column":12},"end":{"line":11,"column":13}},{"start":{},"end":{}}],"line":6},"1":{"loc":{"start":{"line":13,"column":12},"end":{"line":28,"column":13}},"type":"if","locations":[{"start":{"line":13,"column":12},"end":{"line":28,"column":13}},{"start":{},"end":{}}],"line":13},"2":{"loc":{"start":{"line":15,"column":16},"end":{"line":20,"column":17}},"type":"if","locations":[{"start":{"line":15,"column":16},"end":{"line":20,"column":17}},{"start":{},"end":{}}],"line":15},"3":{"loc":{"start":{"line":22,"column":16},"end":{"line":27,"column":17}},"type":"if","locations":[{"start":{"line":22,"column":16},"end":{"line":27,"column":17}},{"start":{},"end":{}}],"line":22}},"s":{"0":7,"1":5,"2":5,"3":5,"4":5,"5":1,"6":4,"7":3,"8":3,"9":1,"10":2,"11":1,"12":2,"13":3,"14":7},"f":{"0":5,"1":5},"b":{"0":[1,4],"1":[3,1],"2":[1,2],"3":[1,1]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"b8f02dc1fb98f4b57421646df8f6fa987258623d"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\middleware\\errorHandler.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\middleware\\errorHandler.js","statementMap":{"0":{"start":{"line":2,"column":4},"end":{"line":44,"column":5}},"1":{"start":{"line":4,"column":12},"end":{"line":6,"column":14}},"2":{"start":{"line":8,"column":12},"end":{"line":10,"column":14}},"3":{"start":{"line":12,"column":12},"end":{"line":14,"column":14}},"4":{"start":{"line":16,"column":12},"end":{"line":18,"column":14}},"5":{"start":{"line":20,"column":12},"end":{"line":22,"column":14}},"6":{"start":{"line":24,"column":12},"end":{"line":26,"column":14}},"7":{"start":{"line":28,"column":12},"end":{"line":30,"column":14}},"8":{"start":{"line":32,"column":12},"end":{"line":34,"column":14}},"9":{"start":{"line":36,"column":12},"end":{"line":38,"column":14}},"10":{"start":{"line":40,"column":12},"end":{"line":42,"column":14}},"11":{"start":{"line":43,"column":12},"end":{"line":43,"column":17}},"12":{"start":{"line":47,"column":0},"end":{"line":47,"column":29}}},"fnMap":{"0":{"name":"errorHandler","decl":{"start":{"line":1,"column":9},"end":{"line":1,"column":21}},"loc":{"start":{"line":1,"column":43},"end":{"line":45,"column":1}},"line":1}},"branchMap":{"0":{"loc":{"start":{"line":2,"column":4},"end":{"line":44,"column":5}},"type":"switch","locations":[{"start":{"line":3,"column":8},"end":{"line":6,"column":14}},{"start":{"line":7,"column":8},"end":{"line":10,"column":14}},{"start":{"line":11,"column":8},"end":{"line":14,"column":14}},{"start":{"line":15,"column":8},"end":{"line":18,"column":14}},{"start":{"line":19,"column":8},"end":{"line":22,"column":14}},{"start":{"line":23,"column":8},"end":{"line":26,"column":14}},{"start":{"line":27,"column":8},"end":{"line":30,"column":14}},{"start":{"line":31,"column":8},"end":{"line":34,"column":14}},{"start":{"line":35,"column":8},"end":{"line":38,"column":14}},{"start":{"line":39,"column":8},"end":{"line":43,"column":17}}],"line":2}},"s":{"0":35,"1":15,"2":5,"3":2,"4":1,"5":1,"6":1,"7":7,"8":1,"9":1,"10":1,"11":1,"12":6},"f":{"0":35},"b":{"0":[15,5,2,1,1,1,7,1,1,1]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"d95963566ebbf063b7e1cea8e1cf73c1061dae6c"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\course.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\course.js","statementMap":{"0":{"start":{"line":4,"column":4},"end":{"line":4,"column":24}},"1":{"start":{"line":5,"column":0},"end":{"line":26,"column":2}},"2":{"start":{"line":16,"column":2},"end":{"line":24,"column":5}},"3":{"start":{"line":25,"column":2},"end":{"line":25,"column":16}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":5,"column":17},"end":{"line":5,"column":18}},"loc":{"start":{"line":5,"column":43},"end":{"line":26,"column":1}},"line":5},"1":{"name":"(anonymous_1)","decl":{"start":{"line":12,"column":4},"end":{"line":12,"column":5}},"loc":{"start":{"line":12,"column":29},"end":{"line":14,"column":5}},"line":12}},"branchMap":{},"s":{"0":6,"1":6,"2":6,"3":6},"f":{"0":6,"1":6},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"3d008ce9bfb81fcdee766ee9210c9dd1723e1907"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\index.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\index.js","statementMap":{"0":{"start":{"line":3,"column":11},"end":{"line":3,"column":24}},"1":{"start":{"line":4,"column":13},"end":{"line":4,"column":28}},"2":{"start":{"line":5,"column":18},"end":{"line":5,"column":38}},"3":{"start":{"line":6,"column":16},"end":{"line":6,"column":34}},"4":{"start":{"line":7,"column":17},"end":{"line":7,"column":42}},"5":{"start":{"line":8,"column":12},"end":{"line":8,"column":49}},"6":{"start":{"line":9,"column":15},"end":{"line":9,"column":65}},"7":{"start":{"line":10,"column":11},"end":{"line":10,"column":13}},"8":{"start":{"line":13,"column":0},"end":{"line":17,"column":1}},"9":{"start":{"line":14,"column":2},"end":{"line":14,"column":74}},"10":{"start":{"line":16,"column":2},"end":{"line":16,"column":87}},"11":{"start":{"line":19,"column":0},"end":{"line":32,"column":5}},"12":{"start":{"line":22,"column":4},"end":{"line":27,"column":6}},"13":{"start":{"line":30,"column":18},"end":{"line":30,"column":85}},"14":{"start":{"line":31,"column":4},"end":{"line":31,"column":27}},"15":{"start":{"line":34,"column":0},"end":{"line":38,"column":3}},"16":{"start":{"line":35,"column":2},"end":{"line":37,"column":3}},"17":{"start":{"line":36,"column":4},"end":{"line":36,"column":32}},"18":{"start":{"line":40,"column":0},"end":{"line":40,"column":25}},"19":{"start":{"line":41,"column":0},"end":{"line":41,"column":25}},"20":{"start":{"line":43,"column":0},"end":{"line":43,"column":20}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":21,"column":10},"end":{"line":21,"column":11}},"loc":{"start":{"line":21,"column":18},"end":{"line":28,"column":3}},"line":21},"1":{"name":"(anonymous_1)","decl":{"start":{"line":29,"column":11},"end":{"line":29,"column":12}},"loc":{"start":{"line":29,"column":19},"end":{"line":32,"column":3}},"line":29},"2":{"name":"(anonymous_2)","decl":{"start":{"line":34,"column":24},"end":{"line":34,"column":25}},"loc":{"start":{"line":34,"column":37},"end":{"line":38,"column":1}},"line":34}},"branchMap":{"0":{"loc":{"start":{"line":8,"column":12},"end":{"line":8,"column":49}},"type":"binary-expr","locations":[{"start":{"line":8,"column":12},"end":{"line":8,"column":32}},{"start":{"line":8,"column":36},"end":{"line":8,"column":49}}],"line":8},"1":{"loc":{"start":{"line":13,"column":0},"end":{"line":17,"column":1}},"type":"if","locations":[{"start":{"line":13,"column":0},"end":{"line":17,"column":1}},{"start":{"line":15,"column":7},"end":{"line":17,"column":1}}],"line":13},"2":{"loc":{"start":{"line":23,"column":6},"end":{"line":26,"column":37}},"type":"binary-expr","locations":[{"start":{"line":23,"column":6},"end":{"line":23,"column":29}},{"start":{"line":24,"column":6},"end":{"line":24,"column":23}},{"start":{"line":25,"column":6},"end":{"line":25,"column":30}},{"start":{"line":26,"column":6},"end":{"line":26,"column":37}}],"line":23},"3":{"loc":{"start":{"line":35,"column":2},"end":{"line":37,"column":3}},"type":"if","locations":[{"start":{"line":35,"column":2},"end":{"line":37,"column":3}},{"start":{},"end":{}}],"line":35}},"s":{"0":6,"1":6,"2":6,"3":6,"4":6,"5":6,"6":6,"7":6,"8":6,"9":0,"10":6,"11":6,"12":30,"13":24,"14":24,"15":6,"16":24,"17":24,"18":6,"19":6,"20":6},"f":{"0":30,"1":24,"2":24},"b":{"0":[6,0],"1":[0,6],"2":[30,30,24,24],"3":[24,0]},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"9b228f725e2f1a2bd161127f95cfa028d2d722a2"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\material.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\material.js","statementMap":{"0":{"start":{"line":4,"column":4},"end":{"line":4,"column":24}},"1":{"start":{"line":5,"column":0},"end":{"line":28,"column":2}},"2":{"start":{"line":16,"column":2},"end":{"line":26,"column":5}},"3":{"start":{"line":27,"column":2},"end":{"line":27,"column":18}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":5,"column":17},"end":{"line":5,"column":18}},"loc":{"start":{"line":5,"column":43},"end":{"line":28,"column":1}},"line":5},"1":{"name":"(anonymous_1)","decl":{"start":{"line":12,"column":4},"end":{"line":12,"column":5}},"loc":{"start":{"line":12,"column":29},"end":{"line":14,"column":5}},"line":12}},"branchMap":{},"s":{"0":6,"1":6,"2":6,"3":6},"f":{"0":6,"1":6},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"2e5c12b7ff3936dfa62b7096cfb2939e440d7d27"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\user.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\user.js","statementMap":{"0":{"start":{"line":4,"column":4},"end":{"line":4,"column":24}},"1":{"start":{"line":5,"column":0},"end":{"line":63,"column":2}},"2":{"start":{"line":14,"column":6},"end":{"line":14,"column":67}},"3":{"start":{"line":17,"column":2},"end":{"line":61,"column":5}},"4":{"start":{"line":62,"column":2},"end":{"line":62,"column":14}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":5,"column":17},"end":{"line":5,"column":18}},"loc":{"start":{"line":5,"column":43},"end":{"line":63,"column":1}},"line":5},"1":{"name":"(anonymous_1)","decl":{"start":{"line":12,"column":4},"end":{"line":12,"column":5}},"loc":{"start":{"line":12,"column":29},"end":{"line":15,"column":5}},"line":12}},"branchMap":{},"s":{"0":6,"1":6,"2":6,"3":6,"4":6},"f":{"0":6,"1":6},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"a8896e18daa17fb289d649d9573d92310ce7c325"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\usercourse.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\models\\usercourse.js","statementMap":{"0":{"start":{"line":4,"column":4},"end":{"line":4,"column":24}},"1":{"start":{"line":5,"column":0},"end":{"line":47,"column":2}},"2":{"start":{"line":16,"column":2},"end":{"line":45,"column":5}},"3":{"start":{"line":46,"column":2},"end":{"line":46,"column":20}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":5,"column":17},"end":{"line":5,"column":18}},"loc":{"start":{"line":5,"column":43},"end":{"line":47,"column":1}},"line":5},"1":{"name":"(anonymous_1)","decl":{"start":{"line":12,"column":4},"end":{"line":12,"column":5}},"loc":{"start":{"line":12,"column":29},"end":{"line":14,"column":5}},"line":12}},"branchMap":{},"s":{"0":6,"1":6,"2":6,"3":6},"f":{"0":6,"1":6},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"ce56a6dbc1a9cd1d70d22e0432fbaf6481ea18f9"}
+,"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\routes\\api.js": {"path":"D:\\repositories\\@Hacktiv8\\p2\\IP-RMT59\\server\\routes\\api.js","statementMap":{"0":{"start":{"line":1,"column":21},"end":{"line":1,"column":59}},"1":{"start":{"line":2,"column":25},"end":{"line":2,"column":67}},"2":{"start":{"line":3,"column":27},"end":{"line":3,"column":71}},"3":{"start":{"line":4,"column":23},"end":{"line":4,"column":63}},"4":{"start":{"line":5,"column":29},"end":{"line":5,"column":75}},"5":{"start":{"line":6,"column":23},"end":{"line":6,"column":62}},"6":{"start":{"line":7,"column":22},"end":{"line":7,"column":60}},"7":{"start":{"line":9,"column":16},"end":{"line":9,"column":34}},"8":{"start":{"line":10,"column":15},"end":{"line":10,"column":31}},"9":{"start":{"line":12,"column":0},"end":{"line":16,"column":2}},"10":{"start":{"line":13,"column":4},"end":{"line":15,"column":6}},"11":{"start":{"line":18,"column":0},"end":{"line":18,"column":43}},"12":{"start":{"line":19,"column":0},"end":{"line":19,"column":56}},"13":{"start":{"line":21,"column":0},"end":{"line":21,"column":49}},"14":{"start":{"line":22,"column":0},"end":{"line":22,"column":56}},"15":{"start":{"line":24,"column":0},"end":{"line":24,"column":62}},"16":{"start":{"line":25,"column":0},"end":{"line":25,"column":65}},"17":{"start":{"line":26,"column":0},"end":{"line":26,"column":84}},"18":{"start":{"line":28,"column":0},"end":{"line":28,"column":72}},"19":{"start":{"line":29,"column":0},"end":{"line":29,"column":75}},"20":{"start":{"line":30,"column":0},"end":{"line":30,"column":71}},"21":{"start":{"line":31,"column":0},"end":{"line":31,"column":82}},"22":{"start":{"line":33,"column":0},"end":{"line":33,"column":81}},"23":{"start":{"line":34,"column":0},"end":{"line":34,"column":80}},"24":{"start":{"line":35,"column":0},"end":{"line":35,"column":87}},"25":{"start":{"line":36,"column":0},"end":{"line":36,"column":89}},"26":{"start":{"line":39,"column":0},"end":{"line":39,"column":23}}},"fnMap":{"0":{"name":"(anonymous_0)","decl":{"start":{"line":12,"column":16},"end":{"line":12,"column":17}},"loc":{"start":{"line":12,"column":30},"end":{"line":16,"column":1}},"line":12}},"branchMap":{},"s":{"0":6,"1":6,"2":6,"3":6,"4":6,"5":6,"6":6,"7":6,"8":6,"9":6,"10":0,"11":6,"12":6,"13":6,"14":6,"15":6,"16":6,"17":6,"18":6,"19":6,"20":6,"21":6,"22":6,"23":6,"24":6,"25":6,"26":6},"f":{"0":0},"b":{},"_coverageSchema":"1a1c01bbd47fc00a2c39e90264f33305004495a9","hash":"f84a62f0dadc575f921f906b7a541c684f1fd262"}
+}
diff --git a/server/coverage/lcov-report/base.css b/server/coverage/lcov-report/base.css
new file mode 100644
index 0000000..f418035
--- /dev/null
+++ b/server/coverage/lcov-report/base.css
@@ -0,0 +1,224 @@
+body, html {
+ margin:0; padding: 0;
+ height: 100%;
+}
+body {
+ font-family: Helvetica Neue, Helvetica, Arial;
+ font-size: 14px;
+ color:#333;
+}
+.small { font-size: 12px; }
+*, *:after, *:before {
+ -webkit-box-sizing:border-box;
+ -moz-box-sizing:border-box;
+ box-sizing:border-box;
+ }
+h1 { font-size: 20px; margin: 0;}
+h2 { font-size: 14px; }
+pre {
+ font: 12px/1.4 Consolas, "Liberation Mono", Menlo, Courier, monospace;
+ margin: 0;
+ padding: 0;
+ -moz-tab-size: 2;
+ -o-tab-size: 2;
+ tab-size: 2;
+}
+a { color:#0074D9; text-decoration:none; }
+a:hover { text-decoration:underline; }
+.strong { font-weight: bold; }
+.space-top1 { padding: 10px 0 0 0; }
+.pad2y { padding: 20px 0; }
+.pad1y { padding: 10px 0; }
+.pad2x { padding: 0 20px; }
+.pad2 { padding: 20px; }
+.pad1 { padding: 10px; }
+.space-left2 { padding-left:55px; }
+.space-right2 { padding-right:20px; }
+.center { text-align:center; }
+.clearfix { display:block; }
+.clearfix:after {
+ content:'';
+ display:block;
+ height:0;
+ clear:both;
+ visibility:hidden;
+ }
+.fl { float: left; }
+@media only screen and (max-width:640px) {
+ .col3 { width:100%; max-width:100%; }
+ .hide-mobile { display:none!important; }
+}
+
+.quiet {
+ color: #7f7f7f;
+ color: rgba(0,0,0,0.5);
+}
+.quiet a { opacity: 0.7; }
+
+.fraction {
+ font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace;
+ font-size: 10px;
+ color: #555;
+ background: #E8E8E8;
+ padding: 4px 5px;
+ border-radius: 3px;
+ vertical-align: middle;
+}
+
+div.path a:link, div.path a:visited { color: #333; }
+table.coverage {
+ border-collapse: collapse;
+ margin: 10px 0 0 0;
+ padding: 0;
+}
+
+table.coverage td {
+ margin: 0;
+ padding: 0;
+ vertical-align: top;
+}
+table.coverage td.line-count {
+ text-align: right;
+ padding: 0 5px 0 20px;
+}
+table.coverage td.line-coverage {
+ text-align: right;
+ padding-right: 10px;
+ min-width:20px;
+}
+
+table.coverage td span.cline-any {
+ display: inline-block;
+ padding: 0 5px;
+ width: 100%;
+}
+.missing-if-branch {
+ display: inline-block;
+ margin-right: 5px;
+ border-radius: 3px;
+ position: relative;
+ padding: 0 4px;
+ background: #333;
+ color: yellow;
+}
+
+.skip-if-branch {
+ display: none;
+ margin-right: 10px;
+ position: relative;
+ padding: 0 4px;
+ background: #ccc;
+ color: white;
+}
+.missing-if-branch .typ, .skip-if-branch .typ {
+ color: inherit !important;
+}
+.coverage-summary {
+ border-collapse: collapse;
+ width: 100%;
+}
+.coverage-summary tr { border-bottom: 1px solid #bbb; }
+.keyline-all { border: 1px solid #ddd; }
+.coverage-summary td, .coverage-summary th { padding: 10px; }
+.coverage-summary tbody { border: 1px solid #bbb; }
+.coverage-summary td { border-right: 1px solid #bbb; }
+.coverage-summary td:last-child { border-right: none; }
+.coverage-summary th {
+ text-align: left;
+ font-weight: normal;
+ white-space: nowrap;
+}
+.coverage-summary th.file { border-right: none !important; }
+.coverage-summary th.pct { }
+.coverage-summary th.pic,
+.coverage-summary th.abs,
+.coverage-summary td.pct,
+.coverage-summary td.abs { text-align: right; }
+.coverage-summary td.file { white-space: nowrap; }
+.coverage-summary td.pic { min-width: 120px !important; }
+.coverage-summary tfoot td { }
+
+.coverage-summary .sorter {
+ height: 10px;
+ width: 7px;
+ display: inline-block;
+ margin-left: 0.5em;
+ background: url(sort-arrow-sprite.png) no-repeat scroll 0 0 transparent;
+}
+.coverage-summary .sorted .sorter {
+ background-position: 0 -20px;
+}
+.coverage-summary .sorted-desc .sorter {
+ background-position: 0 -10px;
+}
+.status-line { height: 10px; }
+/* yellow */
+.cbranch-no { background: yellow !important; color: #111; }
+/* dark red */
+.red.solid, .status-line.low, .low .cover-fill { background:#C21F39 }
+.low .chart { border:1px solid #C21F39 }
+.highlighted,
+.highlighted .cstat-no, .highlighted .fstat-no, .highlighted .cbranch-no{
+ background: #C21F39 !important;
+}
+/* medium red */
+.cstat-no, .fstat-no, .cbranch-no, .cbranch-no { background:#F6C6CE }
+/* light red */
+.low, .cline-no { background:#FCE1E5 }
+/* light green */
+.high, .cline-yes { background:rgb(230,245,208) }
+/* medium green */
+.cstat-yes { background:rgb(161,215,106) }
+/* dark green */
+.status-line.high, .high .cover-fill { background:rgb(77,146,33) }
+.high .chart { border:1px solid rgb(77,146,33) }
+/* dark yellow (gold) */
+.status-line.medium, .medium .cover-fill { background: #f9cd0b; }
+.medium .chart { border:1px solid #f9cd0b; }
+/* light yellow */
+.medium { background: #fff4c2; }
+
+.cstat-skip { background: #ddd; color: #111; }
+.fstat-skip { background: #ddd; color: #111 !important; }
+.cbranch-skip { background: #ddd !important; color: #111; }
+
+span.cline-neutral { background: #eaeaea; }
+
+.coverage-summary td.empty {
+ opacity: .5;
+ padding-top: 4px;
+ padding-bottom: 4px;
+ line-height: 1;
+ color: #888;
+}
+
+.cover-fill, .cover-empty {
+ display:inline-block;
+ height: 12px;
+}
+.chart {
+ line-height: 0;
+}
+.cover-empty {
+ background: white;
+}
+.cover-full {
+ border-right: none !important;
+}
+pre.prettyprint {
+ border: none !important;
+ padding: 0 !important;
+ margin: 0 !important;
+}
+.com { color: #999 !important; }
+.ignore-none { color: #999; font-weight: normal; }
+
+.wrapper {
+ min-height: 100%;
+ height: auto !important;
+ height: 100%;
+ margin: 0 auto -48px;
+}
+.footer, .push {
+ height: 48px;
+}
diff --git a/server/coverage/lcov-report/block-navigation.js b/server/coverage/lcov-report/block-navigation.js
new file mode 100644
index 0000000..cc12130
--- /dev/null
+++ b/server/coverage/lcov-report/block-navigation.js
@@ -0,0 +1,87 @@
+/* eslint-disable */
+var jumpToCode = (function init() {
+ // Classes of code we would like to highlight in the file view
+ var missingCoverageClasses = ['.cbranch-no', '.cstat-no', '.fstat-no'];
+
+ // Elements to highlight in the file listing view
+ var fileListingElements = ['td.pct.low'];
+
+ // We don't want to select elements that are direct descendants of another match
+ var notSelector = ':not(' + missingCoverageClasses.join('):not(') + ') > '; // becomes `:not(a):not(b) > `
+
+ // Selecter that finds elements on the page to which we can jump
+ var selector =
+ fileListingElements.join(', ') +
+ ', ' +
+ notSelector +
+ missingCoverageClasses.join(', ' + notSelector); // becomes `:not(a):not(b) > a, :not(a):not(b) > b`
+
+ // The NodeList of matching elements
+ var missingCoverageElements = document.querySelectorAll(selector);
+
+ var currentIndex;
+
+ function toggleClass(index) {
+ missingCoverageElements
+ .item(currentIndex)
+ .classList.remove('highlighted');
+ missingCoverageElements.item(index).classList.add('highlighted');
+ }
+
+ function makeCurrent(index) {
+ toggleClass(index);
+ currentIndex = index;
+ missingCoverageElements.item(index).scrollIntoView({
+ behavior: 'smooth',
+ block: 'center',
+ inline: 'center'
+ });
+ }
+
+ function goToPrevious() {
+ var nextIndex = 0;
+ if (typeof currentIndex !== 'number' || currentIndex === 0) {
+ nextIndex = missingCoverageElements.length - 1;
+ } else if (missingCoverageElements.length > 1) {
+ nextIndex = currentIndex - 1;
+ }
+
+ makeCurrent(nextIndex);
+ }
+
+ function goToNext() {
+ var nextIndex = 0;
+
+ if (
+ typeof currentIndex === 'number' &&
+ currentIndex < missingCoverageElements.length - 1
+ ) {
+ nextIndex = currentIndex + 1;
+ }
+
+ makeCurrent(nextIndex);
+ }
+
+ return function jump(event) {
+ if (
+ document.getElementById('fileSearch') === document.activeElement &&
+ document.activeElement != null
+ ) {
+ // if we're currently focused on the search input, we don't want to navigate
+ return;
+ }
+
+ switch (event.which) {
+ case 78: // n
+ case 74: // j
+ goToNext();
+ break;
+ case 66: // b
+ case 75: // k
+ case 80: // p
+ goToPrevious();
+ break;
+ }
+ };
+})();
+window.addEventListener('keydown', jumpToCode);
diff --git a/server/coverage/lcov-report/favicon.png b/server/coverage/lcov-report/favicon.png
new file mode 100644
index 0000000..c1525b8
Binary files /dev/null and b/server/coverage/lcov-report/favicon.png differ
diff --git a/server/coverage/lcov-report/index.html b/server/coverage/lcov-report/index.html
new file mode 100644
index 0000000..dd9d357
--- /dev/null
+++ b/server/coverage/lcov-report/index.html
@@ -0,0 +1,206 @@
+
+
+
+
+
+ Code coverage report for All files
+
+
+
+
+
+
+
+
+
+
+
+
All files
+
+
+
+ 95.12%
+ Statements
+ 312/328
+
+
+
+
+ 89.32%
+ Branches
+ 92/103
+
+
+
+
+ 97.29%
+ Functions
+ 36/37
+
+
+
+
+ 95.67%
+ Lines
+ 310/324
+
+
+
+
+
+ Press n or j to go to the next uncovered block, b, p or k for the previous block.
+
+
+
+ Filter:
+
+
+
+
+
+
+
+
+
+ | File |
+ |
+ Statements |
+ |
+ Branches |
+ |
+ Functions |
+ |
+ Lines |
+ |
+
+
+
+ | server |
+
+
+ |
+ 100% |
+ 13/13 |
+ 50% |
+ 1/2 |
+ 100% |
+ 0/0 |
+ 100% |
+ 13/13 |
+
+
+
+ | server/__test__ |
+
+
+ |
+ 100% |
+ 2/2 |
+ 100% |
+ 0/0 |
+ 100% |
+ 0/0 |
+ 100% |
+ 2/2 |
+
+
+
+ | server/controllers |
+
+
+ |
+ 92.26% |
+ 167/181 |
+ 89.06% |
+ 57/64 |
+ 100% |
+ 15/15 |
+ 93.25% |
+ 166/178 |
+
+
+
+ | server/helpers |
+
+
+ |
+ 100% |
+ 26/26 |
+ 100% |
+ 3/3 |
+ 100% |
+ 6/6 |
+ 100% |
+ 25/25 |
+
+
+
+ | server/middleware |
+
+
+ |
+ 100% |
+ 41/41 |
+ 100% |
+ 24/24 |
+ 100% |
+ 4/4 |
+ 100% |
+ 41/41 |
+
+
+
+ | server/models |
+
+
+ |
+ 97.36% |
+ 37/38 |
+ 70% |
+ 7/10 |
+ 100% |
+ 11/11 |
+ 97.36% |
+ 37/38 |
+
+
+
+ | server/routes |
+
+
+ |
+ 96.29% |
+ 26/27 |
+ 100% |
+ 0/0 |
+ 0% |
+ 0/1 |
+ 96.29% |
+ 26/27 |
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/server/coverage/lcov-report/prettify.css b/server/coverage/lcov-report/prettify.css
new file mode 100644
index 0000000..b317a7c
--- /dev/null
+++ b/server/coverage/lcov-report/prettify.css
@@ -0,0 +1 @@
+.pln{color:#000}@media screen{.str{color:#080}.kwd{color:#008}.com{color:#800}.typ{color:#606}.lit{color:#066}.pun,.opn,.clo{color:#660}.tag{color:#008}.atn{color:#606}.atv{color:#080}.dec,.var{color:#606}.fun{color:red}}@media print,projection{.str{color:#060}.kwd{color:#006;font-weight:bold}.com{color:#600;font-style:italic}.typ{color:#404;font-weight:bold}.lit{color:#044}.pun,.opn,.clo{color:#440}.tag{color:#006;font-weight:bold}.atn{color:#404}.atv{color:#060}}pre.prettyprint{padding:2px;border:1px solid #888}ol.linenums{margin-top:0;margin-bottom:0}li.L0,li.L1,li.L2,li.L3,li.L5,li.L6,li.L7,li.L8{list-style-type:none}li.L1,li.L3,li.L5,li.L7,li.L9{background:#eee}
diff --git a/server/coverage/lcov-report/prettify.js b/server/coverage/lcov-report/prettify.js
new file mode 100644
index 0000000..b322523
--- /dev/null
+++ b/server/coverage/lcov-report/prettify.js
@@ -0,0 +1,2 @@
+/* eslint-disable */
+window.PR_SHOULD_USE_CONTINUATION=true;(function(){var h=["break,continue,do,else,for,if,return,while"];var u=[h,"auto,case,char,const,default,double,enum,extern,float,goto,int,long,register,short,signed,sizeof,static,struct,switch,typedef,union,unsigned,void,volatile"];var p=[u,"catch,class,delete,false,import,new,operator,private,protected,public,this,throw,true,try,typeof"];var l=[p,"alignof,align_union,asm,axiom,bool,concept,concept_map,const_cast,constexpr,decltype,dynamic_cast,explicit,export,friend,inline,late_check,mutable,namespace,nullptr,reinterpret_cast,static_assert,static_cast,template,typeid,typename,using,virtual,where"];var x=[p,"abstract,boolean,byte,extends,final,finally,implements,import,instanceof,null,native,package,strictfp,super,synchronized,throws,transient"];var R=[x,"as,base,by,checked,decimal,delegate,descending,dynamic,event,fixed,foreach,from,group,implicit,in,interface,internal,into,is,lock,object,out,override,orderby,params,partial,readonly,ref,sbyte,sealed,stackalloc,string,select,uint,ulong,unchecked,unsafe,ushort,var"];var r="all,and,by,catch,class,else,extends,false,finally,for,if,in,is,isnt,loop,new,no,not,null,of,off,on,or,return,super,then,true,try,unless,until,when,while,yes";var w=[p,"debugger,eval,export,function,get,null,set,undefined,var,with,Infinity,NaN"];var s="caller,delete,die,do,dump,elsif,eval,exit,foreach,for,goto,if,import,last,local,my,next,no,our,print,package,redo,require,sub,undef,unless,until,use,wantarray,while,BEGIN,END";var I=[h,"and,as,assert,class,def,del,elif,except,exec,finally,from,global,import,in,is,lambda,nonlocal,not,or,pass,print,raise,try,with,yield,False,True,None"];var f=[h,"alias,and,begin,case,class,def,defined,elsif,end,ensure,false,in,module,next,nil,not,or,redo,rescue,retry,self,super,then,true,undef,unless,until,when,yield,BEGIN,END"];var H=[h,"case,done,elif,esac,eval,fi,function,in,local,set,then,until"];var A=[l,R,w,s+I,f,H];var e=/^(DIR|FILE|vector|(de|priority_)?queue|list|stack|(const_)?iterator|(multi)?(set|map)|bitset|u?(int|float)\d*)/;var C="str";var z="kwd";var j="com";var O="typ";var G="lit";var L="pun";var F="pln";var m="tag";var E="dec";var J="src";var P="atn";var n="atv";var N="nocode";var M="(?:^^\\.?|[+-]|\\!|\\!=|\\!==|\\#|\\%|\\%=|&|&&|&&=|&=|\\(|\\*|\\*=|\\+=|\\,|\\-=|\\->|\\/|\\/=|:|::|\\;|<|<<|<<=|<=|=|==|===|>|>=|>>|>>=|>>>|>>>=|\\?|\\@|\\[|\\^|\\^=|\\^\\^|\\^\\^=|\\{|\\||\\|=|\\|\\||\\|\\|=|\\~|break|case|continue|delete|do|else|finally|instanceof|return|throw|try|typeof)\\s*";function k(Z){var ad=0;var S=false;var ac=false;for(var V=0,U=Z.length;V122)){if(!(al<65||ag>90)){af.push([Math.max(65,ag)|32,Math.min(al,90)|32])}if(!(al<97||ag>122)){af.push([Math.max(97,ag)&~32,Math.min(al,122)&~32])}}}}af.sort(function(av,au){return(av[0]-au[0])||(au[1]-av[1])});var ai=[];var ap=[NaN,NaN];for(var ar=0;arat[0]){if(at[1]+1>at[0]){an.push("-")}an.push(T(at[1]))}}an.push("]");return an.join("")}function W(al){var aj=al.source.match(new RegExp("(?:\\[(?:[^\\x5C\\x5D]|\\\\[\\s\\S])*\\]|\\\\u[A-Fa-f0-9]{4}|\\\\x[A-Fa-f0-9]{2}|\\\\[0-9]+|\\\\[^ux0-9]|\\(\\?[:!=]|[\\(\\)\\^]|[^\\x5B\\x5C\\(\\)\\^]+)","g"));var ah=aj.length;var an=[];for(var ak=0,am=0;ak=2&&ai==="["){aj[ak]=X(ag)}else{if(ai!=="\\"){aj[ak]=ag.replace(/[a-zA-Z]/g,function(ao){var ap=ao.charCodeAt(0);return"["+String.fromCharCode(ap&~32,ap|32)+"]"})}}}}return aj.join("")}var aa=[];for(var V=0,U=Z.length;V=0;){S[ac.charAt(ae)]=Y}}var af=Y[1];var aa=""+af;if(!ag.hasOwnProperty(aa)){ah.push(af);ag[aa]=null}}ah.push(/[\0-\uffff]/);V=k(ah)})();var X=T.length;var W=function(ah){var Z=ah.sourceCode,Y=ah.basePos;var ad=[Y,F];var af=0;var an=Z.match(V)||[];var aj={};for(var ae=0,aq=an.length;ae=5&&"lang-"===ap.substring(0,5);if(am&&!(ai&&typeof ai[1]==="string")){am=false;ap=J}if(!am){aj[ag]=ap}}var ab=af;af+=ag.length;if(!am){ad.push(Y+ab,ap)}else{var al=ai[1];var ak=ag.indexOf(al);var ac=ak+al.length;if(ai[2]){ac=ag.length-ai[2].length;ak=ac-al.length}var ar=ap.substring(5);B(Y+ab,ag.substring(0,ak),W,ad);B(Y+ab+ak,al,q(ar,al),ad);B(Y+ab+ac,ag.substring(ac),W,ad)}}ah.decorations=ad};return W}function i(T){var W=[],S=[];if(T.tripleQuotedStrings){W.push([C,/^(?:\'\'\'(?:[^\'\\]|\\[\s\S]|\'{1,2}(?=[^\']))*(?:\'\'\'|$)|\"\"\"(?:[^\"\\]|\\[\s\S]|\"{1,2}(?=[^\"]))*(?:\"\"\"|$)|\'(?:[^\\\']|\\[\s\S])*(?:\'|$)|\"(?:[^\\\"]|\\[\s\S])*(?:\"|$))/,null,"'\""])}else{if(T.multiLineStrings){W.push([C,/^(?:\'(?:[^\\\']|\\[\s\S])*(?:\'|$)|\"(?:[^\\\"]|\\[\s\S])*(?:\"|$)|\`(?:[^\\\`]|\\[\s\S])*(?:\`|$))/,null,"'\"`"])}else{W.push([C,/^(?:\'(?:[^\\\'\r\n]|\\.)*(?:\'|$)|\"(?:[^\\\"\r\n]|\\.)*(?:\"|$))/,null,"\"'"])}}if(T.verbatimStrings){S.push([C,/^@\"(?:[^\"]|\"\")*(?:\"|$)/,null])}var Y=T.hashComments;if(Y){if(T.cStyleComments){if(Y>1){W.push([j,/^#(?:##(?:[^#]|#(?!##))*(?:###|$)|.*)/,null,"#"])}else{W.push([j,/^#(?:(?:define|elif|else|endif|error|ifdef|include|ifndef|line|pragma|undef|warning)\b|[^\r\n]*)/,null,"#"])}S.push([C,/^<(?:(?:(?:\.\.\/)*|\/?)(?:[\w-]+(?:\/[\w-]+)+)?[\w-]+\.h|[a-z]\w*)>/,null])}else{W.push([j,/^#[^\r\n]*/,null,"#"])}}if(T.cStyleComments){S.push([j,/^\/\/[^\r\n]*/,null]);S.push([j,/^\/\*[\s\S]*?(?:\*\/|$)/,null])}if(T.regexLiterals){var X=("/(?=[^/*])(?:[^/\\x5B\\x5C]|\\x5C[\\s\\S]|\\x5B(?:[^\\x5C\\x5D]|\\x5C[\\s\\S])*(?:\\x5D|$))+/");S.push(["lang-regex",new RegExp("^"+M+"("+X+")")])}var V=T.types;if(V){S.push([O,V])}var U=(""+T.keywords).replace(/^ | $/g,"");if(U.length){S.push([z,new RegExp("^(?:"+U.replace(/[\s,]+/g,"|")+")\\b"),null])}W.push([F,/^\s+/,null," \r\n\t\xA0"]);S.push([G,/^@[a-z_$][a-z_$@0-9]*/i,null],[O,/^(?:[@_]?[A-Z]+[a-z][A-Za-z_$@0-9]*|\w+_t\b)/,null],[F,/^[a-z_$][a-z_$@0-9]*/i,null],[G,new RegExp("^(?:0x[a-f0-9]+|(?:\\d(?:_\\d+)*\\d*(?:\\.\\d*)?|\\.\\d\\+)(?:e[+\\-]?\\d+)?)[a-z]*","i"),null,"0123456789"],[F,/^\\[\s\S]?/,null],[L,/^.[^\s\w\.$@\'\"\`\/\#\\]*/,null]);return g(W,S)}var K=i({keywords:A,hashComments:true,cStyleComments:true,multiLineStrings:true,regexLiterals:true});function Q(V,ag){var U=/(?:^|\s)nocode(?:\s|$)/;var ab=/\r\n?|\n/;var ac=V.ownerDocument;var S;if(V.currentStyle){S=V.currentStyle.whiteSpace}else{if(window.getComputedStyle){S=ac.defaultView.getComputedStyle(V,null).getPropertyValue("white-space")}}var Z=S&&"pre"===S.substring(0,3);var af=ac.createElement("LI");while(V.firstChild){af.appendChild(V.firstChild)}var W=[af];function ae(al){switch(al.nodeType){case 1:if(U.test(al.className)){break}if("BR"===al.nodeName){ad(al);if(al.parentNode){al.parentNode.removeChild(al)}}else{for(var an=al.firstChild;an;an=an.nextSibling){ae(an)}}break;case 3:case 4:if(Z){var am=al.nodeValue;var aj=am.match(ab);if(aj){var ai=am.substring(0,aj.index);al.nodeValue=ai;var ah=am.substring(aj.index+aj[0].length);if(ah){var ak=al.parentNode;ak.insertBefore(ac.createTextNode(ah),al.nextSibling)}ad(al);if(!ai){al.parentNode.removeChild(al)}}}break}}function ad(ak){while(!ak.nextSibling){ak=ak.parentNode;if(!ak){return}}function ai(al,ar){var aq=ar?al.cloneNode(false):al;var ao=al.parentNode;if(ao){var ap=ai(ao,1);var an=al.nextSibling;ap.appendChild(aq);for(var am=an;am;am=an){an=am.nextSibling;ap.appendChild(am)}}return aq}var ah=ai(ak.nextSibling,0);for(var aj;(aj=ah.parentNode)&&aj.nodeType===1;){ah=aj}W.push(ah)}for(var Y=0;Y=S){ah+=2}if(V>=ap){Z+=2}}}var t={};function c(U,V){for(var S=V.length;--S>=0;){var T=V[S];if(!t.hasOwnProperty(T)){t[T]=U}else{if(window.console){console.warn("cannot override language handler %s",T)}}}}function q(T,S){if(!(T&&t.hasOwnProperty(T))){T=/^\s*]*(?:>|$)/],[j,/^<\!--[\s\S]*?(?:-\->|$)/],["lang-",/^<\?([\s\S]+?)(?:\?>|$)/],["lang-",/^<%([\s\S]+?)(?:%>|$)/],[L,/^(?:<[%?]|[%?]>)/],["lang-",/^]*>([\s\S]+?)<\/xmp\b[^>]*>/i],["lang-js",/^
+
+
+
+