diff --git a/.env.example b/.env.example index 6957df0..d1cfe3a 100644 --- a/.env.example +++ b/.env.example @@ -17,4 +17,6 @@ REDIS_PORT=6379 REDIS_PASSWORD= SMTP_HOST= SMTP_PORT= -RESEND_API_KEY= \ No newline at end of file +RESEND_API_KEY= +OPENSEARCH_HOST=http://localhost +OPENSEARCH_CORE=tethys-records \ No newline at end of file diff --git a/.gitea/workflows/checkReferenceType.yaml b/.gitea/workflows/checkReferenceType.yaml new file mode 100644 index 0000000..ab88380 --- /dev/null +++ b/.gitea/workflows/checkReferenceType.yaml @@ -0,0 +1,78 @@ +# This is a Gitea Actions workflow configuration file for running CI tests on the `feat/checkReferenceType` branch. +# The workflow is named "CI" and runs on the latest Ubuntu environment using a Node.js 20 Docker container. +# It sets up a PostgreSQL service with specified environment variables and health checks. +# The workflow includes the following steps: +# 1. Checkout the repository using the actions/checkout@v3 action. +# 2. Install Node.js dependencies using `npm ci`. +# 3. Create a `.env.test` file by copying from `.env.example`. +# 4. Set up environment variables in the `.env.test` file, including database connection details and other app-specific settings. +# 5. Run functional tests using the `node ace test functional --groups "ReferenceValidation"` command. +name: CI +run-name: Running tests for checkReferenceType branch +on: + push: + branches: + - feat/checkReferenceType + +jobs: + container-job: + runs-on: ubuntu-latest + # Docker Hub image that `container-job` executes in + container: node:20-bullseye + + services: + # Label used to access the service container + postgres: + image: postgres:latest + env: + POSTGRES_USER: alice + POSTGRES_PASSWORD: ${{ secrets.POSTGRES_PASSWORD }} + POSTGRES_DB: tethys_dev + # ports: + # - 5432:5432 + options: | + --health-cmd pg_isready + --health-interval 10s + --health-timeout 5s + --health-retries 5 + + steps: + - name: Checkout repository + uses: actions/checkout@v3 + + # - name: Set up Node.js + # uses: actions/setup-node@v2 + # with: + # node-version: '20' + + - name: Install dependencies + run: npm ci + + - name: Create .env.test file + run: cp .env.example .env.test + + - name: Set up environment variables + run: | + echo "DB_CONNECTION=pg" >> .env.test + echo "PG_HOST=postgres" >> .env.test + echo "PG_PORT=5432" >> .env.test + echo "PG_USER=alice" >> .env.test + echo "PG_PASSWORD=${{ secrets.POSTGRES_PASSWORD }}" >> .env.test + echo "PG_DB_NAME=tethys_dev" >> .env.test + echo "NODE_ENV=test" >> .env.test + echo "ASSETS_DRIVER=fake" >> .env.test + echo "SESSION_DRIVER=memory" >> .env.test + echo "HASH_DRIVER=bcrypt" >> .env.test + echo "HOST=127.0.0.1" >> .env.test + echo "PORT=3333" >> .env.test + echo "APP_NAME=TethysCloud" >> .env.test + echo "APP_URL=http://${HOST}:${PORT}" >> .env.test + echo "CACHE_VIEWS=false" >> .env.test + echo "APP_KEY=pfi5N2ACN4tMJ5d8d8BPHfh3FEuvleej" >> .env.test + echo "DRIVE_DISK=local" >> .env.test + echo "OAI_LIST_SIZE=200" >> .env.test + echo "OPENSEARCH_HOST=${{ secrets.OPENSEARCH_HOST }}" >> .env.test + echo "OPENSEARCH_CORE=tethys-records" >> .env.test + + - name: Run tests + run: node ace test functional --groups "ReferenceValidation" diff --git a/.gitea/workflows/ci.yaml b/.gitea/workflows/ci.yaml index 35a3f43..916f839 100644 --- a/.gitea/workflows/ci.yaml +++ b/.gitea/workflows/ci.yaml @@ -4,7 +4,13 @@ name: CI Pipeline run-name: ${{ github.actor }} is running CI pipeline # trigger build when pushing, or when creating a pull request -on: [push, pull_request] +on: + push: + branches: + - master + pull_request: + branches: + - master jobs: # Label of the container job @@ -12,7 +18,7 @@ jobs: # run build on latest ubuntu runs-on: ubuntu-latest - container: node:18-bullseye + container: node:20-bullseye services: mydb: @@ -70,6 +76,7 @@ jobs: && echo "CACHE_VIEWS=false" >> .env.test && echo "APP_KEY=pfi5N2ACN4tMJ5d8d8BPHfh3FEuvleej" >> .env.test && echo "DRIVE_DISK=local" >> .env.test + && echo "OAI_LIST_SIZE=200" >> .env.test # finally run the tests # - run: npm test @@ -95,3 +102,4 @@ jobs: # uses: coverallsapp/github-action@master # with: # github-token: ${{ secrets.GITHUB_TOKEN }} + diff --git a/.gitignore b/.gitignore index f0bd219..6d6d265 100644 --- a/.gitignore +++ b/.gitignore @@ -7,3 +7,4 @@ coverage tmp docker-compose.yml .env.test +public/assets diff --git a/Dockerfile b/Dockerfile index 0e42158..a5d1263 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,7 +1,7 @@ ################## First Stage - Creating base ######################### # Created a variable to hold our node base image -ARG NODE_IMAGE=node:20-bookworm-slim +ARG NODE_IMAGE=node:22-bookworm-slim FROM $NODE_IMAGE AS base # Install dumb-init and ClamAV, and perform ClamAV database update diff --git a/ace.js b/ace.js index a313518..98fb5cb 100644 --- a/ace.js +++ b/ace.js @@ -15,10 +15,11 @@ /** * Register hook to process TypeScript files using ts-node */ -import { register } from 'node:module' -register('ts-node/esm', import.meta.url) +// import { register } from 'node:module'; +// register('ts-node/esm', import.meta.url); +import 'ts-node-maintained/register/esm'; /** * Import ace console entrypoint */ -await import('./bin/console.js') +await import('./bin/console.js'); diff --git a/adonisrc.ts b/adonisrc.ts index 0c424f9..19e946c 100644 --- a/adonisrc.ts +++ b/adonisrc.ts @@ -1,7 +1,7 @@ -import { defineConfig } from '@adonisjs/core/app' +import { defineConfig } from '@adonisjs/core/app'; export default defineConfig({ - /* + /* |-------------------------------------------------------------------------- | Commands |-------------------------------------------------------------------------- @@ -10,12 +10,11 @@ export default defineConfig({ | will be scanned automatically from the "./commands" directory. */ - commands: [ - () => import('@adonisjs/core/commands'), - () => import('@adonisjs/lucid/commands'), - () => import('@adonisjs/mail/commands') - ], - /* + commands: [ + () => import('@adonisjs/core/commands'), + () => import('@adonisjs/lucid/commands'), + () => import('@adonisjs/mail/commands')], + /* |-------------------------------------------------------------------------- | Preloads |-------------------------------------------------------------------------- @@ -23,19 +22,22 @@ export default defineConfig({ | List of modules to import before starting the application. | */ - preloads: [ - () => import('./start/routes.js'), - () => import('./start/kernel.js'), - () => import('#start/validator'), - () => import('#start/rules/unique'), - () => import('#start/rules/translated_language'), - () => import('#start/rules/unique_person'), - () => import('#start/rules/file_length'), - () => import('#start/rules/file_scan'), - () => import('#start/rules/allowed_extensions_mimetypes'), - () => import('#start/rules/dependent_array_min_length') - ], - /* + preloads: [ + () => import('./start/routes.js'), + () => import('./start/kernel.js'), + () => import('#start/validator'), + () => import('#start/rules/unique'), + () => import('#start/rules/translated_language'), + () => import('#start/rules/unique_person'), + () => import('#start/rules/file_length'), + () => import('#start/rules/file_scan'), + () => import('#start/rules/allowed_extensions_mimetypes'), + () => import('#start/rules/dependent_array_min_length'), + () => import('#start/rules/referenceValidation'), + () => import('#start/rules/valid_mimetype'), + () => import('#start/rules/array_contains_types'), + ], + /* |-------------------------------------------------------------------------- | Service providers |-------------------------------------------------------------------------- @@ -44,48 +46,49 @@ export default defineConfig({ | application | */ - providers: [ - // () => import('./providers/AppProvider.js'), - () => import('@adonisjs/core/providers/app_provider'), - () => import('@adonisjs/core/providers/hash_provider'), - { - file: () => import('@adonisjs/core/providers/repl_provider'), - environment: ['repl', 'test'], - }, - () => import('@adonisjs/session/session_provider'), - () => import('@adonisjs/core/providers/edge_provider'), - () => import('@adonisjs/shield/shield_provider'), - // () => import('@eidellev/inertia-adonisjs'), - // () => import('@adonisjs/inertia/inertia_provider'), - () => import('#providers/app_provider'), - () => import('#providers/inertia_provider'), - () => import('@adonisjs/lucid/database_provider'), - () => import('@adonisjs/auth/auth_provider'), - // () => import('@eidellev/adonis-stardust'), - () => import('@adonisjs/redis/redis_provider'), - () => import('@adonisjs/encore/encore_provider'), - () => import('@adonisjs/static/static_provider'), - () => import('#providers/stardust_provider'), - () => import('#providers/query_builder_provider'), - () => import('#providers/token_worker_provider'), - // () => import('#providers/validator_provider'), - () => import('#providers/drive/provider/drive_provider'), - // () => import('@adonisjs/core/providers/vinejs_provider'), - () => import('#providers/vinejs_provider'), - () => import('@adonisjs/mail/mail_provider') - // () => import('#providers/mail_provider'), - ], - metaFiles: [ - { - pattern: 'public/**', - reloadServer: false, - }, - { - pattern: 'resources/views/**/*.edge', - reloadServer: false, - }, - ], - /* + providers: [ + // () => import('./providers/AppProvider.js'), + () => import('@adonisjs/core/providers/app_provider'), + () => import('@adonisjs/core/providers/hash_provider'), + { + file: () => import('@adonisjs/core/providers/repl_provider'), + environment: ['repl', 'test'], + }, + () => import('@adonisjs/session/session_provider'), + () => import('@adonisjs/core/providers/edge_provider'), + () => import('@adonisjs/shield/shield_provider'), + // () => import('@eidellev/inertia-adonisjs'), + // () => import('@adonisjs/inertia/inertia_provider'), + () => import('#providers/app_provider'), + () => import('#providers/inertia_provider'), + () => import('@adonisjs/lucid/database_provider'), + () => import('@adonisjs/auth/auth_provider'), + // () => import('@eidellev/adonis-stardust'), + () => import('@adonisjs/redis/redis_provider'), + // () => import('@adonisjs/encore/encore_provider'), + () => import('@adonisjs/static/static_provider'), + () => import('#providers/stardust_provider'), + () => import('#providers/query_builder_provider'), + () => import('#providers/token_worker_provider'), + // () => import('#providers/validator_provider'), + // () => import('#providers/drive/provider/drive_provider'), + () => import('@adonisjs/drive/drive_provider'), + // () => import('@adonisjs/core/providers/vinejs_provider'), + () => import('#providers/vinejs_provider'), + () => import('@adonisjs/mail/mail_provider'), + () => import('@adonisjs/vite/vite_provider'), + ], + metaFiles: [ + { + pattern: 'public/**', + reloadServer: false, + }, + { + pattern: 'resources/views/**/*.edge', + reloadServer: false, + }, + ], + /* |-------------------------------------------------------------------------- | Tests |-------------------------------------------------------------------------- @@ -94,22 +97,24 @@ export default defineConfig({ | and add additional suites. | */ - tests: { - suites: [ - { - files: ['tests/unit/**/*.spec(.ts|.js)'], - name: 'unit', - timeout: 2000, - }, - { - files: ['tests/functional/**/*.spec(.ts|.js)'], - name: 'functional', - timeout: 30000, - }, - ], - forceExit: false, - }, - - - -}) + tests: { + suites: [ + { + files: ['tests/unit/**/*.spec(.ts|.js)'], + name: 'unit', + timeout: 2000, + }, + { + files: ['tests/functional/**/*.spec(.ts|.js)'], + name: 'functional', + timeout: 30000, + }, + ], + forceExit: false, + }, + assetsBundler: false, + hooks: { + onBuildStarting: [() => import('@adonisjs/vite/build_hook')], + }, + // assetsBundler: false +}); diff --git a/app/Controllers/Http/Admin/AdminuserController.ts b/app/Controllers/Http/Admin/AdminuserController.ts index ce99724..2962cbd 100644 --- a/app/Controllers/Http/Admin/AdminuserController.ts +++ b/app/Controllers/Http/Admin/AdminuserController.ts @@ -85,7 +85,9 @@ export default class AdminuserController { // return response.badRequest(error.messages); throw error; } - const input = request.only(['login', 'email', 'password', 'first_name', 'last_name']); + + const input: Record = request.only(['login', 'email','first_name', 'last_name']); + input.password = request.input('new_password'); const user = await User.create(input); if (request.input('roles')) { const roles: Array = request.input('roles'); @@ -95,7 +97,6 @@ export default class AdminuserController { session.flash('message', 'User has been created successfully'); return response.redirect().toRoute('settings.user.index'); } - public async show({ request, inertia }: HttpContext) { const id = request.param('id'); const user = await User.query().where('id', id).firstOrFail(); @@ -139,9 +140,11 @@ export default class AdminuserController { }); // password is optional - let input; - if (request.input('password')) { - input = request.only(['login', 'email', 'password', 'first_name', 'last_name']); + let input: Record; + + if (request.input('new_password')) { + input = request.only(['login', 'email', 'first_name', 'last_name']); + input.password = request.input('new_password'); } else { input = request.only(['login', 'email', 'first_name', 'last_name']); } @@ -156,7 +159,6 @@ export default class AdminuserController { session.flash('message', 'User has been updated successfully'); return response.redirect().toRoute('settings.user.index'); } - public async destroy({ request, response, session }: HttpContext) { const id = request.param('id'); const user = await User.findOrFail(id); diff --git a/app/Controllers/Http/Admin/MimetypeController.ts b/app/Controllers/Http/Admin/MimetypeController.ts index 716c4fb..1d9d2a7 100644 --- a/app/Controllers/Http/Admin/MimetypeController.ts +++ b/app/Controllers/Http/Admin/MimetypeController.ts @@ -25,6 +25,7 @@ export default class MimetypeController { const newDatasetSchema = vine.object({ name: vine.string().trim().isUnique({ table: 'mime_types', column: 'name' }), file_extension: vine.array(vine.string()).minLength(1), // define at least one extension for the new mimetype + alternate_mimetype: vine.array(vine.string().isValidMimetype()).distinct().optional(), // define alias mimetypes enabled: vine.boolean(), }); // await request.validate({ schema: newDatasetSchema, messages: this.messages }); @@ -32,18 +33,22 @@ export default class MimetypeController { // Step 2 - Validate request body against the schema // await request.validate({ schema: newDatasetSchema, messages: this.messages }); const validator = vine.compile(newDatasetSchema); - validator.messagesProvider = new SimpleMessagesProvider(this.messages); - await request.validateUsing(validator); + validator.messagesProvider = new SimpleMessagesProvider(this.messages); + await request.validateUsing(validator, { messagesProvider: new SimpleMessagesProvider(this.messages) }); } catch (error) { // Step 3 - Handle errors // return response.badRequest(error.messages); throw error; } - const input = request.only(['name', 'enabled', 'file_extension']); + const input = request.only(['name', 'enabled', 'file_extension', 'alternate_mimetype']); // Concatenate the file_extensions array into a string with '|' as the separator if (Array.isArray(input.file_extension)) { input.file_extension = input.file_extension.join('|'); } + // Concatenate the alias_mimetype array into a string with '|' as the separator + if (Array.isArray(input.alternate_mimetype)) { + input.alternate_mimetype = input.alternate_mimetype.join('|'); + } await MimeType.create(input); // if (request.input('roles')) { // const roles: Array = request.input('roles'); diff --git a/app/Controllers/Http/Api/AuthorsController.ts b/app/Controllers/Http/Api/AuthorsController.ts index 599da80..7209699 100644 --- a/app/Controllers/Http/Api/AuthorsController.ts +++ b/app/Controllers/Http/Api/AuthorsController.ts @@ -9,12 +9,15 @@ export default class AuthorsController { // where exists (select * from gba.documents inner join gba.link_documents_persons on "documents"."id" = "link_documents_persons"."document_id" // where ("link_documents_persons"."role" = 'author') and ("persons"."id" = "link_documents_persons"."person_id")); const authors = await Person.query() + .preload('datasets') + .where('name_type', 'Personal') .whereHas('datasets', (dQuery) => { dQuery.wherePivot('role', 'author'); }) .withCount('datasets', (query) => { query.as('datasets_count'); - }); + }) + .orderBy('datasets_count', 'desc'); return authors; } @@ -25,7 +28,10 @@ export default class AuthorsController { if (request.input('filter')) { // users = users.whereRaw('name like %?%', [request.input('search')]) const searchTerm = request.input('filter'); - authors.whereILike('first_name', `%${searchTerm}%`).orWhereILike('last_name', `%${searchTerm}%`); + authors.andWhere((query) => { + query.whereILike('first_name', `%${searchTerm}%`) + .orWhereILike('last_name', `%${searchTerm}%`); + }); // .orWhere('email', 'like', `%${searchTerm}%`); } diff --git a/app/Controllers/Http/Api/AvatarController.ts b/app/Controllers/Http/Api/AvatarController.ts index 96b158c..74fe135 100644 --- a/app/Controllers/Http/Api/AvatarController.ts +++ b/app/Controllers/Http/Api/AvatarController.ts @@ -1,65 +1,135 @@ import type { HttpContext } from '@adonisjs/core/http'; import { StatusCodes } from 'http-status-codes'; -// import * as fs from 'fs'; -// import * as path from 'path'; +import redis from '@adonisjs/redis/services/main'; -const prefixes = ['von', 'van']; +const PREFIXES = ['von', 'van']; +const DEFAULT_SIZE = 50; +const FONT_SIZE_RATIO = 0.4; +const COLOR_LIGHTENING_PERCENT = 60; +const COLOR_DARKENING_FACTOR = 0.6; -// node ace make:controller Author export default class AvatarController { public async generateAvatar({ request, response }: HttpContext) { try { - const { name, background, textColor, size } = request.only(['name', 'background', 'textColor', 'size']); + const { name, size = DEFAULT_SIZE } = request.only(['name', 'size']); + if (!name) { + return response.status(StatusCodes.BAD_REQUEST).json({ error: 'Name is required' }); + } + + // Build a unique cache key for the given name and size + const cacheKey = `avatar:${name.trim().toLowerCase()}-${size}`; + const cachedSvg = await redis.get(cacheKey); + if (cachedSvg) { + this.setResponseHeaders(response); + return response.send(cachedSvg); + } - // Generate initials - // const initials = name - // .split(' ') - // .map((part) => part.charAt(0).toUpperCase()) - // .join(''); const initials = this.getInitials(name); + const colors = this.generateColors(name); + const svgContent = this.createSvg(size, colors, initials); - // Define SVG content with dynamic values for initials, background color, text color, and size - const svgContent = ` - - - ${initials} - - `; - - // Set response headers for SVG content - response.header('Content-type', 'image/svg+xml'); - response.header('Cache-Control', 'no-cache'); - response.header('Pragma', 'no-cache'); - response.header('Expires', '0'); + // // Cache the generated avatar for future use, e.g. 1 hour expiry + await redis.setex(cacheKey, 3600, svgContent); + this.setResponseHeaders(response); return response.send(svgContent); } catch (error) { - return response.status(StatusCodes.OK).json({ error: error.message }); + return response.status(StatusCodes.INTERNAL_SERVER_ERROR).json({ error: error.message }); } } - private getInitials(name: string) { - const parts = name.split(' '); - let initials = ''; + private getInitials(name: string): string { + const parts = name + .trim() + .split(' ') + .filter((part) => part.length > 0); + + if (parts.length === 0) { + return 'NA'; + } if (parts.length >= 2) { - const firstName = parts[0]; - const lastName = parts[parts.length - 1]; + return this.getMultiWordInitials(parts); + } + return parts[0].substring(0, 2).toUpperCase(); + } - const firstInitial = firstName.charAt(0).toUpperCase(); - const lastInitial = lastName.charAt(0).toUpperCase(); + private getMultiWordInitials(parts: string[]): string { + const firstName = parts[0]; + const lastName = parts[parts.length - 1]; + const firstInitial = firstName.charAt(0).toUpperCase(); + const lastInitial = lastName.charAt(0).toUpperCase(); - if (prefixes.includes(lastName.toLowerCase()) && lastName === lastName.toUpperCase()) { - initials = firstInitial + lastName.charAt(1).toUpperCase(); - } else { - initials = firstInitial + lastInitial; - } - } else if (parts.length === 1) { - initials = parts[0].substring(0, 2).toUpperCase(); + if (PREFIXES.includes(lastName.toLowerCase()) && lastName === lastName.toUpperCase()) { + return firstInitial + lastName.charAt(1).toUpperCase(); + } + return firstInitial + lastInitial; + } + + private generateColors(name: string): { background: string; text: string } { + const baseColor = this.getColorFromName(name); + return { + background: this.lightenColor(baseColor, COLOR_LIGHTENING_PERCENT), + text: this.darkenColor(baseColor), + }; + } + + private createSvg(size: number, colors: { background: string; text: string }, initials: string): string { + const fontSize = size * FONT_SIZE_RATIO; + return ` + + + ${initials} + + `; + } + + private setResponseHeaders(response: HttpContext['response']): void { + response.header('Content-type', 'image/svg+xml'); + response.header('Cache-Control', 'no-cache'); + response.header('Pragma', 'no-cache'); + response.header('Expires', '0'); + } + + private getColorFromName(name: string): string { + let hash = 0; + for (let i = 0; i < name.length; i++) { + hash = name.charCodeAt(i) + ((hash << 5) - hash); } - return initials; + const colorParts = []; + for (let i = 0; i < 3; i++) { + const value = (hash >> (i * 8)) & 0xff; + colorParts.push(value.toString(16).padStart(2, '0')); + } + return colorParts.join(''); + } + + private lightenColor(hexColor: string, percent: number): string { + const r = parseInt(hexColor.substring(0, 2), 16); + const g = parseInt(hexColor.substring(2, 4), 16); + const b = parseInt(hexColor.substring(4, 6), 16); + + const lightenValue = (value: number) => Math.min(255, Math.floor((value * (100 + percent)) / 100)); + + const newR = lightenValue(r); + const newG = lightenValue(g); + const newB = lightenValue(b); + + return ((newR << 16) | (newG << 8) | newB).toString(16).padStart(6, '0'); + } + + private darkenColor(hexColor: string): string { + const r = parseInt(hexColor.slice(0, 2), 16); + const g = parseInt(hexColor.slice(2, 4), 16); + const b = parseInt(hexColor.slice(4, 6), 16); + + const darkenValue = (value: number) => Math.round(value * COLOR_DARKENING_FACTOR); + + const darkerR = darkenValue(r); + const darkerG = darkenValue(g); + const darkerB = darkenValue(b); + + return ((darkerR << 16) + (darkerG << 8) + darkerB).toString(16).padStart(6, '0'); } } diff --git a/app/Controllers/Http/Api/DatasetController.ts b/app/Controllers/Http/Api/DatasetController.ts index 41befe6..b80949d 100644 --- a/app/Controllers/Http/Api/DatasetController.ts +++ b/app/Controllers/Http/Api/DatasetController.ts @@ -6,10 +6,15 @@ import { StatusCodes } from 'http-status-codes'; // node ace make:controller Author export default class DatasetController { public async index({}: HttpContext) { - // select * from gba.persons - // where exists (select * from gba.documents inner join gba.link_documents_persons on "documents"."id" = "link_documents_persons"."document_id" - // where ("link_documents_persons"."role" = 'author') and ("persons"."id" = "link_documents_persons"."person_id")); - const datasets = await Dataset.query().where('server_state', 'published').orWhere('server_state', 'deleted'); + // Select datasets with server_state 'published' or 'deleted' and sort by the last published date + const datasets = await Dataset.query() + .where(function (query) { + query.where('server_state', 'published') + .orWhere('server_state', 'deleted'); + }) + .preload('titles') + .preload('identifier') + .orderBy('server_date_published', 'desc'); return datasets; } diff --git a/app/Controllers/Http/Api/FileController.ts b/app/Controllers/Http/Api/FileController.ts index 3c1f3d4..8080479 100644 --- a/app/Controllers/Http/Api/FileController.ts +++ b/app/Controllers/Http/Api/FileController.ts @@ -14,7 +14,7 @@ export default class FileController { // where: { id: id }, // }); if (file) { - const filePath = '/storage/app/public/' + file.pathName; + const filePath = '/storage/app/data/' + file.pathName; const ext = path.extname(filePath); const fileName = file.label + ext; try { diff --git a/app/Controllers/Http/Api/HomeController.ts b/app/Controllers/Http/Api/HomeController.ts index 020f651..fdec7f9 100644 --- a/app/Controllers/Http/Api/HomeController.ts +++ b/app/Controllers/Http/Api/HomeController.ts @@ -17,7 +17,8 @@ export default class HomeController { // .preload('authors') // .orderBy('server_date_published'); - const datasets = await db.from('documents as doc') + const datasets = await db + .from('documents as doc') .select(['publish_id', 'server_date_published', db.raw(`date_part('year', server_date_published) as pub_year`)]) .where('server_state', serverState) .innerJoin('link_documents_persons as ba', 'doc.id', 'ba.document_id') @@ -59,7 +60,6 @@ export default class HomeController { // const year = params.year; // const from = parseInt(year); try { - // const datasets = await Database.from('documents as doc') // .select([Database.raw(`date_part('month', server_date_published) as pub_month`), Database.raw('COUNT(*) as count')]) // .where('server_state', serverState) @@ -68,9 +68,12 @@ export default class HomeController { // .groupBy('pub_month'); // // .orderBy('server_date_published'); - const years = [2021, 2022, 2023]; // Add the second year + // Calculate the last 4 years including the current year + const currentYear = new Date().getFullYear(); + const years = Array.from({ length: 4 }, (_, i) => currentYear - (i + 1)).reverse(); - const result = await db.from('documents as doc') + const result = await db + .from('documents as doc') .select([ db.raw(`date_part('year', server_date_published) as pub_year`), db.raw(`date_part('month', server_date_published) as pub_month`), @@ -83,7 +86,7 @@ export default class HomeController { .groupBy('pub_year', 'pub_month') .orderBy('pub_year', 'asc') .orderBy('pub_month', 'asc'); - + const labels = Array.from({ length: 12 }, (_, i) => i + 1); // Assuming 12 months const inputDatasets: Map = result.reduce((acc, item) => { @@ -100,15 +103,15 @@ export default class HomeController { acc[pub_year].data[pub_month - 1] = parseInt(count); - return acc ; + return acc; }, {}); const outputDatasets = Object.entries(inputDatasets).map(([year, data]) => ({ data: data.data, label: year, borderColor: data.borderColor, - fill: data.fill - })); + fill: data.fill, + })); const data = { labels: labels, @@ -126,11 +129,11 @@ export default class HomeController { private getRandomHexColor() { const letters = '0123456789ABCDEF'; let color = '#'; - + for (let i = 0; i < 6; i++) { color += letters[Math.floor(Math.random() * 16)]; } - + return color; } } @@ -139,5 +142,4 @@ interface ChartDataset { label: string; borderColor: string; fill: boolean; - } diff --git a/app/Controllers/Http/Api/UserController.ts b/app/Controllers/Http/Api/UserController.ts index 975e5e2..1d7a14a 100644 --- a/app/Controllers/Http/Api/UserController.ts +++ b/app/Controllers/Http/Api/UserController.ts @@ -9,6 +9,24 @@ import BackupCode from '#models/backup_code'; // Here we are generating secret and recovery codes for the user that’s enabling 2FA and storing them to our database. export default class UserController { + public async getSubmitters({ response }: HttpContext) { + try { + const submitters = await User.query() + .preload('roles', (query) => { + query.where('name', 'submitter') + }) + .whereHas('roles', (query) => { + query.where('name', 'submitter') + }) + .exec(); + return submitters; + } catch (error) { + return response.status(StatusCodes.INTERNAL_SERVER_ERROR).json({ + message: 'Invalid TOTP state', + }); + } + } + public async enable({ auth, response, request }: HttpContext) { const user = (await User.find(auth.user?.id)) as User; // await user.load('totp_secret'); diff --git a/app/Controllers/Http/Api/collections_controller.ts b/app/Controllers/Http/Api/collections_controller.ts new file mode 100644 index 0000000..04dcdb2 --- /dev/null +++ b/app/Controllers/Http/Api/collections_controller.ts @@ -0,0 +1,36 @@ +import type { HttpContext } from '@adonisjs/core/http'; +import Collection from '#models/collection'; + +export default class CollectionsController { + public async show({ params, response }: HttpContext) { + // Get the collection id from route parameters + const collectionId = params.id; + + // Find the selected collection by id + const collection = await Collection.find(collectionId); + if (!collection) { + return response.status(404).json({ message: 'Collection not found' }); + } + + // Query for narrower concepts: collections whose parent_id equals the selected collection's id + const narrowerCollections = await Collection.query().where('parent_id', collection.id) || []; + + // For broader concept, if the selected collection has a parent_id fetch that record (otherwise null) + const broaderCollection: Collection[] | never[] | null = await (async () => { + if (collection.parent_id) { + // Try to fetch the parent... + const parent = await Collection.find(collection.parent_id) + // If found, return it wrapped in an array; if not found, return null (or empty array if you prefer) + return parent ? [parent] : null + } + return [] + })() + + // Return the selected collection along with its narrower and broader concepts in JSON format + return response.json({ + selectedCollection: collection, + narrowerCollections, + broaderCollection, + }); + } +} diff --git a/app/Controllers/Http/Auth/AuthController.ts b/app/Controllers/Http/Auth/AuthController.ts index 71256f2..08b1c05 100644 --- a/app/Controllers/Http/Auth/AuthController.ts +++ b/app/Controllers/Http/Auth/AuthController.ts @@ -5,7 +5,7 @@ import BackupCode from '#models/backup_code'; // import InvalidCredentialException from 'App/Exceptions/InvalidCredentialException'; import { authValidator } from '#validators/auth'; import hash from '@adonisjs/core/services/hash'; - +import db from '@adonisjs/lucid/services/db'; import TwoFactorAuthProvider from '#app/services/TwoFactorAuthProvider'; // import { Authenticator } from '@adonisjs/auth'; // import { LoginState } from 'Contracts/enums'; @@ -29,6 +29,10 @@ export default class AuthController { const { email, password } = request.only(['email', 'password']); try { + + await db.connection().rawQuery('SELECT 1') + + // // attempt to verify credential and login user // await auth.use('web').attempt(email, plainPassword); @@ -51,6 +55,9 @@ export default class AuthController { await auth.use('web').login(user); } catch (error) { + if (error.code === 'ECONNREFUSED') { + throw error + } // if login fails, return vague form message and redirect back session.flash('message', 'Your username, email, or password is incorrect'); return response.redirect().back(); diff --git a/app/Controllers/Http/Auth/UserController.ts b/app/Controllers/Http/Auth/UserController.ts index 442aaca..46a8d48 100644 --- a/app/Controllers/Http/Auth/UserController.ts +++ b/app/Controllers/Http/Auth/UserController.ts @@ -6,6 +6,11 @@ import hash from '@adonisjs/core/services/hash'; // import { schema, rules } from '@adonisjs/validator'; import vine from '@vinejs/vine'; import BackupCodeStorage, { SecureRandom } from '#services/backup_code_storage'; +import path from 'path'; +import crypto from 'crypto'; +// import drive from '#services/drive'; +import drive from '@adonisjs/drive/services/main'; +import logger from '@adonisjs/core/services/logger'; // Here we are generating secret and recovery codes for the user that’s enabling 2FA and storing them to our database. export default class UserController { @@ -28,7 +33,7 @@ export default class UserController { user: user, twoFactorEnabled: user.isTwoFactorEnabled, // code: await TwoFactorAuthProvider.generateQrCode(user), - backupState: backupState, + backupState: backupState, }); } @@ -40,10 +45,8 @@ export default class UserController { // }); const passwordSchema = vine.object({ // first step - old_password: vine - .string() - .trim() - .regex(/^[a-zA-Z0-9]+$/), + old_password: vine.string().trim(), + // .regex(/^[a-zA-Z0-9]+$/), new_password: vine.string().confirmed({ confirmationField: 'confirm_password' }).trim().minLength(8).maxLength(255), }); try { @@ -54,9 +57,9 @@ export default class UserController { // return response.badRequest(error.messages); throw error; } - + try { - const user = await auth.user as User; + const user = (await auth.user) as User; const { old_password, new_password } = request.only(['old_password', 'new_password']); // if (!(old_password && new_password && confirm_password)) { @@ -82,6 +85,171 @@ export default class UserController { } } + public async profile({ inertia, auth }: HttpContext) { + const user = await User.find(auth.user?.id); + // let test = await drive.use().getUrl(user?.avatar); + // user?.preload('roles'); + const avatarFullPathUrl = user?.avatar ? await drive.use('public').getUrl(user.avatar) : null; + return inertia.render('profile/show', { + user: user, + defaultUrl: avatarFullPathUrl, + }); + } + + /** + * Update the user's profile information. + * + * @param {HttpContext} ctx - The HTTP context object. + * @returns {Promise} + */ + public async profileUpdate({ auth, request, response, session }: HttpContext) { + if (!auth.user) { + session.flash('error', 'You must be logged in to update your profile.'); + return response.redirect().toRoute('login'); + } + + const updateProfileValidator = vine.withMetaData<{ userId: number }>().compile( + vine.object({ + first_name: vine.string().trim().minLength(4).maxLength(255), + last_name: vine.string().trim().minLength(4).maxLength(255), + login: vine.string().trim().minLength(4).maxLength(255), + email: vine + .string() + .trim() + .maxLength(255) + .email() + .normalizeEmail() + .isUnique({ table: 'accounts', column: 'email', whereNot: (field) => field.meta.userId }), + avatar: vine + .myfile({ + size: '2mb', + extnames: ['jpg', 'jpeg', 'png', 'gif', 'webp', 'svg'], + }) + // .allowedMimetypeExtensions({ + // allowedExtensions: ['jpg', 'jpeg', 'png', 'gif', 'webp', 'svg'], + // }) + .optional(), + }), + ); + + const user = await User.find(auth.user.id); + if (!user) { + session.flash('error', 'User not found.'); + return response.redirect().toRoute('login'); + } + + try { + // validate update form + await request.validateUsing(updateProfileValidator, { + meta: { + userId: user.id, + }, + }); + + const { login, email, first_name, last_name } = request.only(['login', 'email', 'first_name', 'last_name']); + const sanitizedData: { [key: string]: any } = { + login: login?.trim(), + email: email?.toLowerCase().trim(), + first_name: first_name?.trim(), + last_name: last_name?.trim(), + // avatar: "", + }; + const toCamelCase = (str: string) => str.replace(/_([a-z])/g, (g) => g[1].toUpperCase()); + const hasInputChanges = Object.keys(sanitizedData).some((key) => { + const camelKey = toCamelCase(key); + return sanitizedData[key] !== (user.$attributes as { [key: string]: any })[camelKey]; + }); + + let hasAvatarChanged = false; + const avatar = request.file('avatar'); + if (avatar) { + const fileHash = crypto + .createHash('sha256') + .update(avatar.clientName + avatar.size) + .digest('hex'); + const fileName = `avatar-${fileHash}.${avatar.extname}`; + const avatarFullPath = path.join('/uploads', `${user.login}`, fileName); + + if (user.avatar != avatarFullPath) { + if (user.avatar) { + await drive.use('public').delete(user.avatar); + } + hasAvatarChanged = user.avatar !== avatarFullPath; + await avatar.moveToDisk(avatarFullPath, 'public', { + name: fileName, + overwrite: true, // overwrite in case of conflict + disk: 'public', + }); + sanitizedData.avatar = avatarFullPath; + } + } + + if (!hasInputChanges && !hasAvatarChanged) { + session.flash('message', 'No changes were made.'); + return response.redirect().back(); + } + + await user.merge(sanitizedData).save(); + session.flash('message', 'User has been updated successfully'); + return response.redirect().toRoute('settings.profile.edit'); + } catch (error) { + logger.error('Profile update failed:', error); + // session.flash('errors', 'Profile update failed. Please try again.'); + // return response.redirect().back(); + throw error; + } + } + + public async passwordUpdate({ auth, request, response, session }: HttpContext) { + // const passwordSchema = schema.create({ + // old_password: schema.string({ trim: true }, [rules.required()]), + // new_password: schema.string({ trim: true }, [rules.minLength(8), rules.maxLength(255), rules.confirmed('confirm_password')]), + // confirm_password: schema.string({ trim: true }, [rules.required()]), + // }); + const passwordSchema = vine.object({ + // first step + old_password: vine.string().trim(), + // .regex(/^[a-zA-Z0-9]+$/), + new_password: vine.string().confirmed({ confirmationField: 'confirm_password' }).trim().minLength(8).maxLength(255), + }); + try { + // await request.validate({ schema: passwordSchema }); + const validator = vine.compile(passwordSchema); + await request.validateUsing(validator); + } catch (error) { + // return response.badRequest(error.messages); + throw error; + } + + try { + const user = (await auth.user) as User; + const { old_password, new_password } = request.only(['old_password', 'new_password']); + + // if (!(old_password && new_password && confirm_password)) { + // return response.status(400).send({ warning: 'Old password and new password are required.' }); + // } + + // Verify if the provided old password matches the user's current password + const isSame = await hash.verify(user.password, old_password); + if (!isSame) { + session.flash('warning', 'Old password is incorrect.'); + return response.redirect().back(); + // return response.flash('warning', 'Old password is incorrect.').redirect().back(); + } + + // Hash the new password before updating the user's password + user.password = new_password; + await user.save(); + + // return response.status(200).send({ message: 'Password updated successfully.' }); + session.flash({ message: 'Password updated successfully.' }); + return response.redirect().toRoute('settings.profile.edit'); + } catch (error) { + // return response.status(500).send({ message: 'Internal server error.' }); + return response.flash('warning', `Invalid server state. Internal server error.`).redirect().back(); + } + } + public async enableTwoFactorAuthentication({ auth, response, session }: HttpContext): Promise { // const user: User | undefined = auth?.user; const user = (await User.find(auth.user?.id)) as User; @@ -115,7 +283,7 @@ export default class UserController { } else { session.flash('error', 'User not found.'); } - + return response.redirect().back(); // return inertia.render('Auth/AccountInfo', { // // status: { diff --git a/app/Controllers/Http/Editor/DatasetController.ts b/app/Controllers/Http/Editor/DatasetController.ts index acaf505..23466d9 100644 --- a/app/Controllers/Http/Editor/DatasetController.ts +++ b/app/Controllers/Http/Editor/DatasetController.ts @@ -18,9 +18,33 @@ import { HttpException } from 'node-exceptions'; import { ModelQueryBuilderContract } from '@adonisjs/lucid/types/model'; import vine, { SimpleMessagesProvider } from '@vinejs/vine'; import mail from '@adonisjs/mail/services/main'; -// import { resolveMx } from 'dns/promises'; -// import * as net from 'net'; import { validate } from 'deep-email-validator'; +import { + TitleTypes, + DescriptionTypes, + ContributorTypes, + PersonNameTypes, + ReferenceIdentifierTypes, + RelationTypes, + SubjectTypes, + DatasetTypes, +} from '#contracts/enums'; +import { TransactionClientContract } from '@adonisjs/lucid/types/database'; +import db from '@adonisjs/lucid/services/db'; +import Project from '#models/project'; +import License from '#models/license'; +import Language from '#models/language'; +import File from '#models/file'; +import Coverage from '#models/coverage'; +import Title from '#models/title'; +import Description from '#models/description'; +import Subject from '#models/subject'; +import DatasetReference from '#models/dataset_reference'; +import Collection from '#models/collection'; +import CollectionRole from '#models/collection_role'; +import { updateEditorDatasetValidator } from '#validators/dataset'; +import { savePersons } from '#app/utils/utility-functions'; + // Create a new instance of the client const client = new Client({ node: 'http://localhost:9200' }); // replace with your OpenSearch endpoint @@ -63,8 +87,15 @@ export default class DatasetsController { } datasets.orderBy(attribute, sortOrder); } else { - // users.orderBy('created_at', 'desc'); - datasets.orderBy('id', 'asc'); + // datasets.orderBy('id', 'asc'); + // Custom ordering to prioritize rejected_editor state + datasets.orderByRaw(` + CASE + WHEN server_state = 'rejected_reviewer' THEN 0 + ELSE 1 + END ASC, + id ASC + `); } // const users = await User.query().orderBy('login').paginate(page, limit); @@ -217,6 +248,10 @@ export default class DatasetsController { if (dataset.reject_reviewer_note != null) { dataset.reject_reviewer_note = null; } + if (dataset.reject_editor_note != null) { + dataset.reject_editor_note = null; + } + //save main and additional titles const reviewer_id = request.input('reviewer_id', null); @@ -255,70 +290,7 @@ export default class DatasetsController { }); } - // private async checkEmailDomain(email: string): Promise { - // const domain = email.split('@')[1]; - // try { - // // Step 1: Check MX records for the domain - // const mxRecords = await resolveMx(domain); - // if (mxRecords.length === 0) { - // return false; // No MX records, can't send email - // } - - // // Sort MX records by priority - // mxRecords.sort((a, b) => a.priority - b.priority); - - // // Step 2: Attempt SMTP connection to the first available mail server - // const smtpServer = mxRecords[0].exchange; - - // return await this.checkMailboxExists(smtpServer, email); - // } catch (error) { - // console.error('Error during MX lookup or SMTP validation:', error); - // return false; - // } - // } - - //// Helper function to check if the mailbox exists using SMTP - // private async checkMailboxExists(smtpServer: string, email: string): Promise { - // return new Promise((resolve, reject) => { - // const socket = net.createConnection(25, smtpServer); - - // socket.on('connect', () => { - // socket.write(`HELO ${smtpServer}\r\n`); - // socket.write(`MAIL FROM: \r\n`); - // socket.write(`RCPT TO: <${email}>\r\n`); - // }); - - // socket.on('data', (data) => { - // const response = data.toString(); - // if (response.includes('250')) { - // // 250 is an SMTP success code - // socket.end(); - // resolve(true); // Email exists - // } else if (response.includes('550')) { - // // 550 means the mailbox doesn't exist - // socket.end(); - // resolve(false); // Email doesn't exist - // } - // }); - - // socket.on('error', (error) => { - // console.error('SMTP connection error:', error); - // socket.end(); - // resolve(false); - // }); - - // socket.on('end', () => { - // // SMTP connection closed - // }); - - // socket.setTimeout(5000, () => { - // // Timeout after 5 seconds - // socket.end(); - // resolve(false); // Assume email doesn't exist if no response - // }); - // }); - // } public async rejectUpdate({ request, response, auth }: HttpContext) { const authUser = auth.user!; @@ -353,7 +325,7 @@ export default class DatasetsController { return response .flash( `Invalid server state. Dataset with id ${id} cannot be rejected. Datset has server state ${dataset.server_state}.`, - 'warning' + 'warning', ) .redirect() .toRoute('editor.dataset.list'); @@ -388,7 +360,9 @@ export default class DatasetsController { emailStatusMessage = ` A rejection email was successfully sent to ${dataset.user.email}.`; } catch (error) { logger.error(error); - return response.flash('Dataset has not been rejected due to an email error: ' + error.message, 'error').toRoute('editor.dataset.list'); + return response + .flash('Dataset has not been rejected due to an email error: ' + error.message, 'error') + .toRoute('editor.dataset.list'); } } else { emailStatusMessage = ` However, the email could not be sent because the submitter's email address (${dataset.user.email}) is not valid.`; @@ -404,7 +378,7 @@ export default class DatasetsController { .toRoute('editor.dataset.list'); } - public async publish({ request, inertia, response }: HttpContext) { + public async publish({ request, inertia, response, auth }: HttpContext) { const id = request.param('id'); const dataset = await Dataset.query() @@ -428,8 +402,14 @@ export default class DatasetsController { .back(); } + + return inertia.render('Editor/Dataset/Publish', { dataset, + can: { + reject: await auth.user?.can(['dataset-editor-reject']), + publish: await auth.user?.can(['dataset-publish']), + }, }); } @@ -471,6 +451,119 @@ export default class DatasetsController { } } + public async rejectToReviewer({ request, inertia, response }: HttpContext) { + const id = request.param('id'); + const dataset = await Dataset.query() + .where('id', id) + .preload('reviewer', (builder) => { + builder.select('id', 'login', 'email'); + }) + .firstOrFail(); + + const validStates = ['reviewed']; + if (!validStates.includes(dataset.server_state)) { + // session.flash('errors', 'Invalid server state!'); + return response + .flash( + 'warning', + `Invalid server state. Dataset with id ${id} cannot be rejected to the reviewer. Datset has server state ${dataset.server_state}.`, + ) + .redirect() + .toRoute('editor.dataset.list'); + } + + return inertia.render('Editor/Dataset/RejectToReviewer', { + dataset, + }); + } + + public async rejectToReviewerUpdate({ request, response, auth }: HttpContext) { + const authUser = auth.user!; + + const id = request.param('id'); + const dataset = await Dataset.query() + .where('id', id) + .preload('reviewer', (builder) => { + builder.select('id', 'login', 'email'); + }) + .firstOrFail(); + + const newSchema = vine.object({ + server_state: vine.string().trim(), + reject_editor_note: vine.string().trim().minLength(10).maxLength(500), + send_mail: vine.boolean().optional(), + }); + + try { + // await request.validate({ schema: newSchema }); + const validator = vine.compile(newSchema); + await request.validateUsing(validator); + } catch (error) { + // return response.badRequest(error.messages); + throw error; + } + + const validStates = ['reviewed']; + if (!validStates.includes(dataset.server_state)) { + // throw new Error('Invalid server state!'); + // return response.flash('warning', 'Invalid server state. Dataset cannot be released to editor').redirect().back(); + return response + .flash( + `Invalid server state. Dataset with id ${id} cannot be rejected to reviewer. Datset has server state ${dataset.server_state}.`, + 'warning', + ) + .redirect() + .toRoute('editor.dataset.list'); + } + + dataset.server_state = 'rejected_to_reviewer'; + const rejectEditorNote = request.input('reject_editor_note', ''); + dataset.reject_editor_note = rejectEditorNote; + + // add logic for sending reject message + const sendMail = request.input('send_email', false); + // const validRecipientEmail = await this.checkEmailDomain('arno.kaimbacher@outlook.at'); + const validationResult = await validate({ + email: dataset.reviewer.email, + validateSMTP: false, + }); + const validRecipientEmail: boolean = validationResult.valid; + + await dataset.save(); + + let emailStatusMessage = ''; + if (sendMail == true) { + if (dataset.reviewer.email && validRecipientEmail) { + try { + await mail.send((message) => { + message.to(dataset.reviewer.email).subject('Dataset Rejection Notification').html(` +

Dear ${dataset.reviewer.login},

+

Your dataset with ID ${dataset.id} has been rejected.

+

Reason for rejection: ${rejectEditorNote}

+

Best regards,
Your Tethys editor: ${authUser.login}

+ `); + }); + emailStatusMessage = ` A rejection email was successfully sent to ${dataset.reviewer.email}.`; + } catch (error) { + logger.error(error); + return response + .flash('Dataset has not been rejected due to an email error: ' + error.message, 'error') + .toRoute('editor.dataset.list'); + } + } else { + emailStatusMessage = ` However, the email could not be sent because the submitter's email address (${dataset.reviewer.email}) is not valid.`; + } + } + + + return response + .flash( + `You have successfully rejected dataset ${dataset.id} reviewed by ${dataset.reviewer.login}.${emailStatusMessage}`, + 'message', + ) + .toRoute('editor.dataset.list'); + } + public async doiCreate({ request, inertia }: HttpContext) { const id = request.param('id'); const dataset = await Dataset.query() @@ -536,10 +629,375 @@ export default class DatasetsController { public async show({}: HttpContext) {} - public async edit({}: HttpContext) {} + public async edit({ request, inertia, response }: HttpContext) { + const id = request.param('id'); + const datasetQuery = Dataset.query().where('id', id); + datasetQuery + .preload('titles', (query) => query.orderBy('id', 'asc')) + .preload('descriptions', (query) => query.orderBy('id', 'asc')) + .preload('coverage') + .preload('licenses') + .preload('authors', (query) => query.orderBy('pivot_sort_order', 'asc')) + .preload('contributors', (query) => query.orderBy('pivot_sort_order', 'asc')) + // .preload('subjects') + .preload('subjects', (builder) => { + builder.orderBy('id', 'asc').withCount('datasets'); + }) + .preload('references') + .preload('files', (query) => { + query.orderBy('sort_order', 'asc'); // Sort by sort_order column + }); + + const dataset = await datasetQuery.firstOrFail(); + const validStates = ['editor_accepted', 'rejected_reviewer']; + if (!validStates.includes(dataset.server_state)) { + // session.flash('errors', 'Invalid server state!'); + return response + .flash( + `Invalid server state. Dataset with id ${id} cannot be edited. Datset has server state ${dataset.server_state}.`, + 'warning', + ) + .toRoute('editor.dataset.list'); + } + + const titleTypes = Object.entries(TitleTypes) + .filter(([value]) => value !== 'Main') + .map(([key, value]) => ({ value: key, label: value })); + + const descriptionTypes = Object.entries(DescriptionTypes) + .filter(([value]) => value !== 'Abstract') + .map(([key, value]) => ({ value: key, label: value })); + + const languages = await Language.query().where('active', true).pluck('part1', 'part1'); + + // const contributorTypes = Config.get('enums.contributor_types'); + const contributorTypes = Object.entries(ContributorTypes).map(([key, value]) => ({ value: key, label: value })); + + // const nameTypes = Config.get('enums.name_types'); + const nameTypes = Object.entries(PersonNameTypes).map(([key, value]) => ({ value: key, label: value })); + + // const messages = await Database.table('messages') + // .pluck('help_text', 'metadata_element'); + + const projects = await Project.query().pluck('label', 'id'); + + const currentDate = new Date(); + const currentYear = currentDate.getFullYear(); + const years = Array.from({ length: currentYear - 1990 + 1 }, (_, index) => 1990 + index); + + const licenses = await License.query().select('id', 'name_long').where('active', 'true').pluck('name_long', 'id'); + // const userHasRoles = user.roles; + // const datasetHasLicenses = await dataset.related('licenses').query().pluck('id'); + // const checkeds = dataset.licenses.first().id; + + // const doctypes = { + // analysisdata: { label: 'Analysis', value: 'analysisdata' }, + // measurementdata: { label: 'Measurements', value: 'measurementdata' }, + // monitoring: 'Monitoring', + // remotesensing: 'Remote Sensing', + // gis: 'GIS', + // models: 'Models', + // mixedtype: 'Mixed Type', + // }; + + return inertia.render('Editor/Dataset/Edit', { + dataset, + titletypes: titleTypes, + descriptiontypes: descriptionTypes, + contributorTypes, + nameTypes, + languages, + // messages, + projects, + licenses, + // datasetHasLicenses: Object.keys(datasetHasLicenses).map((key) => datasetHasLicenses[key]), //convert object to array with license ids + // checkeds, + years, + // languages, + subjectTypes: SubjectTypes, + referenceIdentifierTypes: Object.entries(ReferenceIdentifierTypes).map(([key, value]) => ({ value: key, label: value })), + relationTypes: Object.entries(RelationTypes).map(([key, value]) => ({ value: key, label: value })), + doctypes: DatasetTypes, + }); + } + + public async update({ request, response, session }: HttpContext) { + // Get the dataset id from the route parameter + const datasetId = request.param('id'); + // Retrieve the dataset and load its existing files + const dataset = await Dataset.findOrFail(datasetId); + await dataset.load('files'); + + let trx: TransactionClientContract | null = null; + try { + await request.validateUsing(updateEditorDatasetValidator); + trx = await db.transaction(); + // const user = (await User.find(auth.user?.id)) as User; + // await this.createDatasetAndAssociations(user, request, trx); + const dataset = await Dataset.findOrFail(datasetId); + + // save the licenses + const licenses: number[] = request.input('licenses', []); + // await dataset.useTransaction(trx).related('licenses').sync(licenses); + await dataset.useTransaction(trx).related('licenses').sync(licenses); + + // save authors and contributors + await dataset.useTransaction(trx).related('authors').sync([]); + await dataset.useTransaction(trx).related('contributors').sync([]); + await savePersons(dataset, request.input('authors', []), 'author', trx); + await savePersons(dataset, request.input('contributors', []), 'contributor', trx); + + //save the titles: + const titles = request.input('titles', []); + // const savedTitles:Array = []; + for (const titleData of titles) { + if (titleData.id) { + const title = await Title.findOrFail(titleData.id); + title.value = titleData.value; + title.language = titleData.language; + title.type = titleData.type; + if (title.$isDirty) { + await title.useTransaction(trx).save(); + // await dataset.useTransaction(trx).related('titles').save(title); + // savedTitles.push(title); + } + } else { + const title = new Title(); + title.fill(titleData); + // savedTitles.push(title); + await dataset.useTransaction(trx).related('titles').save(title); + } + } + + // save the abstracts + const descriptions = request.input('descriptions', []); + // const savedTitles:Array<Title> = []; + for (const descriptionData of descriptions) { + if (descriptionData.id) { + const description = await Description.findOrFail(descriptionData.id); + description.value = descriptionData.value; + description.language = descriptionData.language; + description.type = descriptionData.type; + if (description.$isDirty) { + await description.useTransaction(trx).save(); + // await dataset.useTransaction(trx).related('titles').save(title); + // savedTitles.push(title); + } + } else { + const description = new Description(); + description.fill(descriptionData); + // savedTitles.push(title); + await dataset.useTransaction(trx).related('descriptions').save(description); + } + } + + // Process all subjects/keywords from the request + const subjects = request.input('subjects'); + for (const subjectData of subjects) { + // Case 1: Subject already exists in the database (has an ID) + if (subjectData.id) { + // Retrieve the existing subject + const existingSubject = await Subject.findOrFail(subjectData.id); + + // Update subject properties from the request data + existingSubject.value = subjectData.value; + existingSubject.type = subjectData.type; + existingSubject.external_key = subjectData.external_key; + + // Only save if there are actual changes + if (existingSubject.$isDirty) { + await existingSubject.save(); + } + + // Note: The relationship between dataset and subject is already established, + // so we don't need to attach it again + } + // Case 2: New subject being added (no ID) + else { + // Check if a subject with the same value and type already exists in the database + const subject = await Subject.firstOrNew({ value: subjectData.value, type: subjectData.type }, subjectData); + + if (subject.$isNew === true) { + // If it's a completely new subject, create and associate it with the dataset + await dataset.useTransaction(trx).related('subjects').save(subject); + } else { + // If the subject already exists, just create the relationship + await dataset.useTransaction(trx).related('subjects').attach([subject.id]); + } + } + } + + const subjectsToDelete = request.input('subjectsToDelete', []); + for (const subjectData of subjectsToDelete) { + if (subjectData.id) { + // const subject = await Subject.findOrFail(subjectData.id); + const subject = await Subject.query() + .where('id', subjectData.id) + .preload('datasets', (builder) => { + builder.orderBy('id', 'asc'); + }) + .withCount('datasets') + .firstOrFail(); + + // Check if the subject is used by multiple datasets + if (subject.$extras.datasets_count > 1) { + // If used by multiple datasets, just detach it from the current dataset + await dataset.useTransaction(trx).related('subjects').detach([subject.id]); + } else { + // If only used by this dataset, delete the subject completely + + await dataset.useTransaction(trx).related('subjects').detach([subject.id]); + await subject.useTransaction(trx).delete(); + } + } + } + + // Process references + const references = request.input('references', []); + // First, get existing references to determine which ones to update vs. create + const existingReferences = await dataset.related('references').query(); + const existingReferencesMap: Map<number, DatasetReference> = new Map(existingReferences.map((ref) => [ref.id, ref])); + + for (const referenceData of references) { + if (existingReferencesMap.has(referenceData.id) && referenceData.id) { + // Update existing reference + const reference = existingReferencesMap.get(referenceData.id); + if (reference) { + reference.merge(referenceData); + if (reference.$isDirty) { + await reference.useTransaction(trx).save(); + } + } + } else { + // Create new reference + const dataReference = new DatasetReference(); + dataReference.fill(referenceData); + await dataset.useTransaction(trx).related('references').save(dataReference); + } + } + + // Handle references to delete if provided + const referencesToDelete = request.input('referencesToDelete', []); + for (const referenceData of referencesToDelete) { + if (referenceData.id) { + const reference = await DatasetReference.findOrFail(referenceData.id); + await reference.useTransaction(trx).delete(); + } + } + + // save coverage + const coverageData = request.input('coverage'); + if (coverageData) { + if (coverageData.id) { + const coverage = await Coverage.findOrFail(coverageData.id); + coverage.merge(coverageData); + if (coverage.$isDirty) { + await coverage.useTransaction(trx).save(); + } + } + } + + const input = request.only(['project_id', 'embargo_date', 'language', 'type', 'creating_corporation']); + // dataset.type = request.input('type'); + dataset.merge(input); + // let test: boolean = dataset.$isDirty; + await dataset.useTransaction(trx).save(); + + await trx.commit(); + // console.log('Dataset has been updated successfully'); + + session.flash('message', 'Dataset has been updated successfully'); + // return response.redirect().toRoute('user.index'); + return response.redirect().toRoute('editor.dataset.edit', [dataset.id]); + } catch (error) { + if (trx !== null) { + await trx.rollback(); + } + console.error('Failed to update dataset and related models:', error); + // throw new ValidationException(true, { 'upload error': `failed to create dataset and related models. ${error}` }); + throw error; + } + } + + public async categorize({ inertia, request, response }: HttpContext) { + const id = request.param('id'); + // Preload dataset and its "collections" relation + const dataset = await Dataset.query().where('id', id).preload('collections').firstOrFail(); + const validStates = ['editor_accepted', 'rejected_reviewer']; + if (!validStates.includes(dataset.server_state)) { + // session.flash('errors', 'Invalid server state!'); + return response + .flash( + 'warning', + `Invalid server state. Dataset with id ${id} cannot be edited. Datset has server state ${dataset.server_state}.`, + ) + .redirect() + .toRoute('editor.dataset.list'); + } + + const collectionRoles = await CollectionRole.query() + .preload('collections', (coll: Collection) => { + // preloa only top level collection with noparent_id + coll.whereNull('parent_id').orderBy('number', 'asc'); + }) + .exec(); + + return inertia.render('Editor/Dataset/Category', { + collectionRoles: collectionRoles, + dataset: dataset, + relatedCollections: dataset.collections, + }); + } + + public async categorizeUpdate({ request, response, session }: HttpContext) { + // Get the dataset id from the route parameter + const id = request.param('id'); + const dataset = await Dataset.query().preload('files').where('id', id).firstOrFail(); + + const validStates = ['editor_accepted', 'rejected_reviewer']; + if (!validStates.includes(dataset.server_state)) { + return response + .flash( + 'warning', + `Invalid server state. Dataset with id ${id} cannot be categorized. Dataset has server state ${dataset.server_state}.`, + ) + .redirect() + .toRoute('editor.dataset.list'); + } + + let trx: TransactionClientContract | null = null; + try { + trx = await db.transaction(); + // const user = (await User.find(auth.user?.id)) as User; + // await this.createDatasetAndAssociations(user, request, trx); + + // Retrieve the selected collections from the request. + // This should be an array of collection ids. + const collections: number[] = request.input('collections', []); + + // Synchronize the dataset collections using the transaction. + await dataset.useTransaction(trx).related('collections').sync(collections); + + // Commit the transaction.await trx.commit() + await trx.commit(); + + // Redirect with a success flash message. + // return response.flash('success', 'Dataset collections updated successfully!').redirect().toRoute('dataset.list'); + + session.flash('message', 'Dataset collections updated successfully!'); + return response.redirect().toRoute('editor.dataset.list'); + } catch (error) { + if (trx !== null) { + await trx.rollback(); + } + console.error('Failed tocatgorize dataset collections:', error); + // throw new ValidationException(true, { 'upload error': `failed to create dataset and related models. ${error}` }); + throw error; + } + } // public async update({}: HttpContextContract) {} - public async update({ response }: HttpContext) { + public async updateOpensearch({ response }: HttpContext) { const id = 273; //request.param('id'); const dataset = await Dataset.query().preload('xmlCache').where('id', id).firstOrFail(); // add xml elements @@ -655,6 +1113,19 @@ export default class DatasetsController { } } + public async download({ params, response }: HttpContext) { + const id = params.id; + // Find the file by ID + const file = await File.findOrFail(id); + // const filePath = await drive.use('local').getUrl('/'+ file.filePath) + const filePath = file.filePath; + const fileExt = file.filePath.split('.').pop() || ''; + // Set the response headers and download the file + response.header('Content-Type', file.mime_type || 'application/octet-stream'); + response.attachment(`${file.label}.${fileExt}`); + return response.download(filePath); + } + public async destroy({}: HttpContext) {} private async createXmlRecord(dataset: Dataset, datasetNode: XMLBuilder) { diff --git a/app/Controllers/Http/Oai/OaiController.ts b/app/Controllers/Http/Oai/OaiController.ts index 6048ca4..db49a32 100644 --- a/app/Controllers/Http/Oai/OaiController.ts +++ b/app/Controllers/Http/Oai/OaiController.ts @@ -19,14 +19,13 @@ import XmlModel from '#app/Library/XmlModel'; import logger from '@adonisjs/core/services/logger'; import ResumptionToken from '#app/Library/Oai/ResumptionToken'; // import Config from '@ioc:Adonis/Core/Config'; -import config from '@adonisjs/core/services/config' +import config from '@adonisjs/core/services/config'; // import { inject } from '@adonisjs/fold'; -import { inject } from '@adonisjs/core' +import { inject } from '@adonisjs/core'; // import { TokenWorkerContract } from "MyApp/Models/TokenWorker"; import TokenWorkerContract from '#library/Oai/TokenWorkerContract'; import { ModelQueryBuilderContract } from '@adonisjs/lucid/types/model'; - interface XslTParameter { [key: string]: any; } @@ -35,12 +34,14 @@ interface Dictionary { [index: string]: string; } -interface ListParameter { +interface PagingParameter { cursor: number; - totalIds: number; + totalLength: number; start: number; - reldocIds: (number | null)[]; + nextDocIds: number[]; + activeWorkIds: number[]; metadataPrefix: string; + queryParams: Object; } @inject() @@ -49,6 +50,7 @@ export default class OaiController { private sampleRegEx = /^[A-Za-zäüÄÜß0-9\-_.!~]+$/; private xsltParameter: XslTParameter; + private firstPublishedDataset: Dataset | null; /** * Holds xml representation of document information to be processed. * @@ -57,7 +59,6 @@ export default class OaiController { private xml: XMLBuilder; private proc; - constructor(public tokenWorker: TokenWorkerContract) { // Load the XSLT file this.proc = readFileSync('public/assets2/datasetxml2oai.sef.json'); @@ -85,9 +86,9 @@ export default class OaiController { let earliestDateFromDb; // const oaiRequest: OaiParameter = request.body; try { - const firstPublishedDataset: Dataset | null = await Dataset.earliestPublicationDate(); - firstPublishedDataset != null && - (earliestDateFromDb = firstPublishedDataset.server_date_published.toFormat("yyyy-MM-dd'T'HH:mm:ss'Z'")); + this.firstPublishedDataset = await Dataset.earliestPublicationDate(); + this.firstPublishedDataset != null && + (earliestDateFromDb = this.firstPublishedDataset.server_date_published.toFormat("yyyy-MM-dd'T'HH:mm:ss'Z'")); this.xsltParameter['earliestDatestamp'] = earliestDateFromDb; // start the request await this.handleRequest(oaiRequest, request); @@ -162,22 +163,19 @@ export default class OaiController { } else if (verb == 'GetRecord') { await this.handleGetRecord(oaiRequest); } else if (verb == 'ListRecords') { - await this.handleListRecords(oaiRequest); + // Get browser fingerprint from the request: + const browserFingerprint = this.getBrowserFingerprint(request); + await this.handleListRecords(oaiRequest, browserFingerprint); } else if (verb == 'ListIdentifiers') { - await this.handleListIdentifiers(oaiRequest); + // Get browser fingerprint from the request: + const browserFingerprint = this.getBrowserFingerprint(request); + await this.handleListIdentifiers(oaiRequest, browserFingerprint); } else if (verb == 'ListSets') { await this.handleListSets(); } else { this.handleIllegalVerb(); } } else { - // // try { - // // console.log("Async code example.") - // const err = new PageNotFoundException("verb not found"); - // throw err; - // // } catch (error) { // manually catching - // // next(error); // passing to default middleware error handler - // // } throw new OaiModelException( StatusCodes.INTERNAL_SERVER_ERROR, 'The verb provided in the request is illegal.', @@ -187,11 +185,11 @@ export default class OaiController { } protected handleIdentify() { - const email = process.env.OAI_EMAIL || 'repository@geosphere.at'; - const repositoryName = 'Tethys RDR'; - const repIdentifier = 'tethys.at'; - const sampleIdentifier = 'oai:' + repIdentifier + ':1'; //$this->_configuration->getSampleIdentifier(); - + // Get configuration values from environment or a dedicated configuration service + const email = process.env.OAI_EMAIL ?? 'repository@geosphere.at'; + const repositoryName = process.env.OAI_REPOSITORY_NAME ?? 'Tethys RDR'; + const repIdentifier = process.env.OAI_REP_IDENTIFIER ?? 'tethys.at'; + const sampleIdentifier = `oai:${repIdentifier}:1`; // Dataset::earliestPublicationDate()->server_date_published->format('Y-m-d\TH:i:s\Z') : null; // earliestDateFromDb!= null && (this.xsltParameter['earliestDatestamp'] = earliestDateFromDb?.server_date_published); @@ -216,7 +214,7 @@ export default class OaiController { const sets: { [key: string]: string } = { 'open_access': 'Set for open access licenses', - 'openaire_data': "OpenAIRE", + 'openaire_data': 'OpenAIRE', 'doc-type:ResearchData': 'Set for document type ResearchData', ...(await this.getSetsForDatasetTypes()), ...(await this.getSetsForCollections()), @@ -234,7 +232,15 @@ export default class OaiController { const repIdentifier = 'tethys.at'; this.xsltParameter['repIdentifier'] = repIdentifier; + // Validate that required parameter exists early + if (!('identifier' in oaiRequest)) { + throw new BadOaiModelException('The prefix of the identifier argument is unknown.'); + } + + // Validate and extract the dataset identifier from the request const dataId = this.validateAndGetIdentifier(oaiRequest); + + // Retrieve dataset with associated XML cache and collection roles const dataset = await Dataset.query() .where('publish_id', dataId) .preload('xmlCache') @@ -251,59 +257,61 @@ export default class OaiController { ); } + // Validate and set the metadata prefix parameter const metadataPrefix = this.validateAndGetMetadataPrefix(oaiRequest); this.xsltParameter['oai_metadataPrefix'] = metadataPrefix; - // do not deliver datasets which are restricted by document state defined in deliveringStates + + // Ensure that the dataset is in an exportable state this.validateDatasetState(dataset); - // add xml elements + // Build the XML for the dataset record and add it to the root node const datasetNode = this.xml.root().ele('Datasets'); await this.createXmlRecord(dataset, datasetNode); } - protected async handleListIdentifiers(oaiRequest: Dictionary) { - !this.tokenWorker.isConnected && (await this.tokenWorker.connect()); + protected async handleListIdentifiers(oaiRequest: Dictionary, browserFingerprint: string) { + if (!this.tokenWorker.isConnected) { + await this.tokenWorker.connect(); + } const maxIdentifier: number = config.get('oai.max.listidentifiers', 100); - await this.handleLists(oaiRequest, maxIdentifier); + await this.handleLists(oaiRequest, maxIdentifier, browserFingerprint); } - protected async handleListRecords(oaiRequest: Dictionary) { - !this.tokenWorker.isConnected && (await this.tokenWorker.connect()); + protected async handleListRecords(oaiRequest: Dictionary, browserFingerprint: string) { + if (!this.tokenWorker.isConnected) { + await this.tokenWorker.connect(); + } const maxRecords: number = config.get('oai.max.listrecords', 100); - await this.handleLists(oaiRequest, maxRecords); + await this.handleLists(oaiRequest, maxRecords, browserFingerprint); } - private async handleLists(oaiRequest: Dictionary, maxRecords: number) { - maxRecords = maxRecords || 100; + private async handleLists(oaiRequest: Dictionary, maxRecords: number, browserFingerprint: string) { const repIdentifier = 'tethys.at'; this.xsltParameter['repIdentifier'] = repIdentifier; const datasetNode = this.xml.root().ele('Datasets'); - // list initialisation - const numWrapper: ListParameter = { + const paginationParams: PagingParameter ={ cursor: 0, - totalIds: 0, + totalLength: 0, start: maxRecords + 1, - reldocIds: [], + nextDocIds: [], + activeWorkIds: [], metadataPrefix: '', + queryParams: {}, }; - // resumptionToken is defined if ('resumptionToken' in oaiRequest) { - await this.handleResumptionToken(oaiRequest, maxRecords, numWrapper); + await this.handleResumptionToken(oaiRequest, maxRecords, paginationParams); } else { - // no resumptionToken is given - await this.handleNoResumptionToken(oaiRequest, numWrapper); + await this.handleNoResumptionToken(oaiRequest, paginationParams, maxRecords); } - // handling of document ids - const restIds = numWrapper.reldocIds as number[]; - const workIds = restIds.splice(0, maxRecords) as number[]; // array_splice(restIds, 0, maxRecords); + const nextIds: number[] = paginationParams.nextDocIds; + const workIds: number[] = paginationParams.activeWorkIds; - // no records returned - if (workIds.length == 0) { + if (workIds.length === 0) { throw new OaiModelException( StatusCodes.INTERNAL_SERVER_ERROR, 'The combination of the given values results in an empty list.', @@ -311,169 +319,218 @@ export default class OaiController { ); } - const datasets: Dataset[] = await Dataset.query() + const datasets = await Dataset.query() .whereIn('publish_id', workIds) .preload('xmlCache') .preload('collections', (builder) => { builder.preload('collectionRole'); }) .orderBy('publish_id'); - for (const dataset of datasets) { await this.createXmlRecord(dataset, datasetNode); } - - // store the further Ids in a resumption-file - const countRestIds = restIds.length; //84 - if (countRestIds > 0) { - const token = new ResumptionToken(); - token.startPosition = numWrapper.start; //101 - token.totalIds = numWrapper.totalIds; //184 - token.documentIds = restIds; //101 -184 - token.metadataPrefix = numWrapper.metadataPrefix; - - // $tokenWorker->storeResumptionToken($token); - const res: string = await this.tokenWorker.set(token); - - // set parameters for the resumptionToken-node - // const res = token.ResumptionId; - this.setParamResumption(res, numWrapper.cursor, numWrapper.totalIds); - } + await this.setResumptionToken(nextIds, paginationParams, browserFingerprint); } - private async handleResumptionToken(oaiRequest: Dictionary, maxRecords: number, numWrapper: ListParameter) { - const resParam = oaiRequest['resumptionToken']; //e.g. "158886496600000" + private async handleNoResumptionToken(oaiRequest: Dictionary, paginationParams: PagingParameter, maxRecords: number) { + this.validateMetadataPrefix(oaiRequest, paginationParams); + const finder: ModelQueryBuilderContract<typeof Dataset, Dataset> = Dataset.query().whereIn( + 'server_state', + this.deliveringDocumentStates, + ); + this.applySetFilter(finder, oaiRequest); + this.applyDateFilters(finder, oaiRequest); + await this.fetchAndSetResults(finder, paginationParams, oaiRequest, maxRecords); + } + + private async fetchAndSetResults( + finder: ModelQueryBuilderContract<typeof Dataset, Dataset>, + paginationParams: PagingParameter, + oaiRequest: Dictionary, + maxRecords: number + ) { + const totalResult = await finder + .clone() + .count('* as total') + .first() + .then((res) => res?.$extras.total); + paginationParams.totalLength = Number(totalResult); + + const combinedRecords: Dataset[] = await finder.select('publish_id').orderBy('publish_id').offset(0).limit(maxRecords*2); + + paginationParams.activeWorkIds = combinedRecords.slice(0, 100).map((dat) => Number(dat.publish_id)); + paginationParams.nextDocIds = combinedRecords.slice(100).map((dat) => Number(dat.publish_id)); + + // No resumption token was used – set queryParams from the current oaiRequest + paginationParams.queryParams = { + ...oaiRequest, + deliveringStates: this.deliveringDocumentStates, + }; + + // paginationParams.totalLength = 230; + } + + private async handleResumptionToken(oaiRequest: Dictionary, maxRecords: number, paginationParams: PagingParameter) { + const resParam = oaiRequest['resumptionToken']; const token = await this.tokenWorker.get(resParam); if (!token) { throw new OaiModelException(StatusCodes.INTERNAL_SERVER_ERROR, 'cache is outdated.', OaiErrorCodes.BADRESUMPTIONTOKEN); } - numWrapper.cursor = token.startPosition - 1; //startet dann bei Index 10 - numWrapper.start = token.startPosition + maxRecords; - numWrapper.totalIds = token.totalIds; - numWrapper.reldocIds = token.documentIds; - numWrapper.metadataPrefix = token.metadataPrefix; + // this.setResumptionParameters(token, maxRecords, paginationParams); + paginationParams.cursor = token.startPosition - 1; + paginationParams.start = token.startPosition + maxRecords; + paginationParams.totalLength = token.totalIds; + paginationParams.activeWorkIds = token.documentIds; + paginationParams.metadataPrefix = token.metadataPrefix; + paginationParams.queryParams = token.queryParams; + this.xsltParameter['oai_metadataPrefix'] = token.metadataPrefix; - this.xsltParameter['oai_metadataPrefix'] = numWrapper.metadataPrefix; + const finder = this.buildDatasetQueryViaToken(token); + const nextRecords: Dataset[] = await this.fetchNextRecords(finder, token, maxRecords); + paginationParams.nextDocIds = nextRecords.map((dat) => Number(dat.publish_id)); } - private async handleNoResumptionToken(oaiRequest: Dictionary, numWrapper: ListParameter) { - // no resumptionToken is given - if ('metadataPrefix' in oaiRequest) { - numWrapper.metadataPrefix = oaiRequest['metadataPrefix']; - } else { + private async setResumptionToken(nextIds: number[], paginationParams: PagingParameter, browserFingerprint: string) { + const countRestIds = nextIds.length; + if (countRestIds > 0) { + // const token = this.createResumptionToken(paginationParams, nextIds); + const token = new ResumptionToken(); + token.startPosition = paginationParams.start; + token.totalIds = paginationParams.totalLength; + token.documentIds = nextIds; + token.metadataPrefix = paginationParams.metadataPrefix; + token.queryParams = paginationParams.queryParams; + const res: string = await this.tokenWorker.set(token, browserFingerprint); + this.setParamResumption(res, paginationParams.cursor, paginationParams.totalLength); + } + } + + private buildDatasetQueryViaToken(token: ResumptionToken) { + const finder = Dataset.query(); + const originalQuery = token.queryParams || {}; + const deliveringStates = originalQuery.deliveringStates || this.deliveringDocumentStates; + + finder.whereIn('server_state', deliveringStates); + this.applySetFilter(finder, originalQuery); + this.applyDateFilters(finder, originalQuery); + + return finder; + } + + private async fetchNextRecords(finder: ModelQueryBuilderContract<typeof Dataset, Dataset>, token: ResumptionToken, maxRecords: number) { + return finder + .select('publish_id') + .orderBy('publish_id') + .offset(token.startPosition - 1 + maxRecords) + .limit(100); + } + + private validateMetadataPrefix(oaiRequest: Dictionary, paginationParams: PagingParameter) { + if (!('metadataPrefix' in oaiRequest)) { throw new OaiModelException( StatusCodes.INTERNAL_SERVER_ERROR, 'The prefix of the metadata argument is unknown.', OaiErrorCodes.BADARGUMENT, ); } - this.xsltParameter['oai_metadataPrefix'] = numWrapper.metadataPrefix; + paginationParams.metadataPrefix = oaiRequest['metadataPrefix']; + this.xsltParameter['oai_metadataPrefix'] = paginationParams.metadataPrefix; + } - let finder: ModelQueryBuilderContract<typeof Dataset, Dataset> = Dataset.query(); - // add server state restrictions - finder.whereIn('server_state', this.deliveringDocumentStates); - if ('set' in oaiRequest) { - const set = oaiRequest['set'] as string; - const setArray = set.split(':'); + private applySetFilter(finder: ModelQueryBuilderContract<typeof Dataset, Dataset>, queryParams: any) { + if ('set' in queryParams) { + const [setType, setValue] = queryParams['set'].split(':'); - if (setArray[0] == 'data-type') { - if (setArray.length == 2 && setArray[1]) { - finder.where('type', setArray[1]); - } - } else if (setArray[0] == 'open_access') { - const openAccessLicences = ['CC-BY-4.0', 'CC-BY-SA-4.0']; - finder.andWhereHas('licenses', (query) => { - query.whereIn('name', openAccessLicences); - }); - } else if (setArray[0] == 'ddc') { - if (setArray.length == 2 && setArray[1] != '') { - finder.andWhereHas('collections', (query) => { - query.where('number', setArray[1]); + switch (setType) { + case 'data-type': + setValue && finder.where('type', setValue); + break; + case 'open_access': + finder.andWhereHas('licenses', (query) => { + query.whereIn('name', ['CC-BY-4.0', 'CC-BY-SA-4.0']); }); - } + break; + case 'ddc': + setValue && + finder.andWhereHas('collections', (query) => { + query.where('number', setValue); + }); + break; } } + } - // const timeZone = "Europe/Vienna"; // Canonical time zone name - // &from=2020-09-03&until2020-09-03 - // &from=2020-09-11&until=2021-05-11 - if ('from' in oaiRequest && 'until' in oaiRequest) { - const from = oaiRequest['from'] as string; - let fromDate = dayjs(from); //.tz(timeZone); - const until = oaiRequest['until'] as string; - let untilDate = dayjs(until); //.tz(timeZone); - if (!fromDate.isValid() || !untilDate.isValid()) { - throw new OaiModelException(StatusCodes.INTERNAL_SERVER_ERROR, 'Date Parameter is not valid.', OaiErrorCodes.BADARGUMENT); - } - fromDate = dayjs.tz(from, 'Europe/Vienna'); - untilDate = dayjs.tz(until, 'Europe/Vienna'); + private applyDateFilters(finder: ModelQueryBuilderContract<typeof Dataset, Dataset>, queryParams: any) { + const { from, until } = queryParams; - if (from.length != until.length) { - throw new OaiModelException( - StatusCodes.INTERNAL_SERVER_ERROR, - 'The request has different granularities for the from and until parameters.', - OaiErrorCodes.BADARGUMENT, - ); - } - fromDate.hour() == 0 && (fromDate = fromDate.startOf('day')); - untilDate.hour() == 0 && (untilDate = untilDate.endOf('day')); + if (from && until) { + this.handleFromUntilFilter(finder, from, until); + } else if (from) { + this.handleFromFilter(finder, from); + } else if (until) { + this.handleUntilFilter(finder, until); + } + } - finder.whereBetween('server_date_published', [fromDate.format('YYYY-MM-DD HH:mm:ss'), untilDate.format('YYYY-MM-DD HH:mm:ss')]); - } else if ('from' in oaiRequest && !('until' in oaiRequest)) { - const from = oaiRequest['from'] as string; - let fromDate = dayjs(from); - if (!fromDate.isValid()) { - throw new OaiModelException( - StatusCodes.INTERNAL_SERVER_ERROR, - 'From date parameter is not valid.', - OaiErrorCodes.BADARGUMENT, - ); - } - fromDate = dayjs.tz(from, 'Europe/Vienna'); - fromDate.hour() == 0 && (fromDate = fromDate.startOf('day')); + private handleFromUntilFilter(finder: ModelQueryBuilderContract<typeof Dataset, Dataset>, from: string, until: string) { + const fromDate = this.parseDateWithValidation(from, 'From'); + const untilDate = this.parseDateWithValidation(until, 'Until'); - const now = dayjs(); - if (fromDate.isAfter(now)) { - throw new OaiModelException( - StatusCodes.INTERNAL_SERVER_ERROR, - 'Given from date is greater than now. The given values results in an empty list.', - OaiErrorCodes.NORECORDSMATCH, - ); - } else { - finder.andWhere('server_date_published', '>=', fromDate.format('YYYY-MM-DD HH:mm:ss')); - } - } else if (!('from' in oaiRequest) && 'until' in oaiRequest) { - const until = oaiRequest['until'] as string; - let untilDate = dayjs(until); - if (!untilDate.isValid()) { - throw new OaiModelException( - StatusCodes.INTERNAL_SERVER_ERROR, - 'Until date parameter is not valid.', - OaiErrorCodes.BADARGUMENT, - ); - } - untilDate = dayjs.tz(until, 'Europe/Vienna'); - untilDate.hour() == 0 && (untilDate = untilDate.endOf('day')); - - const firstPublishedDataset: Dataset = (await Dataset.earliestPublicationDate()) as Dataset; - const earliestPublicationDate = dayjs(firstPublishedDataset.server_date_published.toISO()); //format("YYYY-MM-DDThh:mm:ss[Z]")); - if (earliestPublicationDate.isAfter(untilDate)) { - throw new OaiModelException( - StatusCodes.INTERNAL_SERVER_ERROR, - `earliestDatestamp is greater than given until date. - The given values results in an empty list.`, - OaiErrorCodes.NORECORDSMATCH, - ); - } else { - finder.andWhere('server_date_published', '<=', untilDate.format('YYYY-MM-DD HH:mm:ss')); - } + if (from.length !== until.length) { + throw new OaiModelException( + StatusCodes.INTERNAL_SERVER_ERROR, + 'The request has different granularities for the from and until parameters.', + OaiErrorCodes.BADARGUMENT, + ); } - let reldocIdsDocs = await finder.select('publish_id').orderBy('publish_id'); - numWrapper.reldocIds = reldocIdsDocs.map((dat) => dat.publish_id); - numWrapper.totalIds = numWrapper.reldocIds.length; //212 + finder.whereBetween('server_date_published', [fromDate.format('YYYY-MM-DD HH:mm:ss'), untilDate.format('YYYY-MM-DD HH:mm:ss')]); + } + + private handleFromFilter(finder: ModelQueryBuilderContract<typeof Dataset, Dataset>, from: string) { + const fromDate = this.parseDateWithValidation(from, 'From'); + const now = dayjs(); + + if (fromDate.isAfter(now)) { + throw new OaiModelException( + StatusCodes.INTERNAL_SERVER_ERROR, + 'Given from date is greater than now. The given values results in an empty list.', + OaiErrorCodes.NORECORDSMATCH, + ); + } + + finder.andWhere('server_date_published', '>=', fromDate.format('YYYY-MM-DD HH:mm:ss')); + } + + private handleUntilFilter(finder: ModelQueryBuilderContract<typeof Dataset, Dataset>, until: string) { + const untilDate = this.parseDateWithValidation(until, 'Until'); + + const earliestPublicationDate = dayjs(this.firstPublishedDataset?.server_date_published.toISO()); + + if (earliestPublicationDate.isAfter(untilDate)) { + throw new OaiModelException( + StatusCodes.INTERNAL_SERVER_ERROR, + 'earliestDatestamp is greater than given until date. The given values results in an empty list.', + OaiErrorCodes.NORECORDSMATCH, + ); + } + + finder.andWhere('server_date_published', '<=', untilDate.format('YYYY-MM-DD HH:mm:ss')); + } + + private parseDateWithValidation(dateStr: string, label: string) { + let date = dayjs(dateStr); + if (!date.isValid()) { + throw new OaiModelException( + StatusCodes.INTERNAL_SERVER_ERROR, + `${label} date parameter is not valid.`, + OaiErrorCodes.BADARGUMENT, + ); + } + date = dayjs.tz(dateStr, 'Europe/Vienna'); + return date.hour() === 0 ? (label === 'From' ? date.startOf('day') : date.endOf('day')) : date; } private setParamResumption(res: string, cursor: number, totalIds: number) { @@ -641,4 +698,30 @@ export default class OaiController { this.xsltParameter['oai_error_code'] = 'badVerb'; this.xsltParameter['oai_error_message'] = 'The verb provided in the request is illegal.'; } + + /** + * Helper method to build a browser fingerprint by combining: + * - User-Agent header, + * - the IP address, + * - Accept-Language header, + * - current timestamp rounded to the hour. + * + * Every new hour, this will return a different fingerprint. + */ + private getBrowserFingerprint(request: Request): string { + const userAgent = request.header('user-agent') || 'unknown'; + // Check for X-Forwarded-For header to use the client IP from the proxy if available. + const xForwardedFor = request.header('x-forwarded-for'); + let ip = request.ip(); + // console.log(ip); + if (xForwardedFor) { + // X-Forwarded-For may contain a comma-separated list of IPs; the first one is the client IP. + ip = xForwardedFor.split(',')[0].trim(); + // console.log('xforwardedfor ip' + ip); + } + const locale = request.header('accept-language') || 'default'; + // Round the current time to the start of the hour. + const timestampHour = dayjs().startOf('hour').format('YYYY-MM-DDTHH'); + return `${userAgent}-${ip}-${locale}-${timestampHour}`; + } } diff --git a/app/Controllers/Http/Reviewer/DatasetController.ts b/app/Controllers/Http/Reviewer/DatasetController.ts index 66600f4..6b54772 100644 --- a/app/Controllers/Http/Reviewer/DatasetController.ts +++ b/app/Controllers/Http/Reviewer/DatasetController.ts @@ -9,6 +9,7 @@ import vine from '@vinejs/vine'; import mail from '@adonisjs/mail/services/main'; import logger from '@adonisjs/core/services/logger'; import { validate } from 'deep-email-validator'; +import File from '#models/file'; interface Dictionary { [index: string]: string; @@ -38,13 +39,21 @@ export default class DatasetsController { } datasets.orderBy(attribute, sortOrder); } else { - // users.orderBy('created_at', 'desc'); - datasets.orderBy('id', 'asc'); + // datasets.orderBy('id', 'asc'); + // Custom ordering to prioritize rejected_editor state + datasets.orderByRaw(` + CASE + WHEN server_state = 'rejected_to_reviewer' THEN 0 + ELSE 1 + END ASC, + id ASC + `); } // const users = await User.query().orderBy('login').paginate(page, limit); const myDatasets = await datasets - .where('server_state', 'approved') + // .where('server_state', 'approved') + .whereIn('server_state', ['approved', 'rejected_to_reviewer']) .where('reviewer_id', user.id) .preload('titles') @@ -62,7 +71,52 @@ export default class DatasetsController { }); } - public async review({ request, inertia, response }: HttpContext) { + public async review({ request, inertia, response, auth }: HttpContext) { + const id = request.param('id'); + const datasetQuery = Dataset.query().where('id', id); + + datasetQuery + .preload('titles', (query) => query.orderBy('id', 'asc')) + .preload('descriptions', (query) => query.orderBy('id', 'asc')) + .preload('coverage') + .preload('licenses') + .preload('authors', (query) => query.orderBy('pivot_sort_order', 'asc')) + .preload('contributors', (query) => query.orderBy('pivot_sort_order', 'asc')) + // .preload('subjects') + .preload('subjects', (builder) => { + builder.orderBy('id', 'asc').withCount('datasets'); + }) + .preload('references') + .preload('project') + .preload('files', (query) => { + query.orderBy('sort_order', 'asc'); // Sort by sort_order column + }); + + const dataset = await datasetQuery.firstOrFail(); + + const validStates = ['approved', 'rejected_to_reviewer']; + if (!validStates.includes(dataset.server_state)) { + // session.flash('errors', 'Invalid server state!'); + return response + .flash( + 'warning', + `Invalid server state. Dataset with id ${id} cannot be reviewed. Datset has server state ${dataset.server_state}.`, + ) + .redirect() + .toRoute('reviewer.dataset.list'); + } + + return inertia.render('Reviewer/Dataset/Review', { + dataset, + can: { + review: await auth.user?.can(['dataset-review']), + reject: await auth.user?.can(['dataset-review-reject']), + }, + }); + + } + + public async review_old({ request, inertia, response, auth }: HttpContext) { const id = request.param('id'); const dataset = await Dataset.query() .where('id', id) @@ -158,6 +212,10 @@ export default class DatasetsController { return inertia.render('Reviewer/Dataset/Review', { dataset, fields: fields, + can: { + review: await auth.user?.can(['dataset-review']), + reject: await auth.user?.can(['dataset-review-reject']), + }, }); } @@ -166,7 +224,7 @@ export default class DatasetsController { // const { id } = params; const dataset = await Dataset.findOrFail(id); - const validStates = ['approved']; + const validStates = ['approved', 'rejected_to_reviewer']; if (!validStates.includes(dataset.server_state)) { // throw new Error('Invalid server state!'); // return response.flash('warning', 'Invalid server state. Dataset cannot be released to editor').redirect().back(); @@ -180,6 +238,10 @@ export default class DatasetsController { } dataset.server_state = 'reviewed'; + // if editor has rejected to reviewer: + if (dataset.reject_editor_note != null) { + dataset.reject_editor_note = null; + } try { // await dataset.related('editor').associate(user); // speichert schon ab @@ -203,7 +265,7 @@ export default class DatasetsController { }) .firstOrFail(); - const validStates = ['approved']; + const validStates = ['approved', 'rejected_to_reviewer']; if (!validStates.includes(dataset.server_state)) { // session.flash('errors', 'Invalid server state!'); return response @@ -250,12 +312,12 @@ export default class DatasetsController { throw error; } - const validStates = ['approved']; + const validStates = ['approved', 'rejected_to_reviewer']; if (!validStates.includes(dataset.server_state)) { // throw new Error('Invalid server state!'); // return response.flash('warning', 'Invalid server state. Dataset cannot be released to editor').redirect().back(); return response - .flash( + .flash( `Invalid server state. Dataset with id ${id} cannot be rejected. Datset has server state ${dataset.server_state}.`, 'warning', ) @@ -276,7 +338,7 @@ export default class DatasetsController { validateSMTP: false, }); const validRecipientEmail: boolean = validationResult.valid; - let emailStatusMessage = ''; + // let emailStatusMessage = ''; if (sendMail == true) { if (dataset.editor.email && validRecipientEmail) { @@ -289,7 +351,7 @@ export default class DatasetsController { <p>Best regards,<br>Your Tethys reviewer: ${authUser.login}</p> `); }); - emailStatusMessage = ` A rejection email was successfully sent to ${dataset.editor.email}.`; + // emailStatusMessage = ` A rejection email was successfully sent to ${dataset.editor.email}.`; } catch (error) { logger.error(error); return response @@ -297,7 +359,7 @@ export default class DatasetsController { .toRoute('reviewer.dataset.list'); } } else { - emailStatusMessage = ` However, the email could not be sent because the editor's email address (${dataset.editor.email}) is not valid.`; + // emailStatusMessage = ` However, the email could not be sent because the editor's email address (${dataset.editor.email}) is not valid.`; } } @@ -307,4 +369,17 @@ export default class DatasetsController { .toRoute('reviewer.dataset.list') .flash(`You have rejected dataset ${dataset.id}! to editor ${dataset.editor.login}`, 'message'); } + + public async download({ params, response }: HttpContext) { + const id = params.id; + // Find the file by ID + const file = await File.findOrFail(id); + // const filePath = await drive.use('local').getUrl('/'+ file.filePath) + const filePath = file.filePath; + const fileExt = file.filePath.split('.').pop() || ''; + // Set the response headers and download the file + response.header('Content-Type', file.mime_type || 'application/octet-stream'); + response.attachment(`${file.label}.${fileExt}`); + return response.download(filePath); + } } diff --git a/app/Controllers/Http/Submitter/DatasetController.ts b/app/Controllers/Http/Submitter/DatasetController.ts index 41a5733..e6018cc 100644 --- a/app/Controllers/Http/Submitter/DatasetController.ts +++ b/app/Controllers/Http/Submitter/DatasetController.ts @@ -8,6 +8,7 @@ import Description from '#models/description'; import Language from '#models/language'; import Coverage from '#models/coverage'; import Collection from '#models/collection'; +import CollectionRole from '#models/collection_role'; import dayjs from 'dayjs'; import Person from '#models/person'; import db from '@adonisjs/lucid/services/db'; @@ -28,21 +29,30 @@ import { } from '#contracts/enums'; import { ModelQueryBuilderContract } from '@adonisjs/lucid/types/model'; import DatasetReference from '#models/dataset_reference'; -import { cuid } from '@adonisjs/core/helpers'; import File from '#models/file'; import ClamScan from 'clamscan'; -// import { ValidationException } from '@adonisjs/validator'; -// import Drive from '@ioc:Adonis/Core/Drive'; -import drive from '#services/drive'; +import drive from '@adonisjs/drive/services/main'; +import path from 'path'; import { Exception } from '@adonisjs/core/exceptions'; import { MultipartFile } from '@adonisjs/core/types/bodyparser'; import * as crypto from 'crypto'; +import { pipeline } from 'node:stream/promises'; +import { createWriteStream } from 'node:fs'; +import type { Multipart } from '@adonisjs/bodyparser'; +import * as fs from 'fs'; +import { parseBytesSize, getConfigFor, getTmpPath, formatBytes } from '#app/utils/utility-functions'; + interface Dictionary { [index: string]: string; } import vine, { SimpleMessagesProvider, errors } from '@vinejs/vine'; export default class DatasetController { + /** + * Bodyparser config + */ + // config: BodyParserConfig = config.get('bodyparser'); + public async index({ auth, request, inertia }: HttpContext) { const user = (await User.find(auth.user?.id)) as User; const page = request.input('page', 1); @@ -66,8 +76,16 @@ export default class DatasetController { } datasets.orderBy(attribute, sortOrder); } else { - // users.orderBy('created_at', 'desc'); - datasets.orderBy('id', 'asc'); + // datasets.orderBy('id', 'asc'); + // Custom ordering to prioritize rejected_editor state + datasets.orderByRaw(` + CASE + WHEN server_state = 'rejected_editor' THEN 0 + WHEN server_state = 'rejected_reviewer' THEN 1 + ELSE 2 + END ASC, + id ASC + `); } // const results = await Database @@ -188,7 +206,8 @@ export default class DatasetController { .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), }), ) - .minLength(1), + // .minLength(2) + .arrayContainsTypes({ typeA: 'main', typeB: 'translated' }), descriptions: vine .array( vine.object({ @@ -202,7 +221,8 @@ export default class DatasetController { .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), }), ) - .minLength(1), + // .minLength(1), + .arrayContainsTypes({ typeA: 'abstract', typeB: 'translated' }), authors: vine .array( vine.object({ @@ -277,7 +297,8 @@ export default class DatasetController { .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), }), ) - .minLength(1), + // .minLength(2) + .arrayContainsTypes({ typeA: 'main', typeB: 'translated' }), descriptions: vine .array( vine.object({ @@ -291,7 +312,8 @@ export default class DatasetController { .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), }), ) - .minLength(1), + // .minLength(1), + .arrayContainsTypes({ typeA: 'abstract', typeB: 'translated' }), authors: vine .array( vine.object({ @@ -363,7 +385,8 @@ export default class DatasetController { references: vine .array( vine.object({ - value: vine.string().trim().minLength(3).maxLength(255), + // value: vine.string().trim().minLength(3).maxLength(255), + value: vine.string().trim().minLength(3).maxLength(255).validateReference({ typeField: 'type' }), type: vine.enum(Object.values(ReferenceIdentifierTypes)), relation: vine.enum(Object.values(RelationTypes)), label: vine.string().trim().minLength(2).maxLength(255), @@ -398,21 +421,99 @@ export default class DatasetController { } public async store({ auth, request, response, session }: HttpContext) { - // node ace make:validator CreateDataset + // At the top of the store() method, declare an array to hold temporary file paths + const uploadedTmpFiles: string[] = []; + // Aggregated limit example (adjust as needed) + const multipartConfig = getConfigFor('multipart'); + const aggregatedLimit = multipartConfig.limit ? parseBytesSize(multipartConfig.limit) : 100 * 1024 * 1024; + // const aggregatedLimit = 200 * 1024 * 1024; + let totalUploadedSize = 0; + + // // Helper function to format bytes as human-readable text + // function formatBytes(bytes: number): string { + // if (bytes === 0) return '0 Bytes'; + // const k = 1024; + // const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB']; + // const i = Math.floor(Math.log(bytes) / Math.log(k)); + // return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i]; + // } + // const enabledExtensions = await this.getEnabledExtensions(); + const multipart: Multipart = request.multipart; + + multipart.onFile('files', { deferValidations: true }, async (part) => { + // Attach an individual file size accumulator if needed + let fileUploadedSize = 0; + + // Simply accumulate the size in on('data') without performing the expensive check per chunk + part.on('data', (chunk) => { + // reporter(chunk); + // Increase counters using the chunk length + fileUploadedSize += chunk.length; + }); + + // After the file is completely read, update the global counter and check aggregated limit + part.on('end', () => { + totalUploadedSize += fileUploadedSize; + part.file.size = fileUploadedSize; + // Record the temporary file path + if (part.file.tmpPath) { + uploadedTmpFiles.push(part.file.tmpPath); + } + + if (totalUploadedSize > aggregatedLimit) { + // Clean up all temporary files if aggregate limit is exceeded + uploadedTmpFiles.forEach((tmpPath) => { + try { + fs.unlinkSync(tmpPath); + } catch (cleanupError) { + console.error('Error cleaning up temporary file:', cleanupError); + } + }); + const error = new errors.E_VALIDATION_ERROR({ + 'upload error': `Aggregated upload limit of ${formatBytes(aggregatedLimit)} exceeded. The total size of files being uploaded would exceed the limit.`, + }); + request.multipart.abort(error); + } + }); + + part.on('error', (error) => { + // fileUploadError = error; + request.multipart.abort(error); + }); + + // await pipeline(part, createWriteStream(filePath)); + // return { filePath }; + // Process file with error handling + try { + // Extract extension from the client file name, e.g. "Tethys 5 - Ampflwang_dataset.zip" + const ext = path.extname(part.file.clientName).replace('.', ''); + // Attach the extracted extension to the file object for later use + part.file.extname = ext; + + // part.file.sortOrder = part.file.sortOrder; + + const tmpPath = getTmpPath(multipartConfig); + (part.file as any).tmpPath = tmpPath; + + const writeStream = createWriteStream(tmpPath); + await pipeline(part, writeStream); + } catch (error) { + request.multipart.abort(new errors.E_VALIDATION_ERROR({ 'upload error': error.message })); + } + }); + try { - // Step 2 - Validate request body against the schema - // await request.validate({ schema: newDatasetSchema, messages: this.messages }); - // await request.validate(CreateDatasetValidator); - await request.validateUsing(createDatasetValidator); - // console.log({ payload }); + await multipart.process(); + // // Instead of letting an error abort the controller, check if any error occurred } catch (error) { - // Step 3 - Handle errors - // return response.badRequest(error.messages); - throw error; + // This is where you'd expect to catch any errors. + session.flash('errors', error.messages); + return response.redirect().back(); } let trx: TransactionClientContract | null = null; try { + await request.validateUsing(createDatasetValidator); trx = await db.transaction(); const user = (await User.find(auth.user?.id)) as User; @@ -421,6 +522,14 @@ export default class DatasetController { await trx.commit(); console.log('Dataset and related models created successfully'); } catch (error) { + // Clean up temporary files if validation or later steps fail + uploadedTmpFiles.forEach((tmpPath) => { + try { + fs.unlinkSync(tmpPath); + } catch (cleanupError) { + console.error('Error cleaning up temporary file:', cleanupError); + } + }); if (trx !== null) { await trx.rollback(); } @@ -433,14 +542,19 @@ export default class DatasetController { return response.redirect().toRoute('dataset.list'); // return response.redirect().back(); } - - private async createDatasetAndAssociations(user: User, request: HttpContext['request'], trx: TransactionClientContract) { + private async createDatasetAndAssociations( + user: User, + request: HttpContext['request'], + trx: TransactionClientContract, + // uploadedFiles: Array<MultipartFile>, + ) { // Create a new instance of the Dataset model: const dataset = new Dataset(); dataset.type = request.input('type'); dataset.creating_corporation = request.input('creating_corporation'); dataset.language = request.input('language'); dataset.embargo_date = request.input('embargo_date'); + dataset.project_id = request.input('project_id'); //await dataset.related('user').associate(user); // speichert schon ab // Dataset.$getRelation('user').boot(); // Dataset.$getRelation('user').setRelated(dataset, user); @@ -498,7 +612,7 @@ export default class DatasetController { } // save collection - const collection: Collection | null = await Collection.query().where('id', 21).first(); + const collection: Collection | null = await Collection.query().where('id', 594).first(); collection && (await dataset.useTransaction(trx).related('collections').attach([collection.id])); // save coverage @@ -531,18 +645,25 @@ export default class DatasetController { const fileName = this.generateFilename(file.extname as string); const mimeType = file.headers['content-type'] || 'application/octet-stream'; // Fallback to a default MIME type const datasetFolder = `files/${dataset.id}`; + const datasetFullPath = path.join(`${datasetFolder}`, fileName); // const size = file.size; - await file.move(drive.makePath(datasetFolder), { + // await file.move(drive.makePath(datasetFolder), { + // name: fileName, + // overwrite: true, // overwrite in case of conflict + // }); + await file.moveToDisk(datasetFullPath, 'local', { name: fileName, overwrite: true, // overwrite in case of conflict + disk: 'local', }); + // save file metadata into db const newFile = new File(); newFile.pathName = `${datasetFolder}/${fileName}`; newFile.fileSize = file.size; newFile.mimeType = mimeType; newFile.label = file.clientName; - newFile.sortOrder = index; + newFile.sortOrder = index + 1; newFile.visibleInFrontdoor = true; newFile.visibleInOai = true; // let path = coverImage.filePath; @@ -693,6 +814,8 @@ export default class DatasetController { 'files.array.minLength': 'At least {{ min }} file upload is required.', 'files.*.size': 'file size is to big', 'files.*.extnames': 'file extension is not supported', + + 'embargo_date.date.afterOrEqual': `Embargo date must be on or after ${dayjs().add(10, 'day').format('DD.MM.YYYY')}`, }; // public async release({ params, view }) { @@ -803,7 +926,7 @@ export default class DatasetController { // throw new GeneralException(trans('exceptions.publish.release.update_error')); } - public async edit({ request, inertia, response }: HttpContext) { + public async edit({ request, inertia, response, auth }: HttpContext) { const id = request.param('id'); const datasetQuery = Dataset.query().where('id', id); datasetQuery @@ -811,8 +934,8 @@ export default class DatasetController { .preload('descriptions', (query) => query.orderBy('id', 'asc')) .preload('coverage') .preload('licenses') - .preload('authors') - .preload('contributors') + .preload('authors', (query) => query.orderBy('pivot_sort_order', 'asc')) + .preload('contributors', (query) => query.orderBy('pivot_sort_order', 'asc')) // .preload('subjects') .preload('subjects', (builder) => { builder.orderBy('id', 'asc').withCount('datasets'); @@ -828,10 +951,9 @@ export default class DatasetController { // session.flash('errors', 'Invalid server state!'); return response .flash( - 'warning', `Invalid server state. Dataset with id ${id} cannot be edited. Datset has server state ${dataset.server_state}.`, + 'warning', ) - .redirect() .toRoute('dataset.list'); } @@ -865,15 +987,15 @@ export default class DatasetController { // const datasetHasLicenses = await dataset.related('licenses').query().pluck('id'); // const checkeds = dataset.licenses.first().id; - const doctypes = { - analysisdata: { label: 'Analysis', value: 'analysisdata' }, - measurementdata: { label: 'Measurements', value: 'measurementdata' }, - monitoring: 'Monitoring', - remotesensing: 'Remote Sensing', - gis: 'GIS', - models: 'Models', - mixedtype: 'Mixed Type', - }; + // const doctypes = { + // analysisdata: { label: 'Analysis', value: 'analysisdata' }, + // measurementdata: { label: 'Measurements', value: 'measurementdata' }, + // monitoring: 'Monitoring', + // remotesensing: 'Remote Sensing', + // gis: 'GIS', + // models: 'Models', + // mixedtype: 'Mixed Type', + // }; return inertia.render('Submitter/Dataset/Edit', { dataset, @@ -892,25 +1014,95 @@ export default class DatasetController { subjectTypes: SubjectTypes, referenceIdentifierTypes: Object.entries(ReferenceIdentifierTypes).map(([key, value]) => ({ value: key, label: value })), relationTypes: Object.entries(RelationTypes).map(([key, value]) => ({ value: key, label: value })), - doctypes, + doctypes: DatasetTypes, + can: { + edit: await auth.user?.can(['dataset-edit']), + delete: await auth.user?.can(['dataset-delete']), + }, }); } public async update({ request, response, session }: HttpContext) { - try { - // await request.validate(UpdateDatasetValidator); - await request.validateUsing(updateDatasetValidator); - } catch (error) { - // - Handle errors - // return response.badRequest(error.messages); - throw error; - // return response.badRequest(error.messages); + // Get the dataset id from the route parameter + const datasetId = request.param('id'); + // Retrieve the dataset and load its existing files + const dataset = await Dataset.findOrFail(datasetId); + await dataset.load('files'); + // Accumulate the size of the already related files + // const preExistingFileSize = dataset.files.reduce((acc, file) => acc + file.fileSize, 0); + let preExistingFileSize = 0; + for (const file of dataset.files) { + preExistingFileSize += Number(file.fileSize); } - // await request.validate(UpdateDatasetValidator); - const id = request.param('id'); + const uploadedTmpFiles: string[] = []; + // Only process multipart if the request has a multipart content type + const contentType = request.request.headers['content-type'] || ''; + if (contentType.includes('multipart/form-data')) { + const multipart: Multipart = request.multipart; + // Aggregated limit example (adjust as needed) + const multipartConfig = getConfigFor('multipart'); + const aggregatedLimit = multipartConfig.limit ? parseBytesSize(multipartConfig.limit) : 100 * 1024 * 1024; + // Initialize totalUploadedSize with the size of existing files + let totalUploadedSize = preExistingFileSize; + + multipart.onFile('files', { deferValidations: true }, async (part) => { + let fileUploadedSize = 0; + + part.on('data', (chunk) => { + fileUploadedSize += chunk.length; + }); + + part.on('end', () => { + totalUploadedSize += fileUploadedSize; + part.file.size = fileUploadedSize; + if (part.file.tmpPath) { + uploadedTmpFiles.push(part.file.tmpPath); + } + if (totalUploadedSize > aggregatedLimit) { + uploadedTmpFiles.forEach((tmpPath) => { + try { + fs.unlinkSync(tmpPath); + } catch (cleanupError) { + console.error('Error cleaning up temporary file:', cleanupError); + } + }); + const error = new errors.E_VALIDATION_ERROR({ + 'upload error': `Aggregated upload limit of ${formatBytes(aggregatedLimit)} exceeded. The total size of files being uploaded would exceed the limit.`, + }); + request.multipart.abort(error); + } + }); + + part.on('error', (error) => { + request.multipart.abort(error); + }); + + try { + const fileNameWithoutParams = part.file.clientName.split('?')[0]; + const ext = path.extname(fileNameWithoutParams).replace('.', ''); + part.file.extname = ext; + const tmpPath = getTmpPath(multipartConfig); + (part.file as any).tmpPath = tmpPath; + const writeStream = createWriteStream(tmpPath); + await pipeline(part, writeStream); + } catch (error) { + request.multipart.abort(new errors.E_VALIDATION_ERROR({ 'upload error': error.message })); + } + }); + + try { + await multipart.process(); + } catch (error) { + session.flash('errors', error.messages); + return response.redirect().back(); + } + } + + const id = request.param('id'); let trx: TransactionClientContract | null = null; try { + await request.validateUsing(updateDatasetValidator); trx = await db.transaction(); // const user = (await User.find(auth.user?.id)) as User; // await this.createDatasetAndAssociations(user, request, trx); @@ -971,22 +1163,97 @@ export default class DatasetController { } } - // await dataset.useTransaction(trx).related('subjects').sync([]); - const keywords = request.input('subjects'); - for (const keywordData of keywords) { - if (keywordData.id) { - const subject = await Subject.findOrFail(keywordData.id); - // await dataset.useTransaction(trx).related('subjects').attach([keywordData.id]); - subject.value = keywordData.value; - subject.type = keywordData.type; - subject.external_key = keywordData.external_key; - if (subject.$isDirty) { - await subject.save(); + // Process all subjects/keywords from the request + const subjects = request.input('subjects'); + for (const subjectData of subjects) { + // Case 1: Subject already exists in the database (has an ID) + if (subjectData.id) { + // Retrieve the existing subject + const existingSubject = await Subject.findOrFail(subjectData.id); + + // Update subject properties from the request data + existingSubject.value = subjectData.value; + existingSubject.type = subjectData.type; + existingSubject.external_key = subjectData.external_key; + + // Only save if there are actual changes + if (existingSubject.$isDirty) { + await existingSubject.save(); + } + + // Note: The relationship between dataset and subject is already established, + // so we don't need to attach it again + } + // Case 2: New subject being added (no ID) + else { + // Check if a subject with the same value and type already exists in the database + const subject = await Subject.firstOrNew({ value: subjectData.value, type: subjectData.type }, subjectData); + + if (subject.$isNew === true) { + // If it's a completely new subject, create and associate it with the dataset + await dataset.useTransaction(trx).related('subjects').save(subject); + } else { + // If the subject already exists, just create the relationship + await dataset.useTransaction(trx).related('subjects').attach([subject.id]); + } + } + } + + const subjectsToDelete = request.input('subjectsToDelete', []); + for (const subjectData of subjectsToDelete) { + if (subjectData.id) { + // const subject = await Subject.findOrFail(subjectData.id); + const subject = await Subject.query() + .where('id', subjectData.id) + .preload('datasets', (builder) => { + builder.orderBy('id', 'asc'); + }) + .withCount('datasets') + .firstOrFail(); + + // Check if the subject is used by multiple datasets + if (subject.$extras.datasets_count > 1) { + // If used by multiple datasets, just detach it from the current dataset + await dataset.useTransaction(trx).related('subjects').detach([subject.id]); + } else { + // If only used by this dataset, delete the subject completely + + await dataset.useTransaction(trx).related('subjects').detach([subject.id]); + await subject.useTransaction(trx).delete(); + } + } + } + + // Process references + const references = request.input('references', []); + // First, get existing references to determine which ones to update vs. create + const existingReferences = await dataset.related('references').query(); + const existingReferencesMap: Map<number, DatasetReference> = new Map(existingReferences.map((ref) => [ref.id, ref])); + + for (const referenceData of references) { + if (existingReferencesMap.has(referenceData.id) && referenceData.id) { + // Update existing reference + const reference = existingReferencesMap.get(referenceData.id); + if (reference) { + reference.merge(referenceData); + if (reference.$isDirty) { + await reference.useTransaction(trx).save(); + } } } else { - const keyword = new Subject(); - keyword.fill(keywordData); - await dataset.useTransaction(trx).related('subjects').save(keyword, false); + // Create new reference + const dataReference = new DatasetReference(); + dataReference.fill(referenceData); + await dataset.useTransaction(trx).related('references').save(dataReference); + } + } + + // Handle references to delete if provided + const referencesToDelete = request.input('referencesToDelete', []); + for (const referenceData of referencesToDelete) { + if (referenceData.id) { + const reference = await DatasetReference.findOrFail(referenceData.id); + await reference.useTransaction(trx).delete(); } } @@ -1018,9 +1285,9 @@ export default class DatasetController { // handle new uploaded files: const uploadedFiles: MultipartFile[] = request.files('files'); if (Array.isArray(uploadedFiles) && uploadedFiles.length > 0) { - for (const [index, fileData] of uploadedFiles.entries()) { + for (const [index, file] of uploadedFiles.entries()) { try { - await this.scanFileForViruses(fileData.tmpPath); //, 'gitea.lan', 3310); + await this.scanFileForViruses(file.tmpPath); //, 'gitea.lan', 3310); // await this.scanFileForViruses("/tmp/testfile.txt"); } catch (error) { // If the file is infected or there's an error scanning the file, throw a validation exception @@ -1028,23 +1295,29 @@ export default class DatasetController { } // move to disk: - const fileName = `file-${cuid()}.${fileData.extname}`; //'file-ls0jyb8xbzqtrclufu2z2e0c.pdf' + const fileName = this.generateFilename(file.extname as string); const datasetFolder = `files/${dataset.id}`; // 'files/307' - // await fileData.moveToDisk(datasetFolder, { name: fileName, overwrite: true }, 'local'); - await fileData.move(drive.makePath(datasetFolder), { + const datasetFullPath = path.join(`${datasetFolder}`, fileName); + // await file.moveToDisk(datasetFolder, { name: fileName, overwrite: true }, 'local'); + // await file.move(drive.makePath(datasetFolder), { + // name: fileName, + // overwrite: true, // overwrite in case of conflict + // }); + await file.moveToDisk(datasetFullPath, 'local', { name: fileName, overwrite: true, // overwrite in case of conflict + disk: 'local', }); //save to db: - const { clientFileName, sortOrder } = this.extractVariableNameAndSortOrder(fileData.clientName); - const mimeType = fileData.headers['content-type'] || 'application/octet-stream'; // Fallback to a default MIME type + const { clientFileName, sortOrder } = this.extractVariableNameAndSortOrder(file.clientName); + const mimeType = file.headers['content-type'] || 'application/octet-stream'; // Fallback to a default MIME type const newFile = await dataset .useTransaction(trx) .related('files') .create({ pathName: `${datasetFolder}/${fileName}`, - fileSize: fileData.size, + fileSize: file.size, mimeType, label: clientFileName, sortOrder: sortOrder || index, @@ -1084,16 +1357,24 @@ export default class DatasetController { await dataset.useTransaction(trx).save(); await trx.commit(); - console.log('Dataset and related models created successfully'); + console.log('Dataset has been updated successfully'); session.flash('message', 'Dataset has been updated successfully'); // return response.redirect().toRoute('user.index'); return response.redirect().toRoute('dataset.edit', [dataset.id]); } catch (error) { + // Clean up temporary files if validation or later steps fail + uploadedTmpFiles.forEach((tmpPath) => { + try { + fs.unlinkSync(tmpPath); + } catch (cleanupError) { + console.error('Error cleaning up temporary file:', cleanupError); + } + }); if (trx !== null) { await trx.rollback(); } - console.error('Failed to create dataset and related models:', error); + console.error('Failed to update dataset and related models:', error); // throw new ValidationException(true, { 'upload error': `failed to create dataset and related models. ${error}` }); throw error; } @@ -1160,31 +1441,32 @@ export default class DatasetController { if (validStates.includes(dataset.server_state)) { if (dataset.files && dataset.files.length > 0) { for (const file of dataset.files) { - // overwritten delete method also delets file on filespace + // overwritten delete method also delets file on filespace and db object await file.delete(); } } const datasetFolder = `files/${params.id}`; - const folderExists = await drive.exists(datasetFolder); - if (folderExists) { - const dirListing = drive.list(datasetFolder); - const folderContents = await dirListing.toArray(); - if (folderContents.length === 0) { - await drive.delete(datasetFolder); - } - // delete dataset wirh relation in db - await dataset.delete(); - session.flash({ message: 'You have deleted 1 dataset!' }); - return response.redirect().toRoute('dataset.list'); - } else { - // session.flash({ - // warning: `You cannot delete this dataset! Invalid server_state: "${dataset.server_state}"!`, - // }); - return response - .flash({ warning: `You cannot delete this dataset! Dataset folder "${datasetFolder}" doesn't exist!` }) - .redirect() - .back(); - } + // const folderExists = await drive.use('local').exists(datasetFolder); + // if (folderExists) { + // const dirListing = drive.list(datasetFolder); + // const folderContents = await dirListing.toArray(); + // if (folderContents.length === 0) { + // await drive.delete(datasetFolder); + // } + await drive.use('local').deleteAll(datasetFolder); + // delete dataset wirh relation in db + await dataset.delete(); + session.flash({ message: 'You have deleted 1 dataset!' }); + return response.redirect().toRoute('dataset.list'); + // } else { + // // session.flash({ + // // warning: `You cannot delete this dataset! Invalid server_state: "${dataset.server_state}"!`, + // // }); + // return response + // .flash({ warning: `You cannot delete this dataset! Dataset folder "${datasetFolder}" doesn't exist!` }) + // .redirect() + // .back(); + // } } } catch (error) { if (error instanceof errors.E_VALIDATION_ERROR) { @@ -1192,11 +1474,89 @@ export default class DatasetController { throw error; } else if (error instanceof Exception) { // General exception handling - return response.flash('errors', { error: error.message }).redirect().back(); + session.flash({ error: error.message }); + return response.redirect().back(); } else { session.flash({ error: 'An error occurred while deleting the dataset.' }); return response.redirect().back(); } } } + + public async categorize({ inertia, request, response }: HttpContext) { + const id = request.param('id'); + // Preload dataset and its "collections" relation + const dataset = await Dataset.query().where('id', id).preload('collections').firstOrFail(); + const validStates = ['inprogress', 'rejected_editor']; + if (!validStates.includes(dataset.server_state)) { + // session.flash('errors', 'Invalid server state!'); + return response + .flash( + 'warning', + `Invalid server state. Dataset with id ${id} cannot be edited. Datset has server state ${dataset.server_state}.`, + ) + .redirect() + .toRoute('dataset.list'); + } + + const collectionRoles = await CollectionRole.query() + .preload('collections', (coll: Collection) => { + // preloa only top level collection with noparent_id + coll.whereNull('parent_id').orderBy('number', 'asc'); + }) + .exec(); + + return inertia.render('Submitter/Dataset/Category', { + collectionRoles: collectionRoles, + dataset: dataset, + relatedCollections: dataset.collections, + }); + } + + public async categorizeUpdate({ request, response, session }: HttpContext) { + // Get the dataset id from the route parameter + const id = request.param('id'); + const dataset = await Dataset.query().preload('files').where('id', id).firstOrFail(); + + const validStates = ['inprogress', 'rejected_editor']; + if (!validStates.includes(dataset.server_state)) { + return response + .flash( + 'warning', + `Invalid server state. Dataset with id ${id} cannot be categorized. Dataset has server state ${dataset.server_state}.`, + ) + .redirect() + .toRoute('dataset.list'); + } + + let trx: TransactionClientContract | null = null; + try { + trx = await db.transaction(); + // const user = (await User.find(auth.user?.id)) as User; + // await this.createDatasetAndAssociations(user, request, trx); + + // Retrieve the selected collections from the request. + // This should be an array of collection ids. + const collections: number[] = request.input('collections', []); + + // Synchronize the dataset collections using the transaction. + await dataset.useTransaction(trx).related('collections').sync(collections); + + // Commit the transaction.await trx.commit() + await trx.commit(); + + // Redirect with a success flash message. + // return response.flash('success', 'Dataset collections updated successfully!').redirect().toRoute('dataset.list'); + + session.flash('message', 'Dataset collections updated successfully!'); + return response.redirect().toRoute('dataset.list'); + } catch (error) { + if (trx !== null) { + await trx.rollback(); + } + console.error('Failed tocatgorize dataset collections:', error); + // throw new ValidationException(true, { 'upload error': `failed to create dataset and related models. ${error}` }); + throw error; + } + } } diff --git a/app/Library/Doi/DoiClient.ts b/app/Library/Doi/DoiClient.ts index 5b56280..86151e7 100644 --- a/app/Library/Doi/DoiClient.ts +++ b/app/Library/Doi/DoiClient.ts @@ -6,7 +6,7 @@ import DoiClientException from '#app/exceptions/DoiClientException'; import { StatusCodes } from 'http-status-codes'; import logger from '@adonisjs/core/services/logger'; import { AxiosResponse } from 'axios'; -import axios from 'axios'; +import { default as axios } from 'axios'; export class DoiClient implements DoiClientContract { public username: string; @@ -50,7 +50,7 @@ export class DoiClient implements DoiClientContract { 'Content-Type': 'application/xml;charset=UTF-8', }; try { - const metadataResponse = await axios.default.put(`${this.serviceUrl}/metadata/${doiValue}`, xmlMeta, { auth, headers }); + const metadataResponse = await axios.put(`${this.serviceUrl}/metadata/${doiValue}`, xmlMeta, { auth, headers }); // Response Codes // 201 Created: operation successful @@ -65,7 +65,7 @@ export class DoiClient implements DoiClientContract { throw new DoiClientException(metadataResponse.status, message); } - const doiResponse = await axios.default.put(`${this.serviceUrl}/doi/${doiValue}`, `doi=${doiValue}\nurl=${landingPageUrl}`, { + const doiResponse = await axios.put(`${this.serviceUrl}/doi/${doiValue}`, `doi=${doiValue}\nurl=${landingPageUrl}`, { auth, headers, }); diff --git a/app/Library/Oai/ResumptionToken.ts b/app/Library/Oai/ResumptionToken.ts index 5eca661..56bbea4 100644 --- a/app/Library/Oai/ResumptionToken.ts +++ b/app/Library/Oai/ResumptionToken.ts @@ -4,6 +4,7 @@ export default class ResumptionToken { private _resumptionId = ''; private _startPosition = 0; private _totalIds = 0; + private _queryParams: Record<string, any> = {}; get key(): string { return this.metadataPrefix + this.startPosition + this.totalIds; @@ -48,4 +49,12 @@ export default class ResumptionToken { set totalIds(totalIds: number) { this._totalIds = totalIds; } + + get queryParams(): Record<string, any> { + return this._queryParams; + } + + set queryParams(params: Record<string, any>) { + this._queryParams = params; + } } diff --git a/app/Library/Oai/TokenWorkerContract.ts b/app/Library/Oai/TokenWorkerContract.ts index 94dae70..2491817 100644 --- a/app/Library/Oai/TokenWorkerContract.ts +++ b/app/Library/Oai/TokenWorkerContract.ts @@ -6,6 +6,6 @@ export default abstract class TokenWorkerContract { abstract connect(): void; abstract close(): void; abstract get(key: string): Promise<ResumptionToken | null>; - abstract set(token: ResumptionToken): Promise<string>; + abstract set(token: ResumptionToken, browserFingerprint: string): Promise<string>; } diff --git a/app/Library/Oai/TokenWorkerSerice.ts b/app/Library/Oai/TokenWorkerSerice.ts index ee63f97..9cb9ccc 100644 --- a/app/Library/Oai/TokenWorkerSerice.ts +++ b/app/Library/Oai/TokenWorkerSerice.ts @@ -40,14 +40,64 @@ export default class TokenWorkerService implements TokenWorkerContract { return result !== undefined && result !== null; } - public async set(token: ResumptionToken): Promise<string> { - const uniqueName = await this.generateUniqueName(); + /** + * Simplified set method that stores the token using a browser fingerprint key. + * If the token for that fingerprint already exists and its documentIds match the new token, + * then the fingerprint key is simply returned. + */ + public async set(token: ResumptionToken, browserFingerprint: string): Promise<string> { + // Generate a 15-digit unique number string based on the fingerprint + const uniqueNumberKey = this.createUniqueNumberFromFingerprint(browserFingerprint, token.documentIds, token.totalIds); + // Optionally, you could prefix it if desired, e.g. 'rs_' + uniqueNumberKey + const fingerprintKey = uniqueNumberKey; + + // const fingerprintKey = `rs_fp_${browserFingerprint}`; + const existingTokenString = await this.cache.get(fingerprintKey); + + if (existingTokenString) { + const existingToken = this.parseToken(existingTokenString); + if (this.arraysAreEqual(existingToken.documentIds, token.documentIds)) { + return fingerprintKey; + } + } const serialToken = JSON.stringify(token); - await this.cache.setEx(uniqueName, this.ttl, serialToken); - return uniqueName; + await this.cache.setEx(fingerprintKey, this.ttl, serialToken); + return fingerprintKey; } + // Updated helper method to generate a unique key based on fingerprint and documentIds + private createUniqueNumberFromFingerprint(browserFingerprint: string, documentIds: number[], totalIds: number): string { + // Combine the fingerprint, document IDs and totalIds to produce the input string + const combined = browserFingerprint + ':' + documentIds.join('-') + ':' + totalIds; + // Simple hash algorithm + let hash = 0; + for (let i = 0; i < combined.length; i++) { + hash = (hash << 5) - hash + combined.charCodeAt(i); + hash |= 0; // Convert to 32-bit integer + } + // Ensure positive number and limit it to at most 15 digits + const positiveHash = Math.abs(hash) % 1000000000000000; + // Pad with trailing zeros to ensure a 15-digit string + return positiveHash.toString().padEnd(15, '0'); + } + + // Add a helper function to compare two arrays of numbers with identical order + private arraysAreEqual(arr1: number[], arr2: number[]): boolean { + if (arr1.length !== arr2.length) { + return false; + } + return arr1.every((num, index) => num === arr2[index]); + } + + // public async set(token: ResumptionToken): Promise<string> { + // const uniqueName = await this.generateUniqueName(); + + // const serialToken = JSON.stringify(token); + // await this.cache.setEx(uniqueName, this.ttl, serialToken); + // return uniqueName; + // } + private async generateUniqueName(): Promise<string> { let fc = 0; const uniqueId = dayjs().unix().toString(); diff --git a/app/exceptions/db_handler.ts b/app/exceptions/db_handler.ts new file mode 100644 index 0000000..868fec9 --- /dev/null +++ b/app/exceptions/db_handler.ts @@ -0,0 +1,43 @@ +// import { Exception } from '@adonisjs/core/exceptions' +import { HttpContext, ExceptionHandler } from '@adonisjs/core/http'; + +export default class DbHandlerException extends ExceptionHandler { + // constructor() { + // super(Logger) + // } + + async handle(error: any, ctx: HttpContext) { + // Check for AggregateError type + if (error.type === 'AggregateError' && error.aggregateErrors) { + const dbErrors = error.aggregateErrors.some((err: any) => err.code === 'ECONNREFUSED' && err.port === 5432); + + if (dbErrors) { + return ctx.response.status(503).json({ + status: 'error', + message: 'PostgreSQL database connection failed. Please ensure the database service is running.', + details: { + code: error.code, + type: error.type, + ports: error.aggregateErrors.map((err: any) => ({ + port: err.port, + address: err.address, + })), + }, + }); + } + } + + // Handle simple ECONNREFUSED errors + if (error.code === 'ECONNREFUSED') { + return ctx.response.status(503).json({ + status: 'error', + message: 'Database connection failed. Please ensure PostgreSQL is running.', + code: error.code, + }); + } + + return super.handle(error, ctx); + } + + static status = 500; +} diff --git a/app/exceptions/handler.ts b/app/exceptions/handler.ts index d8f1c63..a460c3d 100644 --- a/app/exceptions/handler.ts +++ b/app/exceptions/handler.ts @@ -46,6 +46,7 @@ export default class HttpExceptionHandler extends ExceptionHandler { // return view.render('./errors/server-error', { error }); // }, // }; + protected statusPages: Record<StatusPageRange, StatusPageRenderer> = { '404': (error, { inertia }) => { return inertia.render('Errors/ServerError', { @@ -58,9 +59,47 @@ export default class HttpExceptionHandler extends ExceptionHandler { return inertia.render('Errors/ServerError', { error: error.message, code: error.status, - }); + }); + }, + // '500': (error, { inertia }) => { + // return inertia.render('Errors/postgres_error', { + // status: 'error', + // message: 'PostgreSQL database connection failed. Please ensure the database service is running.', + // details: { + // code: error.code, + // type: error.status, + // ports: error.errors.map((err: any) => ({ + // port: err.port, + // address: err.address, + // })), + // }, + // }); + // }, + '500..599': (error, { inertia }) => { + if (error.code === 'ECONNREFUSED') { + const dbErrors = error.errors.some((err: any) => err.code === 'ECONNREFUSED' && err.port === 5432); + + if (dbErrors) { + return inertia.render('Errors/postgres_error', { + status: 'error', + message: 'PostgreSQL database connection failed. Please ensure the database service is running.', + details: { + code: error.code, + type: error.status, + ports: error.errors.map((err: any) => ({ + port: err.port, + address: err.address, + })), + }, + }); + } + } else { + return inertia.render('Errors/ServerError', { + error: error.message, + code: error.status, + }); + } }, - '500..599': (error, { inertia }) => inertia.render('Errors/ServerError', { error: error.message, code: error.status }), }; // constructor() { @@ -68,7 +107,7 @@ export default class HttpExceptionHandler extends ExceptionHandler { // } public async handle(error: any, ctx: HttpContext) { - const { response, request, session } = ctx; + const { response, request, session, inertia } = ctx; /** * Handle failed authentication attempt @@ -82,6 +121,47 @@ export default class HttpExceptionHandler extends ExceptionHandler { // return response.redirect('/dashboard'); // } + // Handle Axios errors + if (error.code === 'ECONNREFUSED') { + const dbErrors = error.errors.some((err: any) => err.code === 'ECONNREFUSED' && err.port === 5432); + + if (dbErrors) { + // return ctx.response.status(503).json({ + // status: 'error', + // message: 'PostgreSQL database connection failed. Please ensure the database service is running.', + // details: { + // code: error.code, + // type: error.status, + // ports: error.errors.map((err: any) => ({ + // port: err.port, + // address: err.address, + // })), + // }, + // }); + // return inertia.render('Errors/postgres_error', { + // status: 'error', + // message: 'PostgreSQL database connection failed. Please ensure the database service is running.', + // details: { + // code: error.code, + // type: error.status, + // ports: error.errors.map((err: any) => ({ + // port: err.port, + // address: err.address, + // })), + // }, + // }); + } + } + + // Handle simple ECONNREFUSED errors + // if (error.code === 'ECONNREFUSED') { + // return ctx.response.status(503).json({ + // status: 'error', + // message: 'Database connection failed. Please ensure PostgreSQL is running.', + // code: error.code, + // }); + // } + // https://github.com/inertiajs/inertia-laravel/issues/56 // let test = response.getStatus(); //200 // let header = request.header('X-Inertia'); // true @@ -98,12 +178,21 @@ export default class HttpExceptionHandler extends ExceptionHandler { // ->toResponse($request) // ->setStatusCode($response->status()); } + + // Handle simple ECONNREFUSED errors + // if (error.code === 'ECONNREFUSED') { + // return ctx.response.status(503).json({ + // status: 'error', + // message: 'Database connection failed. Please ensure PostgreSQL is running.', + // code: error.code, + // }); + // } // Dynamically change the error templates based on the absence of X-Inertia header // if (!ctx.request.header('X-Inertia')) { // this.statusPages = { - // '401..403': (error, { view }) => view.render('./errors/unauthorized', { error }), - // '404': (error, { view }) => view.render('./errors/not-found', { error }), - // '500..599': (error, { view }) => view.render('./errors/server-error', { error }), + // '401..403': (error, { inertia }) => inertia.render('Errors/ServerError', { error: error.message, code: error.status }), + // '404': (error, { inertia }) => inertia.render('Errors/ServerError', { error: error.message, code: error.status }), + // '500..599': (error, { inertia }) => inertia.render('Errors/ServerError', { error: error.message, code: error.status }), // }; // } diff --git a/app/models/dataset.ts b/app/models/dataset.ts index 73ca3a0..1233a28 100644 --- a/app/models/dataset.ts +++ b/app/models/dataset.ts @@ -209,6 +209,15 @@ export default class Dataset extends DatasetExtension { return mainTitle ? mainTitle.value : null; } + @computed({ + serializeAs: 'doi_identifier', + }) + public get doiIdentifier() { + // return `${this.firstName} ${this.lastName}`; + const identifier: DatasetIdentifier = this.identifier; + return identifier ? identifier.value : null; + } + @manyToMany(() => Person, { pivotForeignKey: 'document_id', pivotRelatedForeignKey: 'person_id', diff --git a/app/models/file.ts b/app/models/file.ts index fc129de..3d9145f 100644 --- a/app/models/file.ts +++ b/app/models/file.ts @@ -3,12 +3,12 @@ import { column, hasMany, belongsTo, SnakeCaseNamingStrategy, computed } from '@ import HashValue from './hash_value.js'; import Dataset from './dataset.js'; import BaseModel from './base_model.js'; -// import { Buffer } from 'buffer'; import * as fs from 'fs'; import crypto from 'crypto'; // import Drive from '@ioc:Adonis/Core/Drive'; // import Drive from '@adonisjs/drive'; -import drive from '#services/drive'; +// import drive from '#services/drive'; +import drive from '@adonisjs/drive/services/main'; import type { HasMany } from "@adonisjs/lucid/types/relations"; import type { BelongsTo } from "@adonisjs/lucid/types/relations"; @@ -88,7 +88,8 @@ export default class File extends BaseModel { serializeAs: 'filePath', }) public get filePath() { - return `/storage/app/public/${this.pathName}`; + // return `/storage/app/public/${this.pathName}`; + return `/storage/app/data/${this.pathName}`; // const mainTitle = this.titles?.find((title) => title.type === 'Main'); // return mainTitle ? mainTitle.value : null; } @@ -165,7 +166,7 @@ export default class File extends BaseModel { public async delete() { if (this.pathName) { // Delete file from additional storage - await drive.delete(this.pathName); + await drive.use('local').delete(this.pathName); } // Call the original delete method of the BaseModel to remove the record from the database diff --git a/app/models/mime_type.ts b/app/models/mime_type.ts index 52d614f..cf62678 100644 --- a/app/models/mime_type.ts +++ b/app/models/mime_type.ts @@ -16,9 +16,14 @@ export default class MimeType extends BaseModel { @column({}) public name: string; + // 1 : n file_extensions are separated by '|' in the database @column({}) public file_extension: string; + // 1 : n alternate_mimetype are separated by '|' in the database + @column({}) + public alternate_mimetype: string; + @column({}) public enabled: boolean; diff --git a/app/models/person.ts b/app/models/person.ts index 3feff8a..cdc612d 100644 --- a/app/models/person.ts +++ b/app/models/person.ts @@ -3,7 +3,7 @@ import { DateTime } from 'luxon'; import dayjs from 'dayjs'; import Dataset from './dataset.js'; import BaseModel from './base_model.js'; -import type { ManyToMany } from "@adonisjs/lucid/types/relations"; +import type { ManyToMany } from '@adonisjs/lucid/types/relations'; export default class Person extends BaseModel { public static namingStrategy = new SnakeCaseNamingStrategy(); @@ -51,7 +51,7 @@ export default class Person extends BaseModel { serializeAs: 'name', }) public get fullName() { - return `${this.firstName} ${this.lastName}`; + return [this.firstName, this.lastName].filter(Boolean).join(' '); } // @computed() @@ -64,10 +64,12 @@ export default class Person extends BaseModel { // return '2023-03-21 08:45:00'; // } - @computed() + @computed({ + serializeAs: 'dataset_count', + }) public get datasetCount() { const stock = this.$extras.datasets_count; //my pivot column name was "stock" - return stock; + return Number(stock); } @computed() @@ -76,6 +78,16 @@ export default class Person extends BaseModel { return contributor_type; } + @computed({ serializeAs: 'allow_email_contact' }) + public get allowEmailContact() { + // If the datasets relation is missing or empty, return false instead of null. + if (!this.datasets || this.datasets.length === 0) { + return false; + } + // Otherwise return the pivot attribute from the first related dataset. + return this.datasets[0].$extras?.pivot_allow_email_contact; + } + @manyToMany(() => Dataset, { pivotForeignKey: 'person_id', pivotRelatedForeignKey: 'document_id', diff --git a/app/models/types.ts b/app/models/types.ts new file mode 100644 index 0000000..1fda1e1 --- /dev/null +++ b/app/models/types.ts @@ -0,0 +1,57 @@ +/** + * Qs module config + */ +type QueryStringConfig = { + depth?: number + allowPrototypes?: boolean + plainObjects?: boolean + parameterLimit?: number + arrayLimit?: number + ignoreQueryPrefix?: boolean + delimiter?: RegExp | string + allowDots?: boolean + charset?: 'utf-8' | 'iso-8859-1' | undefined + charsetSentinel?: boolean + interpretNumericEntities?: boolean + parseArrays?: boolean + comma?: boolean + } +/** + * Base config used by all types + */ +type BodyParserBaseConfig = { + encoding: string + limit: string | number + types: string[] + } + + /** + * Body parser config for parsing JSON requests + */ + export type BodyParserJSONConfig = BodyParserBaseConfig & { + strict: boolean + convertEmptyStringsToNull: boolean + } + + /** + * Parser config for parsing form data + */ + export type BodyParserFormConfig = BodyParserBaseConfig & { + queryString: QueryStringConfig + convertEmptyStringsToNull: boolean + } + + /** + * Parser config for parsing raw body (untouched) + */ + export type BodyParserRawConfig = BodyParserBaseConfig +/** + * Body parser config for all supported form types + */ +export type BodyParserConfig = { + allowedMethods: string[] + json: BodyParserJSONConfig + form: BodyParserFormConfig + raw: BodyParserRawConfig + multipart: BodyParserMultipartConfig + } \ No newline at end of file diff --git a/app/models/user.ts b/app/models/user.ts index 58f1676..5dbaab4 100644 --- a/app/models/user.ts +++ b/app/models/user.ts @@ -1,6 +1,6 @@ import { DateTime } from 'luxon'; import { withAuthFinder } from '@adonisjs/auth/mixins/lucid'; -import { column, manyToMany, hasMany, SnakeCaseNamingStrategy } from '@adonisjs/lucid/orm'; +import { column, manyToMany, hasMany, SnakeCaseNamingStrategy, computed, beforeFetch, beforeFind } from '@adonisjs/lucid/orm'; import hash from '@adonisjs/core/services/hash'; import Role from './role.js'; import db from '@adonisjs/lucid/services/db'; @@ -49,7 +49,6 @@ export default class User extends compose(BaseModel, AuthFinder) { @column() public login: string; - @column() public firstName: string; @@ -87,6 +86,9 @@ export default class User extends compose(BaseModel, AuthFinder) { @column({}) public state: number; + @column({}) + public avatar: string; + // @hasOne(() => TotpSecret, { // foreignKey: 'user_id', // }) @@ -104,6 +106,7 @@ export default class User extends compose(BaseModel, AuthFinder) { // return Boolean(this.totp_secret?.twoFactorSecret); } + @manyToMany(() => Role, { pivotForeignKey: 'account_id', pivotRelatedForeignKey: 'role_id', @@ -121,6 +124,27 @@ export default class User extends compose(BaseModel, AuthFinder) { }) public backupcodes: HasMany<typeof BackupCode>; + @computed({ + serializeAs: 'is_admin', + }) + public get isAdmin(): boolean { + const roles = this.roles; + const isAdmin = roles?.map((role: Role) => role.name).includes('administrator'); + return isAdmin; + } + + // public toJSON() { + // return { + // ...super.toJSON(), + // roles: [] + // }; + // } + @beforeFind() + @beforeFetch() + public static preloadRoles(user: User) { + user.preload('roles') + } + public async getBackupCodes(this: User): Promise<BackupCode[]> { const test = await this.related('backupcodes').query(); // return test.map((role) => role.code); diff --git a/app/utils/utility-functions.ts b/app/utils/utility-functions.ts index 6afb2a0..a281f59 100644 --- a/app/utils/utility-functions.ts +++ b/app/utils/utility-functions.ts @@ -1,3 +1,16 @@ +import { join, isAbsolute } from 'node:path'; +import type { BodyParserConfig } from '#models/types'; +import { createId } from '@paralleldrive/cuid2'; +import { tmpdir } from 'node:os'; +import config from '@adonisjs/core/services/config'; +import Dataset from '#models/dataset'; +import { TransactionClientContract } from '@adonisjs/lucid/types/database'; +import Person from '#models/person'; + +interface Dictionary { + [index: string]: string; +} + export function sum(a: number, b: number): number { return a + b; } @@ -24,3 +37,88 @@ export function preg_match(regex: RegExp, str: string) { const result: boolean = regex.test(str); return result; } + +/** + * Returns the tmp path for storing the files temporarly + */ +export function getTmpPath(config: BodyParserConfig['multipart']): string { + if (typeof config.tmpFileName === 'function') { + const tmpPath = config.tmpFileName(); + return isAbsolute(tmpPath) ? tmpPath : join(tmpdir(), tmpPath); + } + + return join(tmpdir(), createId()); +} +/** + * Returns config for a given type + */ +export function getConfigFor<K extends keyof BodyParserConfig>(type: K): BodyParserConfig[K] { + const bodyParserConfig: BodyParserConfig = config.get('bodyparser'); + const configType = bodyParserConfig[type]; + return configType; +} + +export function parseBytesSize(size: string): number { + const units: Record<string, number> = { + kb: 1024, + mb: 1024 * 1024, + gb: 1024 * 1024 * 1024, + tb: 1024 * 1024 * 1024 * 1024, + }; + + const match = size.match(/^(\d+)(kb|mb|gb|tb)$/i); // Regex to match size format + + if (!match) { + throw new Error('Invalid size format'); + } + + const [, value, unit] = match; + return parseInt(value) * units[unit.toLowerCase()]; +} + +// Helper function to format bytes as human-readable text + +export function formatBytes(bytes: number): string { + if (bytes === 0) return '0 Bytes'; + const k = 1024; + const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB']; + const i = Math.floor(Math.log(bytes) / Math.log(k)); + return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i]; +} + +export async function savePersons(dataset: Dataset, persons: any[], role: string, trx: TransactionClientContract) { + for (const [key, person] of persons.entries()) { + const pivotData = { + role: role, + sort_order: key + 1, + allow_email_contact: false, + ...extractPivotAttributes(person), // Merge pivot attributes here + }; + + if (person.id !== undefined) { + await dataset + .useTransaction(trx) + .related('persons') + .attach({ + [person.id]: pivotData, + }); + } else { + const dataPerson = new Person(); + dataPerson.fill(person); + await dataset.useTransaction(trx).related('persons').save(dataPerson, false, pivotData); + } + } +} + +// Helper function to extract pivot attributes from a person object +function extractPivotAttributes(person: any) { + const pivotAttributes: Dictionary = {}; + for (const key in person) { + if (key.startsWith('pivot_')) { + // pivotAttributes[key] = person[key]; + const cleanKey = key.replace('pivot_', ''); // Remove 'pivot_' prefix + pivotAttributes[cleanKey] = person[key]; + } + } + return pivotAttributes; +} diff --git a/app/validators/dataset.ts b/app/validators/dataset.ts index 2b92348..5417463 100644 --- a/app/validators/dataset.ts +++ b/app/validators/dataset.ts @@ -1,6 +1,7 @@ import vine, { SimpleMessagesProvider } from '@vinejs/vine'; import { TitleTypes, DescriptionTypes, ContributorTypes, ReferenceIdentifierTypes, RelationTypes } from '#contracts/enums'; import dayjs from 'dayjs'; + // import MimeType from '#models/mime_type'; // const enabledExtensions = await MimeType.query().select('file_extension').where('enabled', true).exec(); @@ -39,7 +40,8 @@ export const createDatasetValidator = vine.compile( .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), }), ) - .minLength(1), + // .minLength(2) + .arrayContainsTypes({ typeA: 'main', typeB: 'translated' }), descriptions: vine .array( vine.object({ @@ -53,7 +55,8 @@ export const createDatasetValidator = vine.compile( .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), }), ) - .minLength(1), + // .minLength(1), + .arrayContainsTypes({ typeA: 'abstract', typeB: 'translated' }), authors: vine .array( vine.object({ @@ -125,7 +128,7 @@ export const createDatasetValidator = vine.compile( references: vine .array( vine.object({ - value: vine.string().trim().minLength(3).maxLength(255), + value: vine.string().trim().minLength(3).maxLength(255).validateReference({ typeField: 'type' }), type: vine.enum(Object.values(ReferenceIdentifierTypes)), relation: vine.enum(Object.values(RelationTypes)), label: vine.string().trim().minLength(2).maxLength(255), @@ -155,8 +158,7 @@ export const createDatasetValidator = vine.compile( .fileScan({ removeInfected: true }), ) .minLength(1), - }), -); + }),); /** * Validates the dataset's update action @@ -186,7 +188,8 @@ export const updateDatasetValidator = vine.compile( .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), }), ) - .minLength(1), + // .minLength(2) + .arrayContainsTypes({ typeA: 'main', typeB: 'translated' }), descriptions: vine .array( vine.object({ @@ -200,7 +203,7 @@ export const updateDatasetValidator = vine.compile( .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), }), ) - .minLength(1), + .arrayContainsTypes({ typeA: 'abstract', typeB: 'translated' }), authors: vine .array( vine.object({ @@ -272,7 +275,7 @@ export const updateDatasetValidator = vine.compile( references: vine .array( vine.object({ - value: vine.string().trim().minLength(3).maxLength(255), + value: vine.string().trim().minLength(3).maxLength(255).validateReference({ typeField: 'type' }), type: vine.enum(Object.values(ReferenceIdentifierTypes)), relation: vine.enum(Object.values(RelationTypes)), label: vine.string().trim().minLength(2).maxLength(255), @@ -311,12 +314,137 @@ export const updateDatasetValidator = vine.compile( }), ); -// files: schema.array([rules.minLength(1)]).members( -// schema.file({ -// size: '512mb', -// extnames: ['jpg', 'gif', 'png', 'tif', 'pdf', 'zip', 'fgb', 'nc', 'qml', 'ovr', 'gpkg', 'gml', 'gpx', 'kml', 'kmz', 'json'], -// }), -// ), +export const updateEditorDatasetValidator = vine.compile( + vine.object({ + // first step + language: vine + .string() + .trim() + .regex(/^[a-zA-Z0-9]+$/), + licenses: vine.array(vine.number()).minLength(1), // define at least one license for the new dataset + rights: vine.string().in(['true']), + // second step + type: vine.string().trim().minLength(3).maxLength(255), + creating_corporation: vine.string().trim().minLength(3).maxLength(255), + titles: vine + .array( + vine.object({ + value: vine.string().trim().minLength(3).maxLength(255), + type: vine.enum(Object.values(TitleTypes)), + language: vine + .string() + .trim() + .minLength(2) + .maxLength(255) + .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), + }), + ) + // .minLength(2) + .arrayContainsTypes({ typeA: 'main', typeB: 'translated' }), + descriptions: vine + .array( + vine.object({ + value: vine.string().trim().minLength(3).maxLength(2500), + type: vine.enum(Object.values(DescriptionTypes)), + language: vine + .string() + .trim() + .minLength(2) + .maxLength(255) + .translatedLanguage({ mainLanguageField: 'language', typeField: 'type' }), + }), + ) + .arrayContainsTypes({ typeA: 'abstract', typeB: 'translated' }), + authors: vine + .array( + vine.object({ + email: vine + .string() + .trim() + .maxLength(255) + .email() + .normalizeEmail() + .isUniquePerson({ table: 'persons', column: 'email', idField: 'id' }), + first_name: vine.string().trim().minLength(3).maxLength(255), + last_name: vine.string().trim().minLength(3).maxLength(255), + }), + ) + .minLength(1) + .distinct('email'), + contributors: vine + .array( + vine.object({ + email: vine + .string() + .trim() + .maxLength(255) + .email() + .normalizeEmail() + .isUniquePerson({ table: 'persons', column: 'email', idField: 'id' }), + first_name: vine.string().trim().minLength(3).maxLength(255), + last_name: vine.string().trim().minLength(3).maxLength(255), + pivot_contributor_type: vine.enum(Object.keys(ContributorTypes)), + }), + ) + .distinct('email') + .optional(), + // third step + project_id: vine.number().optional(), + // embargo_date: schema.date.optional({ format: 'yyyy-MM-dd' }, [rules.after(10, 'days')]), + embargo_date: vine + .date({ + formats: ['YYYY-MM-DD'], + }) + .afterOrEqual((_field) => { + return dayjs().add(10, 'day').format('YYYY-MM-DD'); + }) + .optional(), + coverage: vine.object({ + x_min: vine.number(), + x_max: vine.number(), + y_min: vine.number(), + y_max: vine.number(), + elevation_absolut: vine.number().positive().optional(), + elevation_min: vine.number().positive().optional().requiredIfExists('elevation_max'), + elevation_max: vine.number().positive().optional().requiredIfExists('elevation_min'), + // type: vine.enum(Object.values(DescriptionTypes)), + depth_absolut: vine.number().negative().optional(), + depth_min: vine.number().negative().optional().requiredIfExists('depth_max'), + depth_max: vine.number().negative().optional().requiredIfExists('depth_min'), + time_abolute: vine.date({ formats: { utc: true } }).optional(), + time_min: vine + .date({ formats: { utc: true } }) + .beforeField('time_max') + .optional() + .requiredIfExists('time_max'), + time_max: vine + .date({ formats: { utc: true } }) + .afterField('time_min') + .optional() + .requiredIfExists('time_min'), + }), + references: vine + .array( + vine.object({ + value: vine.string().trim().minLength(3).maxLength(255).validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + relation: vine.enum(Object.values(RelationTypes)), + label: vine.string().trim().minLength(2).maxLength(255), + }), + ) + .optional(), + subjects: vine + .array( + vine.object({ + value: vine.string().trim().minLength(3).maxLength(255), + // pivot_contributor_type: vine.enum(Object.keys(ContributorTypes)), + language: vine.string().trim().minLength(2).maxLength(255), + }), + ) + .minLength(3) + .distinct('value'), + }), +); let messagesProvider = new SimpleMessagesProvider({ 'minLength': '{{ field }} must be at least {{ min }} characters long', @@ -368,8 +496,10 @@ let messagesProvider = new SimpleMessagesProvider({ 'files.array.minLength': 'At least {{ min }} file upload is required.', 'files.*.size': 'file size is to big', 'files.*.extnames': 'file extension is not supported', + 'embargo_date.date.afterOrEqual': `Embargo date must be on or after ${dayjs().add(10, 'day').format('DD.MM.YYYY')}`, }); createDatasetValidator.messagesProvider = messagesProvider; updateDatasetValidator.messagesProvider = messagesProvider; +updateEditorDatasetValidator.messagesProvider = messagesProvider; // export default createDatasetValidator; diff --git a/app/validators/user.ts b/app/validators/user.ts index c87d467..b7bbd70 100644 --- a/app/validators/user.ts +++ b/app/validators/user.ts @@ -16,7 +16,7 @@ export const createUserValidator = vine.compile( first_name: vine.string().trim().minLength(3).maxLength(255), last_name: vine.string().trim().minLength(3).maxLength(255), email: vine.string().maxLength(255).email().normalizeEmail().isUnique({ table: 'accounts', column: 'email' }), - password: vine.string().confirmed().trim().minLength(3).maxLength(60), + new_password: vine.string().confirmed({ confirmationField: 'password_confirmation' }).trim().minLength(3).maxLength(60), roles: vine.array(vine.number()).minLength(1), // define at least one role for the new user }), ); @@ -42,7 +42,7 @@ export const updateUserValidator = vine.withMetaData<{ objId: number }>().compil .email() .normalizeEmail() .isUnique({ table: 'accounts', column: 'email', whereNot: (field) => field.meta.objId }), - password: vine.string().confirmed().trim().minLength(3).maxLength(60).optional(), + new_password: vine.string().confirmed({ confirmationField: 'password_confirmation' }).trim().minLength(3).maxLength(60).optional(), roles: vine.array(vine.number()).minLength(1), // define at least one role for the new user }), ); diff --git a/app/validators/vanilla_error_reporter.ts b/app/validators/vanilla_error_reporter.ts index fe145eb..75bd4ff 100644 --- a/app/validators/vanilla_error_reporter.ts +++ b/app/validators/vanilla_error_reporter.ts @@ -142,7 +142,7 @@ export class VanillaErrorReporter implements ErrorReporterContract { // } this.hasErrors = true; - var test = field.getFieldPath(); + // var test = field.getFieldPath(); // this.errors.push(error); // if (this.errors[error.field]) { diff --git a/commands/validate_checksum.ts b/commands/validate_checksum.ts index a2d8096..a11b38b 100644 --- a/commands/validate_checksum.ts +++ b/commands/validate_checksum.ts @@ -88,7 +88,7 @@ export default class ValidateChecksum extends BaseCommand { ); // Construct the file path - const filePath = '/storage/app/public/' + file.pathName; + const filePath = '/storage/app/data/' + file.pathName; try { // Calculate the MD5 checksum of the file diff --git a/config/app.ts b/config/app.ts index 23ad925..c0e7b2b 100644 --- a/config/app.ts +++ b/config/app.ts @@ -80,7 +80,8 @@ export const http = defineConfig({ | headers. | */ - trustProxy: proxyAddr.compile('loopback'), + // trustProxy: proxyAddr.compile('loopback'), + trustProxy: proxyAddr.compile(['127.0.0.1', '::1/128']), /* |-------------------------------------------------------------------------- diff --git a/config/bodyparser.ts b/config/bodyparser.ts index efc7dbb..b7c7d35 100644 --- a/config/bodyparser.ts +++ b/config/bodyparser.ts @@ -128,7 +128,7 @@ allowedMethods: ['POST', 'PUT', 'PATCH', 'DELETE'], | projects/:id/file | ``` */ - processManually: [], + processManually: ['/submitter/dataset/submit', '/submitter/dataset/:id/update'], /* |-------------------------------------------------------------------------- @@ -185,8 +185,8 @@ allowedMethods: ['POST', 'PUT', 'PATCH', 'DELETE'], | and fields data. | */ - // limit: '20mb', - limit: env.get('UPLOAD_LIMIT', '513mb'), + limit: '513mb', + //limit: env.get('UPLOAD_LIMIT', '513mb'), /* |-------------------------------------------------------------------------- diff --git a/config/database.ts b/config/database.ts index 87f302e..1c0fbdc 100644 --- a/config/database.ts +++ b/config/database.ts @@ -47,7 +47,7 @@ const databaseConfig = defineConfig({ migrations: { naturalSort: true, }, - healthCheck: false, + // healthCheck: false, debug: false, pool: { min: 1, max: 100 }, }, diff --git a/config/drive.ts b/config/drive.ts index 5679c62..cfb91f9 100644 --- a/config/drive.ts +++ b/config/drive.ts @@ -1,151 +1,45 @@ -/** - * Config source: https://git.io/JBt3o - * - * Feel free to let us know via PR, if you find something broken in this config - * file. - */ -import { defineConfig } from '#providers/drive/src/types/define_config'; -import env from '#start/env'; -// import { driveConfig } from '@adonisjs/core/build/config'; -// import { driveConfig } from "@adonisjs/drive/build/config.js"; -// import Application from '@ioc:Adonis/Core/Application'; +// import env from '#start/env' +// import app from '@adonisjs/core/services/app' +import { defineConfig, services } from '@adonisjs/drive' -/* -|-------------------------------------------------------------------------- -| Drive Config -|-------------------------------------------------------------------------- -| -| The `DriveConfig` relies on the `DisksList` interface which is -| defined inside the `contracts` directory. -| -*/ -export default defineConfig({ - /* - |-------------------------------------------------------------------------- - | Default disk - |-------------------------------------------------------------------------- - | - | The default disk to use for managing file uploads. The value is driven by - | the `DRIVE_DISK` environment variable. - | - */ - disk: env.get('DRIVE_DISK', 'local'), +const driveConfig = defineConfig({ + + default: 'public', + - disks: { - /* - |-------------------------------------------------------------------------- - | Local - |-------------------------------------------------------------------------- - | - | Uses the local file system to manage files. Make sure to turn off serving - | files when not using this disk. - | - */ - local: { - driver: 'local', - visibility: 'public', - - /* - |-------------------------------------------------------------------------- - | Storage root - Local driver only - |-------------------------------------------------------------------------- - | - | Define an absolute path to the storage directory from where to read the - | files. - | - */ - // root: Application.tmpPath('uploads'), - root: '/storage/app/public', - - /* - |-------------------------------------------------------------------------- - | Serve files - Local driver only - |-------------------------------------------------------------------------- - | - | When this is set to true, AdonisJS will configure a files server to serve - | files from the disk root. This is done to mimic the behavior of cloud - | storage services that has inbuilt capabilities to serve files. - | - */ - serveFiles: true, - - /* - |-------------------------------------------------------------------------- - | Base path - Local driver only - |-------------------------------------------------------------------------- - | - | Base path is always required when "serveFiles = true". Also make sure - | the `basePath` is unique across all the disks using "local" driver and - | you are not registering routes with this prefix. - | - */ - basePath: '/uploads', - }, - - /* - |-------------------------------------------------------------------------- - | S3 Driver - |-------------------------------------------------------------------------- - | - | Uses the S3 cloud storage to manage files. Make sure to install the s3 - | drive separately when using it. - | - |************************************************************************** - | npm i @adonisjs/drive-s3 - |************************************************************************** - | - */ - // s3: { - // driver: 's3', - // visibility: 'public', - // key: Env.get('S3_KEY'), - // secret: Env.get('S3_SECRET'), - // region: Env.get('S3_REGION'), - // bucket: Env.get('S3_BUCKET'), - // endpoint: Env.get('S3_ENDPOINT'), - // - // // For minio to work - // // forcePathStyle: true, - // }, - - /* - |-------------------------------------------------------------------------- - | GCS Driver - |-------------------------------------------------------------------------- - | - | Uses the Google cloud storage to manage files. Make sure to install the GCS - | drive separately when using it. - | - |************************************************************************** - | npm i @adonisjs/drive-gcs - |************************************************************************** - | - */ - // gcs: { - // driver: 'gcs', - // visibility: 'public', - // keyFilename: Env.get('GCS_KEY_FILENAME'), - // bucket: Env.get('GCS_BUCKET'), - - /* - |-------------------------------------------------------------------------- - | Uniform ACL - Google cloud storage only - |-------------------------------------------------------------------------- - | - | When using the Uniform ACL on the bucket, the "visibility" option is - | ignored. Since, the files ACL is managed by the google bucket policies - | directly. - | - |************************************************************************** - | Learn more: https://cloud.google.com/storage/docs/uniform-bucket-level-access - |************************************************************************** - | - | The following option just informs drive whether your bucket is using uniform - | ACL or not. The actual setting needs to be toggled within the Google cloud - | console. - | - */ - // usingUniformAcl: false, - // }, + services: { + + /** + * Persist files on the local filesystem + */ + public: services.fs({ + location: '/storage/app/public/', + serveFiles: true, + routeBasePath: '/public', + visibility: 'public', + }), + local: services.fs({ + location: '/storage/app/data/', + serveFiles: true, + routeBasePath: '/data', + visibility: 'public', + }), + + + /** + * Persist files on Digital Ocean spaces + */ + // spaces: services.s3({ + // credentials: { + // accessKeyId: env.get('SPACES_KEY'), + // secretAccessKey: env.get('SPACES_SECRET'), + // }, + // region: env.get('SPACES_REGION'), + // bucket: env.get('SPACES_BUCKET'), + // endpoint: env.get('SPACES_ENDPOINT'), + // visibility: 'public', + // }), }, -}); + }) + + export default driveConfig \ No newline at end of file diff --git a/config/drive_self.ts b/config/drive_self.ts new file mode 100644 index 0000000..f1c8152 --- /dev/null +++ b/config/drive_self.ts @@ -0,0 +1,233 @@ +/** + * Config source: https://git.io/JBt3o + * + * Feel free to let us know via PR, if you find something broken in this config + * file. + */ +import { defineConfig } from '#providers/drive/src/types/define_config'; +import env from '#start/env'; +// import { driveConfig } from '@adonisjs/core/build/config'; +// import { driveConfig } from "@adonisjs/drive/build/config.js"; +// import Application from '@ioc:Adonis/Core/Application'; + +/* +|-------------------------------------------------------------------------- +| Drive Config +|-------------------------------------------------------------------------- +| +| The `DriveConfig` relies on the `DisksList` interface which is +| defined inside the `contracts` directory. +| +*/ +export default defineConfig({ + /* + |-------------------------------------------------------------------------- + | Default disk + |-------------------------------------------------------------------------- + | + | The default disk to use for managing file uploads. The value is driven by + | the `DRIVE_DISK` environment variable. + | + */ + disk: env.get('DRIVE_DISK', 'local'), + + disks: { + /* + |-------------------------------------------------------------------------- + | Local + |-------------------------------------------------------------------------- + | + | Uses the local file system to manage files. Make sure to turn off serving + | files when not using this disk. + | + */ + local: { + driver: 'local', + visibility: 'public', + + /* + |-------------------------------------------------------------------------- + | Storage root - Local driver only + |-------------------------------------------------------------------------- + | + | Define an absolute path to the storage directory from where to read the + | files. + | + */ + // root: Application.tmpPath('uploads'), + root: '/storage/app/data', + + /* + |-------------------------------------------------------------------------- + | Serve files - Local driver only + |-------------------------------------------------------------------------- + | + | When this is set to true, AdonisJS will configure a files server to serve + | files from the disk root. This is done to mimic the behavior of cloud + | storage services that has inbuilt capabilities to serve files. + | + */ + serveFiles: true, + + /* + |-------------------------------------------------------------------------- + | Base path - Local driver only + |-------------------------------------------------------------------------- + | + | Base path is always required when "serveFiles = true". Also make sure + | the `basePath` is unique across all the disks using "local" driver and + | you are not registering routes with this prefix. + | + */ + basePath: '/files', + }, + + local: { + driver: 'local', + visibility: 'public', + + /* + |-------------------------------------------------------------------------- + | Storage root - Local driver only + |-------------------------------------------------------------------------- + | + | Define an absolute path to the storage directory from where to read the + | files. + | + */ + // root: Application.tmpPath('uploads'), + root: '/storage/app/data', + + /* + |-------------------------------------------------------------------------- + | Serve files - Local driver only + |-------------------------------------------------------------------------- + | + | When this is set to true, AdonisJS will configure a files server to serve + | files from the disk root. This is done to mimic the behavior of cloud + | storage services that has inbuilt capabilities to serve files. + | + */ + serveFiles: true, + + /* + |-------------------------------------------------------------------------- + | Base path - Local driver only + |-------------------------------------------------------------------------- + | + | Base path is always required when "serveFiles = true". Also make sure + | the `basePath` is unique across all the disks using "local" driver and + | you are not registering routes with this prefix. + | + */ + basePath: '/files', + }, + + fs: { + driver: 'local', + visibility: 'public', + + /* + |-------------------------------------------------------------------------- + | Storage root - Local driver only + |-------------------------------------------------------------------------- + | + | Define an absolute path to the storage directory from where to read the + | files. + | + */ + // root: Application.tmpPath('uploads'), + root: '/storage/app/public', + + /* + |-------------------------------------------------------------------------- + | Serve files - Local driver only + |-------------------------------------------------------------------------- + | + | When this is set to true, AdonisJS will configure a files server to serve + | files from the disk root. This is done to mimic the behavior of cloud + | storage services that has inbuilt capabilities to serve files. + | + */ + serveFiles: true, + + /* + |-------------------------------------------------------------------------- + | Base path - Local driver only + |-------------------------------------------------------------------------- + | + | Base path is always required when "serveFiles = true". Also make sure + | the `basePath` is unique across all the disks using "local" driver and + | you are not registering routes with this prefix. + | + */ + basePath: '/public', + }, + + /* + |-------------------------------------------------------------------------- + | S3 Driver + |-------------------------------------------------------------------------- + | + | Uses the S3 cloud storage to manage files. Make sure to install the s3 + | drive separately when using it. + | + |************************************************************************** + | npm i @adonisjs/drive-s3 + |************************************************************************** + | + */ + // s3: { + // driver: 's3', + // visibility: 'public', + // key: Env.get('S3_KEY'), + // secret: Env.get('S3_SECRET'), + // region: Env.get('S3_REGION'), + // bucket: Env.get('S3_BUCKET'), + // endpoint: Env.get('S3_ENDPOINT'), + // + // // For minio to work + // // forcePathStyle: true, + // }, + + /* + |-------------------------------------------------------------------------- + | GCS Driver + |-------------------------------------------------------------------------- + | + | Uses the Google cloud storage to manage files. Make sure to install the GCS + | drive separately when using it. + | + |************************************************************************** + | npm i @adonisjs/drive-gcs + |************************************************************************** + | + */ + // gcs: { + // driver: 'gcs', + // visibility: 'public', + // keyFilename: Env.get('GCS_KEY_FILENAME'), + // bucket: Env.get('GCS_BUCKET'), + + /* + |-------------------------------------------------------------------------- + | Uniform ACL - Google cloud storage only + |-------------------------------------------------------------------------- + | + | When using the Uniform ACL on the bucket, the "visibility" option is + | ignored. Since, the files ACL is managed by the google bucket policies + | directly. + | + |************************************************************************** + | Learn more: https://cloud.google.com/storage/docs/uniform-bucket-level-access + |************************************************************************** + | + | The following option just informs drive whether your bucket is using uniform + | ACL or not. The actual setting needs to be toggled within the Google cloud + | console. + | + */ + // usingUniformAcl: false, + // }, + }, +}); diff --git a/config/inertia.ts b/config/inertia.ts index bcbd170..f203ddd 100644 --- a/config/inertia.ts +++ b/config/inertia.ts @@ -1,7 +1,8 @@ import { defineConfig } from '@adonisjs/inertia'; import type { HttpContext } from '@adonisjs/core/http'; +import type { InferSharedProps } from '@adonisjs/inertia/types' -export default defineConfig({ +const inertiaConfig = defineConfig({ /** * Path to the Edge view that will be used as the root view for Inertia responses */ @@ -52,6 +53,12 @@ export default defineConfig({ }, }); +export default inertiaConfig + +declare module '@adonisjs/inertia/types' { + export interface SharedProps extends InferSharedProps<typeof inertiaConfig> {} + } + // import { InertiaConfig } from '@ioc:EidelLev/Inertia'; // /* diff --git a/config/mail.ts b/config/mail.ts index 909dec2..8016c29 100644 --- a/config/mail.ts +++ b/config/mail.ts @@ -12,7 +12,7 @@ const mailConfig = defineConfig({ mailers: { smtp: transports.smtp({ - socketTimeout: 5000,// Overall timeout (5 seconds) + // socketTimeout: 5000,// Overall timeout (5 seconds) host: env.get('SMTP_HOST', ''), port: env.get('SMTP_PORT'), secure: false, @@ -30,10 +30,10 @@ const mailConfig = defineConfig({ }, */ }), - resend: transports.resend({ - key: env.get('RESEND_API_KEY'), - baseUrl: 'https://api.resend.com', - }), + // resend: transports.resend({ + // key: env.get('RESEND_API_KEY'), + // baseUrl: 'https://api.resend.com', + // }), }, }); diff --git a/config/session.ts b/config/session.ts index dafbad2..baa0adf 100644 --- a/config/session.ts +++ b/config/session.ts @@ -6,7 +6,7 @@ */ import env from '#start/env'; -import app from '@adonisjs/core/services/app'; +// import app from '@adonisjs/core/services/app'; import { defineConfig, stores } from '@adonisjs/session'; const sessionConfig = defineConfig({ diff --git a/config/vite.ts b/config/vite.ts new file mode 100644 index 0000000..4969e53 --- /dev/null +++ b/config/vite.ts @@ -0,0 +1,32 @@ +import { defineConfig } from '@adonisjs/vite'; + +const viteBackendConfig = defineConfig({ + /** + * The output of vite will be written inside this + * directory. The path should be relative from + * the application root. + */ + buildDirectory: 'public/assets', + + /** + * The path to the manifest file generated by the + * "vite build" command. + */ + manifestFile: 'public/assets/.vite/manifest.json', + + /** + * Feel free to change the value of the "assetsUrl" to + * point to a CDN in production. + */ + assetsUrl: '/assets', + + /** + * Add defer attribute to scripts for better performance. + */ + scriptAttributes: { + defer: true, + }, + +}); + +export default viteBackendConfig; diff --git a/contracts/enums.ts b/contracts/enums.ts index 68ec4d9..5772570 100644 --- a/contracts/enums.ts +++ b/contracts/enums.ts @@ -21,6 +21,7 @@ export enum ServerStates { rejected_reviewer = 'rejected_reviewer', rejected_editor = 'rejected_editor', reviewed = 'reviewed', + rejected_to_reviewer = 'rejected_to_reviewer', } // for table dataset_titles diff --git a/database/migrations/acl_4_accounts.ts b/database/migrations/acl_4_accounts.ts index ba2373f..1237635 100644 --- a/database/migrations/acl_4_accounts.ts +++ b/database/migrations/acl_4_accounts.ts @@ -18,6 +18,7 @@ export default class Accounts extends BaseSchema { table.text("two_factor_recovery_codes").nullable(); table.smallint('state').nullable(); table.bigint('last_counter').nullable(); + table.string('avatar').nullable(); }); } @@ -43,6 +44,7 @@ export default class Accounts extends BaseSchema { // two_factor_recovery_codes text COLLATE pg_catalog."default", // state smallint, // last_counter bigint, +// avatar character varying(255), // ) // ALTER TABLE gba.accounts @@ -85,3 +87,6 @@ export default class Accounts extends BaseSchema { // GRANT ALL ON SEQUENCE gba.totp_secrets_id_seq TO tethys_admin; // ALTER TABLE gba.totp_secrets ALTER COLUMN id SET DEFAULT nextval('gba.totp_secrets_id_seq'); + + +// ALTER TABLE "accounts" ADD COLUMN "avatar" VARCHAR(255) NULL diff --git a/database/migrations/1723813974183_create_appconfigs_table.ts b/database/migrations/config_1_appconfigs_table.ts similarity index 100% rename from database/migrations/1723813974183_create_appconfigs_table.ts rename to database/migrations/config_1_appconfigs_table.ts diff --git a/database/migrations/1721999420193_create_backupcodes_table.ts b/database/migrations/config_2_backupcodes_table.ts similarity index 100% rename from database/migrations/1721999420193_create_backupcodes_table.ts rename to database/migrations/config_2_backupcodes_table.ts diff --git a/database/migrations/dataset_2_documents.ts b/database/migrations/dataset_2_documents.ts index 565e1f9..6629689 100644 --- a/database/migrations/dataset_2_documents.ts +++ b/database/migrations/dataset_2_documents.ts @@ -86,3 +86,22 @@ export default class Documents extends BaseSchema { // CONSTRAINT documents_server_state_check CHECK (server_state::text = ANY (ARRAY['deleted'::character varying::text, 'inprogress'::character varying::text, 'published'::character varying::text, 'released'::character varying::text, 'editor_accepted'::character varying::text, 'approved'::character varying::text, 'rejected_reviewer'::character varying::text, 'rejected_editor'::character varying::text, 'reviewed'::character varying::text])), // CONSTRAINT documents_type_check CHECK (type::text = ANY (ARRAY['analysisdata'::character varying::text, 'measurementdata'::character varying::text, 'monitoring'::character varying::text, 'remotesensing'::character varying::text, 'gis'::character varying::text, 'models'::character varying::text, 'mixedtype'::character varying::text])) // ) + + +// ALTER TABLE documents DROP CONSTRAINT documents_server_state_check; + +// ALTER TABLE documents +// ADD CONSTRAINT documents_server_state_check CHECK ( +// server_state::text = ANY (ARRAY[ +// 'deleted', +// 'inprogress', +// 'published', +// 'released', +// 'editor_accepted', +// 'approved', +// 'rejected_reviewer', +// 'rejected_editor', +// 'reviewed', +// 'rejected_to_reviewer' -- new value added +// ]::text[]) +// ); \ No newline at end of file diff --git a/database/migrations/dataset_7_collections.ts b/database/migrations/dataset_7_collections.ts index 7e2b3a9..1e970c5 100644 --- a/database/migrations/dataset_7_collections.ts +++ b/database/migrations/dataset_7_collections.ts @@ -5,7 +5,7 @@ export default class Collections extends BaseSchema { public async up() { this.schema.createTable(this.tableName, (table) => { - table.increments('id').defaultTo("nextval('collections_id_seq')"); + table.increments('id');//.defaultTo("nextval('collections_id_seq')"); table.integer('role_id').unsigned(); table .foreign('role_id', 'collections_role_id_foreign') @@ -54,3 +54,8 @@ export default class Collections extends BaseSchema { // ON UPDATE CASCADE // ON DELETE CASCADE // ) + + +// change to normal intzeger: +// ALTER TABLE collections ALTER COLUMN id DROP DEFAULT; +// DROP SEQUENCE IF EXISTS collections_id_seq; diff --git a/database/migrations/update_1_to_mime_types.ts b/database/migrations/update_1_to_mime_types.ts new file mode 100644 index 0000000..f09210c --- /dev/null +++ b/database/migrations/update_1_to_mime_types.ts @@ -0,0 +1,18 @@ +import { BaseSchema } from "@adonisjs/lucid/schema"; + +export default class AddAlternateMimetypeToMimeTypes extends BaseSchema { + protected tableName = 'mime_types'; + + public async up () { + this.schema.alterTable(this.tableName, (table) => { + table.string('alternate_mimetype').nullable(); + }); + } + + public async down () { + this.schema.alterTable(this.tableName, (table) => { + table.dropColumn('alternate_mimetype'); + }); + } +} +// ALTER TABLE "mime_types" ADD COLUMN "alternate_mimetype" VARCHAR(255) NULL \ No newline at end of file diff --git a/index.d.ts b/index.d.ts index 8165192..f6767a1 100644 --- a/index.d.ts +++ b/index.d.ts @@ -183,3 +183,9 @@ declare module 'saxon-js' { export function transform(options: ITransformOptions): Promise<ITransformOutput> | ITransformOutput; } + +declare global { + interface File { + sort_order?: number; + } + } \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index 609fd32..fa7be33 100644 --- a/package-lock.json +++ b/package-lock.json @@ -8,30 +8,32 @@ "name": "myapp", "version": "1.0.0", "dependencies": { - "@adonisjs/auth": "^9.1.1", - "@adonisjs/core": "^6.3.1", + "@adonisjs/auth": "^9.2.4", + "@adonisjs/bodyparser": "^10.0.1", + "@adonisjs/core": "^6.17.0", "@adonisjs/cors": "^2.2.1", - "@adonisjs/drive": "^2.3.0", - "@adonisjs/encore": "^1.0.0", - "@adonisjs/inertia": "^1.0.0-7", - "@adonisjs/lucid": "^21.1.0", + "@adonisjs/drive": "^3.2.0", + "@adonisjs/inertia": "^2.1.3", + "@adonisjs/lucid": "^21.5.1", "@adonisjs/mail": "^9.2.2", "@adonisjs/redis": "^9.1.0", - "@adonisjs/session": "^7.1.1", + "@adonisjs/session": "^7.5.0", "@adonisjs/shield": "^8.1.1", "@adonisjs/static": "^1.1.1", + "@adonisjs/vite": "^4.0.0", "@eidellev/adonis-stardust": "^3.0.0", "@fontsource/archivo-black": "^5.0.1", "@fontsource/inter": "^5.0.1", "@inertiajs/inertia": "^0.11.1", - "@inertiajs/vue3": "^1.0.0", - "@opensearch-project/opensearch": "^2.4.0", + "@inertiajs/vue3": "^2.0.3", + "@opensearch-project/opensearch": "^3.2.0", "@phc/format": "^1.0.0", - "@vinejs/vine": "^2.0.0", + "@poppinss/manager": "^5.0.2", + "@vinejs/vine": "^3.0.0", + "axios": "^1.7.9", "bcrypt": "^5.1.1", "bcryptjs": "^2.4.3", "clamscan": "^2.1.2", - "crypto": "^1.0.1", "dayjs": "^1.11.7", "deep-email-validator": "^0.1.21", "edge.js": "^6.0.1", @@ -47,7 +49,7 @@ "notiwind": "^2.0.0", "pg": "^8.9.0", "qrcode": "^1.5.3", - "redis": "^4.6.10", + "redis": "^5.0.0", "reflect-metadata": "^0.2.1", "saxon-js": "^2.5.0", "toastify-js": "^1.12.0", @@ -56,51 +58,48 @@ }, "devDependencies": { "@adonisjs/assembler": "^7.1.1", - "@adonisjs/tsconfig": "^1.2.1", - "@babel/core": "^7.20.12", - "@babel/plugin-proposal-class-properties": "^7.18.6", - "@babel/plugin-proposal-decorators": "^7.20.13", - "@babel/plugin-transform-runtime": "^7.19.6", - "@babel/preset-env": "^7.20.2", - "@babel/preset-typescript": "^7.18.6", - "@japa/api-client": "^2.0.3", - "@japa/assert": "^3.0.0", - "@japa/plugin-adonisjs": "^3.0.0", - "@japa/runner": "^3.1.1", + "@adonisjs/tsconfig": "^1.4.0", + "@headlessui/vue": "^1.7.23", + "@japa/assert": "^4.0.1", + "@japa/plugin-adonisjs": "^4.0.0", + "@japa/runner": "^4.2.0", "@mdi/js": "^7.1.96", "@poppinss/utils": "^6.7.2", - "@swc/core": "^1.4.2", - "@symfony/webpack-encore": "^5.0.1", + "@swc/wasm": "^1.10.14", "@tailwindcss/forms": "^0.5.2", "@types/bcryptjs": "^2.4.6", "@types/clamscan": "^2.0.4", "@types/escape-html": "^1.0.4", - "@types/leaflet": "^1.9.3", + "@types/fs-extra": "^11.0.4", + "@types/leaflet": "^1.9.16", "@types/luxon": "^3.4.2", - "@types/node": "^22.5.5", + "@types/node": "^22.10.2", "@types/proxy-addr": "^2.0.0", "@types/qrcode": "^1.5.5", "@types/source-map-support": "^0.5.6", "@types/sprintf-js": "^1.1.4", "@types/supertest": "^6.0.2", + "@vitejs/plugin-vue": "^5.2.1", "autoprefixer": "^10.4.13", "babel-preset-typescript-vue3": "^2.0.17", "chart.js": "^4.2.0", "dotenv-webpack": "^8.0.1", "eslint": "^8.57.1", - "eslint-config-prettier": "^9.0.0", + "eslint-config-prettier": "^10.0.1", "eslint-plugin-adonis": "^2.1.1", "eslint-plugin-prettier": "^5.0.0-alpha.2", + "hot-hook": "^0.4.0", "numeral": "^2.0.6", - "pinia": "^2.0.30", - "pino-pretty": "^11.2.2", + "pinia": "^3.0.2", + "pino-pretty": "^13.0.0", "postcss-loader": "^8.1.1", - "prettier": "^3.0.0", + "prettier": "^3.4.2", "supertest": "^6.3.3", - "tailwindcss": "^3.2.4", + "tailwindcss": "^3.4.17", "ts-loader": "^9.4.2", - "ts-node": "^10.9.2", - "typescript": "^5.1.3", + "ts-node-maintained": "^10.9.5", + "typescript": "~5.7", + "vite": "^6.0.11", "vue": "^3.4.26", "vue-facing-decorator": "^3.0.0", "vue-loader": "^17.0.1", @@ -131,55 +130,23 @@ } }, "node_modules/@adonisjs/application": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/@adonisjs/application/-/application-5.3.0.tgz", - "integrity": "sha512-AruZZXMgOdmmRxJEHUbXoqhgRavPfhkeIR2nQtGyxbn0PCNjqlGraq8ypuLINY1J+wNuH2tt0xCS98EDeMdTOQ==", + "version": "8.3.1", + "resolved": "https://registry.npmjs.org/@adonisjs/application/-/application-8.3.1.tgz", + "integrity": "sha512-hfZBgZ23BQAXvoSHDkc/I0hTSXyFVxypNqHPQ/WCk4VoWlBVWVgGaGnHLvIGhrZ3RMvyoC5NBgC0PR5G+/fGSw==", "license": "MIT", - "peer": true, "dependencies": { - "@adonisjs/config": "^3.0.9", - "@adonisjs/env": "^3.0.9", - "@adonisjs/fold": "^8.2.0", - "@adonisjs/logger": "^4.1.5", - "@adonisjs/profiler": "^6.0.9", - "@poppinss/utils": "^5.0.0", - "semver": "^7.3.8" - } - }, - "node_modules/@adonisjs/application/node_modules/@poppinss/utils": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-5.0.0.tgz", - "integrity": "sha512-SpJL5p4Nx3bRCpCf62KagZLUHLvJD+VDylGpXAeP2G5qb3s6SSOBlpaFmer4GxdyTqLIUt0PRCzF1TbpNU+qZw==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", - "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" - } - }, - "node_modules/@adonisjs/application/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "license": "MIT", - "peer": true, + "@poppinss/hooks": "^7.2.3", + "@poppinss/macroable": "^1.0.2", + "@poppinss/utils": "^6.7.3", + "glob-parent": "^6.0.2", + "tempura": "^0.4.0" + }, "engines": { - "node": ">=8" + "node": ">=18.16.0" + }, + "peerDependencies": { + "@adonisjs/config": "^5.0.0", + "@adonisjs/fold": "^10.0.0" } }, "node_modules/@adonisjs/assembler": { @@ -213,30 +180,14 @@ "typescript": "^4.0.0 || ^5.0.0" } }, - "node_modules/@adonisjs/assembler/node_modules/@adonisjs/env": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/@adonisjs/env/-/env-6.1.0.tgz", - "integrity": "sha512-CzK+njXTH3EK+d/UJPqckyqWocOItmLgHIUbvhpd6WvveBnfv1Dz5j9H3k+ogHqThDSJCXu1RkaRAC+HNym9gA==", - "devOptional": true, - "license": "MIT", - "dependencies": { - "@poppinss/utils": "^6.7.3", - "@poppinss/validator-lite": "^1.0.3", - "dotenv": "^16.4.5", - "split-lines": "^3.0.0" - }, - "engines": { - "node": ">=18.16.0" - } - }, "node_modules/@adonisjs/auth": { - "version": "9.3.0", - "resolved": "https://registry.npmjs.org/@adonisjs/auth/-/auth-9.3.0.tgz", - "integrity": "sha512-fJ6QNiCwWuIWONgwruPrzkkMK4EOj8YoCbrdnvHh6HdMXLTyp9xPR3xbbg2BTb1ssKxAnsaaNR+HPYfGLNYacw==", + "version": "9.4.0", + "resolved": "https://registry.npmjs.org/@adonisjs/auth/-/auth-9.4.0.tgz", + "integrity": "sha512-dzvnJRKY+RcKUXCRT6ebnlGV0wAfejTSGqS0XbgjB97r6Pww14MhsY89EBr1nSydQzvjdbtIR3EDGbU97BVmbQ==", "license": "MIT", "dependencies": { - "@adonisjs/presets": "^2.6.3", - "@poppinss/utils": "^6.8.3", + "@adonisjs/presets": "^2.6.4", + "@poppinss/utils": "^6.9.2", "basic-auth": "^2.0.1" }, "engines": { @@ -246,9 +197,9 @@ "@adonisjs/core": "^6.11.0", "@adonisjs/lucid": "^20.0.0 || ^21.0.1", "@adonisjs/session": "^7.4.1", - "@japa/api-client": "^2.0.3", + "@japa/api-client": "^2.0.3 || ^3.0.0", "@japa/browser-client": "^2.0.3", - "@japa/plugin-adonisjs": "^3.0.1" + "@japa/plugin-adonisjs": "^3.0.1 || ^4.0.0" }, "peerDependenciesMeta": { "@adonisjs/lucid": { @@ -268,80 +219,71 @@ } } }, - "node_modules/@adonisjs/config": { - "version": "3.0.9", - "resolved": "https://registry.npmjs.org/@adonisjs/config/-/config-3.0.9.tgz", - "integrity": "sha512-f+wzrc+0HLvhJyYGEMV2QTHtyJ8sI3PKvH9h/baW/iF8UO3KF+llHH0Cf3/M5dYnpdz9rnmj0VtdTaIDfxrgGg==", + "node_modules/@adonisjs/bodyparser": { + "version": "10.0.3", + "resolved": "https://registry.npmjs.org/@adonisjs/bodyparser/-/bodyparser-10.0.3.tgz", + "integrity": "sha512-rgnAE+w7/+8tlCG+GxT8/99TLyccMFHtUyIP99K4YbSg0af6kTHrbXOJkKE8YHKU8cFwECybu3uZOoIoHOYLyQ==", "license": "MIT", - "peer": true, "dependencies": { - "@poppinss/utils": "^5.0.0" - } - }, - "node_modules/@adonisjs/config/node_modules/@poppinss/utils": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-5.0.0.tgz", - "integrity": "sha512-SpJL5p4Nx3bRCpCf62KagZLUHLvJD+VDylGpXAeP2G5qb3s6SSOBlpaFmer4GxdyTqLIUt0PRCzF1TbpNU+qZw==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", + "@paralleldrive/cuid2": "^2.2.2", + "@poppinss/macroable": "^1.0.4", + "@poppinss/multiparty": "^2.0.1", + "@poppinss/utils": "^6.9.2", + "@types/qs": "^6.9.18", "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" + "file-type": "^20.1.0", + "inflation": "^2.1.0", + "media-typer": "^1.1.0", + "qs": "^6.14.0", + "raw-body": "^3.0.0" + }, + "engines": { + "node": ">=18.16.0" + }, + "peerDependencies": { + "@adonisjs/http-server": "^7.4.0" } }, - "node_modules/@adonisjs/config/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "node_modules/@adonisjs/config": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/@adonisjs/config/-/config-5.0.2.tgz", + "integrity": "sha512-NXjFqDHNGRTZ1EnA4zr20GFEt7qw/JvZ4ZV8/PzFyVc7dPoFprpoyE3bw7kmlKHhcQdBbF7YXCGB4q+HQUnqiQ==", "license": "MIT", - "peer": true, + "dependencies": { + "@poppinss/utils": "^6.7.3" + }, "engines": { - "node": ">=8" + "node": ">=18.16.0" } }, "node_modules/@adonisjs/core": { - "version": "6.17.0", - "resolved": "https://registry.npmjs.org/@adonisjs/core/-/core-6.17.0.tgz", - "integrity": "sha512-x78xF4VpTBi7bbBUvMXREbUddz2Ts3Ypz+WUp/D1R0YnA4lE8x+lwGox10SZAe+izQ41KkZlGKY24RYImFkLEg==", + "version": "6.17.2", + "resolved": "https://registry.npmjs.org/@adonisjs/core/-/core-6.17.2.tgz", + "integrity": "sha512-POT5COID8Z3j37+Dd7Y1EfG01Q6+HPY/tGcSb0Y97W2VIPkFjqcW2ooTE4wFT09u7coNohtXJa19a0feMz9ncw==", "license": "MIT", "dependencies": { "@adonisjs/ace": "^13.3.0", "@adonisjs/application": "^8.3.1", - "@adonisjs/bodyparser": "^10.0.2", + "@adonisjs/bodyparser": "^10.0.3", "@adonisjs/config": "^5.0.2", "@adonisjs/encryption": "^6.0.2", - "@adonisjs/env": "^6.1.0", + "@adonisjs/env": "^6.1.1", "@adonisjs/events": "^9.0.2", "@adonisjs/fold": "^10.1.3", "@adonisjs/hash": "^9.0.5", "@adonisjs/health": "^2.0.0", "@adonisjs/http-server": "^7.4.0", "@adonisjs/logger": "^6.0.5", - "@adonisjs/repl": "^4.0.1", - "@antfu/install-pkg": "^0.5.0", + "@adonisjs/repl": "^4.1.0", + "@antfu/install-pkg": "^1.0.0", "@paralleldrive/cuid2": "^2.2.2", - "@poppinss/colors": "^4.1.3", - "@poppinss/dumper": "^0.6.1", - "@poppinss/macroable": "^1.0.3", - "@poppinss/utils": "^6.8.3", + "@poppinss/colors": "^4.1.4", + "@poppinss/dumper": "^0.6.2", + "@poppinss/macroable": "^1.0.4", + "@poppinss/utils": "^6.9.2", "@sindresorhus/is": "^7.0.1", "@types/he": "^1.2.3", - "error-stack-parser-es": "^0.1.5", + "error-stack-parser-es": "^1.0.5", "he": "^1.2.0", "parse-imports": "^2.2.1", "pretty-hrtime": "^1.0.3", @@ -380,64 +322,61 @@ } } }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/application": { - "version": "8.3.1", - "resolved": "https://registry.npmjs.org/@adonisjs/application/-/application-8.3.1.tgz", - "integrity": "sha512-hfZBgZ23BQAXvoSHDkc/I0hTSXyFVxypNqHPQ/WCk4VoWlBVWVgGaGnHLvIGhrZ3RMvyoC5NBgC0PR5G+/fGSw==", + "node_modules/@adonisjs/core/node_modules/@antfu/install-pkg": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@antfu/install-pkg/-/install-pkg-1.0.0.tgz", + "integrity": "sha512-xvX6P/lo1B3ej0OsaErAjqgFYzYVcJpamjLAFLYh9vRJngBrMoUG7aVnrGTeqM7yxbyTD5p3F2+0/QUEh8Vzhw==", "license": "MIT", "dependencies": { - "@poppinss/hooks": "^7.2.3", - "@poppinss/macroable": "^1.0.2", - "@poppinss/utils": "^6.7.3", - "glob-parent": "^6.0.2", - "tempura": "^0.4.0" + "package-manager-detector": "^0.2.8", + "tinyexec": "^0.3.2" }, + "funding": { + "url": "https://github.com/sponsors/antfu" + } + }, + "node_modules/@adonisjs/cors": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/@adonisjs/cors/-/cors-2.2.1.tgz", + "integrity": "sha512-qnrSG8ylpgTeZBOYEN3yXxY0PBUEg1KGDhgn9VKVFGxLKT+o9GGVOSZxUK3wG341B1zB9w5vuZN1z4M0Jitb6g==", + "license": "MIT", "engines": { "node": ">=18.16.0" }, "peerDependencies": { - "@adonisjs/config": "^5.0.0", - "@adonisjs/fold": "^10.0.0" + "@adonisjs/core": "^6.2.0" } }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/bodyparser": { - "version": "10.0.2", - "resolved": "https://registry.npmjs.org/@adonisjs/bodyparser/-/bodyparser-10.0.2.tgz", - "integrity": "sha512-dkbn+DK5B1dODTwk5367gHPhaD4ZIoGon/jvq47iX2cnHjk3a0SyQrBEjoFhnrNkVOJZ76I3OJ3oixqgMcE+yA==", + "node_modules/@adonisjs/drive": { + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/@adonisjs/drive/-/drive-3.4.1.tgz", + "integrity": "sha512-oDYY4wJ7wDMlO4E+dZPYBu+T3Av7Mj+JL8+J33qgyxtiJylnZgoZDuRfFjZZix/bFNNuWX2sLwTMnyiDcK+YsA==", "license": "MIT", "dependencies": { - "@paralleldrive/cuid2": "^2.2.2", - "@poppinss/macroable": "^1.0.2", - "@poppinss/multiparty": "^2.0.1", - "@poppinss/utils": "^6.7.3", - "@types/qs": "^6.9.15", - "bytes": "^3.1.2", - "file-type": "^19.0.0", - "inflation": "^2.1.0", - "media-typer": "^1.1.0", - "qs": "^6.12.1", - "raw-body": "^2.5.2" + "flydrive": "^1.1.0" }, "engines": { - "node": ">=18.16.0" + "node": ">=20.6.0" }, "peerDependencies": { - "@adonisjs/http-server": "^7.0.2" - } - }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/config": { - "version": "5.0.2", - "resolved": "https://registry.npmjs.org/@adonisjs/config/-/config-5.0.2.tgz", - "integrity": "sha512-NXjFqDHNGRTZ1EnA4zr20GFEt7qw/JvZ4ZV8/PzFyVc7dPoFprpoyE3bw7kmlKHhcQdBbF7YXCGB4q+HQUnqiQ==", - "license": "MIT", - "dependencies": { - "@poppinss/utils": "^6.7.3" + "@adonisjs/core": "^6.2.0", + "@aws-sdk/client-s3": "^3.577.0", + "@aws-sdk/s3-request-presigner": "^3.577.0", + "@google-cloud/storage": "^7.10.2" }, - "engines": { - "node": ">=18.16.0" + "peerDependenciesMeta": { + "@aws-sdk/client-s3": { + "optional": true + }, + "@aws-sdk/s3-request-presigner": { + "optional": true + }, + "@google-cloud/storage": { + "optional": true + } } }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/encryption": { + "node_modules/@adonisjs/encryption": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/@adonisjs/encryption/-/encryption-6.0.2.tgz", "integrity": "sha512-37XqVPsZi6zXMbC0Me1/qlcTP0uE+KAtYOFx7D7Tvtz377NL/6gqxqgpW/BopgOSD+CVDXjzO/Wx3M2UrbkJRQ==", @@ -449,22 +388,22 @@ "node": ">=18.16.0" } }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/env": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/@adonisjs/env/-/env-6.1.0.tgz", - "integrity": "sha512-CzK+njXTH3EK+d/UJPqckyqWocOItmLgHIUbvhpd6WvveBnfv1Dz5j9H3k+ogHqThDSJCXu1RkaRAC+HNym9gA==", + "node_modules/@adonisjs/env": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/@adonisjs/env/-/env-6.2.0.tgz", + "integrity": "sha512-DZ7zQ4sBhzWftjU/SxJ7BstimrEiByCvmtAcMNDpDjOtJnR50172PRz1X7KjM3EqjCVrB19izzRVx/rmpCRPOA==", "license": "MIT", "dependencies": { - "@poppinss/utils": "^6.7.3", - "@poppinss/validator-lite": "^1.0.3", - "dotenv": "^16.4.5", + "@poppinss/utils": "^6.9.2", + "@poppinss/validator-lite": "^2.1.0", + "dotenv": "^16.4.7", "split-lines": "^3.0.0" }, "engines": { "node": ">=18.16.0" } }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/events": { + "node_modules/@adonisjs/events": { "version": "9.0.2", "resolved": "https://registry.npmjs.org/@adonisjs/events/-/events-9.0.2.tgz", "integrity": "sha512-qZn2e9V9C8tF4MNqEWv5JGxMG7gcHSJM8RncGpjuJ4cwFwd2jF4xrN6wkCprTVwoyZSxNS0Cp9NkAonySjG5vg==", @@ -482,7 +421,7 @@ "@adonisjs/fold": "^10.0.1" } }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/events/node_modules/@sindresorhus/is": { + "node_modules/@adonisjs/events/node_modules/@sindresorhus/is": { "version": "6.3.1", "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-6.3.1.tgz", "integrity": "sha512-FX4MfcifwJyFOI2lPoX7PQxCqx8BG1HCho7WdiXwpEQx1Ycij0JxkfYtGK7yqNScrZGSlt6RE6sw8QYoH7eKnQ==", @@ -494,7 +433,7 @@ "url": "https://github.com/sindresorhus/is?sponsor=1" } }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/fold": { + "node_modules/@adonisjs/fold": { "version": "10.1.3", "resolved": "https://registry.npmjs.org/@adonisjs/fold/-/fold-10.1.3.tgz", "integrity": "sha512-wzeuWMXx9SoJkNO4ycoyfxzoSyyMy3umVxb9cbzeWR/sYNVgi50l+vgJc634+lxpCE0RFTpxCv1M235EWDF9SQ==", @@ -506,388 +445,6 @@ "node": ">=18.16.0" } }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/http-server": { - "version": "7.4.0", - "resolved": "https://registry.npmjs.org/@adonisjs/http-server/-/http-server-7.4.0.tgz", - "integrity": "sha512-2Me8ytUu0Sm0jYJs2SAiYQX3ECF6clOJwPE04cswsAwEnqSFLZkflD3c6rApjLjZO6P1wFlo090HNaZCFrlcMQ==", - "license": "MIT", - "dependencies": { - "@paralleldrive/cuid2": "^2.2.2", - "@poppinss/macroable": "^1.0.3", - "@poppinss/matchit": "^3.1.2", - "@poppinss/middleware": "^3.2.4", - "@poppinss/utils": "^6.8.3", - "@sindresorhus/is": "^7.0.1", - "accepts": "^1.3.8", - "content-disposition": "^0.5.4", - "cookie": "^1.0.2", - "destroy": "^1.2.0", - "encodeurl": "^2.0.0", - "etag": "^1.8.1", - "fresh": "^0.5.2", - "mime-types": "^2.1.35", - "on-finished": "^2.4.1", - "proxy-addr": "^2.0.7", - "qs": "^6.13.1", - "tmp-cache": "^1.1.0", - "type-is": "^1.6.18", - "vary": "^1.1.2", - "youch": "^3.3.4" - }, - "engines": { - "node": ">=18.16.0" - }, - "peerDependencies": { - "@adonisjs/application": "^8.0.2", - "@adonisjs/encryption": "^6.0.0", - "@adonisjs/events": "^9.0.0", - "@adonisjs/fold": "^10.0.1", - "@adonisjs/logger": "^6.0.1" - } - }, - "node_modules/@adonisjs/core/node_modules/@adonisjs/logger": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/@adonisjs/logger/-/logger-6.0.5.tgz", - "integrity": "sha512-1QmbLPNC636MeJzqflMA64lUnAn5dbb7W0YQ/ea33papnNqGOfvDQuxqqKlzM6ww9jPZlXTIf/3t7KAWlfHCfQ==", - "license": "MIT", - "dependencies": { - "@poppinss/utils": "^6.8.3", - "abstract-logging": "^2.0.1", - "pino": "^9.5.0" - }, - "engines": { - "node": ">=18.16.0" - } - }, - "node_modules/@adonisjs/core/node_modules/@antfu/install-pkg": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/@antfu/install-pkg/-/install-pkg-0.5.0.tgz", - "integrity": "sha512-dKnk2xlAyC7rvTkpkHmu+Qy/2Zc3Vm/l8PtNyIOGDBtXPY3kThfU4ORNEp3V7SXw5XSOb+tOJaUYpfquPzL/Tg==", - "license": "MIT", - "dependencies": { - "package-manager-detector": "^0.2.5", - "tinyexec": "^0.3.1" - }, - "funding": { - "url": "https://github.com/sponsors/antfu" - } - }, - "node_modules/@adonisjs/core/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/@adonisjs/core/node_modules/encodeurl": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", - "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/@adonisjs/core/node_modules/media-typer": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz", - "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, - "node_modules/@adonisjs/core/node_modules/pino": { - "version": "9.6.0", - "resolved": "https://registry.npmjs.org/pino/-/pino-9.6.0.tgz", - "integrity": "sha512-i85pKRCt4qMjZ1+L7sy2Ag4t1atFcdbEt76+7iRJn1g2BvsnRMGu9p8pivl9fs63M2kF/A0OacFZhTub+m/qMg==", - "license": "MIT", - "dependencies": { - "atomic-sleep": "^1.0.0", - "fast-redact": "^3.1.1", - "on-exit-leak-free": "^2.1.0", - "pino-abstract-transport": "^2.0.0", - "pino-std-serializers": "^7.0.0", - "process-warning": "^4.0.0", - "quick-format-unescaped": "^4.0.3", - "real-require": "^0.2.0", - "safe-stable-stringify": "^2.3.1", - "sonic-boom": "^4.0.1", - "thread-stream": "^3.0.0" - }, - "bin": { - "pino": "bin.js" - } - }, - "node_modules/@adonisjs/core/node_modules/pino-std-serializers": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-7.0.0.tgz", - "integrity": "sha512-e906FRY0+tV27iq4juKzSYPbUj2do2X2JX4EzSca1631EB2QJQUqGbDuERal7LCtOpxl6x3+nvo9NPZcmjkiFA==", - "license": "MIT" - }, - "node_modules/@adonisjs/core/node_modules/process-warning": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/process-warning/-/process-warning-4.0.1.tgz", - "integrity": "sha512-3c2LzQ3rY9d0hc1emcsHhfT9Jwz0cChib/QN89oME2R451w5fy3f0afAhERFZAwrbDU43wk12d0ORBpDVME50Q==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/fastify" - }, - { - "type": "opencollective", - "url": "https://opencollective.com/fastify" - } - ], - "license": "MIT" - }, - "node_modules/@adonisjs/core/node_modules/sonic-boom": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/sonic-boom/-/sonic-boom-4.2.0.tgz", - "integrity": "sha512-INb7TM37/mAcsGmc9hyyI6+QR3rR1zVRu36B0NeGXKnOOLiZOfER5SA+N7X7k3yUYRzLWafduTDvJAfDswwEww==", - "license": "MIT", - "dependencies": { - "atomic-sleep": "^1.0.0" - } - }, - "node_modules/@adonisjs/cors": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/@adonisjs/cors/-/cors-2.2.1.tgz", - "integrity": "sha512-qnrSG8ylpgTeZBOYEN3yXxY0PBUEg1KGDhgn9VKVFGxLKT+o9GGVOSZxUK3wG341B1zB9w5vuZN1z4M0Jitb6g==", - "license": "MIT", - "engines": { - "node": ">=18.16.0" - }, - "peerDependencies": { - "@adonisjs/core": "^6.2.0" - } - }, - "node_modules/@adonisjs/drive": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/@adonisjs/drive/-/drive-2.3.0.tgz", - "integrity": "sha512-3V1kBe2qB/860KcS+dDonv8Xya2YDBdR7291pQgObJeTbV50Vy8RhwdOwtU7ybRfN2kh/svdC4238JGpbQOR9w==", - "license": "MIT", - "dependencies": { - "@poppinss/manager": "^5.0.2", - "@poppinss/utils": "^5.0.0", - "@types/fs-extra": "^9.0.13", - "etag": "^1.8.1", - "fs-extra": "^10.1.0", - "memfs": "^3.4.7" - }, - "peerDependencies": { - "@adonisjs/application": "^5.0.0", - "@adonisjs/http-server": "^5.0.0" - } - }, - "node_modules/@adonisjs/drive/node_modules/@poppinss/utils": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-5.0.0.tgz", - "integrity": "sha512-SpJL5p4Nx3bRCpCf62KagZLUHLvJD+VDylGpXAeP2G5qb3s6SSOBlpaFmer4GxdyTqLIUt0PRCzF1TbpNU+qZw==", - "license": "MIT", - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", - "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" - } - }, - "node_modules/@adonisjs/drive/node_modules/fs-extra": { - "version": "10.1.0", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.1.0.tgz", - "integrity": "sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ==", - "license": "MIT", - "dependencies": { - "graceful-fs": "^4.2.0", - "jsonfile": "^6.0.1", - "universalify": "^2.0.0" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/@adonisjs/drive/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "license": "MIT", - "engines": { - "node": ">=8" - } - }, - "node_modules/@adonisjs/encore": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/@adonisjs/encore/-/encore-1.0.0.tgz", - "integrity": "sha512-pDl1sE3QguS3q94UklSoA5oCU5PTS1fuI9OEVnjo8B4ruI/wXBwuXDEAofYictiE3sXBM4+6MmxbI25MaL9DEA==", - "license": "MIT", - "dependencies": { - "@poppinss/utils": "^6.7.1", - "edge-error": "^4.0.1", - "stringify-attributes": "^4.0.0" - }, - "peerDependencies": { - "@adonisjs/core": "^6.2.1", - "edge.js": "^6.0.1" - } - }, - "node_modules/@adonisjs/encryption": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/@adonisjs/encryption/-/encryption-4.0.8.tgz", - "integrity": "sha512-zMWbIESPHXafsbiLJyON/hlRYwrTIA3PuTil7xC8W4ngC36PgWe86Ra0x0t961u1We/LaSGkT8Vn93DymqB3aA==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/utils": "^4.0.3" - }, - "peerDependencies": { - "@adonisjs/application": "^5.0.0" - } - }, - "node_modules/@adonisjs/encryption/node_modules/@poppinss/utils": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-4.0.4.tgz", - "integrity": "sha512-6LS3mofSVB9IQZqofA4rX6KVVcCpdwUQuNe4efHqOTzgD/Q5HTVvDP0vKg1m994QlzJs4aLW1JwXVcNCThEh4g==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", - "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" - } - }, - "node_modules/@adonisjs/encryption/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "license": "MIT", - "peer": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/@adonisjs/env": { - "version": "3.0.9", - "resolved": "https://registry.npmjs.org/@adonisjs/env/-/env-3.0.9.tgz", - "integrity": "sha512-9lxGmOQuF4FpUQ6NIwL/YQumaXG+2Wt8jQlQptplSUTasy6DHSEp7/SYvtC2RD9vxwn4gsptNCo+f8YRiqUvwQ==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/utils": "^4.0.2", - "dotenv": "^16.0.0", - "validator": "^13.7.0" - } - }, - "node_modules/@adonisjs/env/node_modules/@poppinss/utils": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-4.0.4.tgz", - "integrity": "sha512-6LS3mofSVB9IQZqofA4rX6KVVcCpdwUQuNe4efHqOTzgD/Q5HTVvDP0vKg1m994QlzJs4aLW1JwXVcNCThEh4g==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", - "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" - } - }, - "node_modules/@adonisjs/env/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "license": "MIT", - "peer": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/@adonisjs/fold": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/@adonisjs/fold/-/fold-8.2.0.tgz", - "integrity": "sha512-Uoo2HPp4SShIkGOF3+p3gT09W3j0zpkK+fOpPyYPTqYm7CWAunklTlowqX45b6CAVb5DCcORDUB8ia4D1ijeKg==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/utils": "^4.0.4" - } - }, - "node_modules/@adonisjs/fold/node_modules/@poppinss/utils": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-4.0.4.tgz", - "integrity": "sha512-6LS3mofSVB9IQZqofA4rX6KVVcCpdwUQuNe4efHqOTzgD/Q5HTVvDP0vKg1m994QlzJs4aLW1JwXVcNCThEh4g==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", - "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" - } - }, - "node_modules/@adonisjs/fold/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "license": "MIT", - "peer": true, - "engines": { - "node": ">=8" - } - }, "node_modules/@adonisjs/hash": { "version": "9.0.5", "resolved": "https://registry.npmjs.org/@adonisjs/hash/-/hash-9.0.5.tgz", @@ -927,88 +484,57 @@ } }, "node_modules/@adonisjs/http-server": { - "version": "5.12.0", - "resolved": "https://registry.npmjs.org/@adonisjs/http-server/-/http-server-5.12.0.tgz", - "integrity": "sha512-+9cw/DRlLO2NSoHsccmMe3pFf6c0/8INds2yf73ZAZOmzUROb9DQaXHocJ/iwHX9EVxtDuKWDc5z0jI1SYdqEA==", + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/@adonisjs/http-server/-/http-server-7.6.0.tgz", + "integrity": "sha512-ZrQxdIyTJ4gTtUNXp6S956R1tLS5BT2HNP7SAiQSeVmn6J5CL0fBZ2+hx7L0yrF34MUUgZvykIEeLsm3rusXGQ==", "license": "MIT", - "peer": true, "dependencies": { + "@paralleldrive/cuid2": "^2.2.2", + "@poppinss/macroable": "^1.0.4", "@poppinss/matchit": "^3.1.2", - "@poppinss/utils": "^5.0.0", + "@poppinss/middleware": "^3.2.5", + "@poppinss/utils": "^6.9.2", + "@sindresorhus/is": "^7.0.1", "accepts": "^1.3.8", - "co-compose": "^7.0.2", "content-disposition": "^0.5.4", - "cookie": "^0.5.0", + "cookie": "^1.0.2", "destroy": "^1.2.0", - "encodeurl": "^1.0.2", + "encodeurl": "^2.0.0", "etag": "^1.8.1", "fresh": "^0.5.2", - "haye": "^3.0.0", - "macroable": "^7.0.2", "mime-types": "^2.1.35", - "ms": "^2.1.3", "on-finished": "^2.4.1", - "pluralize": "^8.0.0", "proxy-addr": "^2.0.7", - "qs": "^6.11.0", + "qs": "^6.14.0", "tmp-cache": "^1.1.0", "type-is": "^1.6.18", - "vary": "^1.1.2" + "vary": "^1.1.2", + "youch": "^3.3.4" + }, + "engines": { + "node": ">=18.16.0" }, "peerDependencies": { - "@adonisjs/application": "^5.0.0", - "@adonisjs/encryption": "^4.0.0" - } - }, - "node_modules/@adonisjs/http-server/node_modules/@poppinss/utils": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-5.0.0.tgz", - "integrity": "sha512-SpJL5p4Nx3bRCpCf62KagZLUHLvJD+VDylGpXAeP2G5qb3s6SSOBlpaFmer4GxdyTqLIUt0PRCzF1TbpNU+qZw==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", - "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" - } - }, - "node_modules/@adonisjs/http-server/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "license": "MIT", - "peer": true, - "engines": { - "node": ">=8" + "@adonisjs/application": "^8.0.2", + "@adonisjs/encryption": "^6.0.0", + "@adonisjs/events": "^9.0.0", + "@adonisjs/fold": "^10.0.1", + "@adonisjs/logger": "^6.0.1" } }, "node_modules/@adonisjs/inertia": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/@adonisjs/inertia/-/inertia-1.2.4.tgz", - "integrity": "sha512-brYEkXn7ZNv394Bvqewlh9anoxLnW8yQ8beyJz14I4ylZmEyyOcxis3JpobpUzlpWHGLAU/zssEqpJL5AXFcmg==", + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/@adonisjs/inertia/-/inertia-2.1.3.tgz", + "integrity": "sha512-ah05RXq5xb5FhqvLYQd1CU0UCrhaxC2U+xAftw1qPMXzx0jzX7DBjEK6kr5CEfM8PLOdt8gnOVmbDBsVyNSzbg==", "license": "MIT", "dependencies": { "@poppinss/utils": "^6.8.3", - "@tuyau/utils": "^0.0.4", + "@tuyau/utils": "^0.0.6", "crc-32": "^1.2.2", "edge-error": "^4.0.1", "html-entities": "^2.5.2", "locate-path": "^7.2.0", - "qs": "^6.13.0" + "qs": "^6.13.1" }, "engines": { "node": ">=20.6.0" @@ -1016,7 +542,7 @@ "peerDependencies": { "@adonisjs/core": "^6.9.1", "@adonisjs/session": "^7.4.0", - "@adonisjs/vite": "^3.0.0", + "@adonisjs/vite": "^4.0.0", "@japa/api-client": "^2.0.0", "edge.js": "^6.0.0" }, @@ -1027,72 +553,37 @@ } }, "node_modules/@adonisjs/logger": { - "version": "4.1.6", - "resolved": "https://registry.npmjs.org/@adonisjs/logger/-/logger-4.1.6.tgz", - "integrity": "sha512-lmnx/wGvxnlHLuUEWQ+VpUQuKBEKumdutNWZPythGP/VpBRzunypntDb7O9XfauPva6B6Z9WVjr2+v3ElTcKPQ==", + "version": "6.0.6", + "resolved": "https://registry.npmjs.org/@adonisjs/logger/-/logger-6.0.6.tgz", + "integrity": "sha512-r5mLmmklSezzu3cu9QaXle2/gPNrgKpiIo+utYlwV3ITsW5JeIX/xcwwMTNM/9f1zU+SwOj5NccPTEFD3feRaw==", "license": "MIT", - "peer": true, "dependencies": { - "@poppinss/utils": "^5.0.0", - "@types/pino": "^6.3.12", + "@poppinss/utils": "^6.9.2", "abstract-logging": "^2.0.1", - "pino": "^6.14.0" - } - }, - "node_modules/@adonisjs/logger/node_modules/@poppinss/utils": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-5.0.0.tgz", - "integrity": "sha512-SpJL5p4Nx3bRCpCf62KagZLUHLvJD+VDylGpXAeP2G5qb3s6SSOBlpaFmer4GxdyTqLIUt0PRCzF1TbpNU+qZw==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", - "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" - } - }, - "node_modules/@adonisjs/logger/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "license": "MIT", - "peer": true, + "pino": "^9.6.0" + }, "engines": { - "node": ">=8" + "node": ">=18.16.0" } }, "node_modules/@adonisjs/lucid": { - "version": "21.6.0", - "resolved": "https://registry.npmjs.org/@adonisjs/lucid/-/lucid-21.6.0.tgz", - "integrity": "sha512-wEocH/PsAT4VVTS5qz65VvxDJqEJCZPzTMmVldddNhM9Yiwc5mr5Nu+UQp9Pfnx2Jp3o+nicnF0IM6thJyb6lg==", + "version": "21.6.1", + "resolved": "https://registry.npmjs.org/@adonisjs/lucid/-/lucid-21.6.1.tgz", + "integrity": "sha512-0TLCcPm9GHShJlsDAF5SHilafnvTxW25y5nD3bGJBSMEaNfGXcGRBbnyWoeNs7DsnqMCZ6ociT+0XMcKJWzQrQ==", "license": "MIT", "dependencies": { - "@adonisjs/presets": "^2.6.3", - "@faker-js/faker": "^9.3.0", - "@poppinss/hooks": "^7.2.4", - "@poppinss/macroable": "^1.0.3", - "@poppinss/utils": "^6.8.3", + "@adonisjs/presets": "^2.6.4", + "@faker-js/faker": "^9.6.0", + "@poppinss/hooks": "^7.2.5", + "@poppinss/macroable": "^1.0.4", + "@poppinss/utils": "^6.9.2", "fast-deep-equal": "^3.1.3", "igniculus": "^1.5.0", "kleur": "^4.1.5", "knex": "^3.1.0", "knex-dynamic-connection": "^3.2.0", "pretty-hrtime": "^1.0.3", - "qs": "^6.13.1", + "qs": "^6.14.0", "slash": "^5.1.0", "tarn": "^3.0.2" }, @@ -1165,94 +656,44 @@ } } }, - "node_modules/@adonisjs/profiler": { - "version": "6.0.9", - "resolved": "https://registry.npmjs.org/@adonisjs/profiler/-/profiler-6.0.9.tgz", - "integrity": "sha512-V1bJPPDTn05NzAKUEICnYtWi9fC8NownUToaqxVkWOUovYBO6ubt06qtH1Uv9zvUjB2PKHUn+ieDAOgyHle09A==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/utils": "^4.0.3", - "jest-worker": "^27.5.1" - }, - "peerDependencies": { - "@adonisjs/logger": "^4.0.0" - } - }, - "node_modules/@adonisjs/profiler/node_modules/@poppinss/utils": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/@poppinss/utils/-/utils-4.0.4.tgz", - "integrity": "sha512-6LS3mofSVB9IQZqofA4rX6KVVcCpdwUQuNe4efHqOTzgD/Q5HTVvDP0vKg1m994QlzJs4aLW1JwXVcNCThEh4g==", - "license": "MIT", - "peer": true, - "dependencies": { - "@poppinss/file-generator": "^1.0.2", - "@types/bytes": "^3.1.1", - "@types/he": "^1.1.2", - "bytes": "^3.1.2", - "change-case": "^4.1.2", - "cuid": "^2.1.8", - "flattie": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "he": "^1.2.0", - "kind-of": "^6.0.3", - "lodash": "^4.17.21", - "ms": "^2.1.3", - "pluralize": "^8.0.0", - "require-all": "^3.0.0", - "resolve-from": "^5.0.0", - "slugify": "^1.6.5", - "truncatise": "0.0.8" - } - }, - "node_modules/@adonisjs/profiler/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "license": "MIT", - "peer": true, - "engines": { - "node": ">=8" - } - }, "node_modules/@adonisjs/redis": { - "version": "9.1.0", - "resolved": "https://registry.npmjs.org/@adonisjs/redis/-/redis-9.1.0.tgz", - "integrity": "sha512-cesml/1Libmwm2yjmbbp2xtyGp+LBNkqCe9ehSmPFM+5puRRbJkqNf6ZaHFQfKdjQU1Y7qR9xtyf5uHLU/K0uw==", + "version": "9.2.0", + "resolved": "https://registry.npmjs.org/@adonisjs/redis/-/redis-9.2.0.tgz", + "integrity": "sha512-DUI9NrHDLZ2ISNjMlqWbKJT99ZYj1ZmvhNFTfhVs9lc7K2KJmNKZfK8Y85a8eN7q+ZYMBYSu1uRemxGs6xRaYw==", "license": "MIT", "dependencies": { - "@poppinss/utils": "^6.7.3", - "emittery": "^1.0.3", - "ioredis": "^5.4.1" + "@poppinss/utils": "^6.9.2", + "emittery": "^1.1.0", + "ioredis": "^5.4.2" }, "engines": { - "node": ">=18.16.0" + "node": ">=20.6.0" }, "peerDependencies": { "@adonisjs/core": "^6.2.0" } }, "node_modules/@adonisjs/repl": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/@adonisjs/repl/-/repl-4.0.1.tgz", - "integrity": "sha512-fgDRC5I8RBKHzsJPM4rRQF/OWI0K9cNihCIf4yHdqQt3mhFqWSOUjSi4sXWykdICLiddmyBO86au7i0d0dj5vQ==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/@adonisjs/repl/-/repl-4.1.0.tgz", + "integrity": "sha512-7Ml87uoufDQmpjRZYbJeRTk0/WcD4DllJ96L1r2IWF/jZIsryiVN5o+7Xd7fHlRzd8iapAbs32Tq4a6fVI6UKA==", "license": "MIT", "dependencies": { - "@poppinss/colors": "^4.1.2", - "string-width": "^7.1.0" + "@poppinss/colors": "^4.1.4", + "string-width": "^7.2.0" }, "engines": { "node": ">=18.16.0" } }, "node_modules/@adonisjs/session": { - "version": "7.5.0", - "resolved": "https://registry.npmjs.org/@adonisjs/session/-/session-7.5.0.tgz", - "integrity": "sha512-H5fFrkE/yFSWP/XD8eyWLyUVbtIqTcw9QBP1me4d9vnl4mao4jHrmnNA8rwtzsEFdXu/UgWdH4kljh1G9PDzWw==", + "version": "7.5.1", + "resolved": "https://registry.npmjs.org/@adonisjs/session/-/session-7.5.1.tgz", + "integrity": "sha512-b1E0W/1nnJfAq3Gv8yPywgsxJ7uzzOBJxxulonXI4t1eSdvJzZGNrFScfVLOcjTwlxwrEFA847tULIQxgR4Spw==", "license": "MIT", "dependencies": { - "@poppinss/macroable": "^1.0.3", - "@poppinss/utils": "^6.8.3" + "@poppinss/macroable": "^1.0.4", + "@poppinss/utils": "^6.9.2" }, "engines": { "node": ">=18.16.0" @@ -1262,7 +703,7 @@ "@adonisjs/redis": "^8.0.1 || ^9.0.0", "@aws-sdk/client-dynamodb": "^3.658.0", "@aws-sdk/util-dynamodb": "^3.658.0", - "@japa/api-client": "^2.0.3", + "@japa/api-client": "^2.0.3 || ^3.0.0", "@japa/browser-client": "^2.0.3", "edge.js": "^6.0.2" }, @@ -1288,12 +729,12 @@ } }, "node_modules/@adonisjs/shield": { - "version": "8.1.1", - "resolved": "https://registry.npmjs.org/@adonisjs/shield/-/shield-8.1.1.tgz", - "integrity": "sha512-b/rIypxfG8HKPRvWYJo7qhvAlvYCFXn7/A7eb/QI/PQV4fMmW4iF9tAykxl5peu4WJCHCwXwR3Y6/j0VerQcmQ==", + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/@adonisjs/shield/-/shield-8.2.0.tgz", + "integrity": "sha512-RddRbs92y87GGFUgDSWD/Pg7qYHh8+MctUphFZwtbTblvDckrjZxuYyp+vmVATPuvDvK7sOlatuZHT4HQSz9zQ==", "license": "MIT", "dependencies": { - "@poppinss/utils": "^6.7.1", + "@poppinss/utils": "^6.9.2", "csrf": "^3.1.0", "helmet-csp": "^3.4.0" }, @@ -1304,7 +745,7 @@ "@adonisjs/core": "^6.2.0", "@adonisjs/i18n": "^2.0.0", "@adonisjs/session": "^7.0.0", - "@japa/api-client": "^2.0.2", + "@japa/api-client": "^2.0.2 || ^3.0.0", "edge.js": "^6.0.1" }, "peerDependenciesMeta": { @@ -1342,16 +783,15 @@ "license": "MIT" }, "node_modules/@adonisjs/vite": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@adonisjs/vite/-/vite-3.0.0.tgz", - "integrity": "sha512-E09M0zjGwu5GgMYFTGcA00f0y3DblvqekXgtxccjpOE/zLf5ggOxTwI5iZXgD4lVETYirQ0QdS3azznCW2TYkQ==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@adonisjs/vite/-/vite-4.0.0.tgz", + "integrity": "sha512-5kdE0qLIm2dj+XO0HiCohmh3tfZ+X468wkNXErQ15VS0fkDjZWns2VuiYpoToTKd4tdX/oGlpnpd8aIwAGB4ow==", "license": "MIT", - "peer": true, "dependencies": { - "@poppinss/utils": "^6.7.3", - "@vavite/multibuild": "^4.1.1", + "@poppinss/utils": "^6.8.3", + "@vavite/multibuild": "^5.1.0", "edge-error": "^4.0.1", - "vite-plugin-restart": "^0.4.0" + "vite-plugin-restart": "^0.4.2" }, "engines": { "node": ">=20.6.0" @@ -1360,7 +800,7 @@ "@adonisjs/core": "^6.3.0", "@adonisjs/shield": "^8.0.0", "edge.js": "^6.0.1", - "vite": "^5.1.4" + "vite": "^6.0.0" }, "peerDependenciesMeta": { "@adonisjs/shield": { @@ -1390,6 +830,7 @@ "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", "dev": true, "license": "Apache-2.0", + "peer": true, "dependencies": { "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.24" @@ -1412,56 +853,6 @@ "url": "https://github.com/sponsors/antfu" } }, - "node_modules/@apidevtools/json-schema-ref-parser": { - "version": "9.1.2", - "resolved": "https://registry.npmjs.org/@apidevtools/json-schema-ref-parser/-/json-schema-ref-parser-9.1.2.tgz", - "integrity": "sha512-r1w81DpR+KyRWd3f+rk6TNqMgedmAxZP5v5KWlXQWlgMUUtyEJch0DKEci1SorPMiSeM8XPl7MZ3miJ60JIpQg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jsdevtools/ono": "^7.1.3", - "@types/json-schema": "^7.0.6", - "call-me-maybe": "^1.0.1", - "js-yaml": "^4.1.0" - } - }, - "node_modules/@apidevtools/json-schema-ref-parser/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/@apidevtools/json-schema-ref-parser/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/@apidevtools/openapi-schemas": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/@apidevtools/openapi-schemas/-/openapi-schemas-2.1.0.tgz", - "integrity": "sha512-Zc1AlqrJlX3SlpupFGpiLi2EbteyP7fXmUOGup6/DnkRgjP9bgMM/ag+n91rsv0U1Gpz0H3VILA/o3bW7Ua6BQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - } - }, - "node_modules/@apidevtools/swagger-methods": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/@apidevtools/swagger-methods/-/swagger-methods-3.0.2.tgz", - "integrity": "sha512-QAkD5kK2b1WfjDS/UQn/qQkbwF31uqRjPTrsCs5ZG9BQGAkjwvqGFjjPqAuzac/IYzpPtRzjCP1WrTuAIjMrXg==", - "dev": true, - "license": "MIT" - }, "node_modules/@arr/every": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/@arr/every/-/every-1.0.1.tgz", @@ -1487,32 +878,34 @@ } }, "node_modules/@babel/compat-data": { - "version": "7.26.3", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.26.3.tgz", - "integrity": "sha512-nHIxvKPniQXpmQLb0vhY3VaFb3S0YrTAwpOWJZh1wn3oJPjJk9Asva204PsBdmAE8vpzfHudT8DB0scYvy9q0g==", + "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", + "peer": true, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/core": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.26.0.tgz", - "integrity": "sha512-i1SLeK+DzNnQ3LL/CswPCa/E5u4lh1k6IAEphON8F+cXt0t9euTshDru0q7/IqMa1PMPz5RnHuHscF8/ZJsStg==", + "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", + "peer": true, "dependencies": { "@ampproject/remapping": "^2.2.0", - "@babel/code-frame": "^7.26.0", - "@babel/generator": "^7.26.0", - "@babel/helper-compilation-targets": "^7.25.9", + "@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.0", - "@babel/parser": "^7.26.0", - "@babel/template": "^7.25.9", - "@babel/traverse": "^7.25.9", - "@babel/types": "^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", @@ -1527,25 +920,15 @@ "url": "https://opencollective.com/babel" } }, - "node_modules/@babel/core/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/@babel/generator": { - "version": "7.26.3", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.26.3.tgz", - "integrity": "sha512-6FF/urZvD0sTeO7k6/B15pMLC4CHUv1426lzr3N01aHJTl046uCAh9LXW/fzeXXjPNCJ6iABW5XaWOsIZB93aQ==", + "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.26.3", - "@babel/types": "^7.26.3", + "@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" @@ -1568,13 +951,14 @@ } }, "node_modules/@babel/helper-compilation-targets": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.25.9.tgz", - "integrity": "sha512-j9Db8Suy6yV/VHa4qzrj9yZfZxhLWQdVnRlXxmKLYlhWUVB1sB2G5sxuWYXk/whHD9iW76PmNzxZ4UCnTQTVEQ==", + "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", + "peer": true, "dependencies": { - "@babel/compat-data": "^7.25.9", + "@babel/compat-data": "^7.26.8", "@babel/helper-validator-option": "^7.25.9", "browserslist": "^4.24.0", "lru-cache": "^5.1.1", @@ -1584,29 +968,19 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/helper-compilation-targets/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/@babel/helper-create-class-features-plugin": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.25.9.tgz", - "integrity": "sha512-UTZQMvt0d/rSz6KI+qdu7GQze5TIajwTS++GUozlw8VBJDEOAqSXwm1WvmYEZwqdqSGQshRocPDqrt4HBZB3fQ==", + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.27.0.tgz", + "integrity": "sha512-vSGCvMecvFCd/BdpGlhpXYNhhC4ccxyvQWpbGL4CWbvfEoLFWUZuSuf7s9Aw70flgQF+6vptvgK2IfOnKlRmBg==", "dev": true, "license": "MIT", "dependencies": { "@babel/helper-annotate-as-pure": "^7.25.9", "@babel/helper-member-expression-to-functions": "^7.25.9", "@babel/helper-optimise-call-expression": "^7.25.9", - "@babel/helper-replace-supers": "^7.25.9", + "@babel/helper-replace-supers": "^7.26.5", "@babel/helper-skip-transparent-expression-wrappers": "^7.25.9", - "@babel/traverse": "^7.25.9", + "@babel/traverse": "^7.27.0", "semver": "^6.3.1" }, "engines": { @@ -1616,61 +990,6 @@ "@babel/core": "^7.0.0" } }, - "node_modules/@babel/helper-create-class-features-plugin/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/@babel/helper-create-regexp-features-plugin": { - "version": "7.26.3", - "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.26.3.tgz", - "integrity": "sha512-G7ZRb40uUgdKOQqPLjfD12ZmGA54PzqDFUv2BKImnC9QIfGhIHKvVML0oN8IUiDq4iRqpq74ABpvOaerfWdong==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-annotate-as-pure": "^7.25.9", - "regexpu-core": "^6.2.0", - "semver": "^6.3.1" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/helper-create-regexp-features-plugin/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/@babel/helper-define-polyfill-provider": { - "version": "0.6.3", - "resolved": "https://registry.npmjs.org/@babel/helper-define-polyfill-provider/-/helper-define-polyfill-provider-0.6.3.tgz", - "integrity": "sha512-HK7Bi+Hj6H+VTHA3ZvBis7V/6hu9QuTrnMXNybfUf2iiuU/N97I8VjB+KbhFF8Rld/Lx5MzoCwPCpPjfK+n8Cg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-compilation-targets": "^7.22.6", - "@babel/helper-plugin-utils": "^7.22.5", - "debug": "^4.1.1", - "lodash.debounce": "^4.0.8", - "resolve": "^1.14.2" - }, - "peerDependencies": { - "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" - } - }, "node_modules/@babel/helper-member-expression-to-functions": { "version": "7.25.9", "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.25.9.tgz", @@ -1731,43 +1050,25 @@ } }, "node_modules/@babel/helper-plugin-utils": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.25.9.tgz", - "integrity": "sha512-kSMlyUVdWe25rEsRGviIgOWnoT/nfABVWlqt9N19/dIPWViAOW2s9wznP5tURbs/IDuNk4gPy3YdYRgH3uxhBw==", + "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-remap-async-to-generator": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.25.9.tgz", - "integrity": "sha512-IZtukuUeBbhgOcaW2s06OXTzVNJR0ybm4W5xC1opWFFJMZbwRj5LCk+ByYH7WdZPZTt8KnFwA8pvjN2yqcPlgw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-annotate-as-pure": "^7.25.9", - "@babel/helper-wrap-function": "^7.25.9", - "@babel/traverse": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, "node_modules/@babel/helper-replace-supers": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.25.9.tgz", - "integrity": "sha512-IiDqTOTBQy0sWyeXyGSC5TBJpGFXBkRynjBeXsvbhQFKj2viwJC76Epz35YLU1fpe/Am6Vppb7W7zM4fPQzLsQ==", + "version": "7.26.5", + "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.26.5.tgz", + "integrity": "sha512-bJ6iIVdYX1YooY2X7w1q6VITt+LnUILtNk7zT78ykuwStx8BauCzxvFqFaHjOpW1bVnSUM1PN1f0p5P21wHxvg==", "dev": true, "license": "MIT", "dependencies": { "@babel/helper-member-expression-to-functions": "^7.25.9", "@babel/helper-optimise-call-expression": "^7.25.9", - "@babel/traverse": "^7.25.9" + "@babel/traverse": "^7.26.5" }, "engines": { "node": ">=6.9.0" @@ -1818,42 +1119,28 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/helper-wrap-function": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.25.9.tgz", - "integrity": "sha512-ETzz9UTjQSTmw39GboatdymDq4XIQbR8ySgVrylRhPOFpsd+JrKHIuF0de7GCWmem+T4uC5z7EZguod7Wj4A4g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/template": "^7.25.9", - "@babel/traverse": "^7.25.9", - "@babel/types": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/@babel/helpers": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.26.0.tgz", - "integrity": "sha512-tbhNuIxNcVb21pInl3ZSjksLCvgdZy9KwJ8brv993QtIVKJBBkYXz4q4ZbAv31GdnC+R90np23L5FbEBlthAEw==", + "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", + "peer": true, "dependencies": { - "@babel/template": "^7.25.9", - "@babel/types": "^7.26.0" + "@babel/template": "^7.27.0", + "@babel/types": "^7.27.0" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/parser": { - "version": "7.26.3", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.26.3.tgz", - "integrity": "sha512-WJ/CvmY8Mea8iDXo6a7RK2wbmJITT5fN3BEkRuFlxVyNx8jOKIIhmC4fSkTcPcf8JyavbBwIe6OpiCOBXt/IcA==", + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.27.0.tgz", + "integrity": "sha512-iaepho73/2Pz7w2eMS0Q5f83+0RKI7i4xmiYeBmDzfRVbQtTOG7Ts0S4HzJVsTMGI9keU8rNfuZr8DKfSt7Yyg==", "license": "MIT", "dependencies": { - "@babel/types": "^7.26.3" + "@babel/types": "^7.27.0" }, "bin": { "parser": "bin/babel-parser.js" @@ -1862,187 +1149,6 @@ "node": ">=6.0.0" } }, - "node_modules/@babel/plugin-bugfix-firefox-class-in-computed-class-key": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-bugfix-firefox-class-in-computed-class-key/-/plugin-bugfix-firefox-class-in-computed-class-key-7.25.9.tgz", - "integrity": "sha512-ZkRyVkThtxQ/J6nv3JFYv1RYY+JT5BvU0y3k5bWrmuG4woXypRa4PXmm9RhOwodRkYFWqC0C0cqcJ4OqR7kW+g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/traverse": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/plugin-bugfix-safari-class-field-initializer-scope": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-bugfix-safari-class-field-initializer-scope/-/plugin-bugfix-safari-class-field-initializer-scope-7.25.9.tgz", - "integrity": "sha512-MrGRLZxLD/Zjj0gdU15dfs+HH/OXvnw/U4jJD8vpcP2CJQapPEv1IWwjc/qMg7ItBlPwSv1hRBbb7LeuANdcnw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression/-/plugin-bugfix-safari-id-destructuring-collision-in-function-expression-7.25.9.tgz", - "integrity": "sha512-2qUwwfAFpJLZqxd02YW9btUCZHl+RFvdDkNfZwaIJrvB8Tesjsk8pEQkTvGwZXLqXUx/2oyY3ySRhm6HOXuCug==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining/-/plugin-bugfix-v8-spread-parameters-in-optional-chaining-7.25.9.tgz", - "integrity": "sha512-6xWgLZTJXwilVjlnV7ospI3xi+sl8lN8rXXbBD6vYn3UYDlGsag8wrZkKcSI8G6KgqKP7vNFaDgeDnfAABq61g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-skip-transparent-expression-wrappers": "^7.25.9", - "@babel/plugin-transform-optional-chaining": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.13.0" - } - }, - "node_modules/@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly/-/plugin-bugfix-v8-static-class-fields-redefine-readonly-7.25.9.tgz", - "integrity": "sha512-aLnMXYPnzwwqhYSCyXfKkIkYgJ8zv9RK+roo9DkTXz38ynIhd9XCbN08s3MGvqL2MYGVUGdRQLL/JqBIeJhJBg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/traverse": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/plugin-proposal-class-properties": { - "version": "7.18.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.18.6.tgz", - "integrity": "sha512-cumfXOF0+nzZrrN8Rf0t7M+tF6sZc7vhQwYQck9q1/5w2OExlD+b4v4RpMJFaV1Z7WcDRgO6FqvxqxGlwo+RHQ==", - "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-class-properties instead.", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-class-features-plugin": "^7.18.6", - "@babel/helper-plugin-utils": "^7.18.6" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-proposal-decorators": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-decorators/-/plugin-proposal-decorators-7.25.9.tgz", - "integrity": "sha512-smkNLL/O1ezy9Nhy4CNosc4Va+1wo5w4gzSZeLe6y6dM4mmHfYOCPolXQPHQxonZCF+ZyebxN9vqOolkYrSn5g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-class-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/plugin-syntax-decorators": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-proposal-private-property-in-object": { - "version": "7.21.0-placeholder-for-preset-env.2", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-property-in-object/-/plugin-proposal-private-property-in-object-7.21.0-placeholder-for-preset-env.2.tgz", - "integrity": "sha512-SOSkfJDddaM7mak6cPEpswyTRnuRltl429hMraQEglW+OkovnCzsiszTmsrlY//qLFjCpQDFRvjdm2wA5pPm9w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-syntax-decorators": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-decorators/-/plugin-syntax-decorators-7.25.9.tgz", - "integrity": "sha512-ryzI0McXUPJnRCvMo4lumIKZUzhYUO/ScI+Mz4YVaTLt04DHNSjEUjKVvbzQjZFLuod/cYEc07mJWhzl6v4DPg==", - "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-syntax-import-assertions": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-assertions/-/plugin-syntax-import-assertions-7.26.0.tgz", - "integrity": "sha512-QCWT5Hh830hK5EQa7XzuqIkQU9tT/whqbDz7kuaZMHFl1inRRg7JnuAEOQ0Ur0QUl0NufCk1msK2BeY79Aj/eg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-syntax-import-attributes": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.26.0.tgz", - "integrity": "sha512-e2dttdsJ1ZTpi3B9UYGLw41hifAubg19AtCu/2I/F1QNVclOBr1dYpTdmdyZ84Xiz43BS/tCUkMAZNLv12Pi+A==", - "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-syntax-jsx": { "version": "7.25.9", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.25.9.tgz", @@ -2075,409 +1181,6 @@ "@babel/core": "^7.0.0-0" } }, - "node_modules/@babel/plugin-syntax-unicode-sets-regex": { - "version": "7.18.6", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-unicode-sets-regex/-/plugin-syntax-unicode-sets-regex-7.18.6.tgz", - "integrity": "sha512-727YkEAPwSIQTv5im8QHz3upqp92JTWhidIC81Tdx4VJYIte/VndKf1qKrfnnhPLiPghStWfvC/iFaMCQu7Nqg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-regexp-features-plugin": "^7.18.6", - "@babel/helper-plugin-utils": "^7.18.6" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/plugin-transform-arrow-functions": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.25.9.tgz", - "integrity": "sha512-6jmooXYIwn9ca5/RylZADJ+EnSxVUS5sjeJ9UPk6RWRzXCmOJCy6dqItPJFpw2cuCangPK4OYr5uhGKcmrm5Qg==", - "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-async-generator-functions": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-generator-functions/-/plugin-transform-async-generator-functions-7.25.9.tgz", - "integrity": "sha512-RXV6QAzTBbhDMO9fWwOmwwTuYaiPbggWQ9INdZqAYeSHyG7FzQ+nOZaUUjNwKv9pV3aE4WFqFm1Hnbci5tBCAw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-remap-async-to-generator": "^7.25.9", - "@babel/traverse": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-async-to-generator": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.25.9.tgz", - "integrity": "sha512-NT7Ejn7Z/LjUH0Gv5KsBCxh7BH3fbLTV0ptHvpeMvrt3cPThHfJfst9Wrb7S8EvJ7vRTFI7z+VAvFVEQn/m5zQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-module-imports": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-remap-async-to-generator": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-block-scoped-functions": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.25.9.tgz", - "integrity": "sha512-toHc9fzab0ZfenFpsyYinOX0J/5dgJVA2fm64xPewu7CoYHWEivIWKxkK2rMi4r3yQqLnVmheMXRdG+k239CgA==", - "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-block-scoping": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.25.9.tgz", - "integrity": "sha512-1F05O7AYjymAtqbsFETboN1NvBdcnzMerO+zlMyJBEz6WkMdejvGWw9p05iTSjC85RLlBseHHQpYaM4gzJkBGg==", - "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-class-properties": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-class-properties/-/plugin-transform-class-properties-7.25.9.tgz", - "integrity": "sha512-bbMAII8GRSkcd0h0b4X+36GksxuheLFjP65ul9w6C3KgAamI3JqErNgSrosX6ZPj+Mpim5VvEbawXxJCyEUV3Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-class-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-class-static-block": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-class-static-block/-/plugin-transform-class-static-block-7.26.0.tgz", - "integrity": "sha512-6J2APTs7BDDm+UMqP1useWqhcRAXo0WIoVj26N7kPFB6S73Lgvyka4KTZYIxtgYXiN5HTyRObA72N2iu628iTQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-class-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.12.0" - } - }, - "node_modules/@babel/plugin-transform-classes": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.25.9.tgz", - "integrity": "sha512-mD8APIXmseE7oZvZgGABDyM34GUmK45Um2TXiBUt7PnuAxrgoSVf123qUzPxEr/+/BHrRn5NMZCdE2m/1F8DGg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-annotate-as-pure": "^7.25.9", - "@babel/helper-compilation-targets": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-replace-supers": "^7.25.9", - "@babel/traverse": "^7.25.9", - "globals": "^11.1.0" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-computed-properties": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.25.9.tgz", - "integrity": "sha512-HnBegGqXZR12xbcTHlJ9HGxw1OniltT26J5YpfruGqtUHlz/xKf/G2ak9e+t0rVqrjXa9WOhvYPz1ERfMj23AA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/template": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-destructuring": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.25.9.tgz", - "integrity": "sha512-WkCGb/3ZxXepmMiX101nnGiU+1CAdut8oHyEOHxkKuS1qKpU2SMXE2uSvfz8PBuLd49V6LEsbtyPhWC7fnkgvQ==", - "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-dotall-regex": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.25.9.tgz", - "integrity": "sha512-t7ZQ7g5trIgSRYhI9pIJtRl64KHotutUJsh4Eze5l7olJv+mRSg4/MmbZ0tv1eeqRbdvo/+trvJD/Oc5DmW2cA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-regexp-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-duplicate-keys": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.25.9.tgz", - "integrity": "sha512-LZxhJ6dvBb/f3x8xwWIuyiAHy56nrRG3PeYTpBkkzkYRRQ6tJLu68lEF5VIqMUZiAV7a8+Tb78nEoMCMcqjXBw==", - "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-duplicate-named-capturing-groups-regex": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-named-capturing-groups-regex/-/plugin-transform-duplicate-named-capturing-groups-regex-7.25.9.tgz", - "integrity": "sha512-0UfuJS0EsXbRvKnwcLjFtJy/Sxc5J5jhLHnFhy7u4zih97Hz6tJkLU+O+FMMrNZrosUPxDi6sYxJ/EA8jDiAog==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-regexp-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/plugin-transform-dynamic-import": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dynamic-import/-/plugin-transform-dynamic-import-7.25.9.tgz", - "integrity": "sha512-GCggjexbmSLaFhqsojeugBpeaRIgWNTcgKVq/0qIteFEqY2A+b9QidYadrWlnbWQUrW5fn+mCvf3tr7OeBFTyg==", - "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-exponentiation-operator": { - "version": "7.26.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.26.3.tgz", - "integrity": "sha512-7CAHcQ58z2chuXPWblnn1K6rLDnDWieghSOEmqQsrBenH0P9InCUtOJYD89pvngljmZlJcz3fcmgYsXFNGa1ZQ==", - "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-export-namespace-from": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-export-namespace-from/-/plugin-transform-export-namespace-from-7.25.9.tgz", - "integrity": "sha512-2NsEz+CxzJIVOPx2o9UsW1rXLqtChtLoVnwYHHiB04wS5sgn7mrV45fWMBX0Kk+ub9uXytVYfNP2HjbVbCB3Ww==", - "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-for-of": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.25.9.tgz", - "integrity": "sha512-LqHxduHoaGELJl2uhImHwRQudhCM50pT46rIBNvtT/Oql3nqiS3wOwP+5ten7NpYSXrrVLgtZU3DZmPtWZo16A==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-skip-transparent-expression-wrappers": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-function-name": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.25.9.tgz", - "integrity": "sha512-8lP+Yxjv14Vc5MuWBpJsoUCd3hD6V9DgBon2FVYL4jJgbnVQ9fTgYmonchzZJOVNgzEgbxp4OwAf6xz6M/14XA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-compilation-targets": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/traverse": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-json-strings": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-json-strings/-/plugin-transform-json-strings-7.25.9.tgz", - "integrity": "sha512-xoTMk0WXceiiIvsaquQQUaLLXSW1KJ159KP87VilruQm0LNNGxWzahxSS6T6i4Zg3ezp4vA4zuwiNUR53qmQAw==", - "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-literals": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.25.9.tgz", - "integrity": "sha512-9N7+2lFziW8W9pBl2TzaNht3+pgMIRP74zizeCSrtnSKVdUl8mAjjOP2OOVQAfZ881P2cNjDj1uAMEdeD50nuQ==", - "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-logical-assignment-operators": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-logical-assignment-operators/-/plugin-transform-logical-assignment-operators-7.25.9.tgz", - "integrity": "sha512-wI4wRAzGko551Y8eVf6iOY9EouIDTtPb0ByZx+ktDGHwv6bHFimrgJM/2T021txPZ2s4c7bqvHbd+vXG6K948Q==", - "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-member-expression-literals": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.25.9.tgz", - "integrity": "sha512-PYazBVfofCQkkMzh2P6IdIUaCEWni3iYEerAsRWuVd8+jlM1S9S9cz1dF9hIzyoZ8IA3+OwVYIp9v9e+GbgZhA==", - "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-modules-amd": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.25.9.tgz", - "integrity": "sha512-g5T11tnI36jVClQlMlt4qKDLlWnG5pP9CSM4GhdRciTNMRgkfpo5cR6b4rGIOYPgRRuFAvwjPQ/Yk+ql4dyhbw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-module-transforms": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, "node_modules/@babel/plugin-transform-modules-commonjs": { "version": "7.26.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.26.3.tgz", @@ -2495,414 +1198,16 @@ "@babel/core": "^7.0.0-0" } }, - "node_modules/@babel/plugin-transform-modules-systemjs": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.25.9.tgz", - "integrity": "sha512-hyss7iIlH/zLHaehT+xwiymtPOpsiwIIRlCAOwBB04ta5Tt+lNItADdlXw3jAWZ96VJ2jlhl/c+PNIQPKNfvcA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-module-transforms": "^7.25.9", - "@babel/helper-plugin-utils": "^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-0" - } - }, - "node_modules/@babel/plugin-transform-modules-umd": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.25.9.tgz", - "integrity": "sha512-bS9MVObUgE7ww36HEfwe6g9WakQ0KF07mQF74uuXdkoziUPfKyu/nIm663kz//e5O1nPInPFx36z7WJmJ4yNEw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-module-transforms": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-named-capturing-groups-regex": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.25.9.tgz", - "integrity": "sha512-oqB6WHdKTGl3q/ItQhpLSnWWOpjUJLsOCLVyeFgeTktkBSCiurvPOsyt93gibI9CmuKvTUEtWmG5VhZD+5T/KA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-regexp-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/plugin-transform-new-target": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.25.9.tgz", - "integrity": "sha512-U/3p8X1yCSoKyUj2eOBIx3FOn6pElFOKvAAGf8HTtItuPyB+ZeOqfn+mvTtg9ZlOAjsPdK3ayQEjqHjU/yLeVQ==", - "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-nullish-coalescing-operator": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-nullish-coalescing-operator/-/plugin-transform-nullish-coalescing-operator-7.25.9.tgz", - "integrity": "sha512-ENfftpLZw5EItALAD4WsY/KUWvhUlZndm5GC7G3evUsVeSJB6p0pBeLQUnRnBCBx7zV0RKQjR9kCuwrsIrjWog==", - "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-numeric-separator": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-numeric-separator/-/plugin-transform-numeric-separator-7.25.9.tgz", - "integrity": "sha512-TlprrJ1GBZ3r6s96Yq8gEQv82s8/5HnCVHtEJScUj90thHQbwe+E5MLhi2bbNHBEJuzrvltXSru+BUxHDoog7Q==", - "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-object-rest-spread": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-rest-spread/-/plugin-transform-object-rest-spread-7.25.9.tgz", - "integrity": "sha512-fSaXafEE9CVHPweLYw4J0emp1t8zYTXyzN3UuG+lylqkvYd7RMrsOQ8TYx5RF231be0vqtFC6jnx3UmpJmKBYg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-compilation-targets": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/plugin-transform-parameters": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-object-super": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.25.9.tgz", - "integrity": "sha512-Kj/Gh+Rw2RNLbCK1VAWj2U48yxxqL2x0k10nPtSdRa0O2xnHXalD0s+o1A6a0W43gJ00ANo38jxkQreckOzv5A==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-replace-supers": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-optional-catch-binding": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-optional-catch-binding/-/plugin-transform-optional-catch-binding-7.25.9.tgz", - "integrity": "sha512-qM/6m6hQZzDcZF3onzIhZeDHDO43bkNNlOX0i8n3lR6zLbu0GN2d8qfM/IERJZYauhAHSLHy39NF0Ctdvcid7g==", - "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-optional-chaining": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-optional-chaining/-/plugin-transform-optional-chaining-7.25.9.tgz", - "integrity": "sha512-6AvV0FsLULbpnXeBjrY4dmWF8F7gf8QnvTEoO/wX/5xm/xE1Xo8oPuD3MPS+KS9f9XBEAWN7X1aWr4z9HdOr7A==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-skip-transparent-expression-wrappers": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-parameters": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.25.9.tgz", - "integrity": "sha512-wzz6MKwpnshBAiRmn4jR8LYz/g8Ksg0o80XmwZDlordjwEk9SxBzTWC7F5ef1jhbrbOW2DJ5J6ayRukrJmnr0g==", - "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-private-methods": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-private-methods/-/plugin-transform-private-methods-7.25.9.tgz", - "integrity": "sha512-D/JUozNpQLAPUVusvqMxyvjzllRaF8/nSrP1s2YGQT/W4LHK4xxsMcHjhOGTS01mp9Hda8nswb+FblLdJornQw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-class-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-private-property-in-object": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-private-property-in-object/-/plugin-transform-private-property-in-object-7.25.9.tgz", - "integrity": "sha512-Evf3kcMqzXA3xfYJmZ9Pg1OvKdtqsDMSWBDzZOPLvHiTt36E75jLDQo5w1gtRU95Q4E5PDttrTf25Fw8d/uWLw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-annotate-as-pure": "^7.25.9", - "@babel/helper-create-class-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-property-literals": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.25.9.tgz", - "integrity": "sha512-IvIUeV5KrS/VPavfSM/Iu+RE6llrHrYIKY1yfCzyO/lMXHQ+p7uGhonmGVisv6tSBSVgWzMBohTcvkC9vQcQFA==", - "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-regenerator": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.25.9.tgz", - "integrity": "sha512-vwDcDNsgMPDGP0nMqzahDWE5/MLcX8sv96+wfX7as7LoF/kr97Bo/7fI00lXY4wUXYfVmwIIyG80fGZ1uvt2qg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "regenerator-transform": "^0.15.2" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-regexp-modifiers": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regexp-modifiers/-/plugin-transform-regexp-modifiers-7.26.0.tgz", - "integrity": "sha512-vN6saax7lrA2yA/Pak3sCxuD6F5InBjn9IcrIKQPjpsLvuHYLVroTxjdlVRHjjBWxKOqIwpTXDkOssYT4BFdRw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-regexp-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/plugin-transform-reserved-words": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.25.9.tgz", - "integrity": "sha512-7DL7DKYjn5Su++4RXu8puKZm2XBPHyjWLUidaPEkCUBbE7IPcsrkRHggAOOKydH1dASWdcUBxrkOGNxUv5P3Jg==", - "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-runtime": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.25.9.tgz", - "integrity": "sha512-nZp7GlEl+yULJrClz0SwHPqir3lc0zsPrDHQUcxGspSL7AKrexNSEfTbfqnDNJUO13bgKyfuOLMF8Xqtu8j3YQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-module-imports": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", - "babel-plugin-polyfill-corejs2": "^0.4.10", - "babel-plugin-polyfill-corejs3": "^0.10.6", - "babel-plugin-polyfill-regenerator": "^0.6.1", - "semver": "^6.3.1" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-runtime/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/@babel/plugin-transform-shorthand-properties": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.25.9.tgz", - "integrity": "sha512-MUv6t0FhO5qHnS/W8XCbHmiRWOphNufpE1IVxhK5kuN3Td9FT1x4rx4K42s3RYdMXCXpfWkGSbCSd0Z64xA7Ng==", - "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-spread": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.25.9.tgz", - "integrity": "sha512-oNknIB0TbURU5pqJFVbOOFspVlrpVwo2H1+HUIsVDvp5VauGGDP1ZEvO8Nn5xyMEs3dakajOxlmkNW7kNgSm6A==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-skip-transparent-expression-wrappers": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-sticky-regex": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.25.9.tgz", - "integrity": "sha512-WqBUSgeVwucYDP9U/xNRQam7xV8W5Zf+6Eo7T2SRVUFlhRiMNFdFz58u0KZmCVVqs2i7SHgpRnAhzRNmKfi2uA==", - "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-template-literals": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.25.9.tgz", - "integrity": "sha512-o97AE4syN71M/lxrCtQByzphAdlYluKPDBzDVzMmfCobUjjhAryZV0AIpRPrxN0eAkxXO6ZLEScmt+PNhj2OTw==", - "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-typeof-symbol": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.25.9.tgz", - "integrity": "sha512-v61XqUMiueJROUv66BVIOi0Fv/CUuZuZMl5NkRoCVxLAnMexZ0A3kMe7vvZ0nulxMuMp0Mk6S5hNh48yki08ZA==", - "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-typescript": { - "version": "7.26.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typescript/-/plugin-transform-typescript-7.26.3.tgz", - "integrity": "sha512-6+5hpdr6mETwSKjmJUdYw0EIkATiQhnELWlE3kJFBwSg/BGIVwVaVbX+gOXBCdc7Ln1RXZxyWGecIXhUfnl7oA==", + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typescript/-/plugin-transform-typescript-7.27.0.tgz", + "integrity": "sha512-fRGGjO2UEGPjvEcyAZXRXAS8AfdaQoq7HnxAbJoAoW10B9xOKesmmndJv+Sym2a+9FHWZ9KbyyLCe9s0Sn5jtg==", "dev": true, "license": "MIT", "dependencies": { "@babel/helper-annotate-as-pure": "^7.25.9", - "@babel/helper-create-class-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", + "@babel/helper-create-class-features-plugin": "^7.27.0", + "@babel/helper-plugin-utils": "^7.26.5", "@babel/helper-skip-transparent-expression-wrappers": "^7.25.9", "@babel/plugin-syntax-typescript": "^7.25.9" }, @@ -2913,194 +1218,18 @@ "@babel/core": "^7.0.0-0" } }, - "node_modules/@babel/plugin-transform-unicode-escapes": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.25.9.tgz", - "integrity": "sha512-s5EDrE6bW97LtxOcGj1Khcx5AaXwiMmi4toFWRDP9/y0Woo6pXC+iyPu/KuhKtfSrNFd7jJB+/fkOtZy6aIC6Q==", - "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-unicode-property-regex": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-property-regex/-/plugin-transform-unicode-property-regex-7.25.9.tgz", - "integrity": "sha512-Jt2d8Ga+QwRluxRQ307Vlxa6dMrYEMZCgGxoPR8V52rxPyldHu3hdlHspxaqYmE7oID5+kB+UKUB/eWS+DkkWg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-regexp-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-unicode-regex": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.25.9.tgz", - "integrity": "sha512-yoxstj7Rg9dlNn9UQxzk4fcNivwv4nUYz7fYXBaKxvw/lnmPuOm/ikoELygbYq68Bls3D/D+NBPHiLwZdZZ4HA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-regexp-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/plugin-transform-unicode-sets-regex": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-sets-regex/-/plugin-transform-unicode-sets-regex-7.25.9.tgz", - "integrity": "sha512-8BYqO3GeVNHtx69fdPshN3fnzUNLrWdHhk/icSwigksJGczKSizZ+Z6SBCxTs723Fr5VSNorTIK7a+R2tISvwQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-create-regexp-features-plugin": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0" - } - }, - "node_modules/@babel/preset-env": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.26.0.tgz", - "integrity": "sha512-H84Fxq0CQJNdPFT2DrfnylZ3cf5K43rGfWK4LJGPpjKHiZlk0/RzwEus3PDDZZg+/Er7lCA03MVacueUuXdzfw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/compat-data": "^7.26.0", - "@babel/helper-compilation-targets": "^7.25.9", - "@babel/helper-plugin-utils": "^7.25.9", - "@babel/helper-validator-option": "^7.25.9", - "@babel/plugin-bugfix-firefox-class-in-computed-class-key": "^7.25.9", - "@babel/plugin-bugfix-safari-class-field-initializer-scope": "^7.25.9", - "@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression": "^7.25.9", - "@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining": "^7.25.9", - "@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly": "^7.25.9", - "@babel/plugin-proposal-private-property-in-object": "7.21.0-placeholder-for-preset-env.2", - "@babel/plugin-syntax-import-assertions": "^7.26.0", - "@babel/plugin-syntax-import-attributes": "^7.26.0", - "@babel/plugin-syntax-unicode-sets-regex": "^7.18.6", - "@babel/plugin-transform-arrow-functions": "^7.25.9", - "@babel/plugin-transform-async-generator-functions": "^7.25.9", - "@babel/plugin-transform-async-to-generator": "^7.25.9", - "@babel/plugin-transform-block-scoped-functions": "^7.25.9", - "@babel/plugin-transform-block-scoping": "^7.25.9", - "@babel/plugin-transform-class-properties": "^7.25.9", - "@babel/plugin-transform-class-static-block": "^7.26.0", - "@babel/plugin-transform-classes": "^7.25.9", - "@babel/plugin-transform-computed-properties": "^7.25.9", - "@babel/plugin-transform-destructuring": "^7.25.9", - "@babel/plugin-transform-dotall-regex": "^7.25.9", - "@babel/plugin-transform-duplicate-keys": "^7.25.9", - "@babel/plugin-transform-duplicate-named-capturing-groups-regex": "^7.25.9", - "@babel/plugin-transform-dynamic-import": "^7.25.9", - "@babel/plugin-transform-exponentiation-operator": "^7.25.9", - "@babel/plugin-transform-export-namespace-from": "^7.25.9", - "@babel/plugin-transform-for-of": "^7.25.9", - "@babel/plugin-transform-function-name": "^7.25.9", - "@babel/plugin-transform-json-strings": "^7.25.9", - "@babel/plugin-transform-literals": "^7.25.9", - "@babel/plugin-transform-logical-assignment-operators": "^7.25.9", - "@babel/plugin-transform-member-expression-literals": "^7.25.9", - "@babel/plugin-transform-modules-amd": "^7.25.9", - "@babel/plugin-transform-modules-commonjs": "^7.25.9", - "@babel/plugin-transform-modules-systemjs": "^7.25.9", - "@babel/plugin-transform-modules-umd": "^7.25.9", - "@babel/plugin-transform-named-capturing-groups-regex": "^7.25.9", - "@babel/plugin-transform-new-target": "^7.25.9", - "@babel/plugin-transform-nullish-coalescing-operator": "^7.25.9", - "@babel/plugin-transform-numeric-separator": "^7.25.9", - "@babel/plugin-transform-object-rest-spread": "^7.25.9", - "@babel/plugin-transform-object-super": "^7.25.9", - "@babel/plugin-transform-optional-catch-binding": "^7.25.9", - "@babel/plugin-transform-optional-chaining": "^7.25.9", - "@babel/plugin-transform-parameters": "^7.25.9", - "@babel/plugin-transform-private-methods": "^7.25.9", - "@babel/plugin-transform-private-property-in-object": "^7.25.9", - "@babel/plugin-transform-property-literals": "^7.25.9", - "@babel/plugin-transform-regenerator": "^7.25.9", - "@babel/plugin-transform-regexp-modifiers": "^7.26.0", - "@babel/plugin-transform-reserved-words": "^7.25.9", - "@babel/plugin-transform-shorthand-properties": "^7.25.9", - "@babel/plugin-transform-spread": "^7.25.9", - "@babel/plugin-transform-sticky-regex": "^7.25.9", - "@babel/plugin-transform-template-literals": "^7.25.9", - "@babel/plugin-transform-typeof-symbol": "^7.25.9", - "@babel/plugin-transform-unicode-escapes": "^7.25.9", - "@babel/plugin-transform-unicode-property-regex": "^7.25.9", - "@babel/plugin-transform-unicode-regex": "^7.25.9", - "@babel/plugin-transform-unicode-sets-regex": "^7.25.9", - "@babel/preset-modules": "0.1.6-no-external-plugins", - "babel-plugin-polyfill-corejs2": "^0.4.10", - "babel-plugin-polyfill-corejs3": "^0.10.6", - "babel-plugin-polyfill-regenerator": "^0.6.1", - "core-js-compat": "^3.38.1", - "semver": "^6.3.1" - }, - "engines": { - "node": ">=6.9.0" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0" - } - }, - "node_modules/@babel/preset-env/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/@babel/preset-modules": { - "version": "0.1.6-no-external-plugins", - "resolved": "https://registry.npmjs.org/@babel/preset-modules/-/preset-modules-0.1.6-no-external-plugins.tgz", - "integrity": "sha512-HrcgcIESLm9aIR842yhJ5RWan/gebQUJ6E/E5+rf0y9o6oj7w0Br+sWuL6kEQ/o/AdfvR1Je9jG18/gnpwjEyA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-plugin-utils": "^7.0.0", - "@babel/types": "^7.4.4", - "esutils": "^2.0.2" - }, - "peerDependencies": { - "@babel/core": "^7.0.0-0 || ^8.0.0-0 <8.0.0" - } - }, "node_modules/@babel/preset-typescript": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/preset-typescript/-/preset-typescript-7.26.0.tgz", - "integrity": "sha512-NMk1IGZ5I/oHhoXEElcm+xUnL/szL6xflkFZmoEU9xj1qSJXpiS7rsspYo92B4DRCDvZn2erT5LdsCeXAKNCkg==", + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@babel/preset-typescript/-/preset-typescript-7.27.0.tgz", + "integrity": "sha512-vxaPFfJtHhgeOVXRKuHpHPAOgymmy8V8I65T1q53R7GCZlefKeCaTyDs3zOPHTTbmquvNlQYC5klEvWsBAtrBQ==", "dev": true, "license": "MIT", "dependencies": { - "@babel/helper-plugin-utils": "^7.25.9", + "@babel/helper-plugin-utils": "^7.26.5", "@babel/helper-validator-option": "^7.25.9", "@babel/plugin-syntax-jsx": "^7.25.9", - "@babel/plugin-transform-modules-commonjs": "^7.25.9", - "@babel/plugin-transform-typescript": "^7.25.9" + "@babel/plugin-transform-modules-commonjs": "^7.26.3", + "@babel/plugin-transform-typescript": "^7.27.0" }, "engines": { "node": ">=6.9.0" @@ -3109,46 +1238,33 @@ "@babel/core": "^7.0.0-0" } }, - "node_modules/@babel/runtime": { - "version": "7.26.0", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.26.0.tgz", - "integrity": "sha512-FDSOghenHTiToteC/QRlv2q3DhPZ/oOXTBoirfWNx1Cx3TMVcGWQtMMmQcSvb/JjpNeGzx8Pq/b4fKEJuWm1sw==", - "dev": true, - "license": "MIT", - "dependencies": { - "regenerator-runtime": "^0.14.0" - }, - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/@babel/template": { - "version": "7.25.9", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.25.9.tgz", - "integrity": "sha512-9DGttpmPvIxBb/2uwpVo3dqJ+O6RooAFOS+lB+xDqoE2PVCE8nfoHMdZLpfCQRLwvohzXISPZcgxt80xLfsuwg==", + "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.25.9", - "@babel/parser": "^7.25.9", - "@babel/types": "^7.25.9" + "@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.26.4", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.26.4.tgz", - "integrity": "sha512-fH+b7Y4p3yqvApJALCPJcwb0/XaOSgtK4pzV6WVjPR5GLFQBRI7pfoX2V2iM48NXvX07NUxxm1Vw98YjqTcU5w==", + "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.26.3", - "@babel/parser": "^7.26.3", - "@babel/template": "^7.25.9", - "@babel/types": "^7.26.3", + "@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" }, @@ -3157,9 +1273,9 @@ } }, "node_modules/@babel/types": { - "version": "7.26.3", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.26.3.tgz", - "integrity": "sha512-vN5p+1kl59GVKMvTHt55NzzmYVxprfJD+ql7U9NFIfKCBkYE55LYtS+WtPlaYOyzydrKI8Nezd+aZextrd+FMA==", + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.27.0.tgz", + "integrity": "sha512-H45s8fVLYjbhFH62dIJ3WtmJ6RSPt/3DRO0ZcT2SUiYiQyz3BLVb9ADEnLl91m74aQPS3AzzeajZHYOalWe3bg==", "license": "MIT", "dependencies": { "@babel/helper-string-parser": "^7.25.9", @@ -3203,17 +1319,6 @@ "@jridgewell/sourcemap-codec": "^1.4.10" } }, - "node_modules/@discoveryjs/json-ext": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz", - "integrity": "sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw==", - "dev": true, - "license": "MIT", - "peer": true, - "engines": { - "node": ">=10.0.0" - } - }, "node_modules/@eidellev/adonis-stardust": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/@eidellev/adonis-stardust/-/adonis-stardust-3.0.0.tgz", @@ -3227,9 +1332,9 @@ } }, "node_modules/@esbuild/aix-ppc64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", - "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.3.tgz", + "integrity": "sha512-W8bFfPA8DowP8l//sxjJLSLkD8iEjMc7cBVyP+u4cEv9sM7mdUCkgsj+t0n/BWPFtv7WWCN5Yzj0N6FJNUUqBQ==", "cpu": [ "ppc64" ], @@ -3238,15 +1343,14 @@ "os": [ "aix" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/android-arm": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz", - "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.3.tgz", + "integrity": "sha512-PuwVXbnP87Tcff5I9ngV0lmiSu40xw1At6i3GsU77U7cjDDB4s0X2cyFuBiDa1SBk9DnvWwnGvVaGBqoFWPb7A==", "cpu": [ "arm" ], @@ -3255,15 +1359,14 @@ "os": [ "android" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/android-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz", - "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.3.tgz", + "integrity": "sha512-XelR6MzjlZuBM4f5z2IQHK6LkK34Cvv6Rj2EntER3lwCBFdg6h2lKbtRjpTTsdEjD/WSe1q8UyPBXP1x3i/wYQ==", "cpu": [ "arm64" ], @@ -3272,15 +1375,14 @@ "os": [ "android" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/android-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz", - "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.3.tgz", + "integrity": "sha512-ogtTpYHT/g1GWS/zKM0cc/tIebFjm1F9Aw1boQ2Y0eUQ+J89d0jFY//s9ei9jVIlkYi8AfOjiixcLJSGNSOAdQ==", "cpu": [ "x64" ], @@ -3289,15 +1391,14 @@ "os": [ "android" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/darwin-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz", - "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.3.tgz", + "integrity": "sha512-eESK5yfPNTqpAmDfFWNsOhmIOaQA59tAcF/EfYvo5/QWQCzXn5iUSOnqt3ra3UdzBv073ykTtmeLJZGt3HhA+w==", "cpu": [ "arm64" ], @@ -3306,15 +1407,14 @@ "os": [ "darwin" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/darwin-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz", - "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.3.tgz", + "integrity": "sha512-Kd8glo7sIZtwOLcPbW0yLpKmBNWMANZhrC1r6K++uDR2zyzb6AeOYtI6udbtabmQpFaxJ8uduXMAo1gs5ozz8A==", "cpu": [ "x64" ], @@ -3323,15 +1423,14 @@ "os": [ "darwin" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/freebsd-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz", - "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.3.tgz", + "integrity": "sha512-EJiyS70BYybOBpJth3M0KLOus0n+RRMKTYzhYhFeMwp7e/RaajXvP+BWlmEXNk6uk+KAu46j/kaQzr6au+JcIw==", "cpu": [ "arm64" ], @@ -3340,15 +1439,14 @@ "os": [ "freebsd" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/freebsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz", - "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.3.tgz", + "integrity": "sha512-Q+wSjaLpGxYf7zC0kL0nDlhsfuFkoN+EXrx2KSB33RhinWzejOd6AvgmP5JbkgXKmjhmpfgKZq24pneodYqE8Q==", "cpu": [ "x64" ], @@ -3357,15 +1455,14 @@ "os": [ "freebsd" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-arm": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz", - "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.3.tgz", + "integrity": "sha512-dUOVmAUzuHy2ZOKIHIKHCm58HKzFqd+puLaS424h6I85GlSDRZIA5ycBixb3mFgM0Jdh+ZOSB6KptX30DD8YOQ==", "cpu": [ "arm" ], @@ -3374,15 +1471,14 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz", - "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.3.tgz", + "integrity": "sha512-xCUgnNYhRD5bb1C1nqrDV1PfkwgbswTTBRbAd8aH5PhYzikdf/ddtsYyMXFfGSsb/6t6QaPSzxtbfAZr9uox4A==", "cpu": [ "arm64" ], @@ -3391,15 +1487,14 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-ia32": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz", - "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.3.tgz", + "integrity": "sha512-yplPOpczHOO4jTYKmuYuANI3WhvIPSVANGcNUeMlxH4twz/TeXuzEP41tGKNGWJjuMhotpGabeFYGAOU2ummBw==", "cpu": [ "ia32" ], @@ -3408,15 +1503,14 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-loong64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz", - "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.3.tgz", + "integrity": "sha512-P4BLP5/fjyihmXCELRGrLd793q/lBtKMQl8ARGpDxgzgIKJDRJ/u4r1A/HgpBpKpKZelGct2PGI4T+axcedf6g==", "cpu": [ "loong64" ], @@ -3425,15 +1519,14 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-mips64el": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz", - "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.3.tgz", + "integrity": "sha512-eRAOV2ODpu6P5divMEMa26RRqb2yUoYsuQQOuFUexUoQndm4MdpXXDBbUoKIc0iPa4aCO7gIhtnYomkn2x+bag==", "cpu": [ "mips64el" ], @@ -3442,15 +1535,14 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-ppc64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz", - "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.3.tgz", + "integrity": "sha512-ZC4jV2p7VbzTlnl8nZKLcBkfzIf4Yad1SJM4ZMKYnJqZFD4rTI+pBG65u8ev4jk3/MPwY9DvGn50wi3uhdaghg==", "cpu": [ "ppc64" ], @@ -3459,15 +1551,14 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-riscv64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz", - "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.3.tgz", + "integrity": "sha512-LDDODcFzNtECTrUUbVCs6j9/bDVqy7DDRsuIXJg6so+mFksgwG7ZVnTruYi5V+z3eE5y+BJZw7VvUadkbfg7QA==", "cpu": [ "riscv64" ], @@ -3476,15 +1567,14 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-s390x": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz", - "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.3.tgz", + "integrity": "sha512-s+w/NOY2k0yC2p9SLen+ymflgcpRkvwwa02fqmAwhBRI3SC12uiS10edHHXlVWwfAagYSY5UpmT/zISXPMW3tQ==", "cpu": [ "s390x" ], @@ -3493,15 +1583,14 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/linux-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz", - "integrity": "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.3.tgz", + "integrity": "sha512-nQHDz4pXjSDC6UfOE1Fw9Q8d6GCAd9KdvMZpfVGWSJztYCarRgSDfOVBY5xwhQXseiyxapkiSJi/5/ja8mRFFA==", "cpu": [ "x64" ], @@ -3510,15 +1599,30 @@ "os": [ "linux" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.3.tgz", + "integrity": "sha512-1QaLtOWq0mzK6tzzp0jRN3eccmN3hezey7mhLnzC6oNlJoUJz4nym5ZD7mDnS/LZQgkrhEbEiTn515lPeLpgWA==", + "cpu": [ + "arm64" + ], + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" } }, "node_modules/@esbuild/netbsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz", - "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.3.tgz", + "integrity": "sha512-i5Hm68HXHdgv8wkrt+10Bc50zM0/eonPb/a/OFVfB6Qvpiirco5gBA5bz7S2SHuU+Y4LWn/zehzNX14Sp4r27g==", "cpu": [ "x64" ], @@ -3527,15 +1631,30 @@ "os": [ "netbsd" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.3.tgz", + "integrity": "sha512-zGAVApJEYTbOC6H/3QBr2mq3upG/LBEXr85/pTtKiv2IXcgKV0RT0QA/hSXZqSvLEpXeIxah7LczB4lkiYhTAQ==", + "cpu": [ + "arm64" + ], + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" } }, "node_modules/@esbuild/openbsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz", - "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.3.tgz", + "integrity": "sha512-fpqctI45NnCIDKBH5AXQBsD0NDPbEFczK98hk/aa6HJxbl+UtLkJV2+Bvy5hLSLk3LHmqt0NTkKNso1A9y1a4w==", "cpu": [ "x64" ], @@ -3544,15 +1663,14 @@ "os": [ "openbsd" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/sunos-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz", - "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.3.tgz", + "integrity": "sha512-ROJhm7d8bk9dMCUZjkS8fgzsPAZEjtRJqCAmVgB0gMrvG7hfmPmz9k1rwO4jSiblFjYmNvbECL9uhaPzONMfgA==", "cpu": [ "x64" ], @@ -3561,15 +1679,14 @@ "os": [ "sunos" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/win32-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz", - "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.3.tgz", + "integrity": "sha512-YWcow8peiHpNBiIXHwaswPnAXLsLVygFwCB3A7Bh5jRkIBFWHGmNQ48AlX4xDvQNoMZlPYzjVOQDYEzWCqufMQ==", "cpu": [ "arm64" ], @@ -3578,15 +1695,14 @@ "os": [ "win32" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/win32-ia32": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz", - "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.3.tgz", + "integrity": "sha512-qspTZOIGoXVS4DpNqUYUs9UxVb04khS1Degaw/MnfMe7goQ3lTfQ13Vw4qY/Nj0979BGvMRpAYbs/BAxEvU8ew==", "cpu": [ "ia32" ], @@ -3595,15 +1711,14 @@ "os": [ "win32" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@esbuild/win32-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz", - "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.3.tgz", + "integrity": "sha512-ICgUR+kPimx0vvRzf+N/7L7tVSQeE3BYY+NhHRHXS1kBuPO7z2+7ea2HbhDyZdTephgvNvKrlDDKUexuCVBVvg==", "cpu": [ "x64" ], @@ -3612,15 +1727,14 @@ "os": [ "win32" ], - "peer": true, "engines": { - "node": ">=12" + "node": ">=18" } }, "node_modules/@eslint-community/eslint-utils": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.1.tgz", - "integrity": "sha512-s3O3waFUrMV8P/XaF/+ZTp1X9XBZW1a4B97ZnjQF2KYWaFD2A8KyFBsrsfSjEmjn3RGWAIuvlneuZm3CUK3jbA==", + "version": "4.6.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.6.1.tgz", + "integrity": "sha512-KTsJMmobmbrFLe3LDh0PC2FXpcSYJt/MLjlkh/9LEnmKYLSYmT/0EW9JWANjeoemiuZrmogti0tW5Ch+qNUYDw==", "dev": true, "license": "MIT", "dependencies": { @@ -3670,13 +1784,6 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/@eslint/eslintrc/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/@eslint/eslintrc/node_modules/globals": { "version": "13.24.0", "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", @@ -3693,19 +1800,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/@eslint/eslintrc/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/@eslint/eslintrc/node_modules/type-fest": { "version": "0.20.2", "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", @@ -3730,9 +1824,9 @@ } }, "node_modules/@faker-js/faker": { - "version": "9.3.0", - "resolved": "https://registry.npmjs.org/@faker-js/faker/-/faker-9.3.0.tgz", - "integrity": "sha512-r0tJ3ZOkMd9xsu3VRfqlFR6cz0V/jFYRswAIpC+m/DIfAUXq7g8N7wTAlhSANySXYGKzGryfDXwtwsY8TxEIDw==", + "version": "9.7.0", + "resolved": "https://registry.npmjs.org/@faker-js/faker/-/faker-9.7.0.tgz", + "integrity": "sha512-aozo5vqjCmDoXLNUJarFZx2IN/GgGaogY4TMJ6so/WLZOWpSV7fvj2dmrV6sEAnUm1O7aCrhTibjpzeDFgNqbg==", "funding": [ { "type": "opencollective", @@ -3746,16 +1840,38 @@ } }, "node_modules/@fontsource/archivo-black": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/@fontsource/archivo-black/-/archivo-black-5.1.1.tgz", - "integrity": "sha512-3hmXvgYkXDsQ7msb+YrwU+6B5j4q7LBfVev1G4T8I5yRcRTo6H8OEA8tf0vULvcAGfOOhl38aRK/2I2+RLE/rQ==", - "license": "OFL-1.1" + "version": "5.2.5", + "resolved": "https://registry.npmjs.org/@fontsource/archivo-black/-/archivo-black-5.2.5.tgz", + "integrity": "sha512-tdBRFgA0CgxVqj3mBM96aiXRBoOp51X3IW2e8/t59AVr0NwiBcB+c3C+p5dd7Np/UT/vqdmjb/gK/HaFpulhIA==", + "license": "OFL-1.1", + "funding": { + "url": "https://github.com/sponsors/ayuhito" + } }, "node_modules/@fontsource/inter": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/@fontsource/inter/-/inter-5.1.1.tgz", - "integrity": "sha512-weN3E+rq0Xb3Z93VHJ+Rc7WOQX9ETJPTAJ+gDcaMHtjft67L58sfS65rAjC5tZUXQ2FdZ/V1/sSzCwZ6v05kJw==", - "license": "OFL-1.1" + "version": "5.2.5", + "resolved": "https://registry.npmjs.org/@fontsource/inter/-/inter-5.2.5.tgz", + "integrity": "sha512-kbsPKj0S4p44JdYRFiW78Td8Ge2sBVxi/PIBwmih+RpSXUdvS9nbs1HIiuUSPtRMi14CqLEZ/fbk7dj7vni1Sg==", + "license": "OFL-1.1", + "funding": { + "url": "https://github.com/sponsors/ayuhito" + } + }, + "node_modules/@headlessui/vue": { + "version": "1.7.23", + "resolved": "https://registry.npmjs.org/@headlessui/vue/-/vue-1.7.23.tgz", + "integrity": "sha512-JzdCNqurrtuu0YW6QaDtR2PIYCKPUWq28csDyMvN4zmGccmE7lz40Is6hc3LA4HFeCI7sekZ/PQMTNmn9I/4Wg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@tanstack/vue-virtual": "^3.0.0-beta.60" + }, + "engines": { + "node": ">=10" + }, + "peerDependencies": { + "vue": "^3.2.0" + } }, "node_modules/@humanwhocodes/config-array": { "version": "0.13.0", @@ -3795,27 +1911,28 @@ "dev": true, "license": "BSD-3-Clause" }, - "node_modules/@inertiajs/core": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/@inertiajs/core/-/core-1.3.0.tgz", - "integrity": "sha512-TJ8R1eUYY473m9DaKlCPRdHTdznFWTDuy5VvEzXg3t/hohbDQedLj46yn/uAqziJPEUZJrSftZzPI2NMzL9tQA==", - "license": "MIT", - "dependencies": { - "axios": "^1.6.0", - "deepmerge": "^4.0.0", - "nprogress": "^0.2.0", - "qs": "^6.9.0" + "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==", + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" } }, - "node_modules/@inertiajs/core/node_modules/axios": { - "version": "1.7.9", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.9.tgz", - "integrity": "sha512-LhLcE7Hbiryz8oMDdDptSrWowmB4Bl6RCt6sIJKpRB4XtVf0iEgewX3au/pJqm+Py1kCASkb/FFKjxQaLtxJvw==", + "node_modules/@inertiajs/core": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/@inertiajs/core/-/core-2.0.8.tgz", + "integrity": "sha512-YE+b5FktbSSaWJt4CjCHy7z3t+IV97G/8kD33mkj2Fqqf+Jfsypd/jsOuxrQGSMDpIyAGR6EDoaiuss6+JuIPA==", "license": "MIT", "dependencies": { - "follow-redirects": "^1.15.6", - "form-data": "^4.0.0", - "proxy-from-env": "^1.1.0" + "axios": "^1.8.2", + "es-toolkit": "^1.34.1", + "qs": "^6.9.0" } }, "node_modules/@inertiajs/inertia": { @@ -3829,15 +1946,23 @@ "qs": "^6.9.0" } }, - "node_modules/@inertiajs/vue3": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/@inertiajs/vue3/-/vue3-1.3.0.tgz", - "integrity": "sha512-GizqdCM3u4JWunit3uUbW4fEmTLKQTi1W7VvPRdrNy8XDt4Qy2cCmfFjq+aH5tHBSS3fI/ngYuhN7XvwqNaKvw==", + "node_modules/@inertiajs/inertia/node_modules/axios": { + "version": "0.21.4", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.4.tgz", + "integrity": "sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==", "license": "MIT", "dependencies": { - "@inertiajs/core": "1.3.0", - "lodash.clonedeep": "^4.5.0", - "lodash.isequal": "^4.5.0" + "follow-redirects": "^1.14.0" + } + }, + "node_modules/@inertiajs/vue3": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/@inertiajs/vue3/-/vue3-2.0.8.tgz", + "integrity": "sha512-XzerZJxxiTE40U6X9MggjQivUpHjHNaDnrts8TVnYIO6iRMSzrgqVMW/9DXIZGAuwE1z832Kj58/sWAefx+/BQ==", + "license": "MIT", + "dependencies": { + "@inertiajs/core": "2.0.8", + "es-toolkit": "^1.33.0" }, "peerDependencies": { "vue": "^3.0.0" @@ -3952,75 +2077,38 @@ "url": "https://github.com/chalk/wrap-ansi?sponsor=1" } }, - "node_modules/@japa/api-client": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/@japa/api-client/-/api-client-2.0.4.tgz", - "integrity": "sha512-hD0UbbyjrDG+hyzI1HXVvqYdwAxltX/lK7znVon5el1hu6FpYSbvboVwxRjW40ttRsy0l45EQDIlFBA+lW352A==", - "devOptional": true, - "license": "MIT", - "dependencies": { - "@poppinss/hooks": "^7.2.4", - "@poppinss/macroable": "^1.0.3", - "@types/superagent": "^8.1.9", - "cookie": "^1.0.1", - "set-cookie-parser": "^2.7.1", - "superagent": "^8.1.2" - }, - "engines": { - "node": ">=18.16.0" - }, - "peerDependencies": { - "@japa/assert": "^2.0.0 || ^3.0.0", - "@japa/runner": "^3.1.2" - }, - "peerDependenciesMeta": { - "@japa/assert": { - "optional": true - } - } - }, - "node_modules/@japa/api-client/node_modules/cookie": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-1.0.2.tgz", - "integrity": "sha512-9Kr/j4O16ISv8zBBhJoi4bXOYNTkFLOqSL3UDB0njXxCXNezjeyVrJyGOWtgfs/q2km1gwBcfH8q1yEGoMYunA==", - "devOptional": true, - "license": "MIT", - "engines": { - "node": ">=18" - } - }, "node_modules/@japa/assert": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@japa/assert/-/assert-3.0.0.tgz", - "integrity": "sha512-4Uvixj78PBpRGeNTqO1GN/qYyl4EeWmIwt/cKiQSLLsoZQpQfe8tvF4PO2Z+zteUi3Zv7WR6pluKYbLQrn3vjg==", + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/@japa/assert/-/assert-4.0.1.tgz", + "integrity": "sha512-n/dA9DVLNvM/Bw8DtN8kBdPjYsSHe3XTRjF5+U8vlzDavpW9skUANl2CHR1K/TBWZxwMfGi15SJIjo6UCs09AA==", "dev": true, "license": "MIT", "dependencies": { - "@poppinss/macroable": "^1.0.2", - "@types/chai": "^4.3.14", - "api-contract-validator": "^2.2.8", - "chai": "^5.1.0" + "@poppinss/macroable": "^1.0.4", + "@types/chai": "^5.0.1", + "assertion-error": "^2.0.1", + "chai": "^5.1.2" }, "engines": { "node": ">=18.16.0" }, "peerDependencies": { - "@japa/runner": "^3.1.2" + "@japa/runner": "^3.1.2 || ^4.0.0" } }, "node_modules/@japa/core": { - "version": "9.0.1", - "resolved": "https://registry.npmjs.org/@japa/core/-/core-9.0.1.tgz", - "integrity": "sha512-snngJNbvYC92nn+dB69DT2iyosWZLXPRnOp8NJnVEeotkkKAWSmcDqBKw9qq2+MVdshwClvKFVXTxko4MtmlEQ==", + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/@japa/core/-/core-10.3.0.tgz", + "integrity": "sha512-+vaqMiPnVaxlKH1sAwRQ80AwzlPysPKivhB8q1I2+BGe35lNrfiHKGMC52fuGAZBNuH5W2nInSCxr4cN/BTEIQ==", "devOptional": true, "license": "MIT", "dependencies": { - "@poppinss/cliui": "^6.4.1", - "@poppinss/hooks": "^7.2.3", - "@poppinss/macroable": "^1.0.2", + "@poppinss/hooks": "^7.2.5", + "@poppinss/macroable": "^1.0.4", + "@poppinss/string": "^1.2.0", "async-retry": "^1.3.3", "emittery": "^1.0.3", - "string-width": "^7.1.0", + "string-width": "^7.2.0", "time-span": "^5.1.0" }, "engines": { @@ -4028,36 +2116,51 @@ } }, "node_modules/@japa/errors-printer": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/@japa/errors-printer/-/errors-printer-3.0.4.tgz", - "integrity": "sha512-gqBWkc8X6n5y91HH7H8fXyfe3rKV1+YeMNgE/+CY6hXf0/BS7J55s/QldosKEV2ZiWj/WmE6UPZiFH8W873fGw==", + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/@japa/errors-printer/-/errors-printer-4.1.2.tgz", + "integrity": "sha512-exl/r07ssJhEEsMdFT2sXgP1sV7Tp3mZvYUEDMXZ8YjWZPHTFLLcA7o9q9FJSSB1ITrEIbx2SWTB+2fFUaZ3NA==", "devOptional": true, "license": "MIT", "dependencies": { - "@poppinss/colors": "^4.1.3", + "@poppinss/colors": "^4.1.4", "jest-diff": "^29.7.0", - "supports-color": "^9.4.0", - "youch": "^3.3.3", - "youch-terminal": "^2.2.3" + "supports-color": "^10.0.0", + "youch": "^4.1.0-beta.5" }, "engines": { "node": ">=18.16.0" } }, + "node_modules/@japa/errors-printer/node_modules/youch": { + "version": "4.1.0-beta.7", + "resolved": "https://registry.npmjs.org/youch/-/youch-4.1.0-beta.7.tgz", + "integrity": "sha512-HUn0M24AUTMvjdkoMtH8fJz2FEd+k1xvtR9EoTrDUoVUi6o7xl5X+pST/vjk4T3GEQo2mJ9FlAvhWBm8dIdD4g==", + "devOptional": true, + "license": "MIT", + "dependencies": { + "@poppinss/dumper": "^0.6.3", + "@speed-highlight/core": "^1.2.7", + "cookie": "^1.0.2", + "youch-core": "^0.3.1" + }, + "engines": { + "node": ">=18" + } + }, "node_modules/@japa/plugin-adonisjs": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/@japa/plugin-adonisjs/-/plugin-adonisjs-3.0.1.tgz", - "integrity": "sha512-xUZOzfBXSz2sWRoQT+qs+6LZBtWWE+cCBZ3j9ckz6+nPw3VI0nV6yLaX+oud3AY8Zb+BH+pErABBhaovZYv9dA==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@japa/plugin-adonisjs/-/plugin-adonisjs-4.0.0.tgz", + "integrity": "sha512-M2LUtHhKr4KgBfX73tDHNCD1IOmcXp9dvC+AinmRxsggIFnarsClcfjT/sXc3uNzjZW7Lk31LvcH76AxJHBmJQ==", "devOptional": true, "license": "MIT", "engines": { "node": ">=18.16.0" }, "peerDependencies": { - "@adonisjs/core": "^6.5.0", - "@japa/api-client": "^2.0.3", + "@adonisjs/core": "^6.17.0", + "@japa/api-client": "^2.0.3 || ^3.0.0", "@japa/browser-client": "^2.0.3", - "@japa/runner": "^3.1.2", + "@japa/runner": "^3.1.2 || ^4.0.0", "playwright": "^1.42.1" }, "peerDependenciesMeta": { @@ -4073,23 +2176,23 @@ } }, "node_modules/@japa/runner": { - "version": "3.1.4", - "resolved": "https://registry.npmjs.org/@japa/runner/-/runner-3.1.4.tgz", - "integrity": "sha512-ShaVZLdYq3GbFwyNiqQMCfdEoNq9vgYC0P6Z9gflqPcSUfOmN5jeJTLrLpChCBM5Sx9kYuAm5Bh6cqv1ZrArkQ==", + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@japa/runner/-/runner-4.2.0.tgz", + "integrity": "sha512-e3BFn1rca/OTiagilkmRTrLVhl00iC/LrY5j4Ns/VZDONYHs9BKAbHaImxjD1zoHMEhwQEF+ce7fgMO/BK+lfg==", "devOptional": true, "license": "MIT", "dependencies": { - "@japa/core": "^9.0.1", - "@japa/errors-printer": "^3.0.4", - "@poppinss/colors": "^4.1.3", - "@poppinss/hooks": "^7.2.3", - "fast-glob": "^3.3.2", + "@japa/core": "^10.3.0", + "@japa/errors-printer": "^4.1.2", + "@poppinss/colors": "^4.1.4", + "@poppinss/hooks": "^7.2.5", + "fast-glob": "^3.3.3", "find-cache-dir": "^5.0.0", "getopts": "^2.3.0", "ms": "^2.1.3", - "serialize-error": "^11.0.3", + "serialize-error": "^12.0.0", "slash": "^5.1.0", - "supports-color": "^9.4.0" + "supports-color": "^10.0.0" }, "engines": { "node": ">=18.16.0" @@ -4108,85 +2211,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/@jest/types": { - "version": "25.5.0", - "resolved": "https://registry.npmjs.org/@jest/types/-/types-25.5.0.tgz", - "integrity": "sha512-OXD0RgQ86Tu3MazKo8bnrkDRaDXXMGUqd+kTtLtK1Zb7CRzQcaSRPPPV37SvYTdevXEBVxe0HXylEjs8ibkmCw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/istanbul-lib-coverage": "^2.0.0", - "@types/istanbul-reports": "^1.1.1", - "@types/yargs": "^15.0.0", - "chalk": "^3.0.0" - }, - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/@jest/types/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/@jest/types/node_modules/chalk": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", - "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/@jest/types/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/@jest/types/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/@jest/types/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/@jridgewell/gen-mapping": { "version": "0.3.8", "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.8.tgz", @@ -4228,6 +2252,7 @@ "integrity": "sha512-1ZJTZebgqllO79ue2bm3rIGud/bOe0pP5BjSRCRxxYkEZS8STV7zN84UBbiYu7jy+eCKSnVIUgoWWE/tt+shMQ==", "devOptional": true, "license": "MIT", + "peer": true, "dependencies": { "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.25" @@ -4250,13 +2275,6 @@ "@jridgewell/sourcemap-codec": "^1.4.14" } }, - "node_modules/@jsdevtools/ono": { - "version": "7.1.3", - "resolved": "https://registry.npmjs.org/@jsdevtools/ono/-/ono-7.1.3.tgz", - "integrity": "sha512-4JQNk+3mVzK3xh2rqd6RB4J46qUR19azEHBneZyTZM+c456qOrbbM/5xcR8huNCCcbVt7+UmizG6GuUvPvKUYg==", - "dev": true, - "license": "MIT" - }, "node_modules/@jsonjoy.com/base64": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/@jsonjoy.com/base64/-/base64-1.1.2.tgz", @@ -4275,9 +2293,9 @@ } }, "node_modules/@jsonjoy.com/json-pack": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/@jsonjoy.com/json-pack/-/json-pack-1.1.1.tgz", - "integrity": "sha512-osjeBqMJ2lb/j/M8NCPjs1ylqWIcTRTycIhVB5pt6LgzgeRSb0YRZ7j9RfA8wIUrsr/medIuhVyonXRZWLyfdw==", + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@jsonjoy.com/json-pack/-/json-pack-1.2.0.tgz", + "integrity": "sha512-io1zEbbYcElht3tdlqEOFxZ0dMTYrHz9iMf0gqn1pPjZFTCgM5R4R5IMA20Chb2UPYYsxjzs8CgZ7Nb5n2K2rA==", "dev": true, "license": "Apache-2.0", "dependencies": { @@ -4357,6 +2375,18 @@ "node-pre-gyp": "bin/node-pre-gyp" } }, + "node_modules/@mapbox/node-pre-gyp/node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/@mdi/js": { "version": "7.4.47", "resolved": "https://registry.npmjs.org/@mdi/js/-/js-7.4.47.tgz", @@ -4365,9 +2395,9 @@ "license": "Apache-2.0" }, "node_modules/@noble/hashes": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.7.0.tgz", - "integrity": "sha512-HXydb0DgzTpDPwbVeDGCG1gIu7X6+AuU6Zl6av/E/KG8LMsvPntvq+w17CHRpKBmN6Ybdrt1eP3k4cj8DJa78w==", + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.8.0.tgz", + "integrity": "sha512-jCs9ldd7NwzpgXDIf6P3+NrHh9/sD6CQdxHyjQI+h/6rDNo88ypBxxz45UDuZHz9r3tNz7N/VInSVoVdtXEI4A==", "license": "MIT", "engines": { "node": "^14.21.3 || >=16" @@ -4414,58 +2444,6 @@ "node": ">= 8" } }, - "node_modules/@nuxt/friendly-errors-webpack-plugin": { - "version": "2.6.0", - "resolved": "https://registry.npmjs.org/@nuxt/friendly-errors-webpack-plugin/-/friendly-errors-webpack-plugin-2.6.0.tgz", - "integrity": "sha512-3IZj6MXbzlvUxDncAxgBMLQwGPY/JlNhy2i+AGyOHCAReR5HcBxYjVRBvyaKM9R3s5k4OODYKeHAbrToZH/47w==", - "dev": true, - "license": "MIT", - "dependencies": { - "chalk": "^2.4.2", - "consola": "^3.2.3", - "error-stack-parser": "^2.1.4", - "string-width": "^4.2.3" - }, - "engines": { - "node": ">=14.18.0", - "npm": ">=5.0.0" - }, - "peerDependencies": { - "webpack": "^2.0.0 || ^3.0.0 || ^4.0.0 || ^5.0.0" - } - }, - "node_modules/@nuxt/friendly-errors-webpack-plugin/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/@nuxt/friendly-errors-webpack-plugin/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/@nuxt/friendly-errors-webpack-plugin/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/@oozcitak/dom": { "version": "1.15.10", "resolved": "https://registry.npmjs.org/@oozcitak/dom/-/dom-1.15.10.tgz", @@ -4515,9 +2493,9 @@ } }, "node_modules/@opensearch-project/opensearch": { - "version": "2.13.0", - "resolved": "https://registry.npmjs.org/@opensearch-project/opensearch/-/opensearch-2.13.0.tgz", - "integrity": "sha512-Bu3jJ7pKzumbMMeefu7/npAWAvFu5W9SlbBow1ulhluqUpqc7QoXe0KidDrMy7Dy3BQrkI6llR3cWL4lQTZOFw==", + "version": "3.5.1", + "resolved": "https://registry.npmjs.org/@opensearch-project/opensearch/-/opensearch-3.5.1.tgz", + "integrity": "sha512-6bf+HcuERzAtHZxrm6phjref54ABse39BpkDie/YO3AUFMCBrb3SK5okKSdT5n3+nDRuEEQLhQCl0RQV3s1qpA==", "license": "Apache-2.0", "dependencies": { "aws4": "^1.11.0", @@ -4528,7 +2506,7 @@ "secure-json-parse": "^2.4.0" }, "engines": { - "node": ">=10", + "node": ">=14", "yarn": "^1.22.10" } }, @@ -4562,28 +2540,28 @@ } }, "node_modules/@pkgr/core": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.1.1.tgz", - "integrity": "sha512-cq8o4cWH0ibXh9VGi5P20Tu9XF/0fFXl9EUinr9QfTM7a7p0oTA4iJRCQWppXR1Pg8dSM0UCItCkPwsk9qWWYA==", + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.2.4.tgz", + "integrity": "sha512-ROFF39F6ZrnzSUEmQQZUar0Jt4xVoP9WnDRdWwF4NNcXs3xBTLgBUDoOwW141y1jP+S8nahIbdxbFC7IShw9Iw==", "dev": true, "license": "MIT", "engines": { "node": "^12.20.0 || ^14.18.0 || >=16.0.0" }, "funding": { - "url": "https://opencollective.com/unts" + "url": "https://opencollective.com/pkgr" } }, "node_modules/@poppinss/chokidar-ts": { - "version": "4.1.5", - "resolved": "https://registry.npmjs.org/@poppinss/chokidar-ts/-/chokidar-ts-4.1.5.tgz", - "integrity": "sha512-V8QtYZZMTbpv9aMX/agZSssIVfig7HK2s9grUqs6x221PEB/YirYtasj6g0Jx1o+yg7D38Y9kKPmL7d9MgeXBw==", + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@poppinss/chokidar-ts/-/chokidar-ts-4.1.6.tgz", + "integrity": "sha512-zGH6G+IcqrL2rgbbQtj349YAkaihb6dFwWS4QVgfcANBPjRWUvlWtlYhHMrglmglXfNjKc3K3bPP2HjxD/BQ4Q==", "devOptional": true, "license": "MIT", "dependencies": { "chokidar": "^4.0.3", - "emittery": "^1.0.3", - "memoize": "^10.0.0", + "emittery": "^1.1.0", + "memoize": "^10.1.0", "picomatch": "^4.0.2", "slash": "^5.1.0" }, @@ -4595,9 +2573,9 @@ } }, "node_modules/@poppinss/cliui": { - "version": "6.4.2", - "resolved": "https://registry.npmjs.org/@poppinss/cliui/-/cliui-6.4.2.tgz", - "integrity": "sha512-+zx32scWjFUReNAzi75/QBwTiQrQ70a3khF5TNnyJVA8V2I9wTRPBLPdLWt83E5m1nTufoilF2MI7UBALkFH1Q==", + "version": "6.4.3", + "resolved": "https://registry.npmjs.org/@poppinss/cliui/-/cliui-6.4.3.tgz", + "integrity": "sha512-flAHvbWHP4r7+DVcWMuO9EGnnkaZLJzkei6E4QTVhPsIAKBI78vDplhHXZofUvwG5IrU42QM0gX/pxMOKwQRvg==", "license": "MIT", "dependencies": { "@poppinss/colors": "^4.1.4", @@ -4615,18 +2593,6 @@ "node": ">=18.16.0" } }, - "node_modules/@poppinss/cliui/node_modules/supports-color": { - "version": "10.0.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-10.0.0.tgz", - "integrity": "sha512-HRVVSbCCMbj7/kdWF9Q+bbckjBHLtHMEoJWlkmYzzdwhYMkjkOwubLM6t7NbWKjgKamGDrWL1++KrjUO1t9oAQ==", - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/chalk/supports-color?sponsor=1" - } - }, "node_modules/@poppinss/colors": { "version": "4.1.4", "resolved": "https://registry.npmjs.org/@poppinss/colors/-/colors-4.1.4.tgz", @@ -4640,9 +2606,9 @@ } }, "node_modules/@poppinss/dumper": { - "version": "0.6.2", - "resolved": "https://registry.npmjs.org/@poppinss/dumper/-/dumper-0.6.2.tgz", - "integrity": "sha512-FhE9rY15aZ6Qp6ltQ0NZjseVRhwgWZ7+sg16343FqnjdUQvvBBi5eSeH/aZA4LF1ZOV5779DYrJXTHT42JlHNg==", + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/@poppinss/dumper/-/dumper-0.6.3.tgz", + "integrity": "sha512-iombbn8ckOixMtuV1p3f8jN6vqhXefNjJttoPaJDMeIk/yIGhkkL3OrHkEjE9SRsgoAx1vBUU2GtgggjvA5hCA==", "license": "MIT", "dependencies": { "@poppinss/colors": "^4.1.4", @@ -4650,34 +2616,13 @@ "supports-color": "^10.0.0" } }, - "node_modules/@poppinss/dumper/node_modules/supports-color": { - "version": "10.0.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-10.0.0.tgz", - "integrity": "sha512-HRVVSbCCMbj7/kdWF9Q+bbckjBHLtHMEoJWlkmYzzdwhYMkjkOwubLM6t7NbWKjgKamGDrWL1++KrjUO1t9oAQ==", + "node_modules/@poppinss/exception": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@poppinss/exception/-/exception-1.2.1.tgz", + "integrity": "sha512-aQypoot0HPSJa6gDPEPTntc1GT6QINrSbgRlRhadGW2WaYqUK3tK4Bw9SBMZXhmxd3GeAlZjVcODHgiu+THY7A==", "license": "MIT", "engines": { "node": ">=18" - }, - "funding": { - "url": "https://github.com/chalk/supports-color?sponsor=1" - } - }, - "node_modules/@poppinss/exception": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@poppinss/exception/-/exception-1.2.0.tgz", - "integrity": "sha512-WLneXKQYNClhaMXccO111VQmZahSrcSRDaHRbV6KL5R4pTvK87fMn/MXLUcvOjk0X5dTHDPKF61tM7j826wrjQ==", - "license": "MIT", - "engines": { - "node": ">=20.6.0" - } - }, - "node_modules/@poppinss/file-generator": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/@poppinss/file-generator/-/file-generator-1.0.2.tgz", - "integrity": "sha512-rRob//4jLbUVbDSsNRihloKGgpyVsWdFQWUmONxX/gyv4koT1OlVoc3ccWgk7Y/sEa2cFxj3zrFs+wdT09iXWw==", - "license": "MIT", - "dependencies": { - "bytes": "^3.1.2" } }, "node_modules/@poppinss/hooks": { @@ -4742,26 +2687,6 @@ "uid-safe": "2.1.5" } }, - "node_modules/@poppinss/multiparty/node_modules/safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "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/@poppinss/object-builder": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/@poppinss/object-builder/-/object-builder-1.1.0.tgz", @@ -4839,83 +2764,75 @@ "license": "BSD-3-Clause" }, "node_modules/@poppinss/validator-lite": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/@poppinss/validator-lite/-/validator-lite-1.0.3.tgz", - "integrity": "sha512-u4dmT7PDHwNtxY3q1jHVp/u+hMEEcBlkzd37QwwM4tVt/0mLlEDttSfPQ+TT7sqPG4VEtWKwVSlMInwPUYyJpA==", - "license": "MIT", - "dependencies": { - "validator": "^13.9.0" - } + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@poppinss/validator-lite/-/validator-lite-2.1.0.tgz", + "integrity": "sha512-CfT8EPeB7jKxjCb5+KP32iu/0BB7cKlRRqMBcCwzky0WgFACsFlRtvHsy+CkOszHmNyOdoH3WoyMyoxVCu9qEw==", + "license": "MIT" }, "node_modules/@redis/bloom": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@redis/bloom/-/bloom-1.2.0.tgz", - "integrity": "sha512-HG2DFjYKbpNmVXsa0keLHp/3leGJz1mjh09f2RLGGLQZzSHpkmZWuwJbAvo3QcRY8p80m5+ZdXZdYOSBLlp7Cg==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@redis/bloom/-/bloom-5.0.0.tgz", + "integrity": "sha512-YHlra6a7+brJ3Ustqa+jC68Pdnw2/wbnVLeULN0llP3gHpKxOSmiDyLwuHq5t/jFi6QUgjSprUQi2a8CieKFIA==", "license": "MIT", + "engines": { + "node": ">= 18" + }, "peerDependencies": { - "@redis/client": "^1.0.0" + "@redis/client": "^5.0.0" } }, "node_modules/@redis/client": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/@redis/client/-/client-1.6.0.tgz", - "integrity": "sha512-aR0uffYI700OEEH4gYnitAnv3vzVGXCFvYfdpu/CJKvk4pHfLPEy/JSZyrpQ+15WhXe1yJRXLtfQ84s4mEXnPg==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@redis/client/-/client-5.0.0.tgz", + "integrity": "sha512-1MqfzOOoFBwWprigJ3S5hM+LuzaWwxyyx+74NDzCpmjKWyjjC07xkWsd5E9fr+61NUDtvUivRiyAxGiRIwJCaQ==", "license": "MIT", "dependencies": { - "cluster-key-slot": "1.1.2", - "generic-pool": "3.9.0", - "yallist": "4.0.0" + "cluster-key-slot": "1.1.2" }, "engines": { - "node": ">=14" - } - }, - "node_modules/@redis/client/node_modules/yallist": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", - "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", - "license": "ISC" - }, - "node_modules/@redis/graph": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/@redis/graph/-/graph-1.1.1.tgz", - "integrity": "sha512-FEMTcTHZozZciLRl6GiiIB4zGm5z5F3F6a6FZCyrfxdKOhFlGkiAqlexWMBzCi4DcRoyiOsuLfW+cjlGWyExOw==", - "license": "MIT", - "peerDependencies": { - "@redis/client": "^1.0.0" + "node": ">= 18" } }, "node_modules/@redis/json": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/@redis/json/-/json-1.0.7.tgz", - "integrity": "sha512-6UyXfjVaTBTJtKNG4/9Z8PSpKE6XgSyEb8iwaqDcy+uKrd/DGYHTWkUdnQDyzm727V7p21WUMhsqz5oy65kPcQ==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@redis/json/-/json-5.0.0.tgz", + "integrity": "sha512-8YFjshsvWR9dWjOiQ50TfBxfl+z1mjCae8jDnfOYN1aCvUZWJXqM1OpL56v6mWRyXwZ5C9qJJKDGmHTr4PVskQ==", "license": "MIT", + "engines": { + "node": ">= 18" + }, "peerDependencies": { - "@redis/client": "^1.0.0" + "@redis/client": "^5.0.0" } }, "node_modules/@redis/search": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@redis/search/-/search-1.2.0.tgz", - "integrity": "sha512-tYoDBbtqOVigEDMAcTGsRlMycIIjwMCgD8eR2t0NANeQmgK/lvxNAvYyb6bZDD4frHRhIHkJu2TBRvB0ERkOmw==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@redis/search/-/search-5.0.0.tgz", + "integrity": "sha512-oEop/S/0NB5p7vpTTlp5X5LQO4oTomJnFEOaKhQ5xZWROLXTgEVvPSfXVNBbtr9maU2+OupxXQR63HW3MGeTUg==", "license": "MIT", + "engines": { + "node": ">= 18" + }, "peerDependencies": { - "@redis/client": "^1.0.0" + "@redis/client": "^5.0.0" } }, "node_modules/@redis/time-series": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@redis/time-series/-/time-series-1.1.0.tgz", - "integrity": "sha512-c1Q99M5ljsIuc4YdaCwfUEXsofakb9c8+Zse2qxTadu8TalLXuAESzLvFAvNVbkmSlvlzIQOLpBCmWI9wTOt+g==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/@redis/time-series/-/time-series-5.0.0.tgz", + "integrity": "sha512-DNsP4DH5CRfXlQDyvLJm2DltZm3kjpVKZsNTI3SPdbRxhc9i2c0NehUbM/gDECnEfG+jkNN/LUv6Em4LaLwPBQ==", "license": "MIT", + "engines": { + "node": ">= 18" + }, "peerDependencies": { - "@redis/client": "^1.0.0" + "@redis/client": "^5.0.0" } }, "node_modules/@rollup/rollup-android-arm-eabi": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.30.1.tgz", - "integrity": "sha512-pSWY+EVt3rJ9fQ3IqlrEUtXh3cGqGtPDH1FQlNZehO2yYxCHEX1SPsz1M//NXwYfbTlcKr9WObLnJX9FsS9K1Q==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.40.1.tgz", + "integrity": "sha512-kxz0YeeCrRUHz3zyqvd7n+TVRlNyTifBsmnmNPtk3hQURUyG9eAB+usz6DAwagMusjx/zb3AjvDUvhFGDAexGw==", "cpu": [ "arm" ], @@ -4923,13 +2840,12 @@ "optional": true, "os": [ "android" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-android-arm64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.30.1.tgz", - "integrity": "sha512-/NA2qXxE3D/BRjOJM8wQblmArQq1YoBVJjrjoTSBS09jgUisq7bqxNHJ8kjCHeV21W/9WDGwJEWSN0KQ2mtD/w==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.40.1.tgz", + "integrity": "sha512-PPkxTOisoNC6TpnDKatjKkjRMsdaWIhyuMkA4UsBXT9WEZY4uHezBTjs6Vl4PbqQQeu6oION1w2voYZv9yquCw==", "cpu": [ "arm64" ], @@ -4937,13 +2853,12 @@ "optional": true, "os": [ "android" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-darwin-arm64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.30.1.tgz", - "integrity": "sha512-r7FQIXD7gB0WJ5mokTUgUWPl0eYIH0wnxqeSAhuIwvnnpjdVB8cRRClyKLQr7lgzjctkbp5KmswWszlwYln03Q==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.40.1.tgz", + "integrity": "sha512-VWXGISWFY18v/0JyNUy4A46KCFCb9NVsH+1100XP31lud+TzlezBbz24CYzbnA4x6w4hx+NYCXDfnvDVO6lcAA==", "cpu": [ "arm64" ], @@ -4951,13 +2866,12 @@ "optional": true, "os": [ "darwin" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-darwin-x64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.30.1.tgz", - "integrity": "sha512-x78BavIwSH6sqfP2xeI1hd1GpHL8J4W2BXcVM/5KYKoAD3nNsfitQhvWSw+TFtQTLZ9OmlF+FEInEHyubut2OA==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.40.1.tgz", + "integrity": "sha512-nIwkXafAI1/QCS7pxSpv/ZtFW6TXcNUEHAIA9EIyw5OzxJZQ1YDrX+CL6JAIQgZ33CInl1R6mHet9Y/UZTg2Bw==", "cpu": [ "x64" ], @@ -4965,13 +2879,12 @@ "optional": true, "os": [ "darwin" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-freebsd-arm64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.30.1.tgz", - "integrity": "sha512-HYTlUAjbO1z8ywxsDFWADfTRfTIIy/oUlfIDmlHYmjUP2QRDTzBuWXc9O4CXM+bo9qfiCclmHk1x4ogBjOUpUQ==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.40.1.tgz", + "integrity": "sha512-BdrLJ2mHTrIYdaS2I99mriyJfGGenSaP+UwGi1kB9BLOCu9SR8ZpbkmmalKIALnRw24kM7qCN0IOm6L0S44iWw==", "cpu": [ "arm64" ], @@ -4979,13 +2892,12 @@ "optional": true, "os": [ "freebsd" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-freebsd-x64": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.30.1.tgz", - "integrity": "sha512-1MEdGqogQLccphhX5myCJqeGNYTNcmTyaic9S7CG3JhwuIByJ7J05vGbZxsizQthP1xpVx7kd3o31eOogfEirw==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.40.1.tgz", + "integrity": "sha512-VXeo/puqvCG8JBPNZXZf5Dqq7BzElNJzHRRw3vjBE27WujdzuOPecDPc/+1DcdcTptNBep3861jNq0mYkT8Z6Q==", "cpu": [ "x64" ], @@ -4993,13 +2905,12 @@ "optional": true, "os": [ "freebsd" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-arm-gnueabihf": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.30.1.tgz", - "integrity": "sha512-PaMRNBSqCx7K3Wc9QZkFx5+CX27WFpAMxJNiYGAXfmMIKC7jstlr32UhTgK6T07OtqR+wYlWm9IxzennjnvdJg==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.40.1.tgz", + "integrity": "sha512-ehSKrewwsESPt1TgSE/na9nIhWCosfGSFqv7vwEtjyAqZcvbGIg4JAcV7ZEh2tfj/IlfBeZjgOXm35iOOjadcg==", "cpu": [ "arm" ], @@ -5007,13 +2918,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-arm-musleabihf": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.30.1.tgz", - "integrity": "sha512-B8Rcyj9AV7ZlEFqvB5BubG5iO6ANDsRKlhIxySXcF1axXYUyqwBok+XZPgIYGBgs7LDXfWfifxhw0Ik57T0Yug==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.40.1.tgz", + "integrity": "sha512-m39iO/aaurh5FVIu/F4/Zsl8xppd76S4qoID8E+dSRQvTyZTOI2gVk3T4oqzfq1PtcvOfAVlwLMK3KRQMaR8lg==", "cpu": [ "arm" ], @@ -5021,13 +2931,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-arm64-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.30.1.tgz", - "integrity": "sha512-hqVyueGxAj3cBKrAI4aFHLV+h0Lv5VgWZs9CUGqr1z0fZtlADVV1YPOij6AhcK5An33EXaxnDLmJdQikcn5NEw==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.40.1.tgz", + "integrity": "sha512-Y+GHnGaku4aVLSgrT0uWe2o2Rq8te9hi+MwqGF9r9ORgXhmHK5Q71N757u0F8yU1OIwUIFy6YiJtKjtyktk5hg==", "cpu": [ "arm64" ], @@ -5035,13 +2944,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-arm64-musl": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.30.1.tgz", - "integrity": "sha512-i4Ab2vnvS1AE1PyOIGp2kXni69gU2DAUVt6FSXeIqUCPIR3ZlheMW3oP2JkukDfu3PsexYRbOiJrY+yVNSk9oA==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.40.1.tgz", + "integrity": "sha512-jEwjn3jCA+tQGswK3aEWcD09/7M5wGwc6+flhva7dsQNRZZTe30vkalgIzV4tjkopsTS9Jd7Y1Bsj6a4lzz8gQ==", "cpu": [ "arm64" ], @@ -5049,13 +2957,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-loongarch64-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.30.1.tgz", - "integrity": "sha512-fARcF5g296snX0oLGkVxPmysetwUk2zmHcca+e9ObOovBR++9ZPOhqFUM61UUZ2EYpXVPN1redgqVoBB34nTpQ==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.40.1.tgz", + "integrity": "sha512-ySyWikVhNzv+BV/IDCsrraOAZ3UaC8SZB67FZlqVwXwnFhPihOso9rPOxzZbjp81suB1O2Topw+6Ug3JNegejQ==", "cpu": [ "loong64" ], @@ -5063,13 +2970,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-powerpc64le-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.30.1.tgz", - "integrity": "sha512-GLrZraoO3wVT4uFXh67ElpwQY0DIygxdv0BNW9Hkm3X34wu+BkqrDrkcsIapAY+N2ATEbvak0XQ9gxZtCIA5Rw==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.40.1.tgz", + "integrity": "sha512-BvvA64QxZlh7WZWqDPPdt0GH4bznuL6uOO1pmgPnnv86rpUpc8ZxgZwcEgXvo02GRIZX1hQ0j0pAnhwkhwPqWg==", "cpu": [ "ppc64" ], @@ -5077,13 +2983,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-riscv64-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.30.1.tgz", - "integrity": "sha512-0WKLaAUUHKBtll0wvOmh6yh3S0wSU9+yas923JIChfxOaaBarmb/lBKPF0w/+jTVozFnOXJeRGZ8NvOxvk/jcw==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.40.1.tgz", + "integrity": "sha512-EQSP+8+1VuSulm9RKSMKitTav89fKbHymTf25n5+Yr6gAPZxYWpj3DzAsQqoaHAk9YX2lwEyAf9S4W8F4l3VBQ==", "cpu": [ "riscv64" ], @@ -5091,13 +2996,25 @@ "optional": true, "os": [ "linux" + ] + }, + "node_modules/@rollup/rollup-linux-riscv64-musl": { + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.40.1.tgz", + "integrity": "sha512-n/vQ4xRZXKuIpqukkMXZt9RWdl+2zgGNx7Uda8NtmLJ06NL8jiHxUawbwC+hdSq1rrw/9CghCpEONor+l1e2gA==", + "cpu": [ + "riscv64" ], - "peer": true + "license": "MIT", + "optional": true, + "os": [ + "linux" + ] }, "node_modules/@rollup/rollup-linux-s390x-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.30.1.tgz", - "integrity": "sha512-GWFs97Ruxo5Bt+cvVTQkOJ6TIx0xJDD/bMAOXWJg8TCSTEK8RnFeOeiFTxKniTc4vMIaWvCplMAFBt9miGxgkA==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.40.1.tgz", + "integrity": "sha512-h8d28xzYb98fMQKUz0w2fMc1XuGzLLjdyxVIbhbil4ELfk5/orZlSTpF/xdI9C8K0I8lCkq+1En2RJsawZekkg==", "cpu": [ "s390x" ], @@ -5105,13 +3022,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-x64-gnu": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.30.1.tgz", - "integrity": "sha512-UtgGb7QGgXDIO+tqqJ5oZRGHsDLO8SlpE4MhqpY9Llpzi5rJMvrK6ZGhsRCST2abZdBqIBeXW6WPD5fGK5SDwg==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.40.1.tgz", + "integrity": "sha512-XiK5z70PEFEFqcNj3/zRSz/qX4bp4QIraTy9QjwJAb/Z8GM7kVUsD0Uk8maIPeTyPCP03ChdI+VVmJriKYbRHQ==", "cpu": [ "x64" ], @@ -5119,13 +3035,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-linux-x64-musl": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.30.1.tgz", - "integrity": "sha512-V9U8Ey2UqmQsBT+xTOeMzPzwDzyXmnAoO4edZhL7INkwQcaW1Ckv3WJX3qrrp/VHaDkEWIBWhRwP47r8cdrOow==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.40.1.tgz", + "integrity": "sha512-2BRORitq5rQ4Da9blVovzNCMaUlyKrzMSvkVR0D4qPuOy/+pMCrh1d7o01RATwVy+6Fa1WBw+da7QPeLWU/1mQ==", "cpu": [ "x64" ], @@ -5133,13 +3048,12 @@ "optional": true, "os": [ "linux" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-win32-arm64-msvc": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.30.1.tgz", - "integrity": "sha512-WabtHWiPaFF47W3PkHnjbmWawnX/aE57K47ZDT1BXTS5GgrBUEpvOzq0FI0V/UYzQJgdb8XlhVNH8/fwV8xDjw==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.40.1.tgz", + "integrity": "sha512-b2bcNm9Kbde03H+q+Jjw9tSfhYkzrDUf2d5MAd1bOJuVplXvFhWz7tRtWvD8/ORZi7qSCy0idW6tf2HgxSXQSg==", "cpu": [ "arm64" ], @@ -5147,13 +3061,12 @@ "optional": true, "os": [ "win32" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-win32-ia32-msvc": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.30.1.tgz", - "integrity": "sha512-pxHAU+Zv39hLUTdQQHUVHf4P+0C47y/ZloorHpzs2SXMRqeAWmGghzAhfOlzFHHwjvgokdFAhC4V+6kC1lRRfw==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.40.1.tgz", + "integrity": "sha512-DfcogW8N7Zg7llVEfpqWMZcaErKfsj9VvmfSyRjCyo4BI3wPEfrzTtJkZG6gKP/Z92wFm6rz2aDO7/JfiR/whA==", "cpu": [ "ia32" ], @@ -5161,13 +3074,12 @@ "optional": true, "os": [ "win32" - ], - "peer": true + ] }, "node_modules/@rollup/rollup-win32-x64-msvc": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.30.1.tgz", - "integrity": "sha512-D6qjsXGcvhTjv0kI4fU8tUuBDF/Ueee4SVX79VfNDXZa64TfCW1Slkb6Z7O1p7vflqZjcmOVdZlqf8gvJxc6og==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.40.1.tgz", + "integrity": "sha512-ECyOuDeH3C1I8jH2MK1RtBJW+YPMvSfT0a5NN0nHfQYnDSJ6tUiZH3gzwVP5/Kfh/+Tt7tpWVF9LXNTnhTJ3kA==", "cpu": [ "x64" ], @@ -5175,8 +3087,7 @@ "optional": true, "os": [ "win32" - ], - "peer": true + ] }, "node_modules/@sec-ant/readable-stream": { "version": "0.4.1", @@ -5216,381 +3127,20 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/@swc/core": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core/-/core-1.10.6.tgz", - "integrity": "sha512-zgXXsI6SAVwr6XsXyMnqlyLoa1lT+r09bAWI1xT3679ejWqI1Vnl14eJG0GjWYXCEMKHCNytfMq3OOQ62C39QQ==", - "dev": true, - "hasInstallScript": true, - "license": "Apache-2.0", - "dependencies": { - "@swc/counter": "^0.1.3", - "@swc/types": "^0.1.17" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/swc" - }, - "optionalDependencies": { - "@swc/core-darwin-arm64": "1.10.6", - "@swc/core-darwin-x64": "1.10.6", - "@swc/core-linux-arm-gnueabihf": "1.10.6", - "@swc/core-linux-arm64-gnu": "1.10.6", - "@swc/core-linux-arm64-musl": "1.10.6", - "@swc/core-linux-x64-gnu": "1.10.6", - "@swc/core-linux-x64-musl": "1.10.6", - "@swc/core-win32-arm64-msvc": "1.10.6", - "@swc/core-win32-ia32-msvc": "1.10.6", - "@swc/core-win32-x64-msvc": "1.10.6" - }, - "peerDependencies": { - "@swc/helpers": "*" - }, - "peerDependenciesMeta": { - "@swc/helpers": { - "optional": true - } - } + "node_modules/@speed-highlight/core": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/@speed-highlight/core/-/core-1.2.7.tgz", + "integrity": "sha512-0dxmVj4gxg3Jg879kvFS/msl4s9F3T9UXC1InxgOf7t5NvcPD97u/WTA5vL/IxWHMn7qSxBozqrnnE2wvl1m8g==", + "devOptional": true, + "license": "CC0-1.0" }, - "node_modules/@swc/core-darwin-arm64": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-darwin-arm64/-/core-darwin-arm64-1.10.6.tgz", - "integrity": "sha512-USbMvT8Rw5PvIfF6HyTm+yW84J9c45emzmHBDIWY76vZHkFsS5MepNi+JLQyBzBBgE7ScwBRBNhRx6VNhkSoww==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-darwin-x64": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-darwin-x64/-/core-darwin-x64-1.10.6.tgz", - "integrity": "sha512-7t2IozcZN4r1p27ei+Kb8IjN4aLoBDn107fPi+aPLcVp2uFgJEUzhCDuZXBNW2057Mx1OHcjzrkaleRpECz3Xg==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm-gnueabihf": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.10.6.tgz", - "integrity": "sha512-CPgWT+D0bDp/qhXsLkIJ54LmKU1/zvyGaf/yz8A4iR+YoF6R5CSXENXhNJY8cIrb6+uNWJZzHJ+gefB5V51bpA==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm64-gnu": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.10.6.tgz", - "integrity": "sha512-5qZ6hVnqO/ShETXdGSzvdGUVx372qydlj1YWSYiaxQzTAepEBc8TC1NVUgYtOHOKVRkky1d7p6GQ9lymsd4bHw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm64-musl": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.10.6.tgz", - "integrity": "sha512-hB2xZFmXCKf2iJF5y2z01PSuLqEoUP3jIX/XlIHN+/AIP7PkSKsValE63LnjlnWPnSEI0IxUyRE3T3FzWE/fQQ==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-x64-gnu": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.10.6.tgz", - "integrity": "sha512-PRGPp0I22+oJ8RMGg8M4hXYxEffH3ayu0WoSDPOjfol1F51Wj1tfTWN4wVa2RibzJjkBwMOT0KGLGb/hSEDDXQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-x64-musl": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.10.6.tgz", - "integrity": "sha512-SoNBxlA86lnoV9vIz/TCyakLkdRhFSHx6tFMKNH8wAhz1kKYbZfDmpYoIzeQqdTh0tpx8e/Zu1zdK4smovsZqQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-arm64-msvc": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.10.6.tgz", - "integrity": "sha512-6L5Y2E+FVvM+BtoA+mJFjf/SjpFr73w2kHBxINxwH8/PkjAjkePDr5m0ibQhPXV61bTwX49+1otzTY85EsUW9Q==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-ia32-msvc": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.10.6.tgz", - "integrity": "sha512-kxK3tW8DJwEkAkwy0vhwoBAShRebH1QTe0mvH9tlBQ21rToVZQn+GCV/I44dind80hYPw0Tw2JKFVfoEJyBszg==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-x64-msvc": { - "version": "1.10.6", - "resolved": "https://registry.npmjs.org/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.10.6.tgz", - "integrity": "sha512-4pJka/+t8XcHee12G/R5VWcilkp5poT2EJhrybpuREkpQ7iC/4WOlOVrohbWQ4AhDQmojYQI/iS+gdF2JFLzTQ==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/counter": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/@swc/counter/-/counter-0.1.3.tgz", - "integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==", + "node_modules/@swc/wasm": { + "version": "1.11.24", + "resolved": "https://registry.npmjs.org/@swc/wasm/-/wasm-1.11.24.tgz", + "integrity": "sha512-8Eo70Ns3BUboUaHfHhDfT/e0gA92+qpHggnb8oio+bfESMXBkmBdkQIln8TI27axKGc7uWz9M82kC8k9FJFq0w==", "dev": true, "license": "Apache-2.0" }, - "node_modules/@swc/types": { - "version": "0.1.17", - "resolved": "https://registry.npmjs.org/@swc/types/-/types-0.1.17.tgz", - "integrity": "sha512-V5gRru+aD8YVyCOMAjMpWR1Ui577DD5KSJsHP8RAxopAH22jFz6GZd/qxqjO6MJHQhcsjvjOFXyDhyLQUnMveQ==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@swc/counter": "^0.1.3" - } - }, - "node_modules/@symfony/webpack-encore": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/@symfony/webpack-encore/-/webpack-encore-5.0.1.tgz", - "integrity": "sha512-2l9ssZCJDMKOXi1iggjn7HEaErdYvITvuheLvtXHAgR2mauV2FiE/pNFS+Bqz2sbj1g4pPcqJIl5AwFE9etOgg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@nuxt/friendly-errors-webpack-plugin": "^2.5.1", - "babel-loader": "^9.1.3", - "css-loader": "^7.1.0", - "css-minimizer-webpack-plugin": "^7.0.0", - "fastest-levenshtein": "^1.0.16", - "mini-css-extract-plugin": "^2.6.0", - "picocolors": "^1.1.0", - "pretty-error": "^4.0.0", - "resolve-url-loader": "^5.0.0", - "semver": "^7.3.2", - "style-loader": "^3.3.0", - "tapable": "^2.2.1", - "terser-webpack-plugin": "^5.3.0", - "tmp": "^0.2.1", - "yargs-parser": "^21.0.0" - }, - "bin": { - "encore": "bin/encore.js" - }, - "engines": { - "node": "^18.12.0 || ^20.0.0 || >=22.0" - }, - "peerDependencies": { - "@babel/core": "^7.17.0", - "@babel/plugin-transform-react-jsx": "^7.12.11", - "@babel/preset-env": "^7.16.0", - "@babel/preset-react": "^7.9.0", - "@babel/preset-typescript": "^7.0.0", - "@symfony/stimulus-bridge": "^3.0.0", - "@vue/babel-helper-vue-jsx-merge-props": "^1.0.0", - "@vue/babel-plugin-jsx": "^1.0.0", - "@vue/babel-preset-jsx": "^1.0.0", - "@vue/compiler-sfc": "^2.6 || ^3.0.2", - "file-loader": "^6.0.0", - "fork-ts-checker-webpack-plugin": "^7.0.0 || ^8.0.0 || ^9.0.0", - "handlebars": "^4.7.7", - "handlebars-loader": "^1.7.0", - "less": "^4.0.0", - "less-loader": "^11.0.0 || ^12.2.0", - "postcss": "^8.3.0", - "postcss-loader": "^7.0.0 || ^8.1.0", - "sass": "^1.17.0", - "sass-loader": "^16.0.1", - "stylus-loader": "^7.0.0 || ^8.1.0", - "ts-loader": "^9.0.0", - "typescript": "^5.0.0", - "vue": "^3.2.14", - "vue-loader": "^17.0.0", - "webpack": "^5.72", - "webpack-cli": "^5.1.4", - "webpack-notifier": "^1.15.0" - }, - "peerDependenciesMeta": { - "@babel/core": { - "optional": false - }, - "@babel/plugin-transform-react-jsx": { - "optional": true - }, - "@babel/preset-env": { - "optional": false - }, - "@babel/preset-react": { - "optional": true - }, - "@babel/preset-typescript": { - "optional": true - }, - "@symfony/stimulus-bridge": { - "optional": true - }, - "@vue/babel-helper-vue-jsx-merge-props": { - "optional": true - }, - "@vue/babel-plugin-jsx": { - "optional": true - }, - "@vue/babel-preset-jsx": { - "optional": true - }, - "@vue/compiler-sfc": { - "optional": true - }, - "file-loader": { - "optional": true - }, - "fork-ts-checker-webpack-plugin": { - "optional": true - }, - "handlebars": { - "optional": true - }, - "handlebars-loader": { - "optional": true - }, - "less": { - "optional": true - }, - "less-loader": { - "optional": true - }, - "postcss": { - "optional": true - }, - "postcss-loader": { - "optional": true - }, - "sass": { - "optional": true - }, - "sass-loader": { - "optional": true - }, - "stylus-loader": { - "optional": true - }, - "ts-loader": { - "optional": true - }, - "typescript": { - "optional": true - }, - "vue": { - "optional": true - }, - "vue-loader": { - "optional": true - }, - "webpack": { - "optional": false - }, - "webpack-cli": { - "optional": false - }, - "webpack-dev-server": { - "optional": true - }, - "webpack-notifier": { - "optional": true - } - } - }, "node_modules/@szmarczak/http-timer": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-5.0.1.tgz", @@ -5616,22 +3166,58 @@ "tailwindcss": ">=3.0.0 || >= 3.0.0-alpha.1 || >= 4.0.0-alpha.20 || >= 4.0.0-beta.1" } }, + "node_modules/@tanstack/virtual-core": { + "version": "3.13.6", + "resolved": "https://registry.npmjs.org/@tanstack/virtual-core/-/virtual-core-3.13.6.tgz", + "integrity": "sha512-cnQUeWnhNP8tJ4WsGcYiX24Gjkc9ALstLbHcBj1t3E7EimN6n6kHH+DPV4PpDnuw00NApQp+ViojMj1GRdwYQg==", + "dev": true, + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/tannerlinsley" + } + }, + "node_modules/@tanstack/vue-virtual": { + "version": "3.13.6", + "resolved": "https://registry.npmjs.org/@tanstack/vue-virtual/-/vue-virtual-3.13.6.tgz", + "integrity": "sha512-GYdZ3SJBQPzgxhuCE2fvpiH46qzHiVx5XzBSdtESgiqh4poj8UgckjGWYEhxaBbcVt1oLzh1m3Ql4TyH32TOzQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@tanstack/virtual-core": "3.13.6" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/tannerlinsley" + }, + "peerDependencies": { + "vue": "^2.7.0 || ^3.0.0" + } + }, + "node_modules/@tokenizer/inflate": { + "version": "0.2.7", + "resolved": "https://registry.npmjs.org/@tokenizer/inflate/-/inflate-0.2.7.tgz", + "integrity": "sha512-MADQgmZT1eKjp06jpI2yozxaU9uVs4GzzgSL+uEq7bVcJ9V1ZXQkeGNql1fsSI0gMy1vhvNTNbUqrx+pZfJVmg==", + "license": "MIT", + "dependencies": { + "debug": "^4.4.0", + "fflate": "^0.8.2", + "token-types": "^6.0.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Borewit" + } + }, "node_modules/@tokenizer/token": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/@tokenizer/token/-/token-0.3.0.tgz", "integrity": "sha512-OvjF+z51L3ov0OyAU0duzsYuvO01PH7x4t6DJx+guahgTnBHkhJdG7soQeTSFLWN3efnHyibZ4Z8l2EuWwJN3A==", "license": "MIT" }, - "node_modules/@trysound/sax": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/@trysound/sax/-/sax-0.2.0.tgz", - "integrity": "sha512-L7z9BgrNEcYyUYtF+HaEfiS5ebkh9jXqbszz7pC0hRBPaatV0XjSD3+eHrpqFemQfgwiFF0QPIarnIihIDn7OA==", - "dev": true, - "license": "ISC", - "engines": { - "node": ">=10.13.0" - } - }, "node_modules/@ts-morph/common": { "version": "0.24.0", "resolved": "https://registry.npmjs.org/@ts-morph/common/-/common-0.24.0.tgz", @@ -5716,9 +3302,9 @@ "license": "MIT" }, "node_modules/@tuyau/utils": { - "version": "0.0.4", - "resolved": "https://registry.npmjs.org/@tuyau/utils/-/utils-0.0.4.tgz", - "integrity": "sha512-ex6CAJNLiTuOvx7nUrgs8FwNG/t88Mi8QTLSO3muHbB6vBSpYimZ6iSUkk4cjEFd4XDy0y+24GDgXKoBfGf4ag==", + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/@tuyau/utils/-/utils-0.0.6.tgz", + "integrity": "sha512-X6teHJyGGyjkSHRvBf9tX13K54O6yxaDcNI/NbgXYb/yBdm4Esr4yRBYjOKEvaQYkAChQh/CXyEeGYNPdj+2Zg==", "license": "ISC" }, "node_modules/@types/bcryptjs": { @@ -5756,16 +3342,19 @@ "license": "MIT" }, "node_modules/@types/chai": { - "version": "4.3.20", - "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.3.20.tgz", - "integrity": "sha512-/pC9HAB5I/xMlc5FP77qjCnI16ChlJfW0tGa0IUcFn38VJrTV6DeZ60NU5KZBtaOZqjdpwTWohz5HU1RrhiYxQ==", + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.1.tgz", + "integrity": "sha512-iu1JLYmGmITRzUgNiLMZD3WCoFzpYtueuyAgHTXqgwSRAMIlFTnZqG6/xenkpUGRJEzSfklUTI4GNSzks/dc0w==", "dev": true, - "license": "MIT" + "license": "MIT", + "dependencies": { + "@types/deep-eql": "*" + } }, "node_modules/@types/clamscan": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/@types/clamscan/-/clamscan-2.4.0.tgz", - "integrity": "sha512-wqYy+klgBWqCFAMNZsJrb5Q4d0VIjMPk2lfGsq5jHMPyq8E61YLOV+3VNrvoJdbNmzVIXGIc9npb0Tjw+MXEfw==", + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/@types/clamscan/-/clamscan-2.4.1.tgz", + "integrity": "sha512-KhBHhXsMGDoEkk87VRtHtDsjoqXD3epu+a09c1sjW7xqpvoihScxhZNdNIPegVCLDvPOp4khiIpy02XabseztQ==", "dev": true, "license": "MIT", "dependencies": { @@ -5797,7 +3386,14 @@ "version": "2.1.5", "resolved": "https://registry.npmjs.org/@types/cookiejar/-/cookiejar-2.1.5.tgz", "integrity": "sha512-he+DHOWReW0nghN24E1WUqM0efK4kI9oTqDm6XmK8ZPe2djZ90BSNdGnIyCLzCPw7/pogPlGbzI2wHGGmi4O/Q==", - "devOptional": true, + "dev": true, + "license": "MIT" + }, + "node_modules/@types/deep-eql": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz", + "integrity": "sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==", + "dev": true, "license": "MIT" }, "node_modules/@types/disposable-email-domains": { @@ -5838,11 +3434,10 @@ } }, "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==", - "license": "MIT", - "peer": true + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.7.tgz", + "integrity": "sha512-w28IoSUCJpidD/TGviZwwMJckNESJZXFu7NBZ5YJ4mEUnNraUn9Pm8HSZm/jDF1pDWYKspWE7oVphigUPRakIQ==", + "license": "MIT" }, "node_modules/@types/express": { "version": "4.17.21", @@ -5858,19 +3453,6 @@ } }, "node_modules/@types/express-serve-static-core": { - "version": "5.0.4", - "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-5.0.4.tgz", - "integrity": "sha512-5kz9ScmzBdzTgB/3susoCgfqNDzBjvLL4taparufgSvlwjdLy6UyUy9T/tCpYd2GIdIilCatC4iSQS0QSYHt0w==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*", - "@types/qs": "*", - "@types/range-parser": "*", - "@types/send": "*" - } - }, - "node_modules/@types/express/node_modules/@types/express-serve-static-core": { "version": "4.19.6", "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.19.6.tgz", "integrity": "sha512-N4LZ2xG7DatVqhCZzOGb1Yi5lMbXSZcmdLDe9EzSndPV2HpWYWzRbaerl2n27irrm94EPpprqa8KpskPT085+A==", @@ -5884,18 +3466,20 @@ } }, "node_modules/@types/fs-extra": { - "version": "9.0.13", - "resolved": "https://registry.npmjs.org/@types/fs-extra/-/fs-extra-9.0.13.tgz", - "integrity": "sha512-nEnwB++1u5lVDM2UI4c1+5R+FYaKfaAzS4OococimjVm3nQw3TuzH5UNsocrcTBbhnerblyHj4A49qXbIiZdpA==", + "version": "11.0.4", + "resolved": "https://registry.npmjs.org/@types/fs-extra/-/fs-extra-11.0.4.tgz", + "integrity": "sha512-yTbItCNreRooED33qjunPthRcSjERP1r4MqCZc7wv0u2sUkzTFp45tgUfS5+r7FrZPdmCCNflLhVSP/o+SemsQ==", + "dev": true, "license": "MIT", "dependencies": { + "@types/jsonfile": "*", "@types/node": "*" } }, "node_modules/@types/geojson": { - "version": "7946.0.15", - "resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-7946.0.15.tgz", - "integrity": "sha512-9oSxFzDCT2Rj6DfcHF8G++jxBKS7mBqXl5xrRW+Kbvjry6Uduya2iiwqHPhVXpasAVMBYKkEPGgKhd3+/HZ6xA==", + "version": "7946.0.16", + "resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-7946.0.16.tgz", + "integrity": "sha512-6C8nqWur3j98U6+lXDfTUWIfgvZU+EumvpHKcYjujKH7woYyLj2sUmff0tRhrqM7BohUw7Pz3ZB1jj2gW9Fvmg==", "dev": true, "license": "MIT" }, @@ -5919,43 +3503,15 @@ "license": "MIT" }, "node_modules/@types/http-proxy": { - "version": "1.17.15", - "resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.15.tgz", - "integrity": "sha512-25g5atgiVNTIv0LBDTg1H74Hvayx0ajtJPLLcYE3whFv75J0pWNtOBzaXJQgDTmrX1bx5U9YC2w/n65BN1HwRQ==", + "version": "1.17.16", + "resolved": "https://registry.npmjs.org/@types/http-proxy/-/http-proxy-1.17.16.tgz", + "integrity": "sha512-sdWoUajOB1cd0A8cRRQ1cfyWNbmFKLAqBB89Y8x5iYyG/mkJHc0YUH8pdWBy2omi9qtCpiIgGjuwO0dQST2l5w==", "dev": true, "license": "MIT", "dependencies": { "@types/node": "*" } }, - "node_modules/@types/istanbul-lib-coverage": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.6.tgz", - "integrity": "sha512-2QF/t/auWm0lsy8XtKVPG19v3sSOQlJe/YHZgfjb/KBBHOGSV+J2q/S671rcq9uTBrLAXmZpqJiaQbMT+zNU1w==", - "dev": true, - "license": "MIT" - }, - "node_modules/@types/istanbul-lib-report": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.3.tgz", - "integrity": "sha512-NQn7AHQnk/RSLOxrBbGyJM/aVQ+pjj5HCgasFxc0K/KhoATfQ/47AyUl15I2yBUpihjmas+a+VJBOqecrFH+uA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/istanbul-lib-coverage": "*" - } - }, - "node_modules/@types/istanbul-reports": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-1.1.2.tgz", - "integrity": "sha512-P/W9yOX/3oPZSpaYOCQzGqgCQRXn0FFO/V8bWrCQs+wLmvVVxk6CRBXALEvNs9OHIatlnlFokfhuDo2ug01ciw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/istanbul-lib-coverage": "*", - "@types/istanbul-lib-report": "*" - } - }, "node_modules/@types/json-schema": { "version": "7.0.15", "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", @@ -5963,10 +3519,20 @@ "dev": true, "license": "MIT" }, + "node_modules/@types/jsonfile": { + "version": "6.1.4", + "resolved": "https://registry.npmjs.org/@types/jsonfile/-/jsonfile-6.1.4.tgz", + "integrity": "sha512-D5qGUYwjvnNNextdU59/+fI+spnwtTFmyQP0h+PfIOSkNfpU6AOICUOkm4i0OnSk+NyjdPJrxCDro0sJsWlRpQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, "node_modules/@types/leaflet": { - "version": "1.9.16", - "resolved": "https://registry.npmjs.org/@types/leaflet/-/leaflet-1.9.16.tgz", - "integrity": "sha512-wzZoyySUxkgMZ0ihJ7IaUIblG8Rdc8AbbZKLneyn+QjYsj5q1QU7TEKYqwTr10BGSzY5LI7tJk9Ifo+mEjdFRw==", + "version": "1.9.17", + "resolved": "https://registry.npmjs.org/@types/leaflet/-/leaflet-1.9.17.tgz", + "integrity": "sha512-IJ4K6t7I3Fh5qXbQ1uwL3CFVbCi6haW9+53oLWgdKlLP7EaS21byWFJxxqOx9y8I0AP0actXSJLVMbyvxhkUTA==", "dev": true, "license": "MIT", "dependencies": { @@ -5974,9 +3540,9 @@ } }, "node_modules/@types/luxon": { - "version": "3.4.2", - "resolved": "https://registry.npmjs.org/@types/luxon/-/luxon-3.4.2.tgz", - "integrity": "sha512-TifLZlFudklWlMBfhubvgqTXRzLDI5pCbGa4P8a3wPyUQSW+1xQ5eDsreP9DWHX3tjq1ke96uYG/nwundroWcA==", + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/@types/luxon/-/luxon-3.6.2.tgz", + "integrity": "sha512-R/BdP7OxEMc44l2Ex5lSXHoIXTB2JLNa3y2QISIbr58U/YcsffyQrYW//hZSdrfxrjRZj3GcUoxMPGdO8gSYuw==", "devOptional": true, "license": "MIT" }, @@ -5984,7 +3550,7 @@ "version": "1.1.4", "resolved": "https://registry.npmjs.org/@types/methods/-/methods-1.1.4.tgz", "integrity": "sha512-ymXWVrDiCxTBE3+RIrrP533E70eA+9qu7zdWoHuOmGujkYtzf4HQF96b8nwHLqhuf4ykX61IGRIB38CC6/sImQ==", - "devOptional": true, + "dev": true, "license": "MIT" }, "node_modules/@types/mime": { @@ -5995,12 +3561,12 @@ "license": "MIT" }, "node_modules/@types/node": { - "version": "22.10.5", - "resolved": "https://registry.npmjs.org/@types/node/-/node-22.10.5.tgz", - "integrity": "sha512-F8Q+SeGimwOo86fiovQh8qiXfFEh2/ocYv7tU5pJ3EXMSSxk1Joj5wefpFK2fHTf/N6HKGSxIDBT9f3gCxXPkQ==", + "version": "22.15.3", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.15.3.tgz", + "integrity": "sha512-lX7HFZeHf4QG/J7tBZqrCAXwz9J5RD56Y6MpP0eJkka8p+K0RY/yBTW7CYFJ4VGCclxqOLKmiGP5juQc6MKgcw==", "license": "MIT", "dependencies": { - "undici-types": "~6.20.0" + "undici-types": "~6.21.0" } }, "node_modules/@types/node-forge": { @@ -6022,6 +3588,13 @@ "@types/node": "*" } }, + "node_modules/@types/normalize-package-data": { + "version": "2.4.4", + "resolved": "https://registry.npmjs.org/@types/normalize-package-data/-/normalize-package-data-2.4.4.tgz", + "integrity": "sha512-37i+OaWTh9qeK4LSHPsyRC7NahnGotNuZvjLSgcPzblpHB3rrCJxAOgI5gCdKm7coonsaX1Of0ILiTcnZjbfxA==", + "dev": true, + "license": "MIT" + }, "node_modules/@types/notp": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/@types/notp/-/notp-2.0.5.tgz", @@ -6031,40 +3604,6 @@ "@types/node": "*" } }, - "node_modules/@types/pino": { - "version": "6.3.12", - "resolved": "https://registry.npmjs.org/@types/pino/-/pino-6.3.12.tgz", - "integrity": "sha512-dsLRTq8/4UtVSpJgl9aeqHvbh6pzdmjYD3C092SYgLD2TyoCqHpTJk6vp8DvCTGGc7iowZ2MoiYiVUUCcu7muw==", - "license": "MIT", - "peer": true, - "dependencies": { - "@types/node": "*", - "@types/pino-pretty": "*", - "@types/pino-std-serializers": "*", - "sonic-boom": "^2.1.0" - } - }, - "node_modules/@types/pino-pretty": { - "version": "4.7.5", - "resolved": "https://registry.npmjs.org/@types/pino-pretty/-/pino-pretty-4.7.5.tgz", - "integrity": "sha512-rfHe6VIknk14DymxGqc9maGsRe8/HQSvM2u46EAz2XrS92qsAJnW16dpdFejBuZKD8cRJX6Aw6uVZqIQctMpAg==", - "license": "MIT", - "peer": true, - "dependencies": { - "@types/node": "*", - "@types/pino": "6.3" - } - }, - "node_modules/@types/pino-std-serializers": { - "version": "2.4.1", - "resolved": "https://registry.npmjs.org/@types/pino-std-serializers/-/pino-std-serializers-2.4.1.tgz", - "integrity": "sha512-17XcksO47M24IVTVKPeAByWUd3Oez7EbIjXpSbzMPhXVzgjGtrOa49gKBwxH9hb8dKv58OelsWQ+A1G1l9S3wQ==", - "license": "MIT", - "peer": true, - "dependencies": { - "@types/node": "*" - } - }, "node_modules/@types/pluralize": { "version": "0.0.33", "resolved": "https://registry.npmjs.org/@types/pluralize/-/pluralize-0.0.33.tgz", @@ -6092,9 +3631,9 @@ } }, "node_modules/@types/qs": { - "version": "6.9.17", - "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.17.tgz", - "integrity": "sha512-rX4/bPcfmvxHDv0XjfJELTTr+iB+tn032nPILqHm5wbthUUUuVtNGGqzhya9XUxjTP8Fpr0qYgSZZKxGY++svQ==", + "version": "6.9.18", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.18.tgz", + "integrity": "sha512-kK7dgTYDyGqS+e2Q4aK9X3D7q234CIZ1Bv0q/7Z5IwRDoADNU81xXJK/YVyLbLTZCoIwUoDoffFeF+p/eIklAA==", "license": "MIT" }, "node_modules/@types/range-parser": { @@ -6112,9 +3651,9 @@ "license": "MIT" }, "node_modules/@types/semver": { - "version": "7.5.8", - "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.8.tgz", - "integrity": "sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==", + "version": "7.7.0", + "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.7.0.tgz", + "integrity": "sha512-k107IF4+Xr7UHjwDc7Cfd6PRQfbdkiRabXGRjo07b4WyPahFBZCZ1sE+BNxYIJPPg73UkfOsVOLwqVc/6ETrIA==", "dev": true, "license": "MIT" }, @@ -6182,7 +3721,7 @@ "version": "8.1.9", "resolved": "https://registry.npmjs.org/@types/superagent/-/superagent-8.1.9.tgz", "integrity": "sha512-pTVjI73witn+9ILmoJdajHGW2jkSaOzhiFYF1Rd3EQ94kymLqB9PjD9ISg7WaALC7+dCHT0FGe9T2LktLq/3GQ==", - "devOptional": true, + "dev": true, "license": "MIT", "dependencies": { "@types/cookiejar": "^2.1.5", @@ -6192,9 +3731,9 @@ } }, "node_modules/@types/supertest": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/@types/supertest/-/supertest-6.0.2.tgz", - "integrity": "sha512-137ypx2lk/wTQbW6An6safu9hXmajAifU/s7szAHLN/FeIm5w7yR0Wkl9fdJMRSHwOn4HLAI0DaB2TOORuhPDg==", + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/@types/supertest/-/supertest-6.0.3.tgz", + "integrity": "sha512-8WzXq62EXFhJ7QsH3Ocb/iKQ/Ty9ZVWnVzoTKc9tyyFRRF3a74Tk2+TLFgaFFw364Ere+npzHKEJ6ga2LzIL7w==", "dev": true, "license": "MIT", "dependencies": { @@ -6203,38 +3742,21 @@ } }, "node_modules/@types/validator": { - "version": "13.12.2", - "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.12.2.tgz", - "integrity": "sha512-6SlHBzUW8Jhf3liqrGGXyTJSIFe4nqlJ5A5KaMZ2l/vbM3Wh3KSybots/wfWVzNLK4D1NZluDlSQIbIEPx6oyA==", + "version": "13.15.0", + "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.15.0.tgz", + "integrity": "sha512-nh7nrWhLr6CBq9ldtw0wx+z9wKnnv/uTVLA9g/3/TcOYxbpOSZE+MhKPmWqU+K0NvThjhv12uD8MuqijB0WzEA==", "license": "MIT" }, "node_modules/@types/ws": { - "version": "8.5.13", - "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.13.tgz", - "integrity": "sha512-osM/gWBTPKgHV8XkTunnegTRIsvF6owmf5w+JtAfOw472dptdm0dlGv4xCt6GwQRcC2XVOvvRE/0bAoQcL2QkA==", + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.18.1.tgz", + "integrity": "sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg==", "dev": true, "license": "MIT", "dependencies": { "@types/node": "*" } }, - "node_modules/@types/yargs": { - "version": "15.0.19", - "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-15.0.19.tgz", - "integrity": "sha512-2XUaGVmyQjgyAZldf0D0c14vvo/yv0MhQBSTJcejMMaitsn3nxCB6TmH4G0ZQf+uxROOa9mpanoSm8h6SG/1ZA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/yargs-parser": "*" - } - }, - "node_modules/@types/yargs-parser": { - "version": "21.0.3", - "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.3.tgz", - "integrity": "sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==", - "dev": true, - "license": "MIT" - }, "node_modules/@typescript-eslint/eslint-plugin": { "version": "5.62.0", "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-5.62.0.tgz", @@ -6270,6 +3792,19 @@ } } }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/@typescript-eslint/parser": { "version": "5.62.0", "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-5.62.0.tgz", @@ -6407,6 +3942,19 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/@typescript-eslint/typescript-estree/node_modules/slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -6468,6 +4016,19 @@ "node": ">=4.0" } }, + "node_modules/@typescript-eslint/utils/node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/@typescript-eslint/visitor-keys": { "version": "5.62.0", "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-5.62.0.tgz", @@ -6487,33 +4048,31 @@ } }, "node_modules/@ungap/structured-clone": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.1.tgz", - "integrity": "sha512-fEzPV3hSkSMltkw152tJKNARhOupqbH96MZWyRjNaYZOMIzbrTeQDG+MTc6Mr2pgzFQzFxAfmhGDNP5QK++2ZA==", + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.3.0.tgz", + "integrity": "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g==", "dev": true, "license": "ISC" }, "node_modules/@vavite/multibuild": { - "version": "4.1.3", - "resolved": "https://registry.npmjs.org/@vavite/multibuild/-/multibuild-4.1.3.tgz", - "integrity": "sha512-V+6mskWf4GMQVb53w2fdJ5aR+zVkzpuCE9q3lDDo0v8AHjQApOeXydj/5rTERIFkO46yNHmr3insg2I/tC0TtA==", + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/@vavite/multibuild/-/multibuild-5.1.0.tgz", + "integrity": "sha512-xhJS6oAhQqDCRFFmpZWNCcAJw7145pvlfKX/IOCQX7oqulbw9amH9rdrNXmwz79UeYgOwxXpWfEavyYTPLY1KQ==", "license": "MIT", - "peer": true, "dependencies": { - "@types/node": "^18.19.50", + "@types/node": "^18.19.67", "cac": "^6.7.14", - "picocolors": "^1.1.0" + "picocolors": "^1.1.1" }, "peerDependencies": { - "vite": "^2.8.1 || 3 || 4 || 5" + "vite": "^2.8.1 || 3 || 4 || 5 || 6" } }, "node_modules/@vavite/multibuild/node_modules/@types/node": { - "version": "18.19.70", - "resolved": "https://registry.npmjs.org/@types/node/-/node-18.19.70.tgz", - "integrity": "sha512-RE+K0+KZoEpDUbGGctnGdkrLFwi1eYKTlIHNl2Um98mUkGsm1u2Ff6Ltd0e8DktTtC98uy7rSj+hO8t/QuLoVQ==", + "version": "18.19.87", + "resolved": "https://registry.npmjs.org/@types/node/-/node-18.19.87.tgz", + "integrity": "sha512-OIAAu6ypnVZHmsHCeJ+7CCSub38QNBS9uceMQeg7K5Ur0Jr+wG9wEOEvvMbhp09pxD5czIUy/jND7s7Tb6Nw7A==", "license": "MIT", - "peer": true, "dependencies": { "undici-types": "~5.26.4" } @@ -6522,37 +4081,50 @@ "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", - "peer": true + "license": "MIT" }, "node_modules/@vinejs/compiler": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/@vinejs/compiler/-/compiler-2.5.1.tgz", - "integrity": "sha512-efiO/SCQSMCqz6LDZTI4R3Ceq1ik3K2IqefEbbch+ko4dZncaYmQWJpX/fXVwgmO78jTZuerzD4I2WphPJUCwg==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@vinejs/compiler/-/compiler-3.0.0.tgz", + "integrity": "sha512-v9Lsv59nR56+bmy2p0+czjZxsLHwaibJ+SV5iK9JJfehlJMa501jUJQqqz4X/OqKXrxtE3uTQmSqjUqzF3B2mw==", "license": "MIT", "engines": { "node": ">=18.0.0" } }, "node_modules/@vinejs/vine": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/@vinejs/vine/-/vine-2.1.0.tgz", - "integrity": "sha512-09aJ2OauxpblqiNqd8qC9RAzzm5SV6fTqZhE4e25j4cM7fmNoXRTjM7Oo8llFADMO4eSA44HqYEO3mkRRYdbYw==", + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@vinejs/vine/-/vine-3.0.1.tgz", + "integrity": "sha512-ZtvYkYpZOYdvbws3uaOAvTFuvFXoQGAtmzeiXu+XSMGxi5GVsODpoI9Xu9TplEMuD/5fmAtBbKb9cQHkWkLXDQ==", "license": "MIT", "dependencies": { - "@poppinss/macroable": "^1.0.2", - "@types/validator": "^13.11.9", - "@vinejs/compiler": "^2.5.0", + "@poppinss/macroable": "^1.0.4", + "@types/validator": "^13.12.2", + "@vinejs/compiler": "^3.0.0", "camelcase": "^8.0.0", - "dayjs": "^1.11.11", + "dayjs": "^1.11.13", "dlv": "^1.1.3", "normalize-url": "^8.0.1", - "validator": "^13.11.0" + "validator": "^13.12.0" }, "engines": { "node": ">=18.16.0" } }, + "node_modules/@vitejs/plugin-vue": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-vue/-/plugin-vue-5.2.3.tgz", + "integrity": "sha512-IYSLEQj4LgZZuoVpdSUCw3dIynTWQgPlaRP6iAvMle4My0HdYwr5g5wQAfwOeHQBmYwEkqF70nRpSilr6PoUDg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.0.0 || >=20.0.0" + }, + "peerDependencies": { + "vite": "^5.0.0 || ^6.0.0", + "vue": "^3.2.25" + } + }, "node_modules/@vue/compiler-core": { "version": "3.5.13", "resolved": "https://registry.npmjs.org/@vue/compiler-core/-/compiler-core-3.5.13.tgz", @@ -6604,11 +4176,40 @@ } }, "node_modules/@vue/devtools-api": { - "version": "6.6.4", - "resolved": "https://registry.npmjs.org/@vue/devtools-api/-/devtools-api-6.6.4.tgz", - "integrity": "sha512-sGhTPMuXqZ1rVOk32RylztWkfXTRhuS7vgAKv0zjqk8gbsHkJ7xfFf+jbySxt7tWObEJwyKaHMikV/WGDiQm8g==", + "version": "7.7.6", + "resolved": "https://registry.npmjs.org/@vue/devtools-api/-/devtools-api-7.7.6.tgz", + "integrity": "sha512-b2Xx0KvXZObePpXPYHvBRRJLDQn5nhKjXh7vUhMEtWxz1AYNFOVIsh5+HLP8xDGL7sy+Q7hXeUxPHB/KgbtsPw==", "dev": true, - "license": "MIT" + "license": "MIT", + "dependencies": { + "@vue/devtools-kit": "^7.7.6" + } + }, + "node_modules/@vue/devtools-kit": { + "version": "7.7.6", + "resolved": "https://registry.npmjs.org/@vue/devtools-kit/-/devtools-kit-7.7.6.tgz", + "integrity": "sha512-geu7ds7tem2Y7Wz+WgbnbZ6T5eadOvozHZ23Atk/8tksHMFOFylKi1xgGlQlVn0wlkEf4hu+vd5ctj1G4kFtwA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vue/devtools-shared": "^7.7.6", + "birpc": "^2.3.0", + "hookable": "^5.5.3", + "mitt": "^3.0.1", + "perfect-debounce": "^1.0.0", + "speakingurl": "^14.0.1", + "superjson": "^2.2.2" + } + }, + "node_modules/@vue/devtools-shared": { + "version": "7.7.6", + "resolved": "https://registry.npmjs.org/@vue/devtools-shared/-/devtools-shared-7.7.6.tgz", + "integrity": "sha512-yFEgJZ/WblEsojQQceuyK6FzpFDx4kqrz2ohInxNj5/DnhoX023upTv4OD6lNPLAA5LLkbwPVb10o/7b+Y4FVA==", + "dev": true, + "license": "MIT", + "dependencies": { + "rfdc": "^1.4.1" + } }, "node_modules/@vue/reactivity": { "version": "3.5.13", @@ -6836,56 +4437,6 @@ "@xtuc/long": "4.2.2" } }, - "node_modules/@webpack-cli/configtest": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/@webpack-cli/configtest/-/configtest-2.1.1.tgz", - "integrity": "sha512-wy0mglZpDSiSS0XHrVR+BAdId2+yxPSoJW8fsna3ZpYSlufjvxnP4YbKTCBZnNIcGN4r6ZPXV55X4mYExOfLmw==", - "dev": true, - "license": "MIT", - "peer": true, - "engines": { - "node": ">=14.15.0" - }, - "peerDependencies": { - "webpack": "5.x.x", - "webpack-cli": "5.x.x" - } - }, - "node_modules/@webpack-cli/info": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/@webpack-cli/info/-/info-2.0.2.tgz", - "integrity": "sha512-zLHQdI/Qs1UyT5UBdWNqsARasIA+AaF8t+4u2aS2nEpBQh2mWIVb8qAklq0eUENnC5mOItrIB4LiS9xMtph18A==", - "dev": true, - "license": "MIT", - "peer": true, - "engines": { - "node": ">=14.15.0" - }, - "peerDependencies": { - "webpack": "5.x.x", - "webpack-cli": "5.x.x" - } - }, - "node_modules/@webpack-cli/serve": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@webpack-cli/serve/-/serve-2.0.5.tgz", - "integrity": "sha512-lqaoKnRYBdo1UgDX8uF24AfGMifWK19TxPmM5FHc2vAGxrJ/qtyUyFBWoY1tISZdelsQ5fBcOusifo5o5wSJxQ==", - "dev": true, - "license": "MIT", - "peer": true, - "engines": { - "node": ">=14.15.0" - }, - "peerDependencies": { - "webpack": "5.x.x", - "webpack-cli": "5.x.x" - }, - "peerDependenciesMeta": { - "webpack-dev-server": { - "optional": true - } - } - }, "node_modules/@xtuc/ieee754": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz", @@ -6908,19 +4459,6 @@ "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", "license": "ISC" }, - "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==", - "dev": true, - "license": "MIT", - "dependencies": { - "event-target-shim": "^5.0.0" - }, - "engines": { - "node": ">=6.5" - } - }, "node_modules/abstract-logging": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/abstract-logging/-/abstract-logging-2.0.1.tgz", @@ -6941,9 +4479,9 @@ } }, "node_modules/acorn": { - "version": "8.14.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.0.tgz", - "integrity": "sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==", + "version": "8.14.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.1.tgz", + "integrity": "sha512-OvQ/2pUDKmgfCg++xsTX1wGxfTaszcHVcTctW4UJB4hibJx2HXxxO5UmVgyjMa+ZDsiaf5wWLXYpRWMmBI0QHg==", "license": "MIT", "bin": { "acorn": "bin/acorn" @@ -6975,20 +4513,6 @@ "node": ">=0.4.0" } }, - "node_modules/adjust-sourcemap-loader": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/adjust-sourcemap-loader/-/adjust-sourcemap-loader-4.0.0.tgz", - "integrity": "sha512-OXwN5b9pCUXNQHJpwwD2qP40byEmSgzj8B4ydSN0uMNYWiFmJ6x6KwUllMmfk8Rwu/HJDFR7U8ubsWBoN0Xp0A==", - "dev": true, - "license": "MIT", - "dependencies": { - "loader-utils": "^2.0.0", - "regex-parser": "^2.2.11" - }, - "engines": { - "node": ">=8.9" - } - }, "node_modules/agent-base": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", @@ -7060,17 +4584,6 @@ "dev": true, "license": "MIT" }, - "node_modules/ajv-keywords": { - "version": "3.5.2", - "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", - "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", - "dev": true, - "license": "MIT", - "peer": true, - "peerDependencies": { - "ajv": "^6.9.1" - } - }, "node_modules/ansi-colors": { "version": "4.1.3", "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.3.tgz", @@ -7118,16 +4631,18 @@ } }, "node_modules/ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", "license": "MIT", "dependencies": { - "color-convert": "^1.9.0" + "color-convert": "^2.0.1" }, "engines": { - "node": ">=4" + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" } }, "node_modules/any-promise": { @@ -7164,169 +4679,6 @@ "url": "https://github.com/sponsors/jonschlinkert" } }, - "node_modules/api-contract-validator": { - "version": "2.2.8", - "resolved": "https://registry.npmjs.org/api-contract-validator/-/api-contract-validator-2.2.8.tgz", - "integrity": "sha512-YM3rMcrIp8Thf/WWbVBXBGX793Mm3Phw2pn3VbJpiZkpeTCTtF10huKPrzQ2gSIaK5GjAhTRJMAOyf+rsS7MAw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "api-schema-builder": "^2.0.10", - "chalk": "^3.0.0", - "columnify": "^1.5.4", - "jest-diff": "^25.5.0", - "jest-matcher-utils": "^25.5.0", - "lodash.flatten": "^4.4.0", - "lodash.get": "^4.4.2", - "lodash.set": "^4.3.2", - "uri-js": "^4.4.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/api-contract-validator/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/api-contract-validator/node_modules/chalk": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", - "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/api-contract-validator/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/api-contract-validator/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/api-contract-validator/node_modules/diff-sequences": { - "version": "25.2.6", - "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-25.2.6.tgz", - "integrity": "sha512-Hq8o7+6GaZeoFjtpgvRBUknSXNeJiCx7V9Fr94ZMljNiCr9n9L8H8aJqgWOQiDDGdyn29fRNcDdRVJ5fdyihfg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/api-contract-validator/node_modules/jest-diff": { - "version": "25.5.0", - "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-25.5.0.tgz", - "integrity": "sha512-z1kygetuPiREYdNIumRpAHY6RXiGmp70YHptjdaxTWGmA085W3iCnXNx0DhflK3vwrKmrRWyY1wUpkPMVxMK7A==", - "dev": true, - "license": "MIT", - "dependencies": { - "chalk": "^3.0.0", - "diff-sequences": "^25.2.6", - "jest-get-type": "^25.2.6", - "pretty-format": "^25.5.0" - }, - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/api-contract-validator/node_modules/jest-get-type": { - "version": "25.2.6", - "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-25.2.6.tgz", - "integrity": "sha512-DxjtyzOHjObRM+sM1knti6or+eOgcGU4xVSb2HNP1TqO4ahsT+rqZg+nyqHWJSvWgKC5cG3QjGFBqxLghiF/Ig==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/api-contract-validator/node_modules/pretty-format": { - "version": "25.5.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-25.5.0.tgz", - "integrity": "sha512-kbo/kq2LQ/A/is0PQwsEHM7Ca6//bGPPvU6UnsdDRSKTWxT/ru/xb88v4BJf6a69H+uTytOEsTusT9ksd/1iWQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jest/types": "^25.5.0", - "ansi-regex": "^5.0.0", - "ansi-styles": "^4.0.0", - "react-is": "^16.12.0" - }, - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/api-contract-validator/node_modules/react-is": { - "version": "16.13.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", - "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/api-contract-validator/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/api-schema-builder": { - "version": "2.0.11", - "resolved": "https://registry.npmjs.org/api-schema-builder/-/api-schema-builder-2.0.11.tgz", - "integrity": "sha512-85zbwf8MtPWodhfnmQRW5YD/fuGR12FP+8TbcYai5wbRnoUmPYLftLSbp7NB6zQMPb61Gjz+ApPUSyTdcCos7g==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "ajv": "^6.12.6", - "clone-deep": "^4.0.1", - "decimal.js": "^10.3.1", - "js-yaml": "^3.14.1", - "json-schema-deref-sync": "^0.14.0", - "lodash.get": "^4.4.2", - "openapi-schema-validator": "^3.0.3", - "swagger-parser": "^10.0.3" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/aproba": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/aproba/-/aproba-2.0.0.tgz", @@ -7355,30 +4707,11 @@ "license": "MIT" }, "node_modules/argparse": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", - "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", - "license": "MIT", - "dependencies": { - "sprintf-js": "~1.0.2" - } - }, - "node_modules/array-buffer-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.2.tgz", - "integrity": "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==", + "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": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "is-array-buffer": "^3.0.5" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } + "license": "Python-2.0" }, "node_modules/array-flatten": { "version": "1.1.1", @@ -7397,28 +4730,6 @@ "node": ">=8" } }, - "node_modules/arraybuffer.prototype.slice": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.4.tgz", - "integrity": "sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "array-buffer-byte-length": "^1.0.1", - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "is-array-buffer": "^3.0.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/as-table": { "version": "1.0.55", "resolved": "https://registry.npmjs.org/as-table/-/as-table-1.0.55.tgz", @@ -7432,7 +4743,7 @@ "version": "2.0.6", "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==", - "devOptional": true, + "dev": true, "license": "MIT" }, "node_modules/assertion-error": { @@ -7480,9 +4791,9 @@ } }, "node_modules/autoprefixer": { - "version": "10.4.20", - "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.20.tgz", - "integrity": "sha512-XY25y5xSv/wEoqzDyXXME4AFfkZI0P23z6Fs3YgymDnKJkCGOnkL0iTxCa85UTqaSgfcqyf3UA6+c7wUvx/16g==", + "version": "10.4.21", + "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.21.tgz", + "integrity": "sha512-O+A6LWV5LDHSJD3LjHYoNi4VLsj/Whi7k6zG12xTYaU4cQ8oxQGckXNX8cRHK5yOZ/ppVHe0ZBXGzSV9jXdVbQ==", "dev": true, "funding": [ { @@ -7500,11 +4811,11 @@ ], "license": "MIT", "dependencies": { - "browserslist": "^4.23.3", - "caniuse-lite": "^1.0.30001646", + "browserslist": "^4.24.4", + "caniuse-lite": "^1.0.30001702", "fraction.js": "^4.3.7", "normalize-range": "^0.1.2", - "picocolors": "^1.0.1", + "picocolors": "^1.1.1", "postcss-value-parser": "^4.2.0" }, "bin": { @@ -7517,22 +4828,6 @@ "postcss": "^8.1.0" } }, - "node_modules/available-typed-arrays": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", - "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "possible-typed-array-names": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/aws4": { "version": "1.13.2", "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.13.2.tgz", @@ -7540,99 +4835,14 @@ "license": "MIT" }, "node_modules/axios": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.4.tgz", - "integrity": "sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==", + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.9.0.tgz", + "integrity": "sha512-re4CqKTJaURpzbLHtIi6XpDv20/CnpXOtjRY5/CU32L8gU8ek9UIivcfvSWvmKEngmVbrUtPpdDwWDWL7DNHvg==", "license": "MIT", "dependencies": { - "follow-redirects": "^1.14.0" - } - }, - "node_modules/babel-loader": { - "version": "9.2.1", - "resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-9.2.1.tgz", - "integrity": "sha512-fqe8naHt46e0yIdkjUZYqddSXfej3AHajX+CSO5X7oy0EmPc6o5Xh+RClNoHjnieWz9AW4kZxW9yyFMhVB1QLA==", - "dev": true, - "license": "MIT", - "dependencies": { - "find-cache-dir": "^4.0.0", - "schema-utils": "^4.0.0" - }, - "engines": { - "node": ">= 14.15.0" - }, - "peerDependencies": { - "@babel/core": "^7.12.0", - "webpack": ">=5" - } - }, - "node_modules/babel-loader/node_modules/find-cache-dir": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-4.0.0.tgz", - "integrity": "sha512-9ZonPT4ZAK4a+1pUPVPZJapbi7O5qbbJPdYw/NOQWZZbVLdDTYM3A4R9z/DpAM08IDaFGsvPgiGZ82WEwUDWjg==", - "dev": true, - "license": "MIT", - "dependencies": { - "common-path-prefix": "^3.0.0", - "pkg-dir": "^7.0.0" - }, - "engines": { - "node": ">=14.16" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/babel-plugin-polyfill-corejs2": { - "version": "0.4.12", - "resolved": "https://registry.npmjs.org/babel-plugin-polyfill-corejs2/-/babel-plugin-polyfill-corejs2-0.4.12.tgz", - "integrity": "sha512-CPWT6BwvhrTO2d8QVorhTCQw9Y43zOu7G9HigcfxvepOU6b8o3tcWad6oVgZIsZCTt42FFv97aA7ZJsbM4+8og==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/compat-data": "^7.22.6", - "@babel/helper-define-polyfill-provider": "^0.6.3", - "semver": "^6.3.1" - }, - "peerDependencies": { - "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" - } - }, - "node_modules/babel-plugin-polyfill-corejs2/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/babel-plugin-polyfill-corejs3": { - "version": "0.10.6", - "resolved": "https://registry.npmjs.org/babel-plugin-polyfill-corejs3/-/babel-plugin-polyfill-corejs3-0.10.6.tgz", - "integrity": "sha512-b37+KR2i/khY5sKmWNVQAnitvquQbNdWy6lJdsr0kmquCKEEUgMKK4SboVM3HtfnZilfjr4MMQ7vY58FVWDtIA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-define-polyfill-provider": "^0.6.2", - "core-js-compat": "^3.38.0" - }, - "peerDependencies": { - "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" - } - }, - "node_modules/babel-plugin-polyfill-regenerator": { - "version": "0.6.3", - "resolved": "https://registry.npmjs.org/babel-plugin-polyfill-regenerator/-/babel-plugin-polyfill-regenerator-0.6.3.tgz", - "integrity": "sha512-LiWSbl4CRSIa5x/JAU6jZiG9eit9w6mz+yVMFwDE83LAWvt0AfGBoZ7HS/mkhrKuh2ZlzfVZYKoLjXdqw6Yt7Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/helper-define-polyfill-provider": "^0.6.3" - }, - "peerDependencies": { - "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + "follow-redirects": "^1.15.6", + "form-data": "^4.0.0", + "proxy-from-env": "^1.1.0" } }, "node_modules/babel-preset-typescript-vue3": { @@ -7660,27 +4870,6 @@ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", "license": "MIT" }, - "node_modules/base64-js": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", - "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/basic-auth": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", @@ -7693,6 +4882,12 @@ "node": ">= 0.8" } }, + "node_modules/basic-auth/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "license": "MIT" + }, "node_modules/batch": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/batch/-/batch-0.6.1.tgz", @@ -7720,16 +4915,6 @@ "integrity": "sha512-V/Hy/X9Vt7f3BbPJEi8BdVFMByHi+jNXrYkW3huaybV/kQ0KJg0Y6PkEMbn+zeT+i+SiKZ/HMqJGIIt4LZDqNQ==", "license": "MIT" }, - "node_modules/big.js": { - "version": "5.2.2", - "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", - "integrity": "sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": "*" - } - }, "node_modules/binary-extensions": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", @@ -7743,6 +4928,16 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/birpc": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/birpc/-/birpc-2.3.0.tgz", + "integrity": "sha512-ijbtkn/F3Pvzb6jHypHRyve2QApOCZDR25D/VnkY2G/lBNcXCTsnsCxgY4k4PkVB7zfwzYbY3O9Lcqe3xufS5g==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/antfu" + } + }, "node_modules/body-parser": { "version": "1.20.3", "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz", @@ -7778,6 +4973,19 @@ "ms": "2.0.0" } }, + "node_modules/body-parser/node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/body-parser/node_modules/ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", @@ -7801,6 +5009,22 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/body-parser/node_modules/raw-body": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", + "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", + "dev": true, + "license": "MIT", + "dependencies": { + "bytes": "3.1.2", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, "node_modules/bonjour-service": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/bonjour-service/-/bonjour-service-1.3.0.tgz", @@ -7812,13 +5036,6 @@ "multicast-dns": "^7.2.5" } }, - "node_modules/boolbase": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", - "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==", - "dev": true, - "license": "ISC" - }, "node_modules/brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", @@ -7842,9 +5059,9 @@ } }, "node_modules/browserslist": { - "version": "4.24.3", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.24.3.tgz", - "integrity": "sha512-1CPmv8iobE2fyRMV97dAcMVegvvWKxmq94hkLiAkUGwKVTyDLw33K+ZxiFrREKmmps4rIw6grcCFCnTMSZ/YiA==", + "version": "4.24.4", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.24.4.tgz", + "integrity": "sha512-KDi1Ny1gSePi1vm0q4oxSF8b4DR44GF4BbmS2YdhPLOEqd8pDviZOGH/GsmRwoWJ2+5Lr085X7naowMwKHDG1A==", "dev": true, "funding": [ { @@ -7874,37 +5091,13 @@ "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" } }, - "node_modules/buffer": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-6.0.3.tgz", - "integrity": "sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==", - "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": { - "base64-js": "^1.3.1", - "ieee754": "^1.2.1" - } - }, "node_modules/buffer-from": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", "devOptional": true, - "license": "MIT" + "license": "MIT", + "peer": true }, "node_modules/bundle-name": { "version": "4.1.0", @@ -7936,7 +5129,6 @@ "resolved": "https://registry.npmjs.org/cac/-/cac-6.7.14.tgz", "integrity": "sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==", "license": "MIT", - "peer": true, "engines": { "node": ">=8" } @@ -7968,29 +5160,10 @@ "node": ">=18" } }, - "node_modules/call-bind": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.8.tgz", - "integrity": "sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind-apply-helpers": "^1.0.0", - "es-define-property": "^1.0.0", - "get-intrinsic": "^1.2.4", - "set-function-length": "^1.2.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/call-bind-apply-helpers": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.1.tgz", - "integrity": "sha512-BhYE+WDaywFg2TBWYNXAE+8B1ATnThNBqXHP5nQu0jWJdVvY2hvkpyB3qOmtmDePiS5/BDQ8wASEWGMWRG148g==", + "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", @@ -8001,13 +5174,13 @@ } }, "node_modules/call-bound": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.3.tgz", - "integrity": "sha512-YTd+6wGlNlPxSuri7Y6X8tY2dmm12UMH66RpKMhiX6rsk5wXXnYgbUcOt8kiS31/AjfoTOvCsE+w8nZQLQnzHA==", + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", + "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", "license": "MIT", "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "get-intrinsic": "^1.2.6" + "call-bind-apply-helpers": "^1.0.2", + "get-intrinsic": "^1.3.0" }, "engines": { "node": ">= 0.4" @@ -8016,13 +5189,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/call-me-maybe": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-me-maybe/-/call-me-maybe-1.0.2.tgz", - "integrity": "sha512-HpX65o1Hnr9HH25ojC1YGs7HCQLq0GCOibSaWER0eNpgJ/Z1MZv2mTc7+xh6WOPxbRVcmgbv4hGU+uSQ/2xFZQ==", - "dev": true, - "license": "MIT" - }, "node_modules/callsites": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", @@ -8033,16 +5199,6 @@ "node": ">=6" } }, - "node_modules/camel-case": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-4.1.2.tgz", - "integrity": "sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw==", - "license": "MIT", - "dependencies": { - "pascal-case": "^3.1.2", - "tslib": "^2.0.3" - } - }, "node_modules/camelcase": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-8.0.0.tgz", @@ -8065,23 +5221,10 @@ "node": ">= 6" } }, - "node_modules/caniuse-api": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/caniuse-api/-/caniuse-api-3.0.0.tgz", - "integrity": "sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.0.0", - "caniuse-lite": "^1.0.0", - "lodash.memoize": "^4.1.2", - "lodash.uniq": "^4.5.0" - } - }, "node_modules/caniuse-lite": { - "version": "1.0.30001690", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001690.tgz", - "integrity": "sha512-5ExiE3qQN6oF8Clf8ifIDcMRCRE/dMGcETG/XGMD8/XiXm6HXQgQTh1yZYLXXpSOsEUlJm1Xr7kGULZTuGtP/w==", + "version": "1.0.30001716", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001716.tgz", + "integrity": "sha512-49/c1+x3Kwz7ZIWt+4DvK3aMJy9oYXXG6/97JKsnjdCk/6n9vVyWL8NAwVt95Lwt9eigI10Hl782kDfZUUlRXw==", "dev": true, "funding": [ { @@ -8099,21 +5242,10 @@ ], "license": "CC-BY-4.0" }, - "node_modules/capital-case": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/capital-case/-/capital-case-1.0.4.tgz", - "integrity": "sha512-ds37W8CytHgwnhGGTi88pcPyR15qoNkOpYwmMMfnWqqWgESapLqvDx6huFjQ5vqWSn2Z06173XNA7LtMOeUh1A==", - "license": "MIT", - "dependencies": { - "no-case": "^3.0.4", - "tslib": "^2.0.3", - "upper-case-first": "^2.0.2" - } - }, "node_modules/case-anything": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/case-anything/-/case-anything-3.1.0.tgz", - "integrity": "sha512-rRYnn5Elur8RuNHKoJ2b0tgn+pjYxL7BzWom+JZ7NKKn1lt/yGV/tUNwOovxYa9l9VL5hnXQdMc+mENbhJzosQ==", + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/case-anything/-/case-anything-3.1.2.tgz", + "integrity": "sha512-wljhAjDDIv/hM2FzgJnYQg90AWmZMNtESCjTeLH680qTzdo0nErlCxOmgzgX4ZsZAtIvqHyD87ES8QyriXB+BQ==", "license": "MIT", "engines": { "node": ">=18" @@ -8123,9 +5255,9 @@ } }, "node_modules/chai": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/chai/-/chai-5.1.2.tgz", - "integrity": "sha512-aGtmf24DW6MLHHG5gCx4zaI3uBq3KRtxeVs0DjFH6Z0rDNbsvTxFASFvdj79pxjxZ8/5u3PIiN3IwEIQkiiuPw==", + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/chai/-/chai-5.2.0.tgz", + "integrity": "sha512-mCuXncKXk5iCLhfhwTc0izo0gtEmpz5CtG2y8GiOINBlMVS6v8TMRc5TaLWKS6692m9+dVVfzgeVxR5UxWHTYw==", "dev": true, "license": "MIT", "dependencies": { @@ -8140,77 +5272,39 @@ } }, "node_modules/chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", - "dev": true, + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "devOptional": true, "license": "MIT", "dependencies": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" }, "engines": { - "node": ">=4" - } - }, - "node_modules/chalk/node_modules/has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=4" + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" } }, "node_modules/chalk/node_modules/supports-color": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", - "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", - "dev": true, + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "devOptional": true, "license": "MIT", "dependencies": { - "has-flag": "^3.0.0" + "has-flag": "^4.0.0" }, "engines": { - "node": ">=4" - } - }, - "node_modules/change-case": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/change-case/-/change-case-4.1.2.tgz", - "integrity": "sha512-bSxY2ws9OtviILG1EiY5K7NNxkqg/JnRnFxLtKQ96JaviiIxi7djMrSd0ECT9AC+lttClmYwKw53BWpOMblo7A==", - "license": "MIT", - "dependencies": { - "camel-case": "^4.1.2", - "capital-case": "^1.0.4", - "constant-case": "^3.0.4", - "dot-case": "^3.0.4", - "header-case": "^2.0.4", - "no-case": "^3.0.4", - "param-case": "^3.0.4", - "pascal-case": "^3.1.2", - "path-case": "^3.0.4", - "sentence-case": "^3.0.4", - "snake-case": "^3.0.4", - "tslib": "^2.0.3" - } - }, - "node_modules/charenc": { - "version": "0.0.2", - "resolved": "https://registry.npmjs.org/charenc/-/charenc-0.0.2.tgz", - "integrity": "sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==", - "dev": true, - "license": "BSD-3-Clause", - "engines": { - "node": "*" + "node": ">=8" } }, "node_modules/chart.js": { - "version": "4.4.7", - "resolved": "https://registry.npmjs.org/chart.js/-/chart.js-4.4.7.tgz", - "integrity": "sha512-pwkcKfdzTMAU/+jNosKhNL2bHtJc/sSmYgVbuGTEDhzkrhmyihmP7vUc/5ZK9WopidMDHNe3Wm7jOd/WhuHWuw==", + "version": "4.4.9", + "resolved": "https://registry.npmjs.org/chart.js/-/chart.js-4.4.9.tgz", + "integrity": "sha512-EyZ9wWKgpAU0fLJ43YAEIF8sr5F2W3LqbS40ZJyHIner2lY14ufqv2VMp69MAiZ2rpwxEUxEhIH/0U3xyRynxg==", "dev": true, "license": "MIT", "dependencies": { @@ -8275,22 +5369,6 @@ "node": ">=6.0" } }, - "node_modules/ci-info": { - "version": "3.9.0", - "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", - "integrity": "sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/sibiraj-s" - } - ], - "license": "MIT", - "engines": { - "node": ">=8" - } - }, "node_modules/clamscan": { "version": "2.4.0", "resolved": "https://registry.npmjs.org/clamscan/-/clamscan-2.4.0.tgz", @@ -8404,39 +5482,6 @@ "wrap-ansi": "^6.2.0" } }, - "node_modules/cliui/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==", - "license": "MIT", - "dependencies": { - "color-convert": "^2.0.1" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/cliui/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==", - "license": "MIT", - "dependencies": { - "color-name": "~1.1.4" - }, - "engines": { - "node": ">=7.0.0" - } - }, - "node_modules/cliui/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==", - "license": "MIT" - }, "node_modules/cliui/node_modules/emoji-regex": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", @@ -8480,31 +5525,6 @@ "node": ">=8" } }, - "node_modules/clone": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", - "integrity": "sha512-3Pe/CF1Nn94hyhIYpjtiLhdCoEoz0DqQ+988E9gmeEdQZlojxnOb74wctFyuwWQHzqyf9X7C7MG8juUpqBJT8w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.8" - } - }, - "node_modules/clone-deep": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz", - "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-plain-object": "^2.0.4", - "kind-of": "^6.0.2", - "shallow-clone": "^3.0.0" - }, - "engines": { - "node": ">=6" - } - }, "node_modules/cluster-key-slot": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/cluster-key-slot/-/cluster-key-slot-1.1.2.tgz", @@ -8514,13 +5534,6 @@ "node": ">=0.10.0" } }, - "node_modules/co-compose": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/co-compose/-/co-compose-7.0.3.tgz", - "integrity": "sha512-ZHLSLzeBXe5yaEyIHo9T92uVrbsBRLMXlG0G4/pSm9f6148l4mJTr1cii8Jl9ce+mbLmW5XqHURPC7gZFJNeZA==", - "license": "MIT", - "peer": true - }, "node_modules/code-block-writer": { "version": "13.0.3", "resolved": "https://registry.npmjs.org/code-block-writer/-/code-block-writer-13.0.3.tgz", @@ -8529,20 +5542,21 @@ "license": "MIT" }, "node_modules/color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", "license": "MIT", "dependencies": { - "color-name": "1.1.3" + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" } }, "node_modules/color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", - "dev": true, + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "license": "MIT" }, "node_modules/color-support": { @@ -8554,33 +5568,12 @@ "color-support": "bin.js" } }, - "node_modules/colord": { - "version": "2.9.3", - "resolved": "https://registry.npmjs.org/colord/-/colord-2.9.3.tgz", - "integrity": "sha512-jeC1axXpnb0/2nn/Y1LPuLdgXBLH7aDcHu4KEKfqw3CUhX7ZpfBSlPKyqXE6btIgEzfWtrX3/tyBCaCvXvMkOw==", - "dev": true, - "license": "MIT" - }, "node_modules/colorette": { "version": "2.0.19", "resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.19.tgz", "integrity": "sha512-3tlv/dIP7FWvj3BsbHrGLJ6l/oKh1O3TcgBqMn+yyCagOxc23fyzDS6HypQbgxWbkpDnf52p1LuR4eWDQ/K9WQ==", "license": "MIT" }, - "node_modules/columnify": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/columnify/-/columnify-1.6.0.tgz", - "integrity": "sha512-lomjuFZKfM6MSAnV9aCZC9sc0qGbmZdfygNv+nCpqVkSKdCxCklLtd16O0EILGkImHw9ZpHkAnHaB+8Zxq5W6Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "strip-ansi": "^6.0.1", - "wcwidth": "^1.0.0" - }, - "engines": { - "node": ">=8.0.0" - } - }, "node_modules/combined-stream": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", @@ -8613,7 +5606,7 @@ "version": "1.3.1", "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.1.tgz", "integrity": "sha512-T0+barUSQRTUQASh8bx02dl+DhF54GtIDY13Y3m9oWTklKbb3Wv974meRpeZ3lp1JpLVECWWNHC4vaG2XHXouQ==", - "devOptional": true, + "dev": true, "license": "MIT", "funding": { "url": "https://github.com/sponsors/sindresorhus" @@ -8633,9 +5626,9 @@ } }, "node_modules/compression": { - "version": "1.7.5", - "resolved": "https://registry.npmjs.org/compression/-/compression-1.7.5.tgz", - "integrity": "sha512-bQJ0YRck5ak3LgtnpKkiabX5pNF7tMUh1BSy2ZBOTh0Dim0BUu6aPPwByIns6/A5Prh8PufSPerMDUklpzes2Q==", + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/compression/-/compression-1.8.0.tgz", + "integrity": "sha512-k6WLKfunuqCYD3t6AsuPGvQWaKwuLLh2/xHNcX4qE+vIfDNXpSqnrhwA7O53R7WVQUnt8dVAIW+YHr7xTgOgGA==", "dev": true, "license": "MIT", "dependencies": { @@ -8678,27 +5671,6 @@ "node": ">= 0.6" } }, - "node_modules/compression/node_modules/safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "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/concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", @@ -8715,33 +5687,12 @@ "node": ">=0.8" } }, - "node_modules/consola": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/consola/-/consola-3.3.3.tgz", - "integrity": "sha512-Qil5KwghMzlqd51UXM0b6fyaGHtOC22scxrwrz4A2882LyUMwQjnvaedN1HAeXzphspQ6CpHkzMAWxBTUruDLg==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^14.18.0 || >=16.10.0" - } - }, "node_modules/console-control-strings": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", "integrity": "sha512-ty/fTekppD2fIwRvnZAVdeOiGd1c7YXEixbgJTNzqcxJWKQnjJ/V1bNEEE6hygpM3WjwHFUVK6HTjWSzV4a8sQ==", "license": "ISC" }, - "node_modules/constant-case": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/constant-case/-/constant-case-3.0.4.tgz", - "integrity": "sha512-I2hSBi7Vvs7BEuJDr5dDHfzb/Ruj3FyvFyh7KLilAjNQw3Be+xgqUBA2W6scVEcL0hL1dwPRtIqEPVUCKkSsyQ==", - "license": "MIT", - "dependencies": { - "no-case": "^3.0.4", - "tslib": "^2.0.3", - "upper-case": "^2.0.2" - } - }, "node_modules/content-disposition": { "version": "0.5.4", "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", @@ -8754,26 +5705,6 @@ "node": ">= 0.6" } }, - "node_modules/content-disposition/node_modules/safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "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/content-type": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", @@ -8802,16 +5733,16 @@ "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" + "license": "MIT", + "peer": true }, "node_modules/cookie": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.5.0.tgz", - "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-1.0.2.tgz", + "integrity": "sha512-9Kr/j4O16ISv8zBBhJoi4bXOYNTkFLOqSL3UDB0njXxCXNezjeyVrJyGOWtgfs/q2km1gwBcfH8q1yEGoMYunA==", "license": "MIT", - "peer": true, "engines": { - "node": ">= 0.6" + "node": ">=18" } }, "node_modules/cookie-signature": { @@ -8825,9 +5756,25 @@ "version": "2.1.4", "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.4.tgz", "integrity": "sha512-LDx6oHrK+PhzLKJU9j5S7/Y3jM/mUHvD/DeI1WQmJn652iPC5Y4TBzC9l+5OMOXlyTTA+SmVUPm0HQUwpD5Jqw==", - "devOptional": true, + "dev": true, "license": "MIT" }, + "node_modules/copy-anything": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/copy-anything/-/copy-anything-3.0.5.tgz", + "integrity": "sha512-yCEafptTtb4bk7GLEQoM8KVJpxAfdBJYaXyzQEgQQQgYrZiDp8SJmGKlYza6CYjEDNstAdNdKA3UuoULlEbS6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-what": "^4.1.8" + }, + "engines": { + "node": ">=12.13" + }, + "funding": { + "url": "https://github.com/sponsors/mesqueeb" + } + }, "node_modules/copy-file": { "version": "11.0.0", "resolved": "https://registry.npmjs.org/copy-file/-/copy-file-11.0.0.tgz", @@ -8845,20 +5792,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/core-js-compat": { - "version": "3.40.0", - "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.40.0.tgz", - "integrity": "sha512-0XEDpr5y5mijvw8Lbc6E5AkjrHfp7eEoPlu36SWeAbcL8fn1G1ANe8DBlo2XoNN89oVpxWwOjYIPVzR4ZvsKCQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.24.3" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/core-js" - } - }, "node_modules/core-util-is": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", @@ -8893,26 +5826,6 @@ } } }, - "node_modules/cosmiconfig/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/cosmiconfig/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/cpy": { "version": "11.1.0", "resolved": "https://registry.npmjs.org/cpy/-/cpy-11.1.0.tgz", @@ -8968,23 +5881,6 @@ "node": ">= 8" } }, - "node_modules/crypt": { - "version": "0.0.2", - "resolved": "https://registry.npmjs.org/crypt/-/crypt-0.0.2.tgz", - "integrity": "sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow==", - "dev": true, - "license": "BSD-3-Clause", - "engines": { - "node": "*" - } - }, - "node_modules/crypto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/crypto/-/crypto-1.0.1.tgz", - "integrity": "sha512-VxBKmeNcqQdiUQUW2Tzq0t377b54N2bMtXO/qiLa+6eRRmmC4qT3D4OnTGoT/U6O9aklQ/jTwbOtRMTTY8G0Ig==", - "deprecated": "This package is no longer supported. It's now a built-in Node module. If you've depended on crypto, you should switch to the one that's built-in.", - "license": "ISC" - }, "node_modules/csrf": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/csrf/-/csrf-3.1.0.tgz", @@ -8999,176 +5895,6 @@ "node": ">= 0.8" } }, - "node_modules/css-declaration-sorter": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/css-declaration-sorter/-/css-declaration-sorter-7.2.0.tgz", - "integrity": "sha512-h70rUM+3PNFuaBDTLe8wF/cdWu+dOZmb7pJt8Z2sedYbAcQVQV/tEchueg3GWxwqS0cxtbxmaHEdkNACqcvsow==", - "dev": true, - "license": "ISC", - "engines": { - "node": "^14 || ^16 || >=18" - }, - "peerDependencies": { - "postcss": "^8.0.9" - } - }, - "node_modules/css-loader": { - "version": "7.1.2", - "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-7.1.2.tgz", - "integrity": "sha512-6WvYYn7l/XEGN8Xu2vWFt9nVzrCn39vKyTEFf/ExEyoksJjjSZV/0/35XPlMbpnr6VGhZIUg5yJrL8tGfes/FA==", - "dev": true, - "license": "MIT", - "dependencies": { - "icss-utils": "^5.1.0", - "postcss": "^8.4.33", - "postcss-modules-extract-imports": "^3.1.0", - "postcss-modules-local-by-default": "^4.0.5", - "postcss-modules-scope": "^3.2.0", - "postcss-modules-values": "^4.0.0", - "postcss-value-parser": "^4.2.0", - "semver": "^7.5.4" - }, - "engines": { - "node": ">= 18.12.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/webpack" - }, - "peerDependencies": { - "@rspack/core": "0.x || 1.x", - "webpack": "^5.27.0" - }, - "peerDependenciesMeta": { - "@rspack/core": { - "optional": true - }, - "webpack": { - "optional": true - } - } - }, - "node_modules/css-minimizer-webpack-plugin": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/css-minimizer-webpack-plugin/-/css-minimizer-webpack-plugin-7.0.0.tgz", - "integrity": "sha512-niy66jxsQHqO+EYbhPuIhqRQ1mNcNVUHrMnkzzir9kFOERJUaQDDRhh7dKDz33kBpkWMF9M8Vx0QlDbc5AHOsw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jridgewell/trace-mapping": "^0.3.25", - "cssnano": "^7.0.1", - "jest-worker": "^29.7.0", - "postcss": "^8.4.38", - "schema-utils": "^4.2.0", - "serialize-javascript": "^6.0.2" - }, - "engines": { - "node": ">= 18.12.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/webpack" - }, - "peerDependencies": { - "webpack": "^5.0.0" - }, - "peerDependenciesMeta": { - "@parcel/css": { - "optional": true - }, - "@swc/css": { - "optional": true - }, - "clean-css": { - "optional": true - }, - "csso": { - "optional": true - }, - "esbuild": { - "optional": true - }, - "lightningcss": { - "optional": true - } - } - }, - "node_modules/css-minimizer-webpack-plugin/node_modules/jest-worker": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-29.7.0.tgz", - "integrity": "sha512-eIz2msL/EzL9UFTFFx7jBTkeZfku0yUAyZZZmJ93H2TYEiroIx2PQjEXcwYtYl8zXCxb+PAmA2hLIt/6ZEkPHw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/node": "*", - "jest-util": "^29.7.0", - "merge-stream": "^2.0.0", - "supports-color": "^8.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/css-minimizer-webpack-plugin/node_modules/supports-color": { - "version": "8.1.1", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", - "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-flag": "^4.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/supports-color?sponsor=1" - } - }, - "node_modules/css-select": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/css-select/-/css-select-4.3.0.tgz", - "integrity": "sha512-wPpOYtnsVontu2mODhA19JrqWxNsfdatRKd64kmpRbQgh1KtItko5sTnEpPdpSaJszTOhEMlF/RPz28qj4HqhQ==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "boolbase": "^1.0.0", - "css-what": "^6.0.1", - "domhandler": "^4.3.1", - "domutils": "^2.8.0", - "nth-check": "^2.0.1" - }, - "funding": { - "url": "https://github.com/sponsors/fb55" - } - }, - "node_modules/css-tree": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-2.3.1.tgz", - "integrity": "sha512-6Fv1DV/TYw//QF5IzQdqsNDjx/wc8TrMBZsqjL9eW01tWb7R7k/mq+/VXfJCl7SoD5emsJop9cOByJZfs8hYIw==", - "dev": true, - "license": "MIT", - "dependencies": { - "mdn-data": "2.0.30", - "source-map-js": "^1.0.1" - }, - "engines": { - "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0" - } - }, - "node_modules/css-what": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/css-what/-/css-what-6.1.0.tgz", - "integrity": "sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==", - "dev": true, - "license": "BSD-2-Clause", - "engines": { - "node": ">= 6" - }, - "funding": { - "url": "https://github.com/sponsors/fb55" - } - }, "node_modules/cssesc": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", @@ -9182,201 +5908,18 @@ "node": ">=4" } }, - "node_modules/cssnano": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-7.0.6.tgz", - "integrity": "sha512-54woqx8SCbp8HwvNZYn68ZFAepuouZW4lTwiMVnBErM3VkO7/Sd4oTOt3Zz3bPx3kxQ36aISppyXj2Md4lg8bw==", - "dev": true, - "license": "MIT", - "dependencies": { - "cssnano-preset-default": "^7.0.6", - "lilconfig": "^3.1.2" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/cssnano" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/cssnano-preset-default": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/cssnano-preset-default/-/cssnano-preset-default-7.0.6.tgz", - "integrity": "sha512-ZzrgYupYxEvdGGuqL+JKOY70s7+saoNlHSCK/OGn1vB2pQK8KSET8jvenzItcY+kA7NoWvfbb/YhlzuzNKjOhQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.23.3", - "css-declaration-sorter": "^7.2.0", - "cssnano-utils": "^5.0.0", - "postcss-calc": "^10.0.2", - "postcss-colormin": "^7.0.2", - "postcss-convert-values": "^7.0.4", - "postcss-discard-comments": "^7.0.3", - "postcss-discard-duplicates": "^7.0.1", - "postcss-discard-empty": "^7.0.0", - "postcss-discard-overridden": "^7.0.0", - "postcss-merge-longhand": "^7.0.4", - "postcss-merge-rules": "^7.0.4", - "postcss-minify-font-values": "^7.0.0", - "postcss-minify-gradients": "^7.0.0", - "postcss-minify-params": "^7.0.2", - "postcss-minify-selectors": "^7.0.4", - "postcss-normalize-charset": "^7.0.0", - "postcss-normalize-display-values": "^7.0.0", - "postcss-normalize-positions": "^7.0.0", - "postcss-normalize-repeat-style": "^7.0.0", - "postcss-normalize-string": "^7.0.0", - "postcss-normalize-timing-functions": "^7.0.0", - "postcss-normalize-unicode": "^7.0.2", - "postcss-normalize-url": "^7.0.0", - "postcss-normalize-whitespace": "^7.0.0", - "postcss-ordered-values": "^7.0.1", - "postcss-reduce-initial": "^7.0.2", - "postcss-reduce-transforms": "^7.0.0", - "postcss-svgo": "^7.0.1", - "postcss-unique-selectors": "^7.0.3" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/cssnano-utils": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/cssnano-utils/-/cssnano-utils-5.0.0.tgz", - "integrity": "sha512-Uij0Xdxc24L6SirFr25MlwC2rCFX6scyUmuKpzI+JQ7cyqDEwD42fJ0xfB3yLfOnRDU5LKGgjQ9FA6LYh76GWQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/csso": { - "version": "5.0.5", - "resolved": "https://registry.npmjs.org/csso/-/csso-5.0.5.tgz", - "integrity": "sha512-0LrrStPOdJj+SPCCrGhzryycLjwcgUSHBtxNA8aIDxf0GLsRh1cKYhB00Gd1lDOS4yGH69+SNn13+TWbVHETFQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "css-tree": "~2.2.0" - }, - "engines": { - "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0", - "npm": ">=7.0.0" - } - }, - "node_modules/csso/node_modules/css-tree": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-2.2.1.tgz", - "integrity": "sha512-OA0mILzGc1kCOCSJerOeqDxDQ4HOh+G8NbOJFOTgOCzpw7fCBubk0fEyxp8AgOL/jvLgYA/uV0cMbe43ElF1JA==", - "dev": true, - "license": "MIT", - "dependencies": { - "mdn-data": "2.0.28", - "source-map-js": "^1.0.1" - }, - "engines": { - "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0", - "npm": ">=7.0.0" - } - }, - "node_modules/csso/node_modules/mdn-data": { - "version": "2.0.28", - "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.28.tgz", - "integrity": "sha512-aylIc7Z9y4yzHYAJNuESG3hfhC+0Ibp/MAMiaOZgNv4pmEdFyfZhhhny4MNiAfWdBQ1RQ2mfDWmM1x8SvGyp8g==", - "dev": true, - "license": "CC0-1.0" - }, "node_modules/csstype": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", "license": "MIT" }, - "node_modules/cuid": { - "version": "2.1.8", - "resolved": "https://registry.npmjs.org/cuid/-/cuid-2.1.8.tgz", - "integrity": "sha512-xiEMER6E7TlTPnDxrM4eRiC6TRgjNX9xzEZ5U/Se2YJKr7Mq4pJn/2XEHjl3STcSh96GmkHPcBXLES8M29wyyg==", - "deprecated": "Cuid and other k-sortable and non-cryptographic ids (Ulid, ObjectId, KSUID, all UUIDs) are all insecure. Use @paralleldrive/cuid2 instead.", - "license": "MIT" - }, - "node_modules/dag-map": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/dag-map/-/dag-map-1.0.2.tgz", - "integrity": "sha512-+LSAiGFwQ9dRnRdOeaj7g47ZFJcOUPukAP8J3A3fuZ1g9Y44BG+P1sgApjLXTQPOzC4+7S9Wr8kXsfpINM4jpw==", - "dev": true, - "license": "MIT" - }, "node_modules/data-uri-to-buffer": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-2.0.2.tgz", "integrity": "sha512-ND9qDTLc6diwj+Xe5cdAgVTbLVdXbtxTJRXRhli8Mowuaan+0EJOtdqJ0QCHNSSPyoXGx9HX2/VMnKeC34AChA==", "license": "MIT" }, - "node_modules/data-view-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.2.tgz", - "integrity": "sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/data-view-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.2.tgz", - "integrity": "sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/inspect-js" - } - }, - "node_modules/data-view-byte-offset": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.1.tgz", - "integrity": "sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/dateformat": { "version": "4.6.3", "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-4.6.3.tgz", @@ -9419,13 +5962,6 @@ "node": ">=0.10.0" } }, - "node_modules/decimal.js": { - "version": "10.4.3", - "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.4.3.tgz", - "integrity": "sha512-VBBaLc1MgL5XpzgIP7ny5Z6Nx3UrRkIViUkPUdtl9aya5amy3De1gsUUSB1g3+3sExYNjCAsAznmukyxCb1GRA==", - "dev": true, - "license": "MIT" - }, "node_modules/decompress-response": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", @@ -9545,29 +6081,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/defaults": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.4.tgz", - "integrity": "sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A==", - "dev": true, - "license": "MIT", - "dependencies": { - "clone": "^1.0.2" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/defaults/node_modules/clone": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", - "integrity": "sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.8" - } - }, "node_modules/defer-to-connect": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-2.0.1.tgz", @@ -9577,24 +6090,6 @@ "node": ">=10" } }, - "node_modules/define-data-property": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", - "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-define-property": "^1.0.0", - "es-errors": "^1.3.0", - "gopd": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/define-lazy-prop": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/define-lazy-prop/-/define-lazy-prop-3.0.0.tgz", @@ -9608,24 +6103,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/define-properties": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", - "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.0.1", - "has-property-descriptors": "^1.0.0", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/delayed-stream": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", @@ -9670,9 +6147,9 @@ } }, "node_modules/detect-libc": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz", - "integrity": "sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==", + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.4.tgz", + "integrity": "sha512-3UDv+G9CsCKO1WKMGw9fwq/SWJYbI0c5Y7LU1AXYoDdbhE2AHQ6N6Nb34sG8Fj7T5APy8qXDCKuuIHd1BR0tVA==", "license": "Apache-2.0", "engines": { "node": ">=8" @@ -9689,7 +6166,7 @@ "version": "1.0.4", "resolved": "https://registry.npmjs.org/dezalgo/-/dezalgo-1.0.4.tgz", "integrity": "sha512-rXSP0bf+5n0Qonsb+SVVfNfIsimO4HEtmnIpPHY8Q1UCzKlQrDMfdobr8nJOOsRgWCyMRqeSBQzmWUMq7zvVig==", - "devOptional": true, + "dev": true, "license": "ISC", "dependencies": { "asap": "^2.0.0", @@ -9790,99 +6267,10 @@ "node": ">=6.0.0" } }, - "node_modules/dom-converter": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/dom-converter/-/dom-converter-0.2.0.tgz", - "integrity": "sha512-gd3ypIPfOMr9h5jIKq8E3sHOTCjeirnl0WK5ZdS1AW0Odt0b1PaWaHdJ4Qk4klv+YB9aJBS7mESXjFoDQPu6DA==", - "dev": true, - "license": "MIT", - "dependencies": { - "utila": "~0.4" - } - }, - "node_modules/dom-serializer": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.4.1.tgz", - "integrity": "sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==", - "dev": true, - "license": "MIT", - "dependencies": { - "domelementtype": "^2.0.1", - "domhandler": "^4.2.0", - "entities": "^2.0.0" - }, - "funding": { - "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" - } - }, - "node_modules/dom-serializer/node_modules/entities": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/entities/-/entities-2.2.0.tgz", - "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==", - "dev": true, - "license": "BSD-2-Clause", - "funding": { - "url": "https://github.com/fb55/entities?sponsor=1" - } - }, - "node_modules/domelementtype": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.3.0.tgz", - "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/fb55" - } - ], - "license": "BSD-2-Clause" - }, - "node_modules/domhandler": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.3.1.tgz", - "integrity": "sha512-GrwoxYN+uWlzO8uhUXRl0P+kHE4GtVPfYzVLcUxPL7KNdHKj66vvlhiweIHqYYXWlw+T8iLMp42Lm67ghw4WMQ==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "domelementtype": "^2.2.0" - }, - "engines": { - "node": ">= 4" - }, - "funding": { - "url": "https://github.com/fb55/domhandler?sponsor=1" - } - }, - "node_modules/domutils": { - "version": "2.8.0", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.8.0.tgz", - "integrity": "sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "dom-serializer": "^1.0.1", - "domelementtype": "^2.2.0", - "domhandler": "^4.2.0" - }, - "funding": { - "url": "https://github.com/fb55/domutils?sponsor=1" - } - }, - "node_modules/dot-case": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-3.0.4.tgz", - "integrity": "sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==", - "license": "MIT", - "dependencies": { - "no-case": "^3.0.4", - "tslib": "^2.0.3" - } - }, "node_modules/dotenv": { - "version": "16.4.7", - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", - "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", + "version": "16.5.0", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.5.0.tgz", + "integrity": "sha512-m/C+AwOAr9/W1UOIZUo232ejMNnJAJtYQjUbHoNTBNTJSvqzzDh7vnrei3o3r3m9blf6ZoDkvcw0VmozNRFJxg==", "license": "BSD-2-Clause", "engines": { "node": ">=12" @@ -9949,36 +6337,36 @@ "license": "MIT" }, "node_modules/edge-error": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/edge-error/-/edge-error-4.0.1.tgz", - "integrity": "sha512-z5mNO97k8hRVpJ6Ew1qbkMTfQ44CwuWnl+ShMCrEFgD+b324CnjBS6HbiR+Wh6Wcmw9C+/XsFBHzZ+376PpD/w==", + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/edge-error/-/edge-error-4.0.2.tgz", + "integrity": "sha512-jB76VYn8wapDHKHSOmP3vbKLoa77RJYsTLNmfl8+cuCD69uxZtP3h+kqV+Prw/YkYmN7yHyp4IApE15pDByk0A==", "license": "MIT", "engines": { "node": ">=18.16.0" } }, "node_modules/edge-lexer": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/edge-lexer/-/edge-lexer-6.0.2.tgz", - "integrity": "sha512-C30wqcw66JwpepLnsTqTp0P4JqKa2xEbAfNj3dPOvBYq4zybiYuhlpSzExvNUeoAAnbjgozgVTVAQ38HctyV4g==", + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/edge-lexer/-/edge-lexer-6.0.3.tgz", + "integrity": "sha512-/s15CNnfhZv97bsW+ZgV5rtONULYjhCDYu+usbVLqZ8UQ6b/hQUNvQSIQBXA6Gql9dm72TMBB9sb/eWM2esufg==", "license": "MIT", "dependencies": { - "edge-error": "^4.0.1" + "edge-error": "^4.0.2" }, "engines": { "node": ">=18.16.0" } }, "node_modules/edge-parser": { - "version": "9.0.3", - "resolved": "https://registry.npmjs.org/edge-parser/-/edge-parser-9.0.3.tgz", - "integrity": "sha512-E9W+9wV8QVGLZCtrgKp6k9kIncsUxmrpa/yG+vwVGPpCMBZZZZaShJXwVDHThyL2mkHkWyYvhBpPhuucgW8kiA==", + "version": "9.0.4", + "resolved": "https://registry.npmjs.org/edge-parser/-/edge-parser-9.0.4.tgz", + "integrity": "sha512-vnjzfpqpjM4Mjt9typc1zLoFpC1F6kAObfcdyA6rSy+izIPji2RaQz5jWx5s5iG9hNcuyjtNyGRCLFVfoYhWcA==", "license": "MIT", "dependencies": { - "acorn": "^8.12.1", + "acorn": "^8.14.0", "astring": "^1.9.0", - "edge-error": "^4.0.1", - "edge-lexer": "^6.0.2", + "edge-error": "^4.0.2", + "edge-lexer": "^6.0.3", "js-stringify": "^1.0.2" }, "engines": { @@ -9986,18 +6374,18 @@ } }, "node_modules/edge.js": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/edge.js/-/edge.js-6.2.0.tgz", - "integrity": "sha512-xw82TzdPngccJiFqK6FE/79vO6mUvWVvKe6OEu/VHDOf199SIOW1q022d3UIaKGXcwf60lIXZYPIRqooQuzigA==", + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/edge.js/-/edge.js-6.2.1.tgz", + "integrity": "sha512-me875zh6YA0V429hywgQIpHgMvQkondv5XHaP6EsL2yIBpLcBWCl7Ba1cai0SwYhp8iD0IyV3KjpxLrnW7S2Ag==", "license": "MIT", "dependencies": { "@poppinss/inspect": "^1.0.1", - "@poppinss/macroable": "^1.0.3", - "@poppinss/utils": "^6.8.1", + "@poppinss/macroable": "^1.0.4", + "@poppinss/utils": "^6.9.2", "classnames": "^2.5.1", - "edge-error": "^4.0.1", - "edge-lexer": "^6.0.2", - "edge-parser": "^9.0.3", + "edge-error": "^4.0.2", + "edge-lexer": "^6.0.3", + "edge-parser": "^9.0.4", "fs-readdir-recursive": "^1.1.0", "he": "^1.2.0", "js-stringify": "^1.0.2", @@ -10015,16 +6403,16 @@ "license": "MIT" }, "node_modules/electron-to-chromium": { - "version": "1.5.79", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.79.tgz", - "integrity": "sha512-nYOxJNxQ9Om4EC88BE4pPoNI8xwSFf8pU/BAeOl4Hh/b/i6V4biTAzwV7pXi3ARKeoYO5JZKMIXTryXSVer5RA==", + "version": "1.5.145", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.145.tgz", + "integrity": "sha512-pZ5EcTWRq/055MvSBgoFEyKf2i4apwfoqJbK/ak2jnFq8oHjZ+vzc3AhRcz37Xn+ZJfL58R666FLJx0YOK9yTw==", "dev": true, "license": "ISC" }, "node_modules/emittery": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/emittery/-/emittery-1.0.3.tgz", - "integrity": "sha512-tJdCJitoy2lrC2ldJcqN4vkqJ00lT+tOWNT1hBJjO/3FDMJa5TTIiYGCKGkn/WfCyOzUMObeohbVTj00fhiLiA==", + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-1.1.0.tgz", + "integrity": "sha512-rsX7ktqARv/6UQDgMaLfIqUWAEzzbCQiVh7V9rhDXp6c37yoJcks12NVD+XPkgl4AEavmNhVfrhGoqYwIsMYYA==", "license": "MIT", "engines": { "node": ">=14.16" @@ -10039,20 +6427,10 @@ "integrity": "sha512-EC+0oUMY1Rqm4O6LLrgjtYDvcVYTy7chDnM4Q7030tP4Kwj3u/pR6gP9ygnp2CJMK5Gq+9Q2oqmrFJAz01DXjw==", "license": "MIT" }, - "node_modules/emojis-list": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/emojis-list/-/emojis-list-3.0.0.tgz", - "integrity": "sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 4" - } - }, "node_modules/encodeurl": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", - "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", + "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", "license": "MIT", "engines": { "node": ">= 0.8" @@ -10069,9 +6447,9 @@ } }, "node_modules/enhanced-resolve": { - "version": "5.18.0", - "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.18.0.tgz", - "integrity": "sha512-0/r0MySGYG8YqlayBZ6MuCfECmHFdJ5qyPh8s8wa5Hnm6SaFLSK1VYCbj+NKp090Nm1caZhD+QTnmxO7esYGyQ==", + "version": "5.18.1", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.18.1.tgz", + "integrity": "sha512-ZSW3ma5GkcQBIpwZTSRAI8N71Uuwgs93IezB7mf7R60tC8ZbJideoDNKjHn2O9KIlx6rkGTTEk1xUCK2E1Y2Yg==", "dev": true, "license": "MIT", "dependencies": { @@ -10117,20 +6495,6 @@ "node": ">=6" } }, - "node_modules/envinfo": { - "version": "7.14.0", - "resolved": "https://registry.npmjs.org/envinfo/-/envinfo-7.14.0.tgz", - "integrity": "sha512-CO40UI41xDQzhLB1hWyqUKgFhs250pNcGbyGKe1l/e4FSaI/+YE4IMG76GDt0In67WLPACIITC+sOi08x4wIvg==", - "dev": true, - "license": "MIT", - "peer": true, - "bin": { - "envinfo": "dist/cli.js" - }, - "engines": { - "node": ">=4" - } - }, "node_modules/environment": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/environment/-/environment-1.1.0.tgz", @@ -10153,91 +6517,15 @@ "is-arrayish": "^0.2.1" } }, - "node_modules/error-stack-parser": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/error-stack-parser/-/error-stack-parser-2.1.4.tgz", - "integrity": "sha512-Sk5V6wVazPhq5MhpO+AUxJn5x7XSXGl1R93Vn7i+zS15KDVxQijejNCrz8340/2bgLBjR9GtEG8ZVKONDjcqGQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "stackframe": "^1.3.4" - } - }, "node_modules/error-stack-parser-es": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/error-stack-parser-es/-/error-stack-parser-es-0.1.5.tgz", - "integrity": "sha512-xHku1X40RO+fO8yJ8Wh2f2rZWVjqyhb1zgq1yZ8aZRQkv6OOKhKWRUaht3eSCUbAOBaKIgM+ykwFLE+QUxgGeg==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/error-stack-parser-es/-/error-stack-parser-es-1.0.5.tgz", + "integrity": "sha512-5qucVt2XcuGMcEGgWI7i+yZpmpByQ8J1lHhcL7PwqCwu9FPP3VUXzT4ltHe5i2z9dePwEHcDVOAfSnHsOlCXRA==", "license": "MIT", "funding": { "url": "https://github.com/sponsors/antfu" } }, - "node_modules/es-abstract": { - "version": "1.23.9", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.23.9.tgz", - "integrity": "sha512-py07lI0wjxAC/DcfK1S6G7iANonniZwTISvdPzk9hzeH0IZIshbuuFxLIU96OyF89Yb9hiqWn8M/bY83KY5vzA==", - "dev": true, - "license": "MIT", - "dependencies": { - "array-buffer-byte-length": "^1.0.2", - "arraybuffer.prototype.slice": "^1.0.4", - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "data-view-buffer": "^1.0.2", - "data-view-byte-length": "^1.0.2", - "data-view-byte-offset": "^1.0.1", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "es-set-tostringtag": "^2.1.0", - "es-to-primitive": "^1.3.0", - "function.prototype.name": "^1.1.8", - "get-intrinsic": "^1.2.7", - "get-proto": "^1.0.0", - "get-symbol-description": "^1.1.0", - "globalthis": "^1.0.4", - "gopd": "^1.2.0", - "has-property-descriptors": "^1.0.2", - "has-proto": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "internal-slot": "^1.1.0", - "is-array-buffer": "^3.0.5", - "is-callable": "^1.2.7", - "is-data-view": "^1.0.2", - "is-regex": "^1.2.1", - "is-shared-array-buffer": "^1.0.4", - "is-string": "^1.1.1", - "is-typed-array": "^1.1.15", - "is-weakref": "^1.1.0", - "math-intrinsics": "^1.1.0", - "object-inspect": "^1.13.3", - "object-keys": "^1.1.1", - "object.assign": "^4.1.7", - "own-keys": "^1.0.1", - "regexp.prototype.flags": "^1.5.3", - "safe-array-concat": "^1.1.3", - "safe-push-apply": "^1.0.0", - "safe-regex-test": "^1.1.0", - "set-proto": "^1.0.0", - "string.prototype.trim": "^1.2.10", - "string.prototype.trimend": "^1.0.9", - "string.prototype.trimstart": "^1.0.8", - "typed-array-buffer": "^1.0.3", - "typed-array-byte-length": "^1.0.3", - "typed-array-byte-offset": "^1.0.4", - "typed-array-length": "^1.0.7", - "unbox-primitive": "^1.1.0", - "which-typed-array": "^1.1.18" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/es-define-property": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", @@ -10257,15 +6545,15 @@ } }, "node_modules/es-module-lexer": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.6.0.tgz", - "integrity": "sha512-qqnD1yMU6tk/jnaMosogGySTZP8YtUgAffA9nMN+E/rjxcfRQ6IEk7IiozUjgxKoFHBGjTLnrHB/YC45r/59EQ==", + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.7.0.tgz", + "integrity": "sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA==", "license": "MIT" }, "node_modules/es-object-atoms": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.0.0.tgz", - "integrity": "sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==", + "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" @@ -10278,7 +6566,6 @@ "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==", - "dev": true, "license": "MIT", "dependencies": { "es-errors": "^1.3.0", @@ -10290,61 +6577,54 @@ "node": ">= 0.4" } }, - "node_modules/es-to-primitive": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.3.0.tgz", - "integrity": "sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==", - "dev": true, + "node_modules/es-toolkit": { + "version": "1.36.0", + "resolved": "https://registry.npmjs.org/es-toolkit/-/es-toolkit-1.36.0.tgz", + "integrity": "sha512-5lpkRpDELuTSeAL//Rcg5urg+K/yOD1BobJSiNeCc89snMqgrhckmj8jdljqraDbpREiXTNW311RN518eVHBng==", "license": "MIT", - "dependencies": { - "is-callable": "^1.2.7", - "is-date-object": "^1.0.5", - "is-symbol": "^1.0.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } + "workspaces": [ + "docs", + "benchmarks" + ] }, "node_modules/esbuild": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", - "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==", + "version": "0.25.3", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.3.tgz", + "integrity": "sha512-qKA6Pvai73+M2FtftpNKRxJ78GIjmFXFxd/1DVBqGo/qNhLSfv+G12n9pNoWdytJC8U00TrViOwpjT0zgqQS8Q==", "hasInstallScript": true, "license": "MIT", - "peer": true, "bin": { "esbuild": "bin/esbuild" }, "engines": { - "node": ">=12" + "node": ">=18" }, "optionalDependencies": { - "@esbuild/aix-ppc64": "0.21.5", - "@esbuild/android-arm": "0.21.5", - "@esbuild/android-arm64": "0.21.5", - "@esbuild/android-x64": "0.21.5", - "@esbuild/darwin-arm64": "0.21.5", - "@esbuild/darwin-x64": "0.21.5", - "@esbuild/freebsd-arm64": "0.21.5", - "@esbuild/freebsd-x64": "0.21.5", - "@esbuild/linux-arm": "0.21.5", - "@esbuild/linux-arm64": "0.21.5", - "@esbuild/linux-ia32": "0.21.5", - "@esbuild/linux-loong64": "0.21.5", - "@esbuild/linux-mips64el": "0.21.5", - "@esbuild/linux-ppc64": "0.21.5", - "@esbuild/linux-riscv64": "0.21.5", - "@esbuild/linux-s390x": "0.21.5", - "@esbuild/linux-x64": "0.21.5", - "@esbuild/netbsd-x64": "0.21.5", - "@esbuild/openbsd-x64": "0.21.5", - "@esbuild/sunos-x64": "0.21.5", - "@esbuild/win32-arm64": "0.21.5", - "@esbuild/win32-ia32": "0.21.5", - "@esbuild/win32-x64": "0.21.5" + "@esbuild/aix-ppc64": "0.25.3", + "@esbuild/android-arm": "0.25.3", + "@esbuild/android-arm64": "0.25.3", + "@esbuild/android-x64": "0.25.3", + "@esbuild/darwin-arm64": "0.25.3", + "@esbuild/darwin-x64": "0.25.3", + "@esbuild/freebsd-arm64": "0.25.3", + "@esbuild/freebsd-x64": "0.25.3", + "@esbuild/linux-arm": "0.25.3", + "@esbuild/linux-arm64": "0.25.3", + "@esbuild/linux-ia32": "0.25.3", + "@esbuild/linux-loong64": "0.25.3", + "@esbuild/linux-mips64el": "0.25.3", + "@esbuild/linux-ppc64": "0.25.3", + "@esbuild/linux-riscv64": "0.25.3", + "@esbuild/linux-s390x": "0.25.3", + "@esbuild/linux-x64": "0.25.3", + "@esbuild/netbsd-arm64": "0.25.3", + "@esbuild/netbsd-x64": "0.25.3", + "@esbuild/openbsd-arm64": "0.25.3", + "@esbuild/openbsd-x64": "0.25.3", + "@esbuild/sunos-x64": "0.25.3", + "@esbuild/win32-arm64": "0.25.3", + "@esbuild/win32-ia32": "0.25.3", + "@esbuild/win32-x64": "0.25.3" } }, "node_modules/escalade": { @@ -10375,13 +6655,16 @@ "license": "MIT" }, "node_modules/escape-string-regexp": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "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": ">=0.8.0" + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, "node_modules/eslint": { @@ -10442,9 +6725,9 @@ } }, "node_modules/eslint-config-prettier": { - "version": "9.1.0", - "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-9.1.0.tgz", - "integrity": "sha512-NSWl5BFQWEPi1j4TjVNItzYV7dZXZ+wP6I6ZhrBGpChQhZRUaElihE9uRRkcbRnNb76UMKDF3r+WTmNcGPKsqw==", + "version": "10.1.2", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-10.1.2.tgz", + "integrity": "sha512-Epgp/EofAUeEpIdZkW60MHKvPyru1ruQJxPL+WIycnaPApuseK0Zpkrh/FwL9oIpQvIhJwV7ptOy0DWUjTlCiA==", "dev": true, "license": "MIT", "bin": { @@ -10472,14 +6755,14 @@ } }, "node_modules/eslint-plugin-prettier": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.2.1.tgz", - "integrity": "sha512-gH3iR3g4JfF+yYPaJYkN7jEl9QbweL/YfkoRlNnuIEHEz1vHVlCmWOS+eGGiRuzHQXdJFCOTxRgvju9b8VUmrw==", + "version": "5.2.6", + "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.2.6.tgz", + "integrity": "sha512-mUcf7QG2Tjk7H055Jk0lGBjbgDnfrvqjhXh9t2xLMSCjZVcw9Rb1V6sVNXO0th3jgeO7zllWPTNRil3JW94TnQ==", "dev": true, "license": "MIT", "dependencies": { "prettier-linter-helpers": "^1.0.0", - "synckit": "^0.9.1" + "synckit": "^0.11.0" }, "engines": { "node": "^14.18.0 || >=16.0.0" @@ -10490,7 +6773,7 @@ "peerDependencies": { "@types/eslint": ">=8.0.0", "eslint": ">=8.0.0", - "eslint-config-prettier": "*", + "eslint-config-prettier": ">= 7.0.0 <10.0.0 || >=10.1.0", "prettier": ">=3.0.0" }, "peerDependenciesMeta": { @@ -10532,79 +6815,6 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/eslint/node_modules/ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "dev": true, - "license": "MIT", - "dependencies": { - "color-convert": "^2.0.1" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/eslint/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/eslint/node_modules/chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", - "dev": true, - "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/eslint/node_modules/color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "color-name": "~1.1.4" - }, - "engines": { - "node": ">=7.0.0" - } - }, - "node_modules/eslint/node_modules/color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true, - "license": "MIT" - }, - "node_modules/eslint/node_modules/escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, "node_modules/eslint/node_modules/globals": { "version": "13.24.0", "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", @@ -10621,32 +6831,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/eslint/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/eslint/node_modules/supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-flag": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/eslint/node_modules/type-fest": { "version": "0.20.2", "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", @@ -10761,16 +6945,6 @@ "node": ">= 0.6" } }, - "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==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=6" - } - }, "node_modules/eventemitter3": { "version": "4.0.7", "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz", @@ -10784,6 +6958,7 @@ "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", "dev": true, "license": "MIT", + "peer": true, "engines": { "node": ">=0.8.x" } @@ -10882,16 +7057,6 @@ "ms": "2.0.0" } }, - "node_modules/express/node_modules/encodeurl": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", - "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/express/node_modules/ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", @@ -10915,27 +7080,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/express/node_modules/safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "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/fast-copy": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/fast-copy/-/fast-copy-3.0.2.tgz", @@ -11013,12 +7157,13 @@ "version": "2.1.1", "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz", "integrity": "sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA==", + "dev": true, "license": "MIT" }, "node_modules/fast-uri": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.0.5.tgz", - "integrity": "sha512-5JnBCWpFlMo0a3ciDy/JckMzzv1U9coZrIhedq+HXxxUfDTAiS0LA8OKVao4G9BxmCVck/jtA5r3KAtRWEyD8Q==", + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/fast-uri/-/fast-uri-3.0.6.tgz", + "integrity": "sha512-Atfo14OibSv5wAp4VWNsFYE1AchQRTv9cBGWET4pZWHzYshFSS9NQI6I57rdKn9croWVMbYFbLhJ+yJvmZIIHw==", "dev": true, "funding": [ { @@ -11042,9 +7187,9 @@ } }, "node_modules/fastq": { - "version": "1.18.0", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.18.0.tgz", - "integrity": "sha512-QKHXPW0hD8g4UET03SdOdunzSouc9N4AuHdsX8XNcTsuz+yYFILVNIX4l9yHABMhiEI9Db0JTTIpu0wB+Y1QQw==", + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", + "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", "license": "ISC", "dependencies": { "reusify": "^1.0.4" @@ -11063,6 +7208,26 @@ "node": ">=0.8.0" } }, + "node_modules/fdir": { + "version": "6.4.4", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.4.4.tgz", + "integrity": "sha512-1NZP+GK4GfuAv3PqKvxQRDMjdSRZjnkq7KfhlNrCNNlZ0ygQFpebfrnfnq/W7fpUnAv9aGWmY1zKx7FYL3gwhg==", + "license": "MIT", + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/fflate": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/fflate/-/fflate-0.8.2.tgz", + "integrity": "sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A==", + "license": "MIT" + }, "node_modules/figures": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/figures/-/figures-6.1.0.tgz", @@ -11093,15 +7258,15 @@ } }, "node_modules/file-type": { - "version": "19.6.0", - "resolved": "https://registry.npmjs.org/file-type/-/file-type-19.6.0.tgz", - "integrity": "sha512-VZR5I7k5wkD0HgFnMsq5hOsSc710MJMu5Nc5QYsbe38NN5iPV/XTObYLc/cpttRTf6lX538+5uO1ZQRhYibiZQ==", + "version": "20.5.0", + "resolved": "https://registry.npmjs.org/file-type/-/file-type-20.5.0.tgz", + "integrity": "sha512-BfHZtG/l9iMm4Ecianu7P8HRD2tBHLtjXinm4X62XBOYzi7CYA7jyqfJzOvXHqzVrVPYqBo2/GvbARMaaJkKVg==", "license": "MIT", "dependencies": { - "get-stream": "^9.0.1", - "strtok3": "^9.0.1", + "@tokenizer/inflate": "^0.2.6", + "strtok3": "^10.2.0", "token-types": "^6.0.0", - "uint8array-extras": "^1.3.0" + "uint8array-extras": "^1.4.0" }, "engines": { "node": ">=18" @@ -11151,16 +7316,6 @@ "ms": "2.0.0" } }, - "node_modules/finalhandler/node_modules/encodeurl": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", - "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/finalhandler/node_modules/ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", @@ -11202,6 +7357,19 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/find-up-simple": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/find-up-simple/-/find-up-simple-1.0.1.tgz", + "integrity": "sha512-afd4O7zpqHeRyg4PfDQsXmlDe2PfdHtJt6Akt8jOWaApLOZk5JXs6VMR29lz03pRe9mpykrRCYIYxaJYcfpncQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/find-up/node_modules/locate-path": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", @@ -11263,17 +7431,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/flat": { - "version": "5.0.2", - "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", - "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", - "dev": true, - "license": "BSD-3-Clause", - "peer": true, - "bin": { - "flat": "cli.js" - } - }, "node_modules/flat-cache": { "version": "3.2.0", "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", @@ -11289,17 +7446,10 @@ "node": "^10.12.0 || >=12.0.0" } }, - "node_modules/flatstr": { - "version": "1.0.12", - "resolved": "https://registry.npmjs.org/flatstr/-/flatstr-1.0.12.tgz", - "integrity": "sha512-4zPxDyhCyiN2wIAtSLI6gc82/EjqZc1onI4Mz/l0pWrAlsSfYH/2ZIcU+e3oA2wDwbzIWNKwa23F8rh6+DRWkw==", - "license": "MIT", - "peer": true - }, "node_modules/flatted": { - "version": "3.3.2", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.2.tgz", - "integrity": "sha512-AiwGJM8YcNOaobumgtng+6NHuOqC3A7MixFeDafM3X9cIUM+xUXoS5Vfgf+OihAYe20fxqNM9yPBXJzRtZ/4eA==", + "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" }, @@ -11312,10 +7462,41 @@ "node": ">=8" } }, + "node_modules/flydrive": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/flydrive/-/flydrive-1.2.0.tgz", + "integrity": "sha512-l9ix5MhBE8bVwxyHdFku6z5KhGOCOXQDI9xGNIlACSz9UrDFQxAB1I6W0qffZiOBBDambiJZlEYBCxlvF4U7fw==", + "license": "MIT", + "dependencies": { + "@humanwhocodes/retry": "^0.4.2", + "@poppinss/utils": "^6.9.2", + "etag": "^1.8.1", + "mime-types": "^2.1.35" + }, + "engines": { + "node": ">=20.6.0" + }, + "peerDependencies": { + "@aws-sdk/client-s3": "^3.577.0", + "@aws-sdk/s3-request-presigner": "^3.577.0", + "@google-cloud/storage": "^7.10.2" + }, + "peerDependenciesMeta": { + "@aws-sdk/client-s3": { + "optional": true + }, + "@aws-sdk/s3-request-presigner": { + "optional": true + }, + "@google-cloud/storage": { + "optional": true + } + } + }, "node_modules/focus-trap": { - "version": "7.6.2", - "resolved": "https://registry.npmjs.org/focus-trap/-/focus-trap-7.6.2.tgz", - "integrity": "sha512-9FhUxK1hVju2+AiQIDJ5Dd//9R2n2RAfJ0qfhF4IHGHgcoEUTMpbTeG/zbEuwaiYXfuAH6XE0/aCyxDdRM+W5w==", + "version": "7.6.4", + "resolved": "https://registry.npmjs.org/focus-trap/-/focus-trap-7.6.4.tgz", + "integrity": "sha512-xx560wGBk7seZ6y933idtjJQc1l+ck+pI3sKvhKozdBV1dRZoKhkW5xoCaFv9tQiX5RH1xfSxjuNu6g+lmN/gw==", "license": "MIT", "dependencies": { "tabbable": "^6.2.0" @@ -11341,24 +7522,14 @@ } } }, - "node_modules/for-each": { - "version": "0.3.3", - "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", - "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-callable": "^1.1.3" - } - }, "node_modules/foreground-child": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.0.tgz", - "integrity": "sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==", + "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.0", + "cross-spawn": "^7.0.6", "signal-exit": "^4.0.1" }, "engines": { @@ -11369,13 +7540,14 @@ } }, "node_modules/form-data": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.1.tgz", - "integrity": "sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==", + "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": { @@ -11401,14 +7573,14 @@ } }, "node_modules/formidable": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/formidable/-/formidable-2.1.2.tgz", - "integrity": "sha512-CM3GuJ57US06mlpQ47YcunuUZ9jpm8Vx+P2CGt2j7HpgkKZO/DJYQ0Bobim8G6PFQmK5lOqOOdUXboU+h73A4g==", - "devOptional": true, + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/formidable/-/formidable-2.1.5.tgz", + "integrity": "sha512-Oz5Hwvwak/DCaXVVUtPn4oLMLLy1CdclLKO1LFgU7XzDpVMUU5UjlSLpGMocyQNNk8F6IJW9M/YdooSn2MRI+Q==", + "dev": true, "license": "MIT", "dependencies": { + "@paralleldrive/cuid2": "^2.2.2", "dezalgo": "^1.0.4", - "hexoid": "^1.0.0", "once": "^1.4.0", "qs": "^6.11.0" }, @@ -11449,9 +7621,9 @@ } }, "node_modules/fs-extra": { - "version": "11.2.0", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-11.2.0.tgz", - "integrity": "sha512-PmDi3uwK5nFuXh7XDTlVnS17xJS7vW36is2+w3xcv8SVxiB4NyATf4ctkVY5bkSjX0Y4nbvZCq1/EjtEyr9ktw==", + "version": "11.3.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-11.3.0.tgz", + "integrity": "sha512-Z4XaCL6dUDHfP/jT25jJKMmtxvuwbkrD1vNSMFlo9lNLY2c5FHYSQgHPRZUjAB26TpDEoW9HCOgplrdbaPV/ew==", "license": "MIT", "dependencies": { "graceful-fs": "^4.2.0", @@ -11492,12 +7664,6 @@ "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", "license": "ISC" }, - "node_modules/fs-monkey": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/fs-monkey/-/fs-monkey-1.0.6.tgz", - "integrity": "sha512-b1FMfwetIKymC0eioW7mTywihSQE4oLzQn1dB6rZB5fx/3NpNEdAWeCSMB+60/AeT0TCXsxzAlcYVEFCTAksWg==", - "license": "Unlicense" - }, "node_modules/fs-readdir-recursive": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/fs-readdir-recursive/-/fs-readdir-recursive-1.1.0.tgz", @@ -11533,37 +7699,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/function.prototype.name": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.8.tgz", - "integrity": "sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "functions-have-names": "^1.2.3", - "hasown": "^2.0.2", - "is-callable": "^1.2.7" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/functions-have-names": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", - "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", - "dev": true, - "license": "MIT", - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/gauge": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/gauge/-/gauge-3.0.2.tgz", @@ -11620,21 +7755,13 @@ "node": ">=8" } }, - "node_modules/generic-pool": { - "version": "3.9.0", - "resolved": "https://registry.npmjs.org/generic-pool/-/generic-pool-3.9.0.tgz", - "integrity": "sha512-hymDOu5B53XvN4QT9dBmZxPX4CWhBPPLguTZ9MMFeFa/Kg0xWVfylOVNlJji/E7yTZWFd/q9GO5TxDLq156D7g==", - "license": "MIT", - "engines": { - "node": ">= 4" - } - }, "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", + "peer": true, "engines": { "node": ">=6.9.0" } @@ -11661,17 +7788,17 @@ } }, "node_modules/get-intrinsic": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.7.tgz", - "integrity": "sha512-VW6Pxhsrk0KAOqs3WEd0klDiF/+V7gQOpAvY1jVU/LHmaD/kQO4523aiJuikX/QAKYiW6x8Jh+RJej1almdtCA==", + "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.1", + "call-bind-apply-helpers": "^1.0.2", "es-define-property": "^1.0.1", "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", + "es-object-atoms": "^1.1.1", "function-bind": "^1.1.2", - "get-proto": "^1.0.0", + "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-symbols": "^1.1.0", "hasown": "^2.0.2", @@ -11745,24 +7872,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/get-symbol-description": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.1.0.tgz", - "integrity": "sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/getopts": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/getopts/-/getopts-2.3.0.tgz", @@ -11819,36 +7928,19 @@ "node": ">=4" } }, - "node_modules/globalthis": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.4.tgz", - "integrity": "sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-properties": "^1.2.1", - "gopd": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/globby": { - "version": "14.0.2", - "resolved": "https://registry.npmjs.org/globby/-/globby-14.0.2.tgz", - "integrity": "sha512-s3Fq41ZVh7vbbe2PN3nrW7yC7U7MFVc5c98/iTl9c2GawNMKx/J648KQRW6WKkuU8GIbbh2IXfIRQjOZnXcTnw==", + "version": "14.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-14.1.0.tgz", + "integrity": "sha512-0Ia46fDOaT7k4og1PDW4YbodWWr3scS2vAr2lTbsplOt2WkKp0vQbkI9wKis/T5LV/dqPjO3bpS/z6GTJB82LA==", "devOptional": true, "license": "MIT", "dependencies": { "@sindresorhus/merge-streams": "^2.1.0", - "fast-glob": "^3.3.2", - "ignore": "^5.2.4", - "path-type": "^5.0.0", + "fast-glob": "^3.3.3", + "ignore": "^7.0.3", + "path-type": "^6.0.0", "slash": "^5.1.0", - "unicorn-magic": "^0.1.0" + "unicorn-magic": "^0.3.0" }, "engines": { "node": ">=18" @@ -11870,6 +7962,16 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/globby/node_modules/ignore": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.4.tgz", + "integrity": "sha512-gJzzk+PQNznz8ysRrC0aOkBNVRBDtE1n53IqyqEf3PXrYwomFs5q4pGMizBMJF+ykh03insJ27hB8gSrD2Hn8A==", + "devOptional": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, "node_modules/gopd": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", @@ -11883,9 +7985,9 @@ } }, "node_modules/got": { - "version": "14.4.5", - "resolved": "https://registry.npmjs.org/got/-/got-14.4.5.tgz", - "integrity": "sha512-sq+uET8TnNKRNnjEOPJzMcxeI0irT8BBNmf+GtZcJpmhYsQM1DSKmCROUjPWKsXZ5HzwD5Cf5/RV+QD9BSTxJg==", + "version": "14.4.7", + "resolved": "https://registry.npmjs.org/got/-/got-14.4.7.tgz", + "integrity": "sha512-DI8zV1231tqiGzOiOzQWDhsBmncFW7oQDH6Zgy6pDPrqJuVZMtoSgPLLsBZQj8Jg4JFfwoOsDA8NGtLQLnIx2g==", "license": "MIT", "dependencies": { "@sindresorhus/is": "^7.0.1", @@ -11927,57 +8029,16 @@ "dev": true, "license": "MIT" }, - "node_modules/has-bigints": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.1.0.tgz", - "integrity": "sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==", - "dev": true, - "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==", + "devOptional": true, "license": "MIT", "engines": { "node": ">=8" } }, - "node_modules/has-property-descriptors": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", - "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-define-property": "^1.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-proto": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.2.0.tgz", - "integrity": "sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "dunder-proto": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/has-symbols": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", @@ -11994,7 +8055,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", - "dev": true, "license": "MIT", "dependencies": { "has-symbols": "^1.0.3" @@ -12031,13 +8091,6 @@ "node": ">= 0.4" } }, - "node_modules/haye": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/haye/-/haye-3.0.0.tgz", - "integrity": "sha512-yWxbPdeex78IR3x3X/DdqkZbVG4rP4UaRdUGmpClfnUh1C61mASt7Iav8vk2tXcTMSygBHDDfgoVqk68NJqzhQ==", - "license": "MIT", - "peer": true - }, "node_modules/he": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", @@ -12047,16 +8100,6 @@ "he": "bin/he" } }, - "node_modules/header-case": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/header-case/-/header-case-2.0.4.tgz", - "integrity": "sha512-H/vuk5TEEVZwrR0lp2zed9OCo1uAILMlx0JEMgC26rzyJJ3N1v6XkwHHXJQdR2doSjcGPM6OKPYoJgf0plJ11Q==", - "license": "MIT", - "dependencies": { - "capital-case": "^1.0.4", - "tslib": "^2.0.3" - } - }, "node_modules/helmet-csp": { "version": "3.4.0", "resolved": "https://registry.npmjs.org/helmet-csp/-/helmet-csp-3.4.0.tgz", @@ -12073,14 +8116,45 @@ "dev": true, "license": "MIT" }, - "node_modules/hexoid": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/hexoid/-/hexoid-1.0.0.tgz", - "integrity": "sha512-QFLV0taWQOZtvIRIAdBChesmogZrtuXvVWsFHZTk2SU+anspqZ2vMnoLg7IE1+Uk16N19APic1BuF8bC8c2m5g==", - "devOptional": true, - "license": "MIT", + "node_modules/hookable": { + "version": "5.5.3", + "resolved": "https://registry.npmjs.org/hookable/-/hookable-5.5.3.tgz", + "integrity": "sha512-Yc+BQe8SvoXH1643Qez1zqLRmbA5rCL+sSmk6TVos0LWVfNIB7PGncdlId77WzLGSIB5KaWgTaNTs2lNVEI6VQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/hosted-git-info": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-7.0.2.tgz", + "integrity": "sha512-puUZAUKT5m8Zzvs72XWy3HtvVbTWljRE66cP60bxJzAqf2DgICo7lYTY2IHUmLnNpjYvw5bvmoHvPc0QO2a62w==", + "dev": true, + "license": "ISC", + "dependencies": { + "lru-cache": "^10.0.1" + }, "engines": { - "node": ">=8" + "node": "^16.14.0 || >=18.0.0" + } + }, + "node_modules/hosted-git-info/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/hot-hook": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/hot-hook/-/hot-hook-0.4.0.tgz", + "integrity": "sha512-D36jqIojBHqxfkel6r7QGfmal7HO3cFTnPKeZIpPsBtFdV3QPV7m42JTBDX3B/Ovi53RXbOix7t/uIeV2bfeRA==", + "dev": true, + "license": "MIT", + "dependencies": { + "chokidar": "^4.0.1", + "fast-glob": "^3.3.2", + "parse-imports": "^2.2.1", + "picomatch": "^4.0.2", + "read-package-up": "^11.0.0" } }, "node_modules/hpack.js": { @@ -12096,13 +8170,6 @@ "wbuf": "^1.1.0" } }, - "node_modules/hpack.js/node_modules/isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", - "dev": true, - "license": "MIT" - }, "node_modules/hpack.js/node_modules/readable-stream": { "version": "2.3.8", "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", @@ -12119,6 +8186,13 @@ "util-deprecate": "~1.0.1" } }, + "node_modules/hpack.js/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true, + "license": "MIT" + }, "node_modules/hpack.js/node_modules/string_decoder": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", @@ -12139,9 +8213,9 @@ } }, "node_modules/html-entities": { - "version": "2.5.2", - "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-2.5.2.tgz", - "integrity": "sha512-K//PSRMQk4FZ78Kyau+mZurHn3FH0Vwr+H36eE0rPbeYkRRi9YxceYPhuN60UwWorxyKHhqoAJl2OFKa4BVtaA==", + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-2.6.0.tgz", + "integrity": "sha512-kig+rMn/QOVRvr7c86gQ8lWXq+Hkv6CbAH1hLu+RG338StTpE8Z0b44SDVaqVu7HGKf27frdmUYEs9hTUX/cLQ==", "funding": [ { "type": "github", @@ -12154,36 +8228,6 @@ ], "license": "MIT" }, - "node_modules/htmlparser2": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-6.1.0.tgz", - "integrity": "sha512-gyyPk6rgonLFEDGoeRgQNaEUvdJ4ktTmmUh/h2t7s+M8oPpIPxgNACWa+6ESR57kXstwqPiCut0V8NRpcwgU7A==", - "dev": true, - "funding": [ - "https://github.com/fb55/htmlparser2?sponsor=1", - { - "type": "github", - "url": "https://github.com/sponsors/fb55" - } - ], - "license": "MIT", - "dependencies": { - "domelementtype": "^2.0.1", - "domhandler": "^4.0.0", - "domutils": "^2.5.2", - "entities": "^2.0.0" - } - }, - "node_modules/htmlparser2/node_modules/entities": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/entities/-/entities-2.2.0.tgz", - "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==", - "dev": true, - "license": "BSD-2-Clause", - "funding": { - "url": "https://github.com/fb55/entities?sponsor=1" - } - }, "node_modules/http-cache-semantics": { "version": "4.1.1", "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz", @@ -12214,9 +8258,9 @@ } }, "node_modules/http-parser-js": { - "version": "0.5.8", - "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.8.tgz", - "integrity": "sha512-SGeBX54F94Wgu5RH3X5jsDtf4eHyRogWX1XGT3b4HuW3tQPM4AaBzoUji/4AAJNXCEOWZ5O0DgZmJw1947gD5Q==", + "version": "0.5.10", + "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.10.tgz", + "integrity": "sha512-Pysuw9XpUq5dVc/2SMHpuTY01RFl8fttgcyunjL7eEMhGM3cI4eOmiCycJDVCo/7O7ClfQD3SaI6ftDzqOXYMA==", "dev": true, "license": "MIT" }, @@ -12236,9 +8280,9 @@ } }, "node_modules/http-proxy-middleware": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-2.0.7.tgz", - "integrity": "sha512-fgVY8AV7qU7z/MmXJ/rxwbrtQH4jBQ9m7kp3llF0liB7glmFeVZFBepQb32T3y8n8k2+AEYuMPCpinYW+/CuRA==", + "version": "2.0.9", + "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-2.0.9.tgz", + "integrity": "sha512-c1IyJYLYppU574+YI7R4QyX2ystMtVXZwIdzazUIPIJsHuWNd+mho2j+bKoHftndicGj9yh+xjd+l0yj7VeT1Q==", "dev": true, "license": "MIT", "dependencies": { @@ -12306,9 +8350,9 @@ } }, "node_modules/human-signals": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-8.0.0.tgz", - "integrity": "sha512-/1/GPCpDUCCYwlERiYjxoczfP0zfvZMU/OWgQPMya9AbAE24vseigFdhAMObpc8Q4lc/kjutPfUddDYyAmejnA==", + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-8.0.1.tgz", + "integrity": "sha512-eKCa6bwnJhvxj14kZk5NCPc6Hb6BdsU9DZcOnmQKSnO1VKrfV0zCvtttPZUsBvjmNDn8rpcJfpwSYnHBjc95MQ==", "devOptional": true, "license": "Apache-2.0", "engines": { @@ -12377,30 +8421,17 @@ } }, "node_modules/iconv-lite": { - "version": "0.4.24", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", - "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", "license": "MIT", "dependencies": { - "safer-buffer": ">= 2.1.2 < 3" + "safer-buffer": ">= 2.1.2 < 3.0.0" }, "engines": { "node": ">=0.10.0" } }, - "node_modules/icss-utils": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-5.1.0.tgz", - "integrity": "sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==", - "dev": true, - "license": "ISC", - "engines": { - "node": "^10 || ^12 || >= 14" - }, - "peerDependencies": { - "postcss": "^8.1.0" - } - }, "node_modules/ieee754": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", @@ -12434,16 +8465,16 @@ "version": "5.3.2", "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", - "devOptional": true, + "dev": true, "license": "MIT", "engines": { "node": ">= 4" } }, "node_modules/import-fresh": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", - "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "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": { @@ -12457,101 +8488,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/import-local": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.2.0.tgz", - "integrity": "sha512-2SPlun1JUPWoM6t3F0dw0FkCF/jWY8kttcY4f599GLTSjh2OCuuhdTkJQsEcZzBqbXZGKMK2OqW1oZsjtf/gQA==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "pkg-dir": "^4.2.0", - "resolve-cwd": "^3.0.0" - }, - "bin": { - "import-local-fixture": "fixtures/cli.js" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/import-local/node_modules/find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/import-local/node_modules/locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "p-locate": "^4.1.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/import-local/node_modules/p-limit": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", - "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "p-try": "^2.0.0" - }, - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/import-local/node_modules/p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "p-limit": "^2.2.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/import-local/node_modules/pkg-dir": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", - "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "find-up": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/imurmurhash": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", @@ -12562,6 +8498,19 @@ "node": ">=0.8.19" } }, + "node_modules/index-to-position": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/index-to-position/-/index-to-position-1.1.0.tgz", + "integrity": "sha512-XPdx9Dq4t9Qk1mTMbWONJqU7boCoumEH7fRET37HX5+khDUl3J2W6PdALxhILYlIYx2amlwYcRPp28p0tSiojg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/inflation": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/inflation/-/inflation-2.1.0.tgz", @@ -12588,21 +8537,6 @@ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", "license": "ISC" }, - "node_modules/internal-slot": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.1.0.tgz", - "integrity": "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "hasown": "^2.0.2", - "side-channel": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - } - }, "node_modules/interpret": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/interpret/-/interpret-2.2.0.tgz", @@ -12613,9 +8547,9 @@ } }, "node_modules/ioredis": { - "version": "5.4.2", - "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-5.4.2.tgz", - "integrity": "sha512-0SZXGNGZ+WzISQ67QDyZ2x0+wVxjjUndtD8oSeik/4ajifeiRufed8fCb8QW8VMyi4MXcS+UO1k/0NGhvq1PAg==", + "version": "5.6.1", + "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-5.6.1.tgz", + "integrity": "sha512-UxC0Yv1Y4WRJiGQxQkP0hfdL0/5/6YvdfOOClRgJ0qppSarkhneSa6UvkMkms0AkdGimSH3Ikqm+6mkMmX7vGA==", "license": "MIT", "dependencies": { "@ioredis/commands": "^1.1.1", @@ -12645,24 +8579,6 @@ "node": ">= 0.10" } }, - "node_modules/is-array-buffer": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.5.tgz", - "integrity": "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-arrayish": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", @@ -12670,41 +8586,6 @@ "dev": true, "license": "MIT" }, - "node_modules/is-async-function": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.1.0.tgz", - "integrity": "sha512-GExz9MtyhlZyXYLxzlJRj5WUCE661zhDa1Yna52CN57AJsymh+DvXXjyveSioqSRdxvUrdKdvqB1b5cVKsNpWQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "get-proto": "^1.0.1", - "has-tostringtag": "^1.0.2", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-bigint": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.1.0.tgz", - "integrity": "sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "has-bigints": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-binary-path": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", @@ -12718,43 +8599,6 @@ "node": ">=8" } }, - "node_modules/is-boolean-object": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.2.1.tgz", - "integrity": "sha512-l9qO6eFlUETHtuihLcYOaLKByJ1f+N4kthcU9YjHy3N+B3hWv0y/2Nd0mu/7lTFnRQHTrSdXF50HQ3bl5fEnng==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-buffer": { - "version": "1.1.6", - "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", - "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", - "dev": true, - "license": "MIT" - }, - "node_modules/is-callable": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", - "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-core-module": { "version": "2.16.1", "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", @@ -12770,41 +8614,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/is-data-view": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.2.tgz", - "integrity": "sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "is-typed-array": "^1.1.13" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-date-object": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.1.0.tgz", - "integrity": "sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-docker": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-3.0.0.tgz", @@ -12830,22 +8639,6 @@ "node": ">=0.10.0" } }, - "node_modules/is-finalizationregistry": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.1.1.tgz", - "integrity": "sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-fullwidth-code-point": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-4.0.0.tgz", @@ -12858,25 +8651,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-generator-function": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.1.0.tgz", - "integrity": "sha512-nPUB5km40q9e8UfN/Zc24eLlzdSf9OfKByBw9CIdw4H1giPMeA0OIJvbchsCu4npfI2QcMVBsGEBHKZ7wLTWmQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "get-proto": "^1.0.0", - "has-tostringtag": "^1.0.2", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-glob": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", @@ -12908,55 +8682,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-invalid-path": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/is-invalid-path/-/is-invalid-path-0.1.0.tgz", - "integrity": "sha512-aZMG0T3F34mTg4eTdszcGXx54oiZ4NtHSft3hWNJMGJXUUqdIj3cOZuHcU0nCWWcY3jd7yRe/3AEm3vSNTpBGQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-glob": "^2.0.0" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-invalid-path/node_modules/is-extglob": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz", - "integrity": "sha512-7Q+VbVafe6x2T+Tu6NcOf6sRklazEPmBoB3IWk3WdGZM2iGUwU/Oe3Wtq5lSEkDTTlpp8yx+5t4pzO/i9Ty1ww==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-invalid-path/node_modules/is-glob": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz", - "integrity": "sha512-a1dBeB19NXsf/E0+FHqkagizel/LQw2DjSQpvQrj3zT+jYPpaUCryPnrQajXKFLCMuf4I6FhRpaGtw4lPrG6Eg==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-extglob": "^1.0.0" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-map": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", - "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-network-error": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/is-network-error/-/is-network-error-1.1.0.tgz", @@ -12979,23 +8704,6 @@ "node": ">=0.12.0" } }, - "node_modules/is-number-object": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.1.1.tgz", - "integrity": "sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-path-inside": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", @@ -13019,67 +8727,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-plain-object": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", - "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", - "dev": true, - "license": "MIT", - "dependencies": { - "isobject": "^3.0.1" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-regex": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.2.1.tgz", - "integrity": "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-set": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", - "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-shared-array-buffer": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.4.tgz", - "integrity": "sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-stream": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-4.0.1.tgz", @@ -13092,57 +8739,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-string": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.1.1.tgz", - "integrity": "sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-symbol": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.1.1.tgz", - "integrity": "sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "has-symbols": "^1.1.0", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-typed-array": { - "version": "1.1.15", - "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.15.tgz", - "integrity": "sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "which-typed-array": "^1.1.16" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-unicode-supported": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-2.1.0.tgz", @@ -13156,63 +8752,17 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-valid-path": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/is-valid-path/-/is-valid-path-0.1.1.tgz", - "integrity": "sha512-+kwPrVDu9Ms03L90Qaml+79+6DZHqHyRoANI6IsZJ/g8frhnfchDOBCa0RbQ6/kdHt5CS5OeIEyrYznNuVN+8A==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-invalid-path": "^0.1.0" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-weakmap": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", - "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", + "node_modules/is-what": { + "version": "4.1.16", + "resolved": "https://registry.npmjs.org/is-what/-/is-what-4.1.16.tgz", + "integrity": "sha512-ZhMwEosbFJkA0YhFnNDgTM4ZxDRsS6HqTo7qsZM08fehyRYIYa0yHu5R6mgo1n/8MgaPBXiPimPD77baVFYg+A==", "dev": true, "license": "MIT", "engines": { - "node": ">= 0.4" + "node": ">=12.13" }, "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakref": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.1.0.tgz", - "integrity": "sha512-SXM8Nwyys6nT5WP6pltOwKytLV7FqQ4UiibxVmW+EIosHcmCqkkjViTb5SNssDlkCiEYRP1/pdWUKVvZBmsR2Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakset": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.4.tgz", - "integrity": "sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "url": "https://github.com/sponsors/mesqueeb" } }, "node_modules/is-wsl": { @@ -13232,9 +8782,9 @@ } }, "node_modules/isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", "dev": true, "license": "MIT" }, @@ -13245,16 +8795,6 @@ "devOptional": true, "license": "ISC" }, - "node_modules/isobject": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", - "integrity": "sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/jackspeak": { "version": "3.4.3", "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz", @@ -13287,72 +8827,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-diff/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==", - "devOptional": true, - "license": "MIT", - "dependencies": { - "color-convert": "^2.0.1" - }, - "engines": { - "node": ">=8" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/jest-diff/node_modules/chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", - "devOptional": 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/jest-diff/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==", - "devOptional": true, - "license": "MIT", - "dependencies": { - "color-name": "~1.1.4" - }, - "engines": { - "node": ">=7.0.0" - } - }, - "node_modules/jest-diff/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==", - "devOptional": true, - "license": "MIT" - }, - "node_modules/jest-diff/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==", - "devOptional": true, - "license": "MIT", - "dependencies": { - "has-flag": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/jest-get-type": { "version": "29.6.3", "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", @@ -13363,284 +8837,13 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-matcher-utils": { - "version": "25.5.0", - "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-25.5.0.tgz", - "integrity": "sha512-VWI269+9JS5cpndnpCwm7dy7JtGQT30UHfrnM3mXl22gHGt/b7NkjBqXfbhZ8V4B7ANUsjK18PlSBmG0YH7gjw==", - "dev": true, - "license": "MIT", - "dependencies": { - "chalk": "^3.0.0", - "jest-diff": "^25.5.0", - "jest-get-type": "^25.2.6", - "pretty-format": "^25.5.0" - }, - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/jest-matcher-utils/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/jest-matcher-utils/node_modules/chalk": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", - "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", - "dev": true, - "license": "MIT", - "dependencies": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/jest-matcher-utils/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/jest-matcher-utils/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/jest-matcher-utils/node_modules/diff-sequences": { - "version": "25.2.6", - "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-25.2.6.tgz", - "integrity": "sha512-Hq8o7+6GaZeoFjtpgvRBUknSXNeJiCx7V9Fr94ZMljNiCr9n9L8H8aJqgWOQiDDGdyn29fRNcDdRVJ5fdyihfg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/jest-matcher-utils/node_modules/jest-diff": { - "version": "25.5.0", - "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-25.5.0.tgz", - "integrity": "sha512-z1kygetuPiREYdNIumRpAHY6RXiGmp70YHptjdaxTWGmA085W3iCnXNx0DhflK3vwrKmrRWyY1wUpkPMVxMK7A==", - "dev": true, - "license": "MIT", - "dependencies": { - "chalk": "^3.0.0", - "diff-sequences": "^25.2.6", - "jest-get-type": "^25.2.6", - "pretty-format": "^25.5.0" - }, - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/jest-matcher-utils/node_modules/jest-get-type": { - "version": "25.2.6", - "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-25.2.6.tgz", - "integrity": "sha512-DxjtyzOHjObRM+sM1knti6or+eOgcGU4xVSb2HNP1TqO4ahsT+rqZg+nyqHWJSvWgKC5cG3QjGFBqxLghiF/Ig==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/jest-matcher-utils/node_modules/pretty-format": { - "version": "25.5.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-25.5.0.tgz", - "integrity": "sha512-kbo/kq2LQ/A/is0PQwsEHM7Ca6//bGPPvU6UnsdDRSKTWxT/ru/xb88v4BJf6a69H+uTytOEsTusT9ksd/1iWQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jest/types": "^25.5.0", - "ansi-regex": "^5.0.0", - "ansi-styles": "^4.0.0", - "react-is": "^16.12.0" - }, - "engines": { - "node": ">= 8.3" - } - }, - "node_modules/jest-matcher-utils/node_modules/react-is": { - "version": "16.13.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", - "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==", - "dev": true, - "license": "MIT" - }, - "node_modules/jest-matcher-utils/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/jest-util": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.7.0.tgz", - "integrity": "sha512-z6EbKajIpqGKU56y5KBUgy1dt1ihhQJgWzUlZHArA/+X2ad7Cb5iF+AK1EWVL/Bo7Rz9uurpqw6SiBCefUbCGA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jest/types": "^29.6.3", - "@types/node": "*", - "chalk": "^4.0.0", - "ci-info": "^3.2.0", - "graceful-fs": "^4.2.9", - "picomatch": "^2.2.3" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-util/node_modules/@jest/types": { - "version": "29.6.3", - "resolved": "https://registry.npmjs.org/@jest/types/-/types-29.6.3.tgz", - "integrity": "sha512-u3UPsIilWKOM3F9CXtrG8LEJmNxwoCQC/XVj4IKYXvvpx7QIi/Kg1LI5uDmDpKlac62NUtX7eLjRh+jVZcLOzw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@jest/schemas": "^29.6.3", - "@types/istanbul-lib-coverage": "^2.0.0", - "@types/istanbul-reports": "^3.0.0", - "@types/node": "*", - "@types/yargs": "^17.0.8", - "chalk": "^4.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-util/node_modules/@types/istanbul-reports": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.4.tgz", - "integrity": "sha512-pk2B1NWalF9toCRu6gjBzR69syFjP4Od8WRAX+0mmf9lAjCRicLOWc+ZrxZHx/0XRjotgkF9t6iaMJ+aXcOdZQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/istanbul-lib-report": "*" - } - }, - "node_modules/jest-util/node_modules/@types/yargs": { - "version": "17.0.33", - "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.33.tgz", - "integrity": "sha512-WpxBCKWPLr4xSsHgz511rFJAM+wS28w2zEO1QDNY5zM/S8ok70NNfztH0xwhqKyaK0OHCbN98LDAZuy1ctxDkA==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/yargs-parser": "*" - } - }, - "node_modules/jest-util/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/jest-util/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/jest-util/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/jest-util/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/jest-util/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/jest-util/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/jest-worker": { "version": "27.5.1", "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", + "dev": true, "license": "MIT", + "peer": true, "dependencies": { "@types/node": "*", "merge-stream": "^2.0.0", @@ -13654,7 +8857,9 @@ "version": "8.1.1", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, "license": "MIT", + "peer": true, "dependencies": { "has-flag": "^4.0.0" }, @@ -13669,7 +8874,7 @@ "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, + "devOptional": true, "license": "MIT", "bin": { "jiti": "bin/jiti.js" @@ -13699,13 +8904,13 @@ "license": "MIT" }, "node_modules/js-yaml": { - "version": "3.14.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", - "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "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": "^1.0.7", - "esprima": "^4.0.0" + "argparse": "^2.0.1" }, "bin": { "js-yaml": "bin/js-yaml.js" @@ -13737,26 +8942,6 @@ "dev": true, "license": "MIT" }, - "node_modules/json-schema-deref-sync": { - "version": "0.14.0", - "resolved": "https://registry.npmjs.org/json-schema-deref-sync/-/json-schema-deref-sync-0.14.0.tgz", - "integrity": "sha512-yGR1xmhdiD6R0MSrwWcFxQzAj5b3i5Gb/mt5tvQKgFMMeNe0KZYNEN/jWr7G+xn39Azqgcvk4ZKMs8dQl8e4wA==", - "dev": true, - "license": "MIT", - "dependencies": { - "clone": "^2.1.2", - "dag-map": "~1.0.0", - "is-valid-path": "^0.1.1", - "lodash": "^4.17.13", - "md5": "~2.2.0", - "memory-cache": "~0.2.0", - "traverse": "~0.6.6", - "valid-url": "~1.0.9" - }, - "engines": { - "node": ">=6.0.0" - } - }, "node_modules/json-schema-traverse": { "version": "0.4.1", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", @@ -13772,9 +8957,9 @@ "license": "MIT" }, "node_modules/json11": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/json11/-/json11-2.0.0.tgz", - "integrity": "sha512-VuKJKUSPEJape+daTm70Nx7vdcdorf4S6LCyN2z0jUVH4UrQ4ftXo2kC0bnHpCREmxHuHqCNVPA75BjI3CB6Ag==", + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/json11/-/json11-2.0.2.tgz", + "integrity": "sha512-HIrd50UPYmP6sqLuLbFVm75g16o0oZrVfxrsY0EEys22klz8mRoWlX9KAEDOSOR9Q34rcxsyC8oDveGrCz5uLQ==", "license": "MIT", "bin": { "json11": "dist/cli.mjs" @@ -13786,6 +8971,7 @@ "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", "dev": true, "license": "MIT", + "peer": true, "bin": { "json5": "lib/cli.js" }, @@ -13836,15 +9022,6 @@ "json-buffer": "3.0.1" } }, - "node_modules/kind-of": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", - "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", - "license": "MIT", - "engines": { - "node": ">=0.10.0" - } - }, "node_modules/kleur": { "version": "4.1.5", "resolved": "https://registry.npmjs.org/kleur/-/kleur-4.1.5.tgz", @@ -13951,9 +9128,9 @@ } }, "node_modules/launch-editor": { - "version": "2.9.1", - "resolved": "https://registry.npmjs.org/launch-editor/-/launch-editor-2.9.1.tgz", - "integrity": "sha512-Gcnl4Bd+hRO9P9icCP/RVVT2o8SFlPXofuCxvA2SaZuH45whSvf5p8x5oih5ftLiVhEI4sp5xDY+R+b3zJBh5w==", + "version": "2.10.0", + "resolved": "https://registry.npmjs.org/launch-editor/-/launch-editor-2.10.0.tgz", + "integrity": "sha512-D7dBRJo/qcGX9xlvt/6wUYzQxjh5G1RvZPgPv8vi4KRU99DVQL/oW7tnVOCCTm2HGeo3C5HvGE5Yrh6UBoZ0vA==", "dev": true, "license": "MIT", "dependencies": { @@ -14012,21 +9189,6 @@ "node": ">=6.11.5" } }, - "node_modules/loader-utils": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-2.0.4.tgz", - "integrity": "sha512-xXqpXoINfFhgua9xiqD8fPFHgkoq1mmmpE92WlDbm9rNRd/EbRb+Gqf908T2DMfuHjjJlksiK2RbHVOdD/MqSw==", - "dev": true, - "license": "MIT", - "dependencies": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^2.1.2" - }, - "engines": { - "node": ">=8.9.0" - } - }, "node_modules/locate-path": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-7.2.0.tgz", @@ -14048,58 +9210,18 @@ "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", "license": "MIT" }, - "node_modules/lodash.clonedeep": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", - "integrity": "sha512-H5ZhCF25riFd9uB5UCkVKo61m3S/xZk1x4wA6yp/L3RFP6Z/eHH1ymQcGLo7J3GMPfm0V/7m1tryHuGVxpqEBQ==", - "license": "MIT" - }, - "node_modules/lodash.debounce": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz", - "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==", - "dev": true, - "license": "MIT" - }, "node_modules/lodash.defaults": { "version": "4.2.0", "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz", "integrity": "sha512-qjxPLHd3r5DnsdGacqOMU6pb/avJzdh9tFX2ymgoZE27BmjXrNy/y4LoaiTeAb+O3gL8AfpJGtqfX/ae2leYYQ==", "license": "MIT" }, - "node_modules/lodash.flatten": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz", - "integrity": "sha512-C5N2Z3DgnnKr0LOpv/hKCgKdb7ZZwafIrsesve6lmzvZIRZRGaZ/l6Q8+2W7NaT+ZwO3fFlSCzCzrDCFdJfZ4g==", - "dev": true, - "license": "MIT" - }, - "node_modules/lodash.get": { - "version": "4.4.2", - "resolved": "https://registry.npmjs.org/lodash.get/-/lodash.get-4.4.2.tgz", - "integrity": "sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ==", - "dev": true, - "license": "MIT" - }, "node_modules/lodash.isarguments": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/lodash.isarguments/-/lodash.isarguments-3.1.0.tgz", "integrity": "sha512-chi4NHZlZqZD18a0imDHnZPrDeBbTtVN7GXMwuGdRH9qotxAjYs3aVLKc7zNOG9eddR5Ksd8rvFEBc9SsggPpg==", "license": "MIT" }, - "node_modules/lodash.isequal": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", - "integrity": "sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ==", - "license": "MIT" - }, - "node_modules/lodash.memoize": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", - "integrity": "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==", - "dev": true, - "license": "MIT" - }, "node_modules/lodash.merge": { "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", @@ -14107,20 +9229,6 @@ "dev": true, "license": "MIT" }, - "node_modules/lodash.set": { - "version": "4.3.2", - "resolved": "https://registry.npmjs.org/lodash.set/-/lodash.set-4.3.2.tgz", - "integrity": "sha512-4hNPN5jlm/N/HLMCO43v8BXKq9Z7QdAGc/VGrRD61w8gN9g/6jF9A4L1pbUgBLCffi0w9VsXfTOij5x8iTyFvg==", - "dev": true, - "license": "MIT" - }, - "node_modules/lodash.uniq": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", - "integrity": "sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ==", - "dev": true, - "license": "MIT" - }, "node_modules/log-update": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/log-update/-/log-update-6.1.0.tgz", @@ -14211,21 +9319,12 @@ } }, "node_modules/loupe": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.1.2.tgz", - "integrity": "sha512-23I4pFZHmAemUnz8WZXbYRSKYj801VDaNv9ETuMh7IrMc7VuVVSo+Z9iLE3ni30+U48iDWfi30d3twAXBYmnCg==", + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/loupe/-/loupe-3.1.3.tgz", + "integrity": "sha512-kkIp7XSkP78ZxJEsSxW3712C6teJVoeHHwgo9zJ380de7IYyJ2ISlxojcH2pC5OFLewESmnRi/+XCDIEEVyoug==", "dev": true, "license": "MIT" }, - "node_modules/lower-case": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.2.tgz", - "integrity": "sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==", - "license": "MIT", - "dependencies": { - "tslib": "^2.0.3" - } - }, "node_modules/lowercase-keys": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-3.0.0.tgz", @@ -14244,26 +9343,20 @@ "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", "dev": true, "license": "ISC", + "peer": true, "dependencies": { "yallist": "^3.0.2" } }, "node_modules/luxon": { - "version": "3.5.0", - "resolved": "https://registry.npmjs.org/luxon/-/luxon-3.5.0.tgz", - "integrity": "sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==", + "version": "3.6.1", + "resolved": "https://registry.npmjs.org/luxon/-/luxon-3.6.1.tgz", + "integrity": "sha512-tJLxrKJhO2ukZ5z0gyjY1zPh3Rh88Ej9P7jNrZiHMUXHae1yvI2imgOZtL1TO8TW6biMMKfTtAOoEJANgtWBMQ==", "license": "MIT", "engines": { "node": ">=12" } }, - "node_modules/macroable": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/macroable/-/macroable-7.0.2.tgz", - "integrity": "sha512-QS9p+Q20YBxpE0dJBnF6CPURP7p1GUsxnhTxTWH5nG3A1F5w8Rg3T4Xyh5UlrFSbHp88oOciVP/0agsNLhkHdQ==", - "license": "MIT", - "peer": true - }, "node_modules/magic-string": { "version": "0.30.17", "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.17.tgz", @@ -14293,15 +9386,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/make-dir/node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "license": "ISC", - "bin": { - "semver": "bin/semver.js" - } - }, "node_modules/make-error": { "version": "1.3.6", "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", @@ -14318,54 +9402,43 @@ "node": ">= 0.4" } }, - "node_modules/md5": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/md5/-/md5-2.2.1.tgz", - "integrity": "sha512-PlGG4z5mBANDGCKsYQe0CaUYHdZYZt8ZPZLmEt+Urf0W4GlpTX4HescwHU+dc9+Z/G/vZKYZYFrwgm9VxK6QOQ==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "charenc": "~0.0.1", - "crypt": "~0.0.1", - "is-buffer": "~1.1.1" - } - }, - "node_modules/mdn-data": { - "version": "2.0.30", - "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.30.tgz", - "integrity": "sha512-GaqWWShW4kv/G9IEucWScBx9G1/vsFZZJUO+tD26M8J8z3Kw5RDQjaoZe03YAClgeS/SWPOcb4nkFBTEi5DUEA==", - "dev": true, - "license": "CC0-1.0" - }, "node_modules/media-typer": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", - "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz", + "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==", "license": "MIT", "engines": { - "node": ">= 0.6" + "node": ">= 0.8" } }, "node_modules/memfs": { - "version": "3.5.3", - "resolved": "https://registry.npmjs.org/memfs/-/memfs-3.5.3.tgz", - "integrity": "sha512-UERzLsxzllchadvbPs5aolHh65ISpKpM+ccLbOJ8/vvpBKmAWf+la7dXFy7Mr0ySHbdHrFv5kGFCUHHe6GFEmw==", - "license": "Unlicense", + "version": "4.17.0", + "resolved": "https://registry.npmjs.org/memfs/-/memfs-4.17.0.tgz", + "integrity": "sha512-4eirfZ7thblFmqFjywlTmuWVSvccHAJbn1r8qQLzmTO11qcqpohOjmY2mFce6x7x7WtskzRqApPD0hv+Oa74jg==", + "dev": true, + "license": "Apache-2.0", "dependencies": { - "fs-monkey": "^1.0.4" + "@jsonjoy.com/json-pack": "^1.0.3", + "@jsonjoy.com/util": "^1.3.0", + "tree-dump": "^1.0.1", + "tslib": "^2.0.0" }, "engines": { "node": ">= 4.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/streamich" } }, "node_modules/memoize": { - "version": "10.0.0", - "resolved": "https://registry.npmjs.org/memoize/-/memoize-10.0.0.tgz", - "integrity": "sha512-H6cBLgsi6vMWOcCpvVCdFFnl3kerEXbrYh9q+lY6VXvQSmM6CkmV08VOwT+WE2tzIEqRPFfAq3fm4v/UIW6mSA==", + "version": "10.1.0", + "resolved": "https://registry.npmjs.org/memoize/-/memoize-10.1.0.tgz", + "integrity": "sha512-MMbFhJzh4Jlg/poq1si90XRlTZRDHVqdlz2mPyGJ6kqMpyHUyVpDd5gpFAvVehW64+RA1eKE9Yt8aSLY7w2Kgg==", "devOptional": true, "license": "MIT", "dependencies": { - "mimic-function": "^5.0.0" + "mimic-function": "^5.0.1" }, "engines": { "node": ">=18" @@ -14374,13 +9447,6 @@ "url": "https://github.com/sindresorhus/memoize?sponsor=1" } }, - "node_modules/memory-cache": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/memory-cache/-/memory-cache-0.2.0.tgz", - "integrity": "sha512-OcjA+jzjOYzKmKS6IQVALHLVz+rNTMPoJvCztFaZxwG14wtAW7VRZjwTQu06vKCYOxh4jVnik7ya0SXTB0W+xA==", - "dev": true, - "license": "BSD-2-Clause" - }, "node_modules/merge-descriptors": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", @@ -14395,7 +9461,9 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", - "license": "MIT" + "dev": true, + "license": "MIT", + "peer": true }, "node_modules/merge2": { "version": "1.4.1", @@ -14411,7 +9479,7 @@ "version": "1.1.2", "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", - "devOptional": true, + "dev": true, "license": "MIT", "engines": { "node": ">= 0.6" @@ -14443,9 +9511,9 @@ } }, "node_modules/mime": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/mime/-/mime-4.0.6.tgz", - "integrity": "sha512-4rGt7rvQHBbaSOF9POGkk1ocRP16Md1x36Xma8sz8h8/vfCUI2OtEIeCqe4Ofes853x4xDoPiFLIT47J5fI/7A==", + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/mime/-/mime-4.0.7.tgz", + "integrity": "sha512-2OfDPL+e03E0LrXaGYOtTFIYhiuzep94NSsuhrNULq+stylcJedcHdzHtz0atMUuGwJfFYs0YL5xeC/Ca2x0eQ==", "funding": [ "https://github.com/sponsors/broofa" ], @@ -14502,27 +9570,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/mini-css-extract-plugin": { - "version": "2.9.2", - "resolved": "https://registry.npmjs.org/mini-css-extract-plugin/-/mini-css-extract-plugin-2.9.2.tgz", - "integrity": "sha512-GJuACcS//jtq4kCtd5ii/M0SZf7OZRH+BxdqXZHaJfb8TJiVl+NgQRPwiYt2EuqeSkNydn/7vP+bcE27C5mb9w==", - "dev": true, - "license": "MIT", - "dependencies": { - "schema-utils": "^4.0.0", - "tapable": "^2.2.1" - }, - "engines": { - "node": ">= 12.13.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/webpack" - }, - "peerDependencies": { - "webpack": "^5.0.0" - } - }, "node_modules/mini-svg-data-uri": { "version": "1.4.4", "resolved": "https://registry.npmjs.org/mini-svg-data-uri/-/mini-svg-data-uri-1.4.4.tgz", @@ -14662,9 +9709,9 @@ } }, "node_modules/nanoid": { - "version": "3.3.8", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz", - "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==", + "version": "3.3.11", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz", + "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==", "funding": [ { "type": "github", @@ -14710,16 +9757,6 @@ "license": "MIT", "peer": true }, - "node_modules/no-case": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.4.tgz", - "integrity": "sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==", - "license": "MIT", - "dependencies": { - "lower-case": "^2.0.2", - "tslib": "^2.0.3" - } - }, "node_modules/node-2fa": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/node-2fa/-/node-2fa-2.0.3.tgz", @@ -14782,9 +9819,9 @@ "license": "MIT" }, "node_modules/nodemailer": { - "version": "6.9.16", - "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.9.16.tgz", - "integrity": "sha512-psAuZdTIRN08HKVd/E8ObdV6NO7NTBY3KsC30F7M4H1OnmLCUNaS56FpYxyb26zWLSyYF9Ozch9KYHhHegsiOQ==", + "version": "6.10.1", + "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-6.10.1.tgz", + "integrity": "sha512-Z+iLaBGVaSjbIzQ4pX6XV41HrooLsQ10ZWPUehGmuantvzWoDVBnmsdUcOIDM1t+yPor5pDhVlDESgOMEGxhHA==", "license": "MIT-0", "engines": { "node": ">=6.0.0" @@ -14805,6 +9842,34 @@ "node": ">=6" } }, + "node_modules/normalize-package-data": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-6.0.2.tgz", + "integrity": "sha512-V6gygoYb/5EmNI+MEGrWkC+e6+Rr7mTmfHrxDbLzxQogBkgzo76rkok0Am6thgSF7Mv2nLOajAJj5vDJZEFn7g==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "hosted-git-info": "^7.0.0", + "semver": "^7.3.5", + "validate-npm-package-license": "^3.0.4" + }, + "engines": { + "node": "^16.14.0 || >=18.0.0" + } + }, + "node_modules/normalize-package-data/node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/normalize-path": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", @@ -14887,19 +9952,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/npm-run-path/node_modules/unicorn-magic": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/unicorn-magic/-/unicorn-magic-0.3.0.tgz", - "integrity": "sha512-+QBBXBCvifc56fsbuxZQ6Sic3wqqc3WWaqxs58gvJrcOuN83HGTCwz3oS5phzU9LthRNE9VrJCFCLUgHeeFnfA==", - "devOptional": true, - "license": "MIT", - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, "node_modules/npmlog": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-5.0.1.tgz", @@ -14913,25 +9965,6 @@ "set-blocking": "^2.0.0" } }, - "node_modules/nprogress": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/nprogress/-/nprogress-0.2.0.tgz", - "integrity": "sha512-I19aIingLgR1fmhftnbWWO3dXc0hSxqHQHQb3H8m+K3TnEn/iSeTZZOyvKXWqQESMwuUVnatlCnZdLBZZt2VSA==", - "license": "MIT" - }, - "node_modules/nth-check": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz", - "integrity": "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "boolbase": "^1.0.0" - }, - "funding": { - "url": "https://github.com/fb55/nth-check?sponsor=1" - } - }, "node_modules/numeral": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/numeral/-/numeral-2.0.6.tgz", @@ -14962,9 +9995,9 @@ } }, "node_modules/object-inspect": { - "version": "1.13.3", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.3.tgz", - "integrity": "sha512-kDCGIbxkDSXE3euJZZXzc6to7fCrKHNI/hSRQnRuQ+BWjFNzZwiFF8fj/6o2t2G9/jTj8PSIYTfCLelLZEeRpA==", + "version": "1.13.4", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", + "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", "license": "MIT", "engines": { "node": ">= 0.4" @@ -14973,37 +10006,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/object-keys": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", - "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object.assign": { - "version": "4.1.7", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.7.tgz", - "integrity": "sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0", - "has-symbols": "^1.1.0", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/obuf": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/obuf/-/obuf-1.1.2.tgz", @@ -15067,9 +10069,9 @@ } }, "node_modules/open": { - "version": "10.1.0", - "resolved": "https://registry.npmjs.org/open/-/open-10.1.0.tgz", - "integrity": "sha512-mnkeQ1qP5Ue2wd+aivTD3NHd/lZ96Lu0jgf0pwktLPtx6cTZiH7tyeGRRHs0zX0rbrahXPnXlUnbeXyaBBuIaw==", + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/open/-/open-10.1.1.tgz", + "integrity": "sha512-zy1wx4+P3PfhXSEPJNtZmJXfhkkIaxU1VauWIrDZw1O7uJRDRJtKr9n3Ic4NgbA16KyOxOXO2ng9gYwCdXuSXA==", "dev": true, "license": "MIT", "dependencies": { @@ -15085,26 +10087,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/openapi-schema-validator": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/openapi-schema-validator/-/openapi-schema-validator-3.0.3.tgz", - "integrity": "sha512-KKpeNEvAmpy6B2JCfyrM4yWjL6vggDCVbBoR8Yfkj0Jltc6PCW+dBbcg+1yrTCuDv80qBQJ6w0ejA71DlOFegA==", - "dev": true, - "license": "MIT", - "dependencies": { - "ajv": "^6.5.2", - "lodash.merge": "^4.6.1", - "openapi-types": "1.3.4", - "swagger-schema-official": "2.0.0-bab6bed" - } - }, - "node_modules/openapi-types": { - "version": "1.3.4", - "resolved": "https://registry.npmjs.org/openapi-types/-/openapi-types-1.3.4.tgz", - "integrity": "sha512-h8rADpW3k/wepLdERKF0VKMAPdoFYNQCLGPmc/f8sgQ2dxUy+7sY4WAX2XDUDjhKTjbJVbxxofLkzy7f1/tE4g==", - "dev": true, - "license": "MIT" - }, "node_modules/optionator": { "version": "0.9.4", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", @@ -15123,24 +10105,6 @@ "node": ">= 0.8.0" } }, - "node_modules/own-keys": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/own-keys/-/own-keys-1.0.1.tgz", - "integrity": "sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "get-intrinsic": "^1.2.6", - "object-keys": "^1.1.1", - "safe-push-apply": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/p-cancelable": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-4.0.1.tgz", @@ -15273,19 +10237,12 @@ "license": "BlueOak-1.0.0" }, "node_modules/package-manager-detector": { - "version": "0.2.8", - "resolved": "https://registry.npmjs.org/package-manager-detector/-/package-manager-detector-0.2.8.tgz", - "integrity": "sha512-ts9KSdroZisdvKMWVAVCXiKqnqNfXz4+IbrBG8/BWx/TR5le+jfenvoBuIZ6UWM9nz47W7AbD9qYfAwfWMIwzA==", - "license": "MIT" - }, - "node_modules/param-case": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/param-case/-/param-case-3.0.4.tgz", - "integrity": "sha512-RXlj7zCYokReqWpOPH9oYivUzLYZ5vAPIfEmCTNViosC78F8F0H9y7T7gG2M39ymgutxF5gcFEsyZQSph9Bp3A==", + "version": "0.2.11", + "resolved": "https://registry.npmjs.org/package-manager-detector/-/package-manager-detector-0.2.11.tgz", + "integrity": "sha512-BEnLolu+yuz22S56CU1SUKq3XC3PkwD5wv4ikR4MfGvnRVcmzXR9DwSlW2fEamyTPyXHomBJRzgapeuBvRNzJQ==", "license": "MIT", "dependencies": { - "dot-case": "^3.0.4", - "tslib": "^2.0.3" + "quansync": "^0.2.7" } }, "node_modules/parent-module": { @@ -15355,16 +10312,6 @@ "node": ">= 0.8" } }, - "node_modules/pascal-case": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-3.1.2.tgz", - "integrity": "sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g==", - "license": "MIT", - "dependencies": { - "no-case": "^3.0.4", - "tslib": "^2.0.3" - } - }, "node_modules/path-browserify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-1.0.1.tgz", @@ -15372,16 +10319,6 @@ "devOptional": true, "license": "MIT" }, - "node_modules/path-case": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/path-case/-/path-case-3.0.4.tgz", - "integrity": "sha512-qO4qCFjXqVTrcbPt/hQfhTQ+VhFsqNKOPtytgNKkKxSoEp3XPUQ8ObFuePylOIok5gjn69ry8XiULxCwot3Wfg==", - "license": "MIT", - "dependencies": { - "dot-case": "^3.0.4", - "tslib": "^2.0.3" - } - }, "node_modules/path-exists": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", @@ -15448,13 +10385,13 @@ "license": "MIT" }, "node_modules/path-type": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/path-type/-/path-type-5.0.0.tgz", - "integrity": "sha512-5HviZNaZcfqP95rwpv+1HDgUamezbqdSYTyzjTvwtJSnIH+3vnbmWsItli8OFEndS984VT55M3jduxZbX351gg==", + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-6.0.0.tgz", + "integrity": "sha512-Vj7sf++t5pBD637NSfkxpHSMfWaeig5+DKWLhcqIYx6mWQz5hdJTGDVMQiJcw1ZYkhs7AazKDGpRVji1LJCZUQ==", "devOptional": true, "license": "MIT", "engines": { - "node": ">=12" + "node": ">=18" }, "funding": { "url": "https://github.com/sponsors/sindresorhus" @@ -15471,27 +10408,34 @@ } }, "node_modules/peek-readable": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/peek-readable/-/peek-readable-5.3.1.tgz", - "integrity": "sha512-GVlENSDW6KHaXcd9zkZltB7tCLosKB/4Hg0fqBJkAoBgYG2Tn1xtMgXtSUuMU9AK/gCm/tTdT8mgAeF4YNeeqw==", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/peek-readable/-/peek-readable-7.0.0.tgz", + "integrity": "sha512-nri2TO5JE3/mRryik9LlHFT53cgHfRK0Lt0BAZQXku/AW3E6XLt2GaY8siWi7dvW/m1z0ecn+J+bpDa9ZN3IsQ==", "license": "MIT", "engines": { - "node": ">=14.16" + "node": ">=18" }, "funding": { "type": "github", "url": "https://github.com/sponsors/Borewit" } }, + "node_modules/perfect-debounce": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/perfect-debounce/-/perfect-debounce-1.0.0.tgz", + "integrity": "sha512-xCy9V055GLEqoFaHoC1SoLIaLmWctgCUaBaWxDZ7/Zx4CTyX7cJQLJOok/orfjZAh9kEYpjJa4d0KcJmCbctZA==", + "dev": true, + "license": "MIT" + }, "node_modules/pg": { - "version": "8.13.1", - "resolved": "https://registry.npmjs.org/pg/-/pg-8.13.1.tgz", - "integrity": "sha512-OUir1A0rPNZlX//c7ksiu7crsGZTKSOXJPgtNiHGIlC9H0lO+NC6ZDYksSgBYY/thSWhnSRBv8w1lieNNGATNQ==", + "version": "8.15.6", + "resolved": "https://registry.npmjs.org/pg/-/pg-8.15.6.tgz", + "integrity": "sha512-yvao7YI3GdmmrslNVsZgx9PfntfWrnXwtR+K/DjI0I/sTKif4Z623um+sjVZ1hk5670B+ODjvHDAckKdjmPTsg==", "license": "MIT", "dependencies": { - "pg-connection-string": "^2.7.0", - "pg-pool": "^3.7.0", - "pg-protocol": "^1.7.0", + "pg-connection-string": "^2.8.5", + "pg-pool": "^3.9.6", + "pg-protocol": "^1.9.5", "pg-types": "^2.1.0", "pgpass": "1.x" }, @@ -15499,7 +10443,7 @@ "node": ">= 8.0.0" }, "optionalDependencies": { - "pg-cloudflare": "^1.1.1" + "pg-cloudflare": "^1.2.5" }, "peerDependencies": { "pg-native": ">=3.0.1" @@ -15511,9 +10455,9 @@ } }, "node_modules/pg-cloudflare": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/pg-cloudflare/-/pg-cloudflare-1.1.1.tgz", - "integrity": "sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==", + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/pg-cloudflare/-/pg-cloudflare-1.2.5.tgz", + "integrity": "sha512-OOX22Vt0vOSRrdoUPKJ8Wi2OpE/o/h9T8X1s4qSkCedbNah9ei2W2765be8iMVxQUsvgT7zIAT2eIa9fs5+vtg==", "license": "MIT", "optional": true }, @@ -15533,18 +10477,18 @@ } }, "node_modules/pg-pool": { - "version": "3.7.0", - "resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.7.0.tgz", - "integrity": "sha512-ZOBQForurqh4zZWjrgSwwAtzJ7QiRX0ovFkZr2klsen3Nm0aoh33Ls0fzfv3imeH/nw/O27cjdz5kzYJfeGp/g==", + "version": "3.9.6", + "resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.9.6.tgz", + "integrity": "sha512-rFen0G7adh1YmgvrmE5IPIqbb+IgEzENUm+tzm6MLLDSlPRoZVhzU1WdML9PV2W5GOdRA9qBKURlbt1OsXOsPw==", "license": "MIT", "peerDependencies": { "pg": ">=8.0" } }, "node_modules/pg-protocol": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.7.0.tgz", - "integrity": "sha512-hTK/mE36i8fDDhgDFjy6xNOG+LCorxLG3WO17tku+ij6sVHXh1jQUJ8hYAnRhNla4QVD2H8er/FOjc/+EgC6yQ==", + "version": "1.9.5", + "resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.9.5.tgz", + "integrity": "sha512-DYTWtWpfd5FOro3UnAfwvhD8jh59r2ig8bPtc9H8Ds7MscE/9NYruUQWFAOuraRl29jwcT2kyMFQ3MxeaVjUhg==", "license": "MIT" }, "node_modules/pg-types": { @@ -15564,9 +10508,9 @@ } }, "node_modules/pg/node_modules/pg-connection-string": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/pg-connection-string/-/pg-connection-string-2.7.0.tgz", - "integrity": "sha512-PI2W9mv53rXJQEOb8xNR8lH7Hr+EKa6oJa38zsK0S/ky2er16ios1wLKhZyxzD7jUReiWokc9WK5nxSnC7W1TA==", + "version": "2.8.5", + "resolved": "https://registry.npmjs.org/pg-connection-string/-/pg-connection-string-2.8.5.tgz", + "integrity": "sha512-Ni8FuZ8yAF+sWZzojvtLE2b03cqjO5jNULcHFfM9ZZ0/JXrgom5pBREbtnAw7oxsxJqHw9Nz/XWORUEL3/IFow==", "license": "MIT" }, "node_modules/pgpass": { @@ -15588,7 +10532,6 @@ "version": "4.0.2", "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz", "integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==", - "devOptional": true, "license": "MIT", "engines": { "node": ">=12" @@ -15608,14 +10551,13 @@ } }, "node_modules/pinia": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pinia/-/pinia-2.3.0.tgz", - "integrity": "sha512-ohZj3jla0LL0OH5PlLTDMzqKiVw2XARmC1XYLdLWIPBMdhDW/123ZWr4zVAhtJm+aoSkFa13pYXskAvAscIkhQ==", + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/pinia/-/pinia-3.0.2.tgz", + "integrity": "sha512-sH2JK3wNY809JOeiiURUR0wehJ9/gd9qFN2Y828jCbxEzKEmEt0pzCXwqiSTfuRsK9vQsOflSdnbdBOGrhtn+g==", "dev": true, "license": "MIT", "dependencies": { - "@vue/devtools-api": "^6.6.3", - "vue-demi": "^0.14.10" + "@vue/devtools-api": "^7.7.2" }, "funding": { "url": "https://github.com/sponsors/posva" @@ -15631,19 +10573,22 @@ } }, "node_modules/pino": { - "version": "6.14.0", - "resolved": "https://registry.npmjs.org/pino/-/pino-6.14.0.tgz", - "integrity": "sha512-iuhEDel3Z3hF9Jfe44DPXR8l07bhjuFY3GMHIXbjnY9XcafbyDDwl2sN2vw2GjMPf5Nkoe+OFao7ffn9SXaKDg==", + "version": "9.6.0", + "resolved": "https://registry.npmjs.org/pino/-/pino-9.6.0.tgz", + "integrity": "sha512-i85pKRCt4qMjZ1+L7sy2Ag4t1atFcdbEt76+7iRJn1g2BvsnRMGu9p8pivl9fs63M2kF/A0OacFZhTub+m/qMg==", "license": "MIT", - "peer": true, "dependencies": { - "fast-redact": "^3.0.0", - "fast-safe-stringify": "^2.0.8", - "flatstr": "^1.0.12", - "pino-std-serializers": "^3.1.0", - "process-warning": "^1.0.0", + "atomic-sleep": "^1.0.0", + "fast-redact": "^3.1.1", + "on-exit-leak-free": "^2.1.0", + "pino-abstract-transport": "^2.0.0", + "pino-std-serializers": "^7.0.0", + "process-warning": "^4.0.0", "quick-format-unescaped": "^4.0.3", - "sonic-boom": "^1.0.2" + "real-require": "^0.2.0", + "safe-stable-stringify": "^2.3.1", + "sonic-boom": "^4.0.1", + "thread-stream": "^3.0.0" }, "bin": { "pino": "bin.js" @@ -15659,9 +10604,9 @@ } }, "node_modules/pino-pretty": { - "version": "11.3.0", - "resolved": "https://registry.npmjs.org/pino-pretty/-/pino-pretty-11.3.0.tgz", - "integrity": "sha512-oXwn7ICywaZPHmu3epHGU2oJX4nPmKvHvB/bwrJHlGcbEWaVcotkpyVHMKLKmiVryWYByNp0jpgAcXpFJDXJzA==", + "version": "13.0.0", + "resolved": "https://registry.npmjs.org/pino-pretty/-/pino-pretty-13.0.0.tgz", + "integrity": "sha512-cQBBIVG3YajgoUjo1FdKVRX6t9XPxwB9lcNJVD5GCnNM4Y6T12YYx8c6zEejxQsU0wrg9TwmDulcE9LR7qcJqA==", "dev": true, "license": "MIT", "dependencies": { @@ -15675,7 +10620,6 @@ "on-exit-leak-free": "^2.1.0", "pino-abstract-transport": "^2.0.0", "pump": "^3.0.0", - "readable-stream": "^4.0.0", "secure-json-parse": "^2.4.0", "sonic-boom": "^4.0.1", "strip-json-comments": "^3.1.1" @@ -15684,55 +10628,16 @@ "pino-pretty": "bin.js" } }, - "node_modules/pino-pretty/node_modules/readable-stream": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-4.7.0.tgz", - "integrity": "sha512-oIGGmcpTLwPga8Bn6/Z75SVaH1z5dUut2ibSyAMVhmUggWpmDn2dapB0n7f8nwaSiRtepAsfJyfXIO5DCVAODg==", - "dev": true, - "license": "MIT", - "dependencies": { - "abort-controller": "^3.0.0", - "buffer": "^6.0.3", - "events": "^3.3.0", - "process": "^0.11.10", - "string_decoder": "^1.3.0" - }, - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - } - }, - "node_modules/pino-pretty/node_modules/sonic-boom": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/sonic-boom/-/sonic-boom-4.2.0.tgz", - "integrity": "sha512-INb7TM37/mAcsGmc9hyyI6+QR3rR1zVRu36B0NeGXKnOOLiZOfER5SA+N7X7k3yUYRzLWafduTDvJAfDswwEww==", - "dev": true, - "license": "MIT", - "dependencies": { - "atomic-sleep": "^1.0.0" - } - }, "node_modules/pino-std-serializers": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-3.2.0.tgz", - "integrity": "sha512-EqX4pwDPrt3MuOAAUBMU0Tk5kR/YcCM5fNPEzgCO2zJ5HfX0vbiH9HbJglnyeQsN96Kznae6MWD47pZB5avTrg==", - "license": "MIT", - "peer": true - }, - "node_modules/pino/node_modules/sonic-boom": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/sonic-boom/-/sonic-boom-1.4.1.tgz", - "integrity": "sha512-LRHh/A8tpW7ru89lrlkU4AszXt1dbwSjVWguGrmlxE7tawVmDBlI1PILMkXAxJTwqhgsEeTHzj36D5CmHgQmNg==", - "license": "MIT", - "peer": true, - "dependencies": { - "atomic-sleep": "^1.0.0", - "flatstr": "^1.0.12" - } + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-7.0.0.tgz", + "integrity": "sha512-e906FRY0+tV27iq4juKzSYPbUj2do2X2JX4EzSca1631EB2QJQUqGbDuERal7LCtOpxl6x3+nvo9NPZcmjkiFA==", + "license": "MIT" }, "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==", + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.7.tgz", + "integrity": "sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==", "dev": true, "license": "MIT", "engines": { @@ -15800,20 +10705,10 @@ "node": ">=10.13.0" } }, - "node_modules/possible-typed-array-names": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", - "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.4" - } - }, "node_modules/postcss": { - "version": "8.4.49", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.49.tgz", - "integrity": "sha512-OCVPnIObs4N29kxTjzLfUryOkvZEq+pf8jTF0lg8E7uETuWHA+v7j3c/xJmiqpX450191LlmZfUKkXxkTry7nA==", + "version": "8.5.3", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.3.tgz", + "integrity": "sha512-dle9A3yYxlBSrt8Fu+IpjGT8SY8hN0mlaA6GY8t0P5PjIOZemULz/E2Bnm/2dcUOena75OTNkHI76uZBNUUq3A==", "funding": [ { "type": "opencollective", @@ -15830,7 +10725,7 @@ ], "license": "MIT", "dependencies": { - "nanoid": "^3.3.7", + "nanoid": "^3.3.8", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" }, @@ -15838,128 +10733,6 @@ "node": "^10 || ^12 || >=14" } }, - "node_modules/postcss-calc": { - "version": "10.1.0", - "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-10.1.0.tgz", - "integrity": "sha512-uQ/LDGsf3mgsSUEXmAt3VsCSHR3aKqtEIkmB+4PhzYwRYOW5MZs/GhCCFpsOtJJkP6EC6uGipbrnaTjqaJZcJw==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-selector-parser": "^7.0.0", - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12 || ^20.9 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.38" - } - }, - "node_modules/postcss-colormin": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-7.0.2.tgz", - "integrity": "sha512-YntRXNngcvEvDbEjTdRWGU606eZvB5prmHG4BF0yLmVpamXbpsRJzevyy6MZVyuecgzI2AWAlvFi8DAeCqwpvA==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.23.3", - "caniuse-api": "^3.0.0", - "colord": "^2.9.3", - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-convert-values": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/postcss-convert-values/-/postcss-convert-values-7.0.4.tgz", - "integrity": "sha512-e2LSXPqEHVW6aoGbjV9RsSSNDO3A0rZLCBxN24zvxF25WknMPpX8Dm9UxxThyEbaytzggRuZxaGXqaOhxQ514Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.23.3", - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-discard-comments": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/postcss-discard-comments/-/postcss-discard-comments-7.0.3.tgz", - "integrity": "sha512-q6fjd4WU4afNhWOA2WltHgCbkRhZPgQe7cXF74fuVB/ge4QbM9HEaOIzGSiMvM+g/cOsNAUGdf2JDzqA2F8iLA==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-selector-parser": "^6.1.2" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-discard-comments/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-discard-duplicates": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/postcss-discard-duplicates/-/postcss-discard-duplicates-7.0.1.tgz", - "integrity": "sha512-oZA+v8Jkpu1ct/xbbrntHRsfLGuzoP+cpt0nJe5ED2FQF8n8bJtn7Bo28jSmBYwqgqnqkuSXJfSUEE7if4nClQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-discard-empty": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-discard-empty/-/postcss-discard-empty-7.0.0.tgz", - "integrity": "sha512-e+QzoReTZ8IAwhnSdp/++7gBZ/F+nBq9y6PomfwORfP7q9nBpK5AMP64kOt0bA+lShBFbBDcgpJ3X4etHg4lzA==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-discard-overridden": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-discard-overridden/-/postcss-discard-overridden-7.0.0.tgz", - "integrity": "sha512-GmNAzx88u3k2+sBTZrJSDauR0ccpE24omTQCVmaTTZFz1du6AasspjaUPMJ2ud4RslZpoFKyf+6MSPETLojc6w==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, "node_modules/postcss-import": { "version": "15.1.0", "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz", @@ -16066,200 +10839,17 @@ } } }, - "node_modules/postcss-merge-longhand": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/postcss-merge-longhand/-/postcss-merge-longhand-7.0.4.tgz", - "integrity": "sha512-zer1KoZA54Q8RVHKOY5vMke0cCdNxMP3KBfDerjH/BYHh4nCIh+1Yy0t1pAEQF18ac/4z3OFclO+ZVH8azjR4A==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0", - "stylehacks": "^7.0.4" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-merge-rules": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/postcss-merge-rules/-/postcss-merge-rules-7.0.4.tgz", - "integrity": "sha512-ZsaamiMVu7uBYsIdGtKJ64PkcQt6Pcpep/uO90EpLS3dxJi6OXamIobTYcImyXGoW0Wpugh7DSD3XzxZS9JCPg==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.23.3", - "caniuse-api": "^3.0.0", - "cssnano-utils": "^5.0.0", - "postcss-selector-parser": "^6.1.2" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-merge-rules/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-minify-font-values": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-minify-font-values/-/postcss-minify-font-values-7.0.0.tgz", - "integrity": "sha512-2ckkZtgT0zG8SMc5aoNwtm5234eUx1GGFJKf2b1bSp8UflqaeFzR50lid4PfqVI9NtGqJ2J4Y7fwvnP/u1cQog==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-minify-gradients": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-minify-gradients/-/postcss-minify-gradients-7.0.0.tgz", - "integrity": "sha512-pdUIIdj/C93ryCHew0UgBnL2DtUS3hfFa5XtERrs4x+hmpMYGhbzo6l/Ir5de41O0GaKVpK1ZbDNXSY6GkXvtg==", - "dev": true, - "license": "MIT", - "dependencies": { - "colord": "^2.9.3", - "cssnano-utils": "^5.0.0", - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-minify-params": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/postcss-minify-params/-/postcss-minify-params-7.0.2.tgz", - "integrity": "sha512-nyqVLu4MFl9df32zTsdcLqCFfE/z2+f8GE1KHPxWOAmegSo6lpV2GNy5XQvrzwbLmiU7d+fYay4cwto1oNdAaQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.23.3", - "cssnano-utils": "^5.0.0", - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-minify-selectors": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/postcss-minify-selectors/-/postcss-minify-selectors-7.0.4.tgz", - "integrity": "sha512-JG55VADcNb4xFCf75hXkzc1rNeURhlo7ugf6JjiiKRfMsKlDzN9CXHZDyiG6x/zGchpjQS+UAgb1d4nqXqOpmA==", - "dev": true, - "license": "MIT", - "dependencies": { - "cssesc": "^3.0.0", - "postcss-selector-parser": "^6.1.2" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-minify-selectors/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-modules-extract-imports": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-3.1.0.tgz", - "integrity": "sha512-k3kNe0aNFQDAZGbin48pL2VNidTF0w4/eASDsxlyspobzU3wZQLOGj7L9gfRe0Jo9/4uud09DsjFNH7winGv8Q==", + "node_modules/postcss-loader/node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", "dev": true, "license": "ISC", - "engines": { - "node": "^10 || ^12 || >= 14" - }, - "peerDependencies": { - "postcss": "^8.1.0" - } - }, - "node_modules/postcss-modules-local-by-default": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-4.2.0.tgz", - "integrity": "sha512-5kcJm/zk+GJDSfw+V/42fJ5fhjL5YbFDl8nVdXkJPLLW+Vf9mTD5Xe0wqIaDnLuL2U6cDNpTr+UQ+v2HWIBhzw==", - "dev": true, - "license": "MIT", - "dependencies": { - "icss-utils": "^5.0.0", - "postcss-selector-parser": "^7.0.0", - "postcss-value-parser": "^4.1.0" + "bin": { + "semver": "bin/semver.js" }, "engines": { - "node": "^10 || ^12 || >= 14" - }, - "peerDependencies": { - "postcss": "^8.1.0" - } - }, - "node_modules/postcss-modules-scope": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-3.2.1.tgz", - "integrity": "sha512-m9jZstCVaqGjTAuny8MdgE88scJnCiQSlSrOWcTQgM2t32UBe+MUmFSO5t7VMSfAf/FJKImAxBav8ooCHJXCJA==", - "dev": true, - "license": "ISC", - "dependencies": { - "postcss-selector-parser": "^7.0.0" - }, - "engines": { - "node": "^10 || ^12 || >= 14" - }, - "peerDependencies": { - "postcss": "^8.1.0" - } - }, - "node_modules/postcss-modules-values": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-4.0.0.tgz", - "integrity": "sha512-RDxHkAiEGI78gS2ofyvCsu7iycRv7oqw5xMWn9iMoR0N/7mf9D50ecQqUo5BZ9Zh2vH4bCUR/ktCqbB9m8vJjQ==", - "dev": true, - "license": "ISC", - "dependencies": { - "icss-utils": "^5.0.0" - }, - "engines": { - "node": "^10 || ^12 || >= 14" - }, - "peerDependencies": { - "postcss": "^8.1.0" + "node": ">=10" } }, "node_modules/postcss-nested": { @@ -16288,260 +10878,7 @@ "postcss": "^8.2.14" } }, - "node_modules/postcss-nested/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-normalize-charset": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-normalize-charset/-/postcss-normalize-charset-7.0.0.tgz", - "integrity": "sha512-ABisNUXMeZeDNzCQxPxBCkXexvBrUHV+p7/BXOY+ulxkcjUZO0cp8ekGBwvIh2LbCwnWbyMPNJVtBSdyhM2zYQ==", - "dev": true, - "license": "MIT", - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-normalize-display-values": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-normalize-display-values/-/postcss-normalize-display-values-7.0.0.tgz", - "integrity": "sha512-lnFZzNPeDf5uGMPYgGOw7v0BfB45+irSRz9gHQStdkkhiM0gTfvWkWB5BMxpn0OqgOQuZG/mRlZyJxp0EImr2Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-normalize-positions": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-normalize-positions/-/postcss-normalize-positions-7.0.0.tgz", - "integrity": "sha512-I0yt8wX529UKIGs2y/9Ybs2CelSvItfmvg/DBIjTnoUSrPxSV7Z0yZ8ShSVtKNaV/wAY+m7bgtyVQLhB00A1NQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-normalize-repeat-style": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-7.0.0.tgz", - "integrity": "sha512-o3uSGYH+2q30ieM3ppu9GTjSXIzOrRdCUn8UOMGNw7Af61bmurHTWI87hRybrP6xDHvOe5WlAj3XzN6vEO8jLw==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-normalize-string": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-normalize-string/-/postcss-normalize-string-7.0.0.tgz", - "integrity": "sha512-w/qzL212DFVOpMy3UGyxrND+Kb0fvCiBBujiaONIihq7VvtC7bswjWgKQU/w4VcRyDD8gpfqUiBQ4DUOwEJ6Qg==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-normalize-timing-functions": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-7.0.0.tgz", - "integrity": "sha512-tNgw3YV0LYoRwg43N3lTe3AEWZ66W7Dh7lVEpJbHoKOuHc1sLrzMLMFjP8SNULHaykzsonUEDbKedv8C+7ej6g==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-normalize-unicode": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/postcss-normalize-unicode/-/postcss-normalize-unicode-7.0.2.tgz", - "integrity": "sha512-ztisabK5C/+ZWBdYC+Y9JCkp3M9qBv/XFvDtSw0d/XwfT3UaKeW/YTm/MD/QrPNxuecia46vkfEhewjwcYFjkg==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.23.3", - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-normalize-url": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-normalize-url/-/postcss-normalize-url-7.0.0.tgz", - "integrity": "sha512-+d7+PpE+jyPX1hDQZYG+NaFD+Nd2ris6r8fPTBAjE8z/U41n/bib3vze8x7rKs5H1uEw5ppe9IojewouHk0klQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-normalize-whitespace": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-normalize-whitespace/-/postcss-normalize-whitespace-7.0.0.tgz", - "integrity": "sha512-37/toN4wwZErqohedXYqWgvcHUGlT8O/m2jVkAfAe9Bd4MzRqlBmXrJRePH0e9Wgnz2X7KymTgTOaaFizQe3AQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-ordered-values": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/postcss-ordered-values/-/postcss-ordered-values-7.0.1.tgz", - "integrity": "sha512-irWScWRL6nRzYmBOXReIKch75RRhNS86UPUAxXdmW/l0FcAsg0lvAXQCby/1lymxn/o0gVa6Rv/0f03eJOwHxw==", - "dev": true, - "license": "MIT", - "dependencies": { - "cssnano-utils": "^5.0.0", - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-reduce-initial": { - "version": "7.0.2", - "resolved": "https://registry.npmjs.org/postcss-reduce-initial/-/postcss-reduce-initial-7.0.2.tgz", - "integrity": "sha512-pOnu9zqQww7dEKf62Nuju6JgsW2V0KRNBHxeKohU+JkHd/GAH5uvoObqFLqkeB2n20mr6yrlWDvo5UBU5GnkfA==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.23.3", - "caniuse-api": "^3.0.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-reduce-transforms": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-reduce-transforms/-/postcss-reduce-transforms-7.0.0.tgz", - "integrity": "sha512-pnt1HKKZ07/idH8cpATX/ujMbtOGhUfE+m8gbqwJE05aTaNw8gbo34a2e3if0xc0dlu75sUOiqvwCGY3fzOHew==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, "node_modules/postcss-selector-parser": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-7.0.0.tgz", - "integrity": "sha512-9RbEr1Y7FFfptd/1eEdntyjMwLeghW1bHX9GWjXo19vx4ytPQhANltvVxDggzJl7mnWM+dX28kb6cyS/4iQjlQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "cssesc": "^3.0.0", - "util-deprecate": "^1.0.2" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/postcss-svgo": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/postcss-svgo/-/postcss-svgo-7.0.1.tgz", - "integrity": "sha512-0WBUlSL4lhD9rA5k1e5D8EN5wCEyZD6HJk0jIvRxl+FDVOMlJ7DePHYWGGVc5QRqrJ3/06FTXM0bxjmJpmTPSA==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-value-parser": "^4.2.0", - "svgo": "^3.3.2" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >= 18" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-unique-selectors": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/postcss-unique-selectors/-/postcss-unique-selectors-7.0.3.tgz", - "integrity": "sha512-J+58u5Ic5T1QjP/LDV9g3Cx4CNOgB5vz+kM6+OxHHhFACdcDeKhBXjQmB7fnIZM12YSTvsL0Opwco83DmacW2g==", - "dev": true, - "license": "MIT", - "dependencies": { - "postcss-selector-parser": "^6.1.2" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/postcss-unique-selectors/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==", @@ -16612,9 +10949,9 @@ } }, "node_modules/prettier": { - "version": "3.4.2", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.2.tgz", - "integrity": "sha512-e9MewbtFo+Fevyuxn/4rrcDAaq0IYxPGLvObpQjiZBMAzB9IGmzlnG9RZy3FFas+eBMu2vA0CszMeduow5dIuQ==", + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.5.3.tgz", + "integrity": "sha512-QQtaxnoDJeAkDvDKWCLiwIXkTgRhwYDEQCghU9Z6q03iyek/rxRh/2lC3HB7P8sWT2xC/y5JDctPLBIGzHKbhw==", "dev": true, "license": "MIT", "bin": { @@ -16640,17 +10977,6 @@ "node": ">=6.0.0" } }, - "node_modules/pretty-error": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/pretty-error/-/pretty-error-4.0.0.tgz", - "integrity": "sha512-AoJ5YMAcXKYxKhuJGdcvse+Voc6v1RgnsR3nWcYU7q4t6z0Q6T86sv5Zq8VIRbOWWFpvdGE83LtdSMNd+6Y0xw==", - "dev": true, - "license": "MIT", - "dependencies": { - "lodash": "^4.17.20", - "renderkid": "^3.0.0" - } - }, "node_modules/pretty-format": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", @@ -16710,16 +11036,6 @@ "integrity": "sha512-dKp+C4iXWK4vVYZmYSd0KBH5F/h1HoZRsbJ82AVKRO3PEo8L4lBS/vLwhVtpwwuYcoIsVY+1JYKR268yn480uQ==", "license": "Unlicense" }, - "node_modules/process": { - "version": "0.11.10", - "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", - "integrity": "sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 0.6.0" - } - }, "node_modules/process-nextick-args": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", @@ -16728,11 +11044,20 @@ "license": "MIT" }, "node_modules/process-warning": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/process-warning/-/process-warning-1.0.0.tgz", - "integrity": "sha512-du4wfLyj4yCZq1VupnVSZmRsPJsNuxoDQFdCFHLaYiEbFBD7QE0a+I4D7hOxrVnh78QE/YipFAj9lXHiXocV+Q==", - "license": "MIT", - "peer": true + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/process-warning/-/process-warning-4.0.1.tgz", + "integrity": "sha512-3c2LzQ3rY9d0hc1emcsHhfT9Jwz0cChib/QN89oME2R451w5fy3f0afAhERFZAwrbDU43wk12d0ORBpDVME50Q==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/fastify" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/fastify" + } + ], + "license": "MIT" }, "node_modules/property-information": { "version": "6.5.0", @@ -16802,12 +11127,12 @@ } }, "node_modules/qs": { - "version": "6.13.1", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.1.tgz", - "integrity": "sha512-EJPeIn0CYrGu+hli1xilKAPXODtJ12T0sP63Ijx2/khC2JtuaN3JyNIpvmnkmaEtha9ocbG4A4cMcr+TvqvwQg==", + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.0.tgz", + "integrity": "sha512-YWWTjgABSKcvs/nWBi9PycY/JiPJqOD4JA6o9Sej2AtvSGarXxKC3OQSk4pAarbdQlKAh5D4FCQkJNkW+GAn3w==", "license": "BSD-3-Clause", "dependencies": { - "side-channel": "^1.0.6" + "side-channel": "^1.1.0" }, "engines": { "node": ">=0.6" @@ -16816,6 +11141,22 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/quansync": { + "version": "0.2.10", + "resolved": "https://registry.npmjs.org/quansync/-/quansync-0.2.10.tgz", + "integrity": "sha512-t41VRkMYbkHyCYmOvx/6URnN80H7k4X0lLdBMGsz+maAwrJQYB1djpV6vHrQIBE0WBSGqhtEHrK9U3DWWH8v7A==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/antfu" + }, + { + "type": "individual", + "url": "https://github.com/sponsors/sxzz" + } + ], + "license": "MIT" + }, "node_modules/queue-microtask": { "version": "1.2.3", "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", @@ -16870,6 +11211,7 @@ "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "safe-buffer": "^5.1.0" } @@ -16884,14 +11226,14 @@ } }, "node_modules/raw-body": { - "version": "2.5.2", - "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", - "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.0.tgz", + "integrity": "sha512-RmkhL8CAyCRPXCE28MMH0z2PNWQBNk2Q09ZdxM9IOOXwxwZbN+qbWaatPkdkWIKL2ZVDImrN/pK5HTRz2PcS4g==", "license": "MIT", "dependencies": { "bytes": "3.1.2", "http-errors": "2.0.0", - "iconv-lite": "0.4.24", + "iconv-lite": "0.6.3", "unpipe": "1.0.0" }, "engines": { @@ -16915,6 +11257,75 @@ "pify": "^2.3.0" } }, + "node_modules/read-package-up": { + "version": "11.0.0", + "resolved": "https://registry.npmjs.org/read-package-up/-/read-package-up-11.0.0.tgz", + "integrity": "sha512-MbgfoNPANMdb4oRBNg5eqLbB2t2r+o5Ua1pNt8BqGp4I0FJZhuVSOj3PaBPni4azWuSzEdNn2evevzVmEk1ohQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "find-up-simple": "^1.0.0", + "read-pkg": "^9.0.0", + "type-fest": "^4.6.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/read-pkg": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-9.0.1.tgz", + "integrity": "sha512-9viLL4/n1BJUCT1NXVTdS1jtm80yDEgR5T4yCelII49Mbj0v1rZdKqj7zCiYdbB0CuCgdrvHcNogAKTFPBocFA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/normalize-package-data": "^2.4.3", + "normalize-package-data": "^6.0.0", + "parse-json": "^8.0.0", + "type-fest": "^4.6.0", + "unicorn-magic": "^0.1.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/read-pkg/node_modules/parse-json": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-8.3.0.tgz", + "integrity": "sha512-ybiGyvspI+fAoRQbIPRddCcSTV9/LsJbf0e/S85VLowVGzRmokfneg2kwVW/KU5rOXrPSbF1qAKPMgNTqqROQQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.26.2", + "index-to-position": "^1.1.0", + "type-fest": "^4.39.1" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/read-pkg/node_modules/unicorn-magic": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/unicorn-magic/-/unicorn-magic-0.1.0.tgz", + "integrity": "sha512-lRfVq8fE8gz6QMBuDM6a+LO3IAzTi05H6gCVaUpir2E1Rwpo4ZUog45KpNXKC/Mn3Yb9UDuHumeFTo9iV/D9FQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/readable-stream": { "version": "3.6.2", "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", @@ -16930,13 +11341,13 @@ } }, "node_modules/readdirp": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.0.2.tgz", - "integrity": "sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==", + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-4.1.2.tgz", + "integrity": "sha512-GDhwkLfywWL2s6vEjyhri+eXmfH6j1L7JE27WhqLeYzoh/A3DBaYGEj2H/HFZCn/kMfim73FXxEJTw06WtxQwg==", "devOptional": true, "license": "MIT", "engines": { - "node": ">= 14.16.0" + "node": ">= 14.18.0" }, "funding": { "type": "individual", @@ -16965,20 +11376,19 @@ } }, "node_modules/redis": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/redis/-/redis-4.7.0.tgz", - "integrity": "sha512-zvmkHEAdGMn+hMRXuMBtu4Vo5P6rHQjLoHftu+lBqq8ZTA3RCVC/WzD790bkKKiNFp7d5/9PcSD19fJyyRvOdQ==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/redis/-/redis-5.0.0.tgz", + "integrity": "sha512-J/fzf0cYeFw5NP4aYIvv9owYOcNUsaDqF4qmwbSuaV4yKBLaIHJQIFbAKLgjn99GDXdJBbfqCRXE7+BIlkpATA==", "license": "MIT", - "workspaces": [ - "./packages/*" - ], "dependencies": { - "@redis/bloom": "1.2.0", - "@redis/client": "1.6.0", - "@redis/graph": "1.1.1", - "@redis/json": "1.0.7", - "@redis/search": "1.2.0", - "@redis/time-series": "1.1.0" + "@redis/bloom": "5.0.0", + "@redis/client": "5.0.0", + "@redis/json": "5.0.0", + "@redis/search": "5.0.0", + "@redis/time-series": "5.0.0" + }, + "engines": { + "node": ">= 18" } }, "node_modules/redis-errors": { @@ -17008,168 +11418,6 @@ "integrity": "sha512-urBwgfrvVP/eAyXx4hluJivBKzuEbSQs9rKWCrCkbSxNv8mxPcUZKeuoF3Uy4mJl3Lwprp6yy5/39VWigZ4K6Q==", "license": "Apache-2.0" }, - "node_modules/reflect.getprototypeof": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.10.tgz", - "integrity": "sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.9", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.7", - "get-proto": "^1.0.1", - "which-builtin-type": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/regenerate": { - "version": "1.4.2", - "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.2.tgz", - "integrity": "sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==", - "dev": true, - "license": "MIT" - }, - "node_modules/regenerate-unicode-properties": { - "version": "10.2.0", - "resolved": "https://registry.npmjs.org/regenerate-unicode-properties/-/regenerate-unicode-properties-10.2.0.tgz", - "integrity": "sha512-DqHn3DwbmmPVzeKj9woBadqmXxLvQoQIwu7nopMc72ztvxVmVk2SBhSnx67zuye5TP+lJsb/TBQsjLKhnDf3MA==", - "dev": true, - "license": "MIT", - "dependencies": { - "regenerate": "^1.4.2" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/regenerator-runtime": { - "version": "0.14.1", - "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz", - "integrity": "sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==", - "dev": true, - "license": "MIT" - }, - "node_modules/regenerator-transform": { - "version": "0.15.2", - "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.15.2.tgz", - "integrity": "sha512-hfMp2BoF0qOk3uc5V20ALGDS2ddjQaLrdl7xrGXvAIow7qeWRM2VA2HuCHkUKk9slq3VwEwLNK3DFBqDfPGYtg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@babel/runtime": "^7.8.4" - } - }, - "node_modules/regex-parser": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/regex-parser/-/regex-parser-2.3.0.tgz", - "integrity": "sha512-TVILVSz2jY5D47F4mA4MppkBrafEaiUWJO/TcZHEIuI13AqoZMkK1WMA4Om1YkYbTx+9Ki1/tSUXbceyr9saRg==", - "dev": true, - "license": "MIT" - }, - "node_modules/regexp.prototype.flags": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.4.tgz", - "integrity": "sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-errors": "^1.3.0", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "set-function-name": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/regexpu-core": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-6.2.0.tgz", - "integrity": "sha512-H66BPQMrv+V16t8xtmq+UC0CBpiTBA60V8ibS1QVReIp8T1z8hwFxqcGzm9K6lgsN7sB5edVH8a+ze6Fqm4weA==", - "dev": true, - "license": "MIT", - "dependencies": { - "regenerate": "^1.4.2", - "regenerate-unicode-properties": "^10.2.0", - "regjsgen": "^0.8.0", - "regjsparser": "^0.12.0", - "unicode-match-property-ecmascript": "^2.0.0", - "unicode-match-property-value-ecmascript": "^2.1.0" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/regjsgen": { - "version": "0.8.0", - "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.8.0.tgz", - "integrity": "sha512-RvwtGe3d7LvWiDQXeQw8p5asZUmfU1G/l6WbUXeHta7Y2PEIvBTwH6E2EfmYUK8pxcxEdEmaomqyp0vZZ7C+3Q==", - "dev": true, - "license": "MIT" - }, - "node_modules/regjsparser": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.12.0.tgz", - "integrity": "sha512-cnE+y8bz4NhMjISKbgeVJtqNbtf5QpjZP+Bslo+UqkIt9QPnX9q095eiRRASJG1/tz6dlNr6Z5NsBiWYokp6EQ==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "jsesc": "~3.0.2" - }, - "bin": { - "regjsparser": "bin/parser" - } - }, - "node_modules/regjsparser/node_modules/jsesc": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.0.2.tgz", - "integrity": "sha512-xKqzzWXDttJuOcawBt4KnKHHIf5oQ/Cxax+0PWFG+DFDgHNAdi+TXECADI+RYiFUMmx8792xsMbbgXj4CwnP4g==", - "dev": true, - "license": "MIT", - "bin": { - "jsesc": "bin/jsesc" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/renderkid": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/renderkid/-/renderkid-3.0.0.tgz", - "integrity": "sha512-q/7VIQA8lmM1hF+jn+sFSPWGlMkSAeNYcPLmDQx2zzuiDfaLrOmumR8iaUKlenFgh0XRPIUeSPlH3A+AW3Z5pg==", - "dev": true, - "license": "MIT", - "dependencies": { - "css-select": "^4.1.3", - "dom-converter": "^0.2.0", - "htmlparser2": "^6.1.0", - "lodash": "^4.17.21", - "strip-ansi": "^6.0.1" - } - }, - "node_modules/require-all": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/require-all/-/require-all-3.0.0.tgz", - "integrity": "sha512-jPGN876lc5exWYrMcgZSd7U42P0PmVQzxnQB13fCSzmyGnqQWW4WUz5DosZ/qe24hz+5o9lSvW2epBNZ1xa6Fw==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", @@ -17228,31 +11476,6 @@ "integrity": "sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g==", "license": "MIT" }, - "node_modules/resolve-cwd": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", - "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "resolve-from": "^5.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/resolve-cwd/node_modules/resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "dev": true, - "license": "MIT", - "peer": true, - "engines": { - "node": ">=8" - } - }, "node_modules/resolve-from": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", @@ -17263,30 +11486,6 @@ "node": ">=4" } }, - "node_modules/resolve-url-loader": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-url-loader/-/resolve-url-loader-5.0.0.tgz", - "integrity": "sha512-uZtduh8/8srhBoMx//5bwqjQ+rfYOUq8zC9NrMUGtjBiGTtFJM42s58/36+hTqeqINcnYe08Nj3LkK9lW4N8Xg==", - "dev": true, - "license": "MIT", - "dependencies": { - "adjust-sourcemap-loader": "^4.0.0", - "convert-source-map": "^1.7.0", - "loader-utils": "^2.0.0", - "postcss": "^8.2.14", - "source-map": "0.6.1" - }, - "engines": { - "node": ">=12" - } - }, - "node_modules/resolve-url-loader/node_modules/convert-source-map": { - "version": "1.9.0", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.9.0.tgz", - "integrity": "sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==", - "dev": true, - "license": "MIT" - }, "node_modules/responselike": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/responselike/-/responselike-3.0.0.tgz", @@ -17329,15 +11528,22 @@ } }, "node_modules/reusify": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", - "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", + "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", "license": "MIT", "engines": { "iojs": ">=1.0.0", "node": ">=0.10.0" } }, + "node_modules/rfdc": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/rfdc/-/rfdc-1.4.1.tgz", + "integrity": "sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA==", + "dev": true, + "license": "MIT" + }, "node_modules/rimraf": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", @@ -17361,13 +11567,12 @@ "license": "MIT" }, "node_modules/rollup": { - "version": "4.30.1", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.30.1.tgz", - "integrity": "sha512-mlJ4glW020fPuLi7DkM/lN97mYEZGWeqBnrljzN0gs7GLctqX3lNWxKQ7Gl712UAX+6fog/L3jh4gb7R6aVi3w==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.40.1.tgz", + "integrity": "sha512-C5VvvgCCyfyotVITIAv+4efVytl5F7wt+/I2i9q9GZcEXW9BP52YYOXC58igUi+LFZVHukErIIqQSWwv/M3WRw==", "license": "MIT", - "peer": true, "dependencies": { - "@types/estree": "1.0.6" + "@types/estree": "1.0.7" }, "bin": { "rollup": "dist/bin/rollup" @@ -17377,25 +11582,26 @@ "npm": ">=8.0.0" }, "optionalDependencies": { - "@rollup/rollup-android-arm-eabi": "4.30.1", - "@rollup/rollup-android-arm64": "4.30.1", - "@rollup/rollup-darwin-arm64": "4.30.1", - "@rollup/rollup-darwin-x64": "4.30.1", - "@rollup/rollup-freebsd-arm64": "4.30.1", - "@rollup/rollup-freebsd-x64": "4.30.1", - "@rollup/rollup-linux-arm-gnueabihf": "4.30.1", - "@rollup/rollup-linux-arm-musleabihf": "4.30.1", - "@rollup/rollup-linux-arm64-gnu": "4.30.1", - "@rollup/rollup-linux-arm64-musl": "4.30.1", - "@rollup/rollup-linux-loongarch64-gnu": "4.30.1", - "@rollup/rollup-linux-powerpc64le-gnu": "4.30.1", - "@rollup/rollup-linux-riscv64-gnu": "4.30.1", - "@rollup/rollup-linux-s390x-gnu": "4.30.1", - "@rollup/rollup-linux-x64-gnu": "4.30.1", - "@rollup/rollup-linux-x64-musl": "4.30.1", - "@rollup/rollup-win32-arm64-msvc": "4.30.1", - "@rollup/rollup-win32-ia32-msvc": "4.30.1", - "@rollup/rollup-win32-x64-msvc": "4.30.1", + "@rollup/rollup-android-arm-eabi": "4.40.1", + "@rollup/rollup-android-arm64": "4.40.1", + "@rollup/rollup-darwin-arm64": "4.40.1", + "@rollup/rollup-darwin-x64": "4.40.1", + "@rollup/rollup-freebsd-arm64": "4.40.1", + "@rollup/rollup-freebsd-x64": "4.40.1", + "@rollup/rollup-linux-arm-gnueabihf": "4.40.1", + "@rollup/rollup-linux-arm-musleabihf": "4.40.1", + "@rollup/rollup-linux-arm64-gnu": "4.40.1", + "@rollup/rollup-linux-arm64-musl": "4.40.1", + "@rollup/rollup-linux-loongarch64-gnu": "4.40.1", + "@rollup/rollup-linux-powerpc64le-gnu": "4.40.1", + "@rollup/rollup-linux-riscv64-gnu": "4.40.1", + "@rollup/rollup-linux-riscv64-musl": "4.40.1", + "@rollup/rollup-linux-s390x-gnu": "4.40.1", + "@rollup/rollup-linux-x64-gnu": "4.40.1", + "@rollup/rollup-linux-x64-musl": "4.40.1", + "@rollup/rollup-win32-arm64-msvc": "4.40.1", + "@rollup/rollup-win32-ia32-msvc": "4.40.1", + "@rollup/rollup-win32-x64-msvc": "4.40.1", "fsevents": "~2.3.2" } }, @@ -17436,67 +11642,26 @@ "queue-microtask": "^1.2.2" } }, - "node_modules/safe-array-concat": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.3.tgz", - "integrity": "sha512-AURm5f0jYEOydBj7VQlVvDrjeFgthDdEF5H1dP+6mNpoXOMo1quQqJ4wvJDyRZ9+pO3kGWoOdmV08cSv2aJV6Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "has-symbols": "^1.1.0", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">=0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "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/safe-push-apply": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-push-apply/-/safe-push-apply-1.0.0.tgz", - "integrity": "sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA==", - "dev": true, - "license": "MIT", - "dependencies": { - "es-errors": "^1.3.0", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/safe-regex-test": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.1.0.tgz", - "integrity": "sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-regex": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/safe-stable-stringify": { "version": "2.5.0", "resolved": "https://registry.npmjs.org/safe-stable-stringify/-/safe-stable-stringify-2.5.0.tgz", @@ -17521,21 +11686,10 @@ "axios": "^1.5.1" } }, - "node_modules/saxon-js/node_modules/axios": { - "version": "1.7.9", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.9.tgz", - "integrity": "sha512-LhLcE7Hbiryz8oMDdDptSrWowmB4Bl6RCt6sIJKpRB4XtVf0iEgewX3au/pJqm+Py1kCASkb/FFKjxQaLtxJvw==", - "license": "MIT", - "dependencies": { - "follow-redirects": "^1.15.6", - "form-data": "^4.0.0", - "proxy-from-env": "^1.1.0" - } - }, "node_modules/schema-utils": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.3.0.tgz", - "integrity": "sha512-Gf9qqc58SpCA/xdziiHz35F4GNIWYWZrEshUc/G/r5BnLph6xpKuLeoJoQuj5WfBIx/eQLf+hmVPYHaxJu7V2g==", + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.3.2.tgz", + "integrity": "sha512-Gn/JaSk/Mt9gYubxTtSn/QCV4em9mpAPiR1rqy/Ocu19u/G9J5WWdNoUT4SiV6mFC3y6cxyFcFwdzPM3FgxGAQ==", "dev": true, "license": "MIT", "dependencies": { @@ -17617,15 +11771,12 @@ } }, "node_modules/semver": { - "version": "7.6.3", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", - "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", "license": "ISC", "bin": { "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" } }, "node_modules/send": { @@ -17667,6 +11818,15 @@ "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", "license": "MIT" }, + "node_modules/send/node_modules/encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, "node_modules/send/node_modules/mime": { "version": "1.6.0", "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", @@ -17679,41 +11839,17 @@ "node": ">=4" } }, - "node_modules/sentence-case": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/sentence-case/-/sentence-case-3.0.4.tgz", - "integrity": "sha512-8LS0JInaQMCRoQ7YUytAo/xUu5W2XnQxV2HI/6uM6U7CITS1RqPElr30V6uIqyMKM9lJGRVFy5/4CuzcixNYSg==", - "license": "MIT", - "dependencies": { - "no-case": "^3.0.4", - "tslib": "^2.0.3", - "upper-case-first": "^2.0.2" - } - }, "node_modules/serialize-error": { - "version": "11.0.3", - "resolved": "https://registry.npmjs.org/serialize-error/-/serialize-error-11.0.3.tgz", - "integrity": "sha512-2G2y++21dhj2R7iHAdd0FIzjGwuKZld+7Pl/bTU6YIkrC2ZMbVUjm+luj6A6V34Rv9XfKJDKpTWu9W4Gse1D9g==", + "version": "12.0.0", + "resolved": "https://registry.npmjs.org/serialize-error/-/serialize-error-12.0.0.tgz", + "integrity": "sha512-ZYkZLAvKTKQXWuh5XpBw7CdbSzagarX39WyZ2H07CDLC5/KfsRGlIXV8d4+tfqX1M7916mRqR1QfNHSij+c9Pw==", "devOptional": true, "license": "MIT", "dependencies": { - "type-fest": "^2.12.2" + "type-fest": "^4.31.0" }, "engines": { - "node": ">=14.16" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/serialize-error/node_modules/type-fest": { - "version": "2.19.0", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-2.19.0.tgz", - "integrity": "sha512-RAH822pAdBgcNMAfWnCBU3CFZcfZ/i1eZjwFU/dsLKumyuuP3niueg2UAukXYF0E2AAoc82ZSSf9J0WQBinzHA==", - "devOptional": true, - "license": "(MIT OR CC0-1.0)", - "engines": { - "node": ">=12.20" + "node": ">=18" }, "funding": { "url": "https://github.com/sponsors/sindresorhus" @@ -17725,6 +11861,7 @@ "integrity": "sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==", "dev": true, "license": "BSD-3-Clause", + "peer": true, "dependencies": { "randombytes": "^2.1.0" } @@ -17830,96 +11967,18 @@ "node": ">= 0.8.0" } }, - "node_modules/serve-static/node_modules/encodeurl": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", - "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", - "license": "MIT", - "engines": { - "node": ">= 0.8" - } - }, "node_modules/set-blocking": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", "integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==", "license": "ISC" }, - "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==", - "devOptional": true, - "license": "MIT" - }, - "node_modules/set-function-length": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", - "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "function-bind": "^1.1.2", - "get-intrinsic": "^1.2.4", - "gopd": "^1.0.1", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/set-function-name": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", - "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "functions-have-names": "^1.2.3", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/set-proto": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/set-proto/-/set-proto-1.0.0.tgz", - "integrity": "sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw==", - "dev": true, - "license": "MIT", - "dependencies": { - "dunder-proto": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - } - }, "node_modules/setprototypeof": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", "license": "ISC" }, - "node_modules/shallow-clone": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz", - "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", - "dev": true, - "license": "MIT", - "dependencies": { - "kind-of": "^6.0.2" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", @@ -18095,16 +12154,6 @@ "node": ">=8.0.0" } }, - "node_modules/snake-case": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/snake-case/-/snake-case-3.0.4.tgz", - "integrity": "sha512-LAOh4z89bGQvl9pFfNF8V146i7o7/CqFPbqzYgP+yYzDIDeS9HaNFtXABamRW+AQzEVODcvE79ljJ+8a9YSdMg==", - "license": "MIT", - "dependencies": { - "dot-case": "^3.0.4", - "tslib": "^2.0.3" - } - }, "node_modules/sockjs": { "version": "0.3.24", "resolved": "https://registry.npmjs.org/sockjs/-/sockjs-0.3.24.tgz", @@ -18118,11 +12167,10 @@ } }, "node_modules/sonic-boom": { - "version": "2.8.0", - "resolved": "https://registry.npmjs.org/sonic-boom/-/sonic-boom-2.8.0.tgz", - "integrity": "sha512-kuonw1YOYYNOve5iHdSahXPOK49GqwA+LZhI6Wz/l0rP57iKyXXIHaRagOBHAPmGwJC6od2Z9zgvZ5loSgMlVg==", + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/sonic-boom/-/sonic-boom-4.2.0.tgz", + "integrity": "sha512-INb7TM37/mAcsGmc9hyyI6+QR3rR1zVRu36B0NeGXKnOOLiZOfER5SA+N7X7k3yUYRzLWafduTDvJAfDswwEww==", "license": "MIT", - "peer": true, "dependencies": { "atomic-sleep": "^1.0.0" } @@ -18157,11 +12205,48 @@ "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", "devOptional": true, "license": "MIT", + "peer": true, "dependencies": { "buffer-from": "^1.0.0", "source-map": "^0.6.0" } }, + "node_modules/spdx-correct": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.2.0.tgz", + "integrity": "sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "spdx-expression-parse": "^3.0.0", + "spdx-license-ids": "^3.0.0" + } + }, + "node_modules/spdx-exceptions": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.5.0.tgz", + "integrity": "sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w==", + "dev": true, + "license": "CC-BY-3.0" + }, + "node_modules/spdx-expression-parse": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", + "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "spdx-exceptions": "^2.1.0", + "spdx-license-ids": "^3.0.0" + } + }, + "node_modules/spdx-license-ids": { + "version": "3.0.21", + "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.21.tgz", + "integrity": "sha512-Bvg/8F5XephndSK3JffaRqdT+gyhfqIPwDHpX80tJrF8QQRYMo8sNMeaZ2Dp5+jhwKnUmIOyFFQfHRkjJm5nXg==", + "dev": true, + "license": "CC0-1.0" + }, "node_modules/spdy": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/spdy/-/spdy-4.0.2.tgz", @@ -18194,6 +12279,16 @@ "wbuf": "^1.7.3" } }, + "node_modules/speakingurl": { + "version": "14.0.1", + "resolved": "https://registry.npmjs.org/speakingurl/-/speakingurl-14.0.1.tgz", + "integrity": "sha512-1POYv7uv2gXoyGFpBCmpDVSNV74IfsWlDW216UPjbWufNf+bSU6GdbDsxdcxtfwb4xlI3yxzOTKClUosxARYrQ==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/split-lines": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/split-lines/-/split-lines-3.0.0.tgz", @@ -18221,13 +12316,6 @@ "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", "license": "BSD-3-Clause" }, - "node_modules/stackframe": { - "version": "1.3.4", - "resolved": "https://registry.npmjs.org/stackframe/-/stackframe-1.3.4.tgz", - "integrity": "sha512-oeVtt7eWQS+Na6F//S4kJ2K2VbRlS9D43mAlMyVpVWovy9o+jfgH8O9agzANzaiLjclA0oYzUXEM4PurhSUChw==", - "dev": true, - "license": "MIT" - }, "node_modules/stacktracey": { "version": "2.1.8", "resolved": "https://registry.npmjs.org/stacktracey/-/stacktracey-2.1.8.tgz", @@ -18262,26 +12350,6 @@ "safe-buffer": "~5.2.0" } }, - "node_modules/string_decoder/node_modules/safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "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/string-width": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/string-width/-/string-width-7.2.0.tgz", @@ -18359,65 +12427,6 @@ "url": "https://github.com/chalk/strip-ansi?sponsor=1" } }, - "node_modules/string.prototype.trim": { - "version": "1.2.10", - "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.10.tgz", - "integrity": "sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-data-property": "^1.1.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-object-atoms": "^1.0.0", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimend": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.9.tgz", - "integrity": "sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimstart": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", - "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/stringify-attributes": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/stringify-attributes/-/stringify-attributes-4.0.0.tgz", @@ -18486,70 +12495,22 @@ } }, "node_modules/strtok3": { - "version": "9.1.1", - "resolved": "https://registry.npmjs.org/strtok3/-/strtok3-9.1.1.tgz", - "integrity": "sha512-FhwotcEqjr241ZbjFzjlIYg6c5/L/s4yBGWSMvJ9UoExiSqL+FnFA/CaeZx17WGaZMS/4SOZp8wH18jSS4R4lw==", + "version": "10.2.2", + "resolved": "https://registry.npmjs.org/strtok3/-/strtok3-10.2.2.tgz", + "integrity": "sha512-Xt18+h4s7Z8xyZ0tmBoRmzxcop97R4BAh+dXouUDCYn+Em+1P3qpkUfI5ueWLT8ynC5hZ+q4iPEmGG1urvQGBg==", "license": "MIT", "dependencies": { "@tokenizer/token": "^0.3.0", - "peek-readable": "^5.3.1" + "peek-readable": "^7.0.0" }, "engines": { - "node": ">=16" + "node": ">=18" }, "funding": { "type": "github", "url": "https://github.com/sponsors/Borewit" } }, - "node_modules/style-loader": { - "version": "3.3.4", - "resolved": "https://registry.npmjs.org/style-loader/-/style-loader-3.3.4.tgz", - "integrity": "sha512-0WqXzrsMTyb8yjZJHDqwmnwRJvhALK9LfRtRc6B4UTWe8AijYLZYZ9thuJTZc2VfQWINADW/j+LiJnfy2RoC1w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 12.13.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/webpack" - }, - "peerDependencies": { - "webpack": "^5.0.0" - } - }, - "node_modules/stylehacks": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/stylehacks/-/stylehacks-7.0.4.tgz", - "integrity": "sha512-i4zfNrGMt9SB4xRK9L83rlsFCgdGANfeDAYacO1pkqcE7cRHPdWHwnKZVz7WY17Veq/FvyYsRAU++Ga+qDFIww==", - "dev": true, - "license": "MIT", - "dependencies": { - "browserslist": "^4.23.3", - "postcss-selector-parser": "^6.1.2" - }, - "engines": { - "node": "^18.12.0 || ^20.9.0 || >=22.0" - }, - "peerDependencies": { - "postcss": "^8.4.31" - } - }, - "node_modules/stylehacks/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/sucrase": { "version": "3.35.0", "resolved": "https://registry.npmjs.org/sucrase/-/sucrase-3.35.0.tgz", @@ -18645,7 +12606,7 @@ "resolved": "https://registry.npmjs.org/superagent/-/superagent-8.1.2.tgz", "integrity": "sha512-6WTxW1EB6yCxV5VFOIPQruWGHqc3yI7hEmZK6h+pyk69Lk/Ut7rLUY6W/ONF2MjBuGjvmMiIpsrVJ2vjrHlslA==", "deprecated": "Please upgrade to v9.0.0+ as we have fixed a public vulnerability with formidable dependency. Note that v9.0.0+ requires Node.js v14.18.0+. See https://github.com/ladjs/superagent/pull/1800 for insight. This project is supported and maintained by the team at Forward Email @ https://forwardemail.net", - "devOptional": true, + "dev": true, "license": "MIT", "dependencies": { "component-emitter": "^1.3.0", @@ -18667,7 +12628,7 @@ "version": "2.6.0", "resolved": "https://registry.npmjs.org/mime/-/mime-2.6.0.tgz", "integrity": "sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg==", - "devOptional": true, + "dev": true, "license": "MIT", "bin": { "mime": "cli.js" @@ -18676,6 +12637,32 @@ "node": ">=4.0.0" } }, + "node_modules/superagent/node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/superjson": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/superjson/-/superjson-2.2.2.tgz", + "integrity": "sha512-5JRxVqC8I8NuOUjzBbvVJAKNM8qoVuH0O77h4WInc/qC2q5IreqKxYwgkga3PfA22OayK2ikceb/B26dztPl+Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "copy-anything": "^3.0.2" + }, + "engines": { + "node": ">=16" + } + }, "node_modules/supertest": { "version": "6.3.4", "resolved": "https://registry.npmjs.org/supertest/-/supertest-6.3.4.tgz", @@ -18691,13 +12678,12 @@ } }, "node_modules/supports-color": { - "version": "9.4.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-9.4.0.tgz", - "integrity": "sha512-VL+lNrEoIXww1coLPOmiEmK/0sGigko5COxI09KzHc2VJXJsQ37UaQ+8quuxjDeA7+KnLGTWRyOXSLLR2Wb4jw==", - "devOptional": true, + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-10.0.0.tgz", + "integrity": "sha512-HRVVSbCCMbj7/kdWF9Q+bbckjBHLtHMEoJWlkmYzzdwhYMkjkOwubLM6t7NbWKjgKamGDrWL1++KrjUO1t9oAQ==", "license": "MIT", "engines": { - "node": ">=12" + "node": ">=18" }, "funding": { "url": "https://github.com/chalk/supports-color?sponsor=1" @@ -18715,166 +12701,21 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/svgo": { - "version": "3.3.2", - "resolved": "https://registry.npmjs.org/svgo/-/svgo-3.3.2.tgz", - "integrity": "sha512-OoohrmuUlBs8B8o6MB2Aevn+pRIH9zDALSR+6hhqVfa6fRwG/Qw9VUMSMW9VNg2CFc/MTIfabtdOVl9ODIJjpw==", - "dev": true, - "license": "MIT", - "dependencies": { - "@trysound/sax": "0.2.0", - "commander": "^7.2.0", - "css-select": "^5.1.0", - "css-tree": "^2.3.1", - "css-what": "^6.1.0", - "csso": "^5.0.5", - "picocolors": "^1.0.0" - }, - "bin": { - "svgo": "bin/svgo" - }, - "engines": { - "node": ">=14.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/svgo" - } - }, - "node_modules/svgo/node_modules/commander": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", - "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">= 10" - } - }, - "node_modules/svgo/node_modules/css-select": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/css-select/-/css-select-5.1.0.tgz", - "integrity": "sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "boolbase": "^1.0.0", - "css-what": "^6.1.0", - "domhandler": "^5.0.2", - "domutils": "^3.0.1", - "nth-check": "^2.0.1" - }, - "funding": { - "url": "https://github.com/sponsors/fb55" - } - }, - "node_modules/svgo/node_modules/dom-serializer": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-2.0.0.tgz", - "integrity": "sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==", - "dev": true, - "license": "MIT", - "dependencies": { - "domelementtype": "^2.3.0", - "domhandler": "^5.0.2", - "entities": "^4.2.0" - }, - "funding": { - "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" - } - }, - "node_modules/svgo/node_modules/domhandler": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-5.0.3.tgz", - "integrity": "sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "domelementtype": "^2.3.0" - }, - "engines": { - "node": ">= 4" - }, - "funding": { - "url": "https://github.com/fb55/domhandler?sponsor=1" - } - }, - "node_modules/svgo/node_modules/domutils": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.2.tgz", - "integrity": "sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==", - "dev": true, - "license": "BSD-2-Clause", - "dependencies": { - "dom-serializer": "^2.0.0", - "domelementtype": "^2.3.0", - "domhandler": "^5.0.3" - }, - "funding": { - "url": "https://github.com/fb55/domutils?sponsor=1" - } - }, - "node_modules/swagger-parser": { - "version": "10.0.3", - "resolved": "https://registry.npmjs.org/swagger-parser/-/swagger-parser-10.0.3.tgz", - "integrity": "sha512-nF7oMeL4KypldrQhac8RyHerJeGPD1p2xDh900GPvc+Nk7nWP6jX2FcC7WmkinMoAmoO774+AFXcWsW8gMWEIg==", - "dev": true, - "license": "MIT", - "dependencies": { - "@apidevtools/swagger-parser": "10.0.3" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/swagger-parser/node_modules/@apidevtools/swagger-parser": { - "version": "10.0.3", - "resolved": "https://registry.npmjs.org/@apidevtools/swagger-parser/-/swagger-parser-10.0.3.tgz", - "integrity": "sha512-sNiLY51vZOmSPFZA5TF35KZ2HbgYklQnTSDnkghamzLb3EkNtcQnrBQEj5AOCxHpTtXpqMCRM1CrmV2rG6nw4g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@apidevtools/json-schema-ref-parser": "^9.0.6", - "@apidevtools/openapi-schemas": "^2.0.4", - "@apidevtools/swagger-methods": "^3.0.2", - "@jsdevtools/ono": "^7.1.3", - "call-me-maybe": "^1.0.1", - "z-schema": "^5.0.1" - }, - "peerDependencies": { - "openapi-types": ">=7" - } - }, - "node_modules/swagger-parser/node_modules/openapi-types": { - "version": "12.1.3", - "resolved": "https://registry.npmjs.org/openapi-types/-/openapi-types-12.1.3.tgz", - "integrity": "sha512-N4YtSYJqghVu4iek2ZUvcN/0aqH1kRDuNqzcycDxhOUpg7GdvLa2F3DgS6yBNhInhv2r/6I0Flkn7CqL8+nIcw==", - "dev": true, - "license": "MIT", - "peer": true - }, - "node_modules/swagger-schema-official": { - "version": "2.0.0-bab6bed", - "resolved": "https://registry.npmjs.org/swagger-schema-official/-/swagger-schema-official-2.0.0-bab6bed.tgz", - "integrity": "sha512-rCC0NWGKr/IJhtRuPq/t37qvZHI/mH4I4sxflVM+qgVe5Z2uOCivzWaVbuioJaB61kvm5UvB7b49E+oBY0M8jA==", - "dev": true, - "license": "ISC" - }, "node_modules/synckit": { - "version": "0.9.2", - "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.9.2.tgz", - "integrity": "sha512-vrozgXDQwYO72vHjUb/HnFbQx1exDjoKzqx23aXEg2a9VIg2TSFZ8FmeZpTjUCFMYw7mpX4BE2SFu8wI7asYsw==", + "version": "0.11.4", + "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.11.4.tgz", + "integrity": "sha512-Q/XQKRaJiLiFIBNN+mndW7S/RHxvwzuZS6ZwmRzUBqJBv/5QIKCEwkBC8GBf8EQJKYnaFs0wOZbKTXBPj8L9oQ==", "dev": true, "license": "MIT", "dependencies": { - "@pkgr/core": "^0.1.0", - "tslib": "^2.6.2" + "@pkgr/core": "^0.2.3", + "tslib": "^2.8.1" }, "engines": { "node": "^14.18.0 || >=16.0.0" }, "funding": { - "url": "https://opencollective.com/unts" + "url": "https://opencollective.com/synckit" } }, "node_modules/tabbable": { @@ -18972,20 +12813,6 @@ "url": "https://github.com/sponsors/jonschlinkert" } }, - "node_modules/tailwindcss/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/tailwindcss/node_modules/readdirp": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", @@ -19063,11 +12890,12 @@ } }, "node_modules/terser": { - "version": "5.37.0", - "resolved": "https://registry.npmjs.org/terser/-/terser-5.37.0.tgz", - "integrity": "sha512-B8wRRkmre4ERucLM/uXx4MOV5cbnOlVAqUst+1+iLKPI0dOgFO28f84ptoQt9HEI537PMzfYa/d+GEPKTRXmYA==", + "version": "5.39.0", + "resolved": "https://registry.npmjs.org/terser/-/terser-5.39.0.tgz", + "integrity": "sha512-LBAhFyLho16harJoWMg/nZsQYgTrg5jXOn2nCYjRUcZZEdE3qa2zb8QEDRUGVZBW4rlazf2fxkg8tztybTaqWw==", "devOptional": true, "license": "BSD-2-Clause", + "peer": true, "dependencies": { "@jridgewell/source-map": "^0.3.3", "acorn": "^8.8.2", @@ -19082,11 +12910,12 @@ } }, "node_modules/terser-webpack-plugin": { - "version": "5.3.11", - "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.11.tgz", - "integrity": "sha512-RVCsMfuD0+cTt3EwX8hSl2Ks56EbFHWmhluwcqoPKtBnfjiT6olaq7PRIRfhyU8nnC2MrnDrBLfrD/RGE+cVXQ==", + "version": "5.3.14", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.14.tgz", + "integrity": "sha512-vkZjpUjb6OMS7dhV+tILUW6BhpDR7P2L/aQSAv+Uwk+m8KATX9EccViHTJR2qDtACKPIYndLGCyl3FMo+r2LMw==", "dev": true, "license": "MIT", + "peer": true, "dependencies": { "@jridgewell/trace-mapping": "^0.3.25", "jest-worker": "^27.4.5", @@ -19121,7 +12950,8 @@ "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", "devOptional": true, - "license": "MIT" + "license": "MIT", + "peer": true }, "node_modules/text-table": { "version": "0.2.0", @@ -19221,14 +13051,20 @@ "integrity": "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA==", "license": "MIT" }, - "node_modules/tmp": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.3.tgz", - "integrity": "sha512-nZD7m9iCPC5g0pYmcaxogYKggSfLsdxl8of3Q/oIbqCqLLIO9IAF0GWjX1z9NZRHPiXv8Wex4yDCaZsgEw0Y8w==", - "dev": true, + "node_modules/tinyglobby": { + "version": "0.2.13", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.13.tgz", + "integrity": "sha512-mEwzpUgrLySlveBwEVDMKk5B57bhLPYovRfPAXD5gA/98Opn0rCDj3GtLwFvCvH5RK9uPCExUROW5NjDwvqkxw==", "license": "MIT", + "dependencies": { + "fdir": "^6.4.4", + "picomatch": "^4.0.2" + }, "engines": { - "node": ">=14.14" + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" } }, "node_modules/tmp-cache": { @@ -19290,24 +13126,6 @@ "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", "license": "MIT" }, - "node_modules/traverse": { - "version": "0.6.10", - "resolved": "https://registry.npmjs.org/traverse/-/traverse-0.6.10.tgz", - "integrity": "sha512-hN4uFRxbK+PX56DxYiGHsTn2dME3TVr9vbNqlQGcGcPhJAn+tdP126iA+TArMpI4YSgnTkMWyoLl5bf81Hi5TA==", - "dev": true, - "license": "MIT", - "dependencies": { - "gopd": "^1.0.1", - "typedarray.prototype.slice": "^1.0.3", - "which-typed-array": "^1.1.15" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/tree-dump": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/tree-dump/-/tree-dump-1.0.2.tgz", @@ -19339,9 +13157,9 @@ "license": "Apache-2.0" }, "node_modules/ts-loader": { - "version": "9.5.1", - "resolved": "https://registry.npmjs.org/ts-loader/-/ts-loader-9.5.1.tgz", - "integrity": "sha512-rNH3sK9kGZcH9dYzC7CewQm4NtxJTjSEVRJ2DyBZR7f8/wcta+iV44UPCXc5+nzDzivKtlzV6c9P4e+oFhDLYg==", + "version": "9.5.2", + "resolved": "https://registry.npmjs.org/ts-loader/-/ts-loader-9.5.2.tgz", + "integrity": "sha512-Qo4piXvOTWcMGIgRiuFa6nHNm+54HbYaZCKqc9eeZCLRy3XqafQgwX2F7mofrbJG3g7EEb+lkiR+z2Lic2s3Zw==", "dev": true, "license": "MIT", "dependencies": { @@ -19359,59 +13177,19 @@ "webpack": "^5.0.0" } }, - "node_modules/ts-loader/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==", + "node_modules/ts-loader/node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", "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/ts-loader/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" + "license": "ISC", + "bin": { + "semver": "bin/semver.js" }, "engines": { "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/chalk?sponsor=1" } }, - "node_modules/ts-loader/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/ts-loader/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/ts-loader/node_modules/source-map": { "version": "0.7.4", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.4.tgz", @@ -19422,19 +13200,6 @@ "node": ">= 8" } }, - "node_modules/ts-loader/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/ts-morph": { "version": "23.0.0", "resolved": "https://registry.npmjs.org/ts-morph/-/ts-morph-23.0.0.tgz", @@ -19446,10 +13211,10 @@ "code-block-writer": "^13.0.1" } }, - "node_modules/ts-node": { - "version": "10.9.2", - "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.2.tgz", - "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", + "node_modules/ts-node-maintained": { + "version": "10.9.5", + "resolved": "https://registry.npmjs.org/ts-node-maintained/-/ts-node-maintained-10.9.5.tgz", + "integrity": "sha512-p8LJFxBTE3YZYGcOMxWKEpaY2nz55NyOg+mTDIOW/MrOIUTAAb7+UkleRu5z90P/fCVVv5pXN1/nb92G/tSNyw==", "dev": true, "license": "MIT", "dependencies": { @@ -19490,7 +13255,7 @@ } } }, - "node_modules/ts-node/node_modules/arg": { + "node_modules/ts-node-maintained/node_modules/arg": { "version": "4.1.3", "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", @@ -19549,9 +13314,9 @@ } }, "node_modules/type-fest": { - "version": "4.31.0", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-4.31.0.tgz", - "integrity": "sha512-yCxltHW07Nkhv/1F6wWBr8kz+5BGMfP+RbRSYFnegVb0qV/UMT0G0ElBloPVerqn4M2ZV80Ir1FtCcYv1cT6vQ==", + "version": "4.40.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-4.40.1.tgz", + "integrity": "sha512-9YvLNnORDpI+vghLU/Nf+zSv0kL47KbVJ1o3sKgoTefl6i+zebxbiDQWoe/oWWqPhIgQdRZRT1KA9sCPL810SA==", "license": "(MIT OR CC0-1.0)", "engines": { "node": ">=16" @@ -19573,111 +13338,19 @@ "node": ">= 0.6" } }, - "node_modules/typed-array-buffer": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.3.tgz", - "integrity": "sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==", - "dev": true, + "node_modules/type-is/node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-typed-array": "^1.1.14" - }, "engines": { - "node": ">= 0.4" - } - }, - "node_modules/typed-array-byte-length": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.3.tgz", - "integrity": "sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.14" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-byte-offset": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.4.tgz", - "integrity": "sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ==", - "dev": true, - "license": "MIT", - "dependencies": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.15", - "reflect.getprototypeof": "^1.0.9" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-length": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.7.tgz", - "integrity": "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.7", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "is-typed-array": "^1.1.13", - "possible-typed-array-names": "^1.0.0", - "reflect.getprototypeof": "^1.0.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typedarray.prototype.slice": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/typedarray.prototype.slice/-/typedarray.prototype.slice-1.0.5.tgz", - "integrity": "sha512-q7QNVDGTdl702bVFiI5eY4l/HkgCM6at9KhcFbgUAzezHFbOVy4+0O/lCjsABEQwbZPravVfBIiBVGo89yzHFg==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.9", - "es-errors": "^1.3.0", - "get-proto": "^1.0.1", - "math-intrinsics": "^1.1.0", - "typed-array-buffer": "^1.0.3", - "typed-array-byte-offset": "^1.0.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">= 0.6" } }, "node_modules/typescript": { - "version": "5.7.2", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.7.2.tgz", - "integrity": "sha512-i5t66RHxDvVN40HfDd1PsEThGNnlMCMT3jMUuoh9/0TaqWevNontacunWyN02LA9/fIbEWlcHZcgTKb9QoaLfg==", + "version": "5.7.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.7.3.tgz", + "integrity": "sha512-84MVSjMEHP+FQRPy3pX9sTVV/INIex71s9TL2Gm5FG/WG1SqXeKyZ0k7/blY/4FdOzI12CBy1vGc4og/eus0fw==", "devOptional": true, "license": "Apache-2.0", "bin": { @@ -19712,79 +13385,16 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/unbox-primitive": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.1.0.tgz", - "integrity": "sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.3", - "has-bigints": "^1.0.2", - "has-symbols": "^1.1.0", - "which-boxed-primitive": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/undici-types": { - "version": "6.20.0", - "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.20.0.tgz", - "integrity": "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg==", + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz", + "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==", "license": "MIT" }, - "node_modules/unicode-canonical-property-names-ecmascript": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.1.tgz", - "integrity": "sha512-dA8WbNeb2a6oQzAQ55YlT5vQAWGV9WXOsi3SskE3bcCdM0P4SDd+24zS/OCacdRq5BkdsRj9q3Pg6YyQoxIGqg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=4" - } - }, - "node_modules/unicode-match-property-ecmascript": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-2.0.0.tgz", - "integrity": "sha512-5kaZCrbp5mmbz5ulBkDkbY0SsPOjKqVS35VpL9ulMPfSl0J0Xsm+9Evphv9CoIZFwre7aJoa94AY6seMKGVN5Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "unicode-canonical-property-names-ecmascript": "^2.0.0", - "unicode-property-aliases-ecmascript": "^2.0.0" - }, - "engines": { - "node": ">=4" - } - }, - "node_modules/unicode-match-property-value-ecmascript": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-2.2.0.tgz", - "integrity": "sha512-4IehN3V/+kkr5YeSSDDQG8QLqO26XpL2XP3GQtqwlT/QYSECAwFztxVHjlbh0+gjJ3XmNLS0zDsbgs9jWKExLg==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=4" - } - }, - "node_modules/unicode-property-aliases-ecmascript": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-2.1.0.tgz", - "integrity": "sha512-6t3foTQI9qne+OZoVQB/8x8rk2k1eVy1gRXhV3oFQ5T6R1dqQ1xtin3XqSlx3+ATBkliTaR/hHyJBm+LVPNM8w==", - "dev": true, - "license": "MIT", - "engines": { - "node": ">=4" - } - }, "node_modules/unicorn-magic": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/unicorn-magic/-/unicorn-magic-0.1.0.tgz", - "integrity": "sha512-lRfVq8fE8gz6QMBuDM6a+LO3IAzTi05H6gCVaUpir2E1Rwpo4ZUog45KpNXKC/Mn3Yb9UDuHumeFTo9iV/D9FQ==", + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/unicorn-magic/-/unicorn-magic-0.3.0.tgz", + "integrity": "sha512-+QBBXBCvifc56fsbuxZQ6Sic3wqqc3WWaqxs58gvJrcOuN83HGTCwz3oS5phzU9LthRNE9VrJCFCLUgHeeFnfA==", "devOptional": true, "license": "MIT", "engines": { @@ -19813,9 +13423,9 @@ } }, "node_modules/update-browserslist-db": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.1.tgz", - "integrity": "sha512-R8UzCaa9Az+38REPiJ1tXlImTJXlVfgHZsglwBD/k6nj76ctsH1E3q4doGrukiLQd3sGQYu56r5+lo5r94l29A==", + "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": [ { @@ -19834,7 +13444,7 @@ "license": "MIT", "dependencies": { "escalade": "^3.2.0", - "picocolors": "^1.1.0" + "picocolors": "^1.1.1" }, "bin": { "update-browserslist-db": "cli.js" @@ -19843,24 +13453,6 @@ "browserslist": ">= 4.21.0" } }, - "node_modules/upper-case": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/upper-case/-/upper-case-2.0.2.tgz", - "integrity": "sha512-KgdgDGJt2TpuwBUIjgG6lzw2GWFRCW9Qkfkiv0DxqHHLYJHmtmdUIKcZd8rHgFSjopVTlw6ggzCm1b8MFQwikg==", - "license": "MIT", - "dependencies": { - "tslib": "^2.0.3" - } - }, - "node_modules/upper-case-first": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/upper-case-first/-/upper-case-first-2.0.2.tgz", - "integrity": "sha512-514ppYHBaKwfJRK/pNC6c/OxfGa0obSnAl106u97Ed0I625Nin96KAjttZF6ZL3e1XLtphxnqrOi9iWgm+u+bg==", - "license": "MIT", - "dependencies": { - "tslib": "^2.0.3" - } - }, "node_modules/uri-js": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", @@ -19877,13 +13469,6 @@ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", "license": "MIT" }, - "node_modules/utila": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/utila/-/utila-0.4.0.tgz", - "integrity": "sha512-Z0DbgELS9/L/75wZbro8xAnT50pBVFQZ+hUEueGDU5FN51YSCYM+jdxsfCiHjwNP/4LCDD0i/graKpeBnOXKRA==", - "dev": true, - "license": "MIT" - }, "node_modules/utils-merge": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", @@ -19917,16 +13502,21 @@ "dev": true, "license": "MIT" }, - "node_modules/valid-url": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/valid-url/-/valid-url-1.0.9.tgz", - "integrity": "sha512-QQDsV8OnSf5Uc30CKSwG9lnhMPe6exHtTXLRYX8uMwKENy640pU+2BgBL0LRbDh/eYRahNCS7aewCx0wf3NYVA==", - "dev": true + "node_modules/validate-npm-package-license": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", + "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "spdx-correct": "^3.0.0", + "spdx-expression-parse": "^3.0.0" + } }, "node_modules/validator": { - "version": "13.12.0", - "resolved": "https://registry.npmjs.org/validator/-/validator-13.12.0.tgz", - "integrity": "sha512-c1Q0mCiPlgdTVVVIJIrBuxNicYE+t/7oKeI9MWLj3fh/uq2Pxh/3eeWbVZ4OcGW1TUf53At0njHw5SMdA3tmMg==", + "version": "13.15.0", + "resolved": "https://registry.npmjs.org/validator/-/validator-13.15.0.tgz", + "integrity": "sha512-36B2ryl4+oL5QxZ3AzD0t5SsMNGvTtQHpjgFO5tbNxfXbMFkY822ktCDe1MnlqV3301QQI9SLHDNJokDI+Z9pA==", "license": "MIT", "engines": { "node": ">= 0.10" @@ -19942,21 +13532,23 @@ } }, "node_modules/vite": { - "version": "5.4.11", - "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.11.tgz", - "integrity": "sha512-c7jFQRklXua0mTzneGW9QVyxFjUgwcihC4bXEtujIo2ouWCe1Ajt/amn2PCxYnhYfd5k09JX3SB7OYWFKYqj8Q==", + "version": "6.3.4", + "resolved": "https://registry.npmjs.org/vite/-/vite-6.3.4.tgz", + "integrity": "sha512-BiReIiMS2fyFqbqNT/Qqt4CVITDU9M9vE+DKcVAsB+ZV0wvTKd+3hMbkpxz1b+NmEDMegpVbisKiAZOnvO92Sw==", "license": "MIT", - "peer": true, "dependencies": { - "esbuild": "^0.21.3", - "postcss": "^8.4.43", - "rollup": "^4.20.0" + "esbuild": "^0.25.0", + "fdir": "^6.4.4", + "picomatch": "^4.0.2", + "postcss": "^8.5.3", + "rollup": "^4.34.9", + "tinyglobby": "^0.2.13" }, "bin": { "vite": "bin/vite.js" }, "engines": { - "node": "^18.0.0 || >=20.0.0" + "node": "^18.0.0 || ^20.0.0 || >=22.0.0" }, "funding": { "url": "https://github.com/vitejs/vite?sponsor=1" @@ -19965,19 +13557,25 @@ "fsevents": "~2.3.3" }, "peerDependencies": { - "@types/node": "^18.0.0 || >=20.0.0", + "@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.4.0" + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" }, "peerDependenciesMeta": { "@types/node": { "optional": true }, + "jiti": { + "optional": true + }, "less": { "optional": true }, @@ -19998,6 +13596,12 @@ }, "terser": { "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true } } }, @@ -20006,7 +13610,6 @@ "resolved": "https://registry.npmjs.org/vite-plugin-restart/-/vite-plugin-restart-0.4.2.tgz", "integrity": "sha512-9aWN2ScJ8hbT7aC8SDeZnsbWapnslz1vhNq6Vgf2GU9WdN4NExlrWhtnu7pmtOUG3Guj8y6lPcUZ+ls7SVP33w==", "license": "MIT", - "peer": true, "dependencies": { "micromatch": "^4.0.8" }, @@ -20038,33 +13641,6 @@ } } }, - "node_modules/vue-demi": { - "version": "0.14.10", - "resolved": "https://registry.npmjs.org/vue-demi/-/vue-demi-0.14.10.tgz", - "integrity": "sha512-nMZBOwuzabUO0nLgIcc6rycZEebF6eeUfaiQx9+WSk8e29IbLvPU9feI6tqW4kTo3hvoYAJkMh8n8D0fuISphg==", - "dev": true, - "hasInstallScript": true, - "license": "MIT", - "bin": { - "vue-demi-fix": "bin/vue-demi-fix.js", - "vue-demi-switch": "bin/vue-demi-switch.js" - }, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/antfu" - }, - "peerDependencies": { - "@vue/composition-api": "^1.0.0-rc.1", - "vue": "^3.0.0-0 || ^2.6.0" - }, - "peerDependenciesMeta": { - "@vue/composition-api": { - "optional": true - } - } - }, "node_modules/vue-facing-decorator": { "version": "3.0.4", "resolved": "https://registry.npmjs.org/vue-facing-decorator/-/vue-facing-decorator-3.0.4.tgz", @@ -20098,72 +13674,6 @@ } } }, - "node_modules/vue-loader/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/vue-loader/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/vue-loader/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/vue-loader/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/vue-loader/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/vuedraggable": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/vuedraggable/-/vuedraggable-4.1.0.tgz", @@ -20200,16 +13710,6 @@ "minimalistic-assert": "^1.0.0" } }, - "node_modules/wcwidth": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/wcwidth/-/wcwidth-1.0.1.tgz", - "integrity": "sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==", - "dev": true, - "license": "MIT", - "dependencies": { - "defaults": "^1.0.3" - } - }, "node_modules/webidl-conversions": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", @@ -20217,15 +13717,16 @@ "license": "BSD-2-Clause" }, "node_modules/webpack": { - "version": "5.97.1", - "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.97.1.tgz", - "integrity": "sha512-EksG6gFY3L1eFMROS/7Wzgrii5mBAFe4rIr3r2BTfo7bcc+DWwFZ4OJ/miOuHJO/A85HwyI4eQ0F6IKXesO7Fg==", + "version": "5.99.7", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.99.7.tgz", + "integrity": "sha512-CNqKBRMQjwcmKR0idID5va1qlhrqVUKpovi+Ec79ksW8ux7iS1+A6VqzfZXgVYCFRKl7XL5ap3ZoMpwBJxcg0w==", "dev": true, "license": "MIT", "peer": true, "dependencies": { "@types/eslint-scope": "^3.7.7", "@types/estree": "^1.0.6", + "@types/json-schema": "^7.0.15", "@webassemblyjs/ast": "^1.14.1", "@webassemblyjs/wasm-edit": "^1.14.1", "@webassemblyjs/wasm-parser": "^1.14.1", @@ -20242,9 +13743,9 @@ "loader-runner": "^4.2.0", "mime-types": "^2.1.27", "neo-async": "^2.6.2", - "schema-utils": "^3.2.0", + "schema-utils": "^4.3.2", "tapable": "^2.1.1", - "terser-webpack-plugin": "^5.3.10", + "terser-webpack-plugin": "^5.3.11", "watchpack": "^2.4.1", "webpack-sources": "^3.2.3" }, @@ -20264,64 +13765,6 @@ } } }, - "node_modules/webpack-cli": { - "version": "5.1.4", - "resolved": "https://registry.npmjs.org/webpack-cli/-/webpack-cli-5.1.4.tgz", - "integrity": "sha512-pIDJHIEI9LR0yxHXQ+Qh95k2EvXpWzZ5l+d+jIo+RdSm9MiHfzazIxwwni/p7+x4eJZuvG1AJwgC4TNQ7NRgsg==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "@discoveryjs/json-ext": "^0.5.0", - "@webpack-cli/configtest": "^2.1.1", - "@webpack-cli/info": "^2.0.2", - "@webpack-cli/serve": "^2.0.5", - "colorette": "^2.0.14", - "commander": "^10.0.1", - "cross-spawn": "^7.0.3", - "envinfo": "^7.7.3", - "fastest-levenshtein": "^1.0.12", - "import-local": "^3.0.2", - "interpret": "^3.1.1", - "rechoir": "^0.8.0", - "webpack-merge": "^5.7.3" - }, - "bin": { - "webpack-cli": "bin/cli.js" - }, - "engines": { - "node": ">=14.15.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/webpack" - }, - "peerDependencies": { - "webpack": "5.x.x" - }, - "peerDependenciesMeta": { - "@webpack-cli/generators": { - "optional": true - }, - "webpack-bundle-analyzer": { - "optional": true - }, - "webpack-dev-server": { - "optional": true - } - } - }, - "node_modules/webpack-cli/node_modules/interpret": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/interpret/-/interpret-3.1.1.tgz", - "integrity": "sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ==", - "dev": true, - "license": "MIT", - "peer": true, - "engines": { - "node": ">=10.13.0" - } - }, "node_modules/webpack-dev-middleware": { "version": "7.4.2", "resolved": "https://registry.npmjs.org/webpack-dev-middleware/-/webpack-dev-middleware-7.4.2.tgz", @@ -20352,36 +13795,17 @@ } } }, - "node_modules/webpack-dev-middleware/node_modules/memfs": { - "version": "4.15.3", - "resolved": "https://registry.npmjs.org/memfs/-/memfs-4.15.3.tgz", - "integrity": "sha512-vR/g1SgqvKJgAyYla+06G4p/EOcEmwhYuVb1yc1ixcKf8o/sh7Zngv63957ZSNd1xrZJoinmNyDf2LzuP8WJXw==", - "dev": true, - "license": "Apache-2.0", - "dependencies": { - "@jsonjoy.com/json-pack": "^1.0.3", - "@jsonjoy.com/util": "^1.3.0", - "tree-dump": "^1.0.1", - "tslib": "^2.0.0" - }, - "engines": { - "node": ">= 4.0.0" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/streamich" - } - }, "node_modules/webpack-dev-server": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-5.2.0.tgz", - "integrity": "sha512-90SqqYXA2SK36KcT6o1bvwvZfJFcmoamqeJY7+boioffX9g9C0wjjJRGUrQIuh43pb0ttX7+ssavmj/WN2RHtA==", + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-5.2.1.tgz", + "integrity": "sha512-ml/0HIj9NLpVKOMq+SuBPLHcmbG+TGIjXRHsYfZwocUBIqEvws8NnS/V9AFQ5FKP+tgn5adwVwRrTEpGL33QFQ==", "dev": true, "license": "MIT", "dependencies": { "@types/bonjour": "^3.5.13", "@types/connect-history-api-fallback": "^1.5.4", "@types/express": "^4.17.21", + "@types/express-serve-static-core": "^4.17.21", "@types/serve-index": "^1.9.4", "@types/serve-static": "^1.15.5", "@types/sockjs": "^0.3.36", @@ -20503,22 +13927,6 @@ "node": ">=8.10.0" } }, - "node_modules/webpack-merge": { - "version": "5.10.0", - "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-5.10.0.tgz", - "integrity": "sha512-+4zXKdx7UnO+1jaN4l2lHVD+mFvnlZQP/6ljaJVb4SZiwIKeUnrT5l0gkT8z+n4hKpC+jpOv6O9R+gLtag7pSA==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "clone-deep": "^4.0.1", - "flat": "^5.0.2", - "wildcard": "^2.0.0" - }, - "engines": { - "node": ">=10.0.0" - } - }, "node_modules/webpack-sources": { "version": "3.2.3", "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-3.2.3.tgz", @@ -20556,26 +13964,6 @@ "node": ">=4.0" } }, - "node_modules/webpack/node_modules/schema-utils": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.3.0.tgz", - "integrity": "sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==", - "dev": true, - "license": "MIT", - "peer": true, - "dependencies": { - "@types/json-schema": "^7.0.8", - "ajv": "^6.12.5", - "ajv-keywords": "^3.5.2" - }, - "engines": { - "node": ">= 10.13.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/webpack" - } - }, "node_modules/websocket-driver": { "version": "0.7.4", "resolved": "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.7.4.tgz", @@ -20627,100 +14015,12 @@ "node": ">= 8" } }, - "node_modules/which-boxed-primitive": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.1.1.tgz", - "integrity": "sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-bigint": "^1.1.0", - "is-boolean-object": "^1.2.1", - "is-number-object": "^1.1.1", - "is-string": "^1.1.1", - "is-symbol": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-builtin-type": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.2.1.tgz", - "integrity": "sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "call-bound": "^1.0.2", - "function.prototype.name": "^1.1.6", - "has-tostringtag": "^1.0.2", - "is-async-function": "^2.0.0", - "is-date-object": "^1.1.0", - "is-finalizationregistry": "^1.1.0", - "is-generator-function": "^1.0.10", - "is-regex": "^1.2.1", - "is-weakref": "^1.0.2", - "isarray": "^2.0.5", - "which-boxed-primitive": "^1.1.0", - "which-collection": "^1.0.2", - "which-typed-array": "^1.1.16" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-collection": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", - "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", - "dev": true, - "license": "MIT", - "dependencies": { - "is-map": "^2.0.3", - "is-set": "^2.0.3", - "is-weakmap": "^2.0.2", - "is-weakset": "^2.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/which-module": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.1.tgz", "integrity": "sha512-iBdZ57RDvnOR9AGBhML2vFZf7h8vmBjhoaZqODJBFWHVtKkDmKuHai3cx5PgVMrX5YDNp27AofYbAwctSS+vhQ==", "license": "ISC" }, - "node_modules/which-typed-array": { - "version": "1.1.18", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.18.tgz", - "integrity": "sha512-qEcY+KJYlWyLH9vNbsr6/5j59AXk5ni5aakf8ldzBvGde6Iz4sxZGkJyWSAueTG7QhOvNRYb1lDdFmL5Td0QKA==", - "dev": true, - "license": "MIT", - "dependencies": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/wide-align": { "version": "1.1.5", "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.5.tgz", @@ -20759,14 +14059,6 @@ "node": ">=8" } }, - "node_modules/wildcard": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/wildcard/-/wildcard-2.0.1.tgz", - "integrity": "sha512-CC1bOL87PIWSBhDcTrdeLo6eGT7mCFtrg0uIJtqJUFyK+eJnzl8A1niH56uu7KMa5XFrtiV+AQuHO3n7DsHnLQ==", - "dev": true, - "license": "MIT", - "peer": true - }, "node_modules/word-wrap": { "version": "1.2.5", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", @@ -20819,42 +14111,6 @@ "url": "https://github.com/chalk/wrap-ansi?sponsor=1" } }, - "node_modules/wrap-ansi-cjs/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/wrap-ansi-cjs/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/wrap-ansi-cjs/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/wrap-ansi-cjs/node_modules/emoji-regex": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", @@ -20933,9 +14189,9 @@ "license": "ISC" }, "node_modules/ws": { - "version": "8.18.0", - "resolved": "https://registry.npmjs.org/ws/-/ws-8.18.0.tgz", - "integrity": "sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==", + "version": "8.18.1", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.18.1.tgz", + "integrity": "sha512-RKW2aJZMXeMxVpnZ6bck+RswznaxmzdULiBr6KY7XkTnW8uvt0iT9H5DkHUChXrc+uurzwa0rVI16n/Xzjdz1w==", "dev": true, "license": "MIT", "engines": { @@ -20969,6 +14225,28 @@ "node": ">=12.0" } }, + "node_modules/xmlbuilder2/node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "license": "MIT", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/xmlbuilder2/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "license": "MIT", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, "node_modules/xslt3": { "version": "2.7.0", "resolved": "https://registry.npmjs.org/xslt3/-/xslt3-2.7.0.tgz", @@ -20983,18 +14261,6 @@ "xslt3": "xslt3.js" } }, - "node_modules/xslt3/node_modules/axios": { - "version": "1.7.9", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.7.9.tgz", - "integrity": "sha512-LhLcE7Hbiryz8oMDdDptSrWowmB4Bl6RCt6sIJKpRB4XtVf0iEgewX3au/pJqm+Py1kCASkb/FFKjxQaLtxJvw==", - "dev": true, - "license": "MIT", - "dependencies": { - "follow-redirects": "^1.15.6", - "form-data": "^4.0.0", - "proxy-from-env": "^1.1.0" - } - }, "node_modules/xtend": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", @@ -21015,13 +14281,14 @@ "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", "dev": true, - "license": "ISC" + "license": "ISC", + "peer": true }, "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, + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.7.1.tgz", + "integrity": "sha512-10ULxpnOCQXxJvBgxsn9ptjq6uviG/htZKk9veJGhlqn3w/DxQ631zFF+nlQXLwmImeS5amR2dl2U8sg6U9jsQ==", + "devOptional": true, "license": "ISC", "bin": { "yaml": "bin.mjs" @@ -21175,9 +14442,9 @@ } }, "node_modules/yocto-queue": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-1.1.1.tgz", - "integrity": "sha512-b4JR1PFR10y1mKjhHY9LaGo6tmrgjit7hxVIeAmyMw3jegXR4dhYqLaQF5zMXZxY7tLpMyJeLjr1C4rLmkVe8g==", + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-1.2.1.tgz", + "integrity": "sha512-AyeEbWOu/TAXdxlV9wmGcR0+yh2j3vYPGOECcIj2S7MkrLyC7ne+oye2BKTItt0ii2PHk4cDy+95+LshzbXnGg==", "license": "MIT", "engines": { "node": ">=12.20" @@ -21210,6 +14477,20 @@ "stacktracey": "^2.1.8" } }, + "node_modules/youch-core": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/youch-core/-/youch-core-0.3.2.tgz", + "integrity": "sha512-fusrlIMLeRvTFYLUjJ9KzlGC3N+6MOPJ68HNj/yJv2nz7zq8t4HEviLms2gkdRPUS7F5rZ5n+pYx9r88m6IE1g==", + "devOptional": true, + "license": "MIT", + "dependencies": { + "@poppinss/exception": "^1.2.0", + "error-stack-parser-es": "^1.0.5" + }, + "engines": { + "node": ">=18" + } + }, "node_modules/youch-terminal": { "version": "2.2.3", "resolved": "https://registry.npmjs.org/youch-terminal/-/youch-terminal-2.2.3.tgz", @@ -21258,38 +14539,6 @@ "engines": { "node": ">= 0.6" } - }, - "node_modules/z-schema": { - "version": "5.0.5", - "resolved": "https://registry.npmjs.org/z-schema/-/z-schema-5.0.5.tgz", - "integrity": "sha512-D7eujBWkLa3p2sIpJA0d1pr7es+a7m0vFAnZLlCEKq/Ij2k0MLi9Br2UPxoxdYystm5K1yeBGzub0FlYUEWj2Q==", - "dev": true, - "license": "MIT", - "dependencies": { - "lodash.get": "^4.4.2", - "lodash.isequal": "^4.5.0", - "validator": "^13.7.0" - }, - "bin": { - "z-schema": "bin/z-schema" - }, - "engines": { - "node": ">=8.0.0" - }, - "optionalDependencies": { - "commander": "^9.4.1" - } - }, - "node_modules/z-schema/node_modules/commander": { - "version": "9.5.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-9.5.0.tgz", - "integrity": "sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==", - "dev": true, - "license": "MIT", - "optional": true, - "engines": { - "node": "^12.20.0 || >=14" - } } } } diff --git a/package.json b/package.json index 9f4fb23..c858691 100644 --- a/package.json +++ b/package.json @@ -4,7 +4,8 @@ "private": true, "scripts": { "type-check": "tsc --noEmit", - "dev": "node ace serve --watch", + "dev": "node ace serve", + "devInspect": "node ace serve --watch --node-args='--inspect'", "compress:xslt": "./node_modules/xslt3/xslt3.js -xsl:public/assets2/datasetxml2oai-pmh.xslt -export:public/assets2/datasetxml2oai.sef.json -t -nogo '-ns:##html5'", "compress:solr": "./node_modules/xslt3/xslt3.js -xsl:public/assets2/solr.xslt -export:public/assets2/solr.sef.json -t -nogo '-ns:##html5'", "compress:doi": "./node_modules/xslt3/xslt3.js -xsl:public/assets2/doi_datacite.xslt -export:public/assets2/doi_datacite.sef.json -t -nogo '-ns:##html5'", @@ -15,59 +16,58 @@ "format-check": "prettier --check ./**/*.{ts,js}", "test": "node ace test" }, - "eslintIgnore": [ - "build" - ], + "eslintConfig": { + "ignorePatterns": [ + "build" + ] + }, "alias": { "vue": "./node_modules/vue/dist/vue.esm-bundler.js" }, "devDependencies": { "@adonisjs/assembler": "^7.1.1", - "@adonisjs/tsconfig": "^1.2.1", - "@babel/core": "^7.20.12", - "@babel/plugin-proposal-class-properties": "^7.18.6", - "@babel/plugin-proposal-decorators": "^7.20.13", - "@babel/plugin-transform-runtime": "^7.19.6", - "@babel/preset-env": "^7.20.2", - "@babel/preset-typescript": "^7.18.6", - "@japa/api-client": "^2.0.3", - "@japa/assert": "^3.0.0", - "@japa/plugin-adonisjs": "^3.0.0", - "@japa/runner": "^3.1.1", + "@adonisjs/tsconfig": "^1.4.0", + "@headlessui/vue": "^1.7.23", + "@japa/assert": "^4.0.1", + "@japa/plugin-adonisjs": "^4.0.0", + "@japa/runner": "^4.2.0", "@mdi/js": "^7.1.96", "@poppinss/utils": "^6.7.2", - "@swc/core": "^1.4.2", - "@symfony/webpack-encore": "^5.0.1", + "@swc/wasm": "^1.10.14", "@tailwindcss/forms": "^0.5.2", "@types/bcryptjs": "^2.4.6", "@types/clamscan": "^2.0.4", "@types/escape-html": "^1.0.4", - "@types/leaflet": "^1.9.3", + "@types/fs-extra": "^11.0.4", + "@types/leaflet": "^1.9.16", "@types/luxon": "^3.4.2", - "@types/node": "^22.5.5", + "@types/node": "^22.10.2", "@types/proxy-addr": "^2.0.0", "@types/qrcode": "^1.5.5", "@types/source-map-support": "^0.5.6", "@types/sprintf-js": "^1.1.4", "@types/supertest": "^6.0.2", + "@vitejs/plugin-vue": "^5.2.1", "autoprefixer": "^10.4.13", "babel-preset-typescript-vue3": "^2.0.17", "chart.js": "^4.2.0", "dotenv-webpack": "^8.0.1", "eslint": "^8.57.1", - "eslint-config-prettier": "^9.0.0", + "eslint-config-prettier": "^10.0.1", "eslint-plugin-adonis": "^2.1.1", "eslint-plugin-prettier": "^5.0.0-alpha.2", + "hot-hook": "^0.4.0", "numeral": "^2.0.6", - "pinia": "^2.0.30", - "pino-pretty": "^11.2.2", + "pinia": "^3.0.2", + "pino-pretty": "^13.0.0", "postcss-loader": "^8.1.1", - "prettier": "^3.0.0", + "prettier": "^3.4.2", "supertest": "^6.3.3", - "tailwindcss": "^3.2.4", + "tailwindcss": "^3.4.17", "ts-loader": "^9.4.2", - "ts-node": "^10.9.2", - "typescript": "^5.1.3", + "ts-node-maintained": "^10.9.5", + "typescript": "~5.7", + "vite": "^6.0.11", "vue": "^3.4.26", "vue-facing-decorator": "^3.0.0", "vue-loader": "^17.0.1", @@ -75,30 +75,32 @@ "xslt3": "^2.5.0" }, "dependencies": { - "@adonisjs/auth": "^9.1.1", - "@adonisjs/core": "^6.3.1", + "@adonisjs/auth": "^9.2.4", + "@adonisjs/bodyparser": "^10.0.1", + "@adonisjs/core": "^6.17.0", "@adonisjs/cors": "^2.2.1", - "@adonisjs/drive": "^2.3.0", - "@adonisjs/encore": "^1.0.0", - "@adonisjs/inertia": "^1.0.0-7", - "@adonisjs/lucid": "^21.1.0", + "@adonisjs/drive": "^3.2.0", + "@adonisjs/inertia": "^2.1.3", + "@adonisjs/lucid": "^21.5.1", "@adonisjs/mail": "^9.2.2", "@adonisjs/redis": "^9.1.0", - "@adonisjs/session": "^7.1.1", + "@adonisjs/session": "^7.5.0", "@adonisjs/shield": "^8.1.1", "@adonisjs/static": "^1.1.1", + "@adonisjs/vite": "^4.0.0", "@eidellev/adonis-stardust": "^3.0.0", "@fontsource/archivo-black": "^5.0.1", "@fontsource/inter": "^5.0.1", "@inertiajs/inertia": "^0.11.1", - "@inertiajs/vue3": "^1.0.0", - "@opensearch-project/opensearch": "^2.4.0", + "@inertiajs/vue3": "^2.0.3", + "@opensearch-project/opensearch": "^3.2.0", "@phc/format": "^1.0.0", - "@vinejs/vine": "^2.0.0", + "@poppinss/manager": "^5.0.2", + "@vinejs/vine": "^3.0.0", + "axios": "^1.7.9", "bcrypt": "^5.1.1", "bcryptjs": "^2.4.3", "clamscan": "^2.1.2", - "crypto": "^1.0.1", "dayjs": "^1.11.7", "deep-email-validator": "^0.1.21", "edge.js": "^6.0.1", @@ -114,13 +116,19 @@ "notiwind": "^2.0.0", "pg": "^8.9.0", "qrcode": "^1.5.3", - "redis": "^4.6.10", + "redis": "^5.0.0", "reflect-metadata": "^0.2.1", "saxon-js": "^2.5.0", "toastify-js": "^1.12.0", "vuedraggable": "^4.1.0", "xmlbuilder2": "^3.1.1" }, + "hotHook": { + "boundaries": [ + "./app/Controllers/**/*.ts", + "./app/middleware/*.ts" + ] + }, "type": "module", "imports": { "#controllers/*": "./app/Controllers/*.js", diff --git a/postcss.config.cjs b/postcss.config.cjs index b6568f7..4ae9206 100644 --- a/postcss.config.cjs +++ b/postcss.config.cjs @@ -1,7 +1,10 @@ module.exports = { plugins: { // 'postcss-import': {}, - 'tailwindcss/nesting': {}, + // 'postcss-nesting': {}, + 'tailwindcss/nesting': {}, + // "@tailwindcss/postcss": {}, + // tailwindcss: {}, tailwindcss: {}, autoprefixer: {}, }, diff --git a/providers/drive/drivers/local.ts b/providers/drive/drivers/local.ts index b008752..7073cd0 100644 --- a/providers/drive/drivers/local.ts +++ b/providers/drive/drivers/local.ts @@ -74,7 +74,8 @@ export class LocalDriver implements LocalDriverContract { */ public async exists(location: string): Promise<boolean> { try { - return await this.adapter.pathExists(this.makePath(location)); + let path_temp = this.makePath(location); //'/storage/app/files/421' + return await this.adapter.pathExists(path_temp); } catch (error) { throw CannotGetMetaDataException.invoke(location, 'exists', error); } diff --git a/providers/mail_provider.ts b/providers/mail_provider.ts index bb775d8..ba140bb 100644 --- a/providers/mail_provider.ts +++ b/providers/mail_provider.ts @@ -69,7 +69,7 @@ export default class MailProvider { const mailConfigProvider = this.app.config.get('mail'); const config = await configProvider.resolve<any>(this.app, mailConfigProvider); - const iwas = await config.mailers.smtp(); + await config.mailers.smtp(); // iwas.config.host = 'hhhost'; // this.app.config.set('mail.mailers.smtp.host', 'xhost'); // const iwas = await config.mailers.smtp(); diff --git a/providers/query_builder_provider.ts b/providers/query_builder_provider.ts index 6509fae..ed7cfb7 100644 --- a/providers/query_builder_provider.ts +++ b/providers/query_builder_provider.ts @@ -63,6 +63,15 @@ export default class QueryBuilderProvider { public register() { // Register your own bindings + // const ModelQueryBuilder = this.app.container.bind('@adonisjs/lucid/orm/ModelQueryBuilder'); + + // ModelQueryBuilder.macro('whereTrue', function (columnName: string) { + // return this.where(columnName, true); + // }); + + // ModelQueryBuilder.macro('whereFalse', function (columnName: string) { + // return this.where(columnName, false); + // }); } public async boot() { @@ -73,15 +82,14 @@ export default class QueryBuilderProvider { // let rolesPluck = {}; let rolesPluck: { [key: number]: any } = {}; const result = await this.exec(); - result.forEach((user, index) => { - let idc; + result.forEach((user: { [key: string]: any }, index: number) => { + let idc: number; if (!id) { idc = index; } else { idc = user[id]; } - const value = user[valueColumn]; - // rolesPluck[idc] = user.name; + const value: any = user[valueColumn]; rolesPluck[idc] = value; }); return rolesPluck; diff --git a/providers/vinejs_provider.ts b/providers/vinejs_provider.ts index f139594..e4aad4c 100644 --- a/providers/vinejs_provider.ts +++ b/providers/vinejs_provider.ts @@ -4,14 +4,15 @@ |-------------------------------------------------------------------------- |*/ import type { ApplicationService } from '@adonisjs/core/types'; -import vine, { BaseLiteralType, Vine } from '@vinejs/vine'; -import type { Validation, FieldContext, FieldOptions } from '@vinejs/vine/types'; +import vine, { symbols, BaseLiteralType, Vine } from '@vinejs/vine'; +import type { FieldContext, FieldOptions } from '@vinejs/vine/types'; // import type { MultipartFile, FileValidationOptions } from '@adonisjs/bodyparser/types'; import type { MultipartFile } from '@adonisjs/core/bodyparser'; import type { FileValidationOptions } from '@adonisjs/core/types/bodyparser'; import { Request, RequestValidator } from '@adonisjs/core/http'; import MimeType from '#models/mime_type'; + /** * Validation options accepted by the "file" rule */ @@ -28,8 +29,7 @@ declare module '@vinejs/vine' { * Extend HTTP request class */ declare module '@adonisjs/core/http' { - interface Request extends RequestValidator { - } + interface Request extends RequestValidator {} } /** @@ -48,7 +48,7 @@ export async function getEnabledExtensions() { .flat(); return extensions; -}; +} /** * VineJS validation rule that validates the file to be an * instance of BodyParser MultipartFile class. @@ -79,11 +79,13 @@ const isMultipartFile = vine.createRule(async (file: MultipartFile | unknown, op // if (validatedFile.allowedExtensions === undefined && validationOptions.extnames) { // validatedFile.allowedExtensions = validationOptions.extnames; // } - if (validatedFile.allowedExtensions === undefined && validationOptions.extnames) { + if (validatedFile.allowedExtensions === undefined && validationOptions.extnames !== undefined) { + validatedFile.allowedExtensions = validationOptions.extnames; // await getEnabledExtensions(); + } else if (validatedFile.allowedExtensions === undefined && validationOptions.extnames === undefined) { validatedFile.allowedExtensions = await getEnabledExtensions(); } - /** - * wieder löschen + /** + * wieder löschen * Set extensions when it's defined in the options and missing * on the file instance */ @@ -102,7 +104,20 @@ const isMultipartFile = vine.createRule(async (file: MultipartFile | unknown, op }); }); +const MULTIPART_FILE: typeof symbols.SUBTYPE = symbols.SUBTYPE; + export class VineMultipartFile extends BaseLiteralType<MultipartFile, MultipartFile, MultipartFile> { + + [MULTIPART_FILE]: string; + // constructor(validationOptions?: FileRuleValidationOptions, options?: FieldOptions) { + // super(options, [isMultipartFile(validationOptions || {})]); + // this.validationOptions = validationOptions; + // this.#private = true; + // } + + // clone(): this { + // return new VineMultipartFile(this.validationOptions, this.cloneOptions()) as this; + // } // #private; // constructor(validationOptions?: FileRuleValidationOptions, options?: FieldOptions, validations?: Validation<any>[]); // clone(): this; @@ -111,14 +126,16 @@ export class VineMultipartFile extends BaseLiteralType<MultipartFile, MultipartF // extnames: (18) ['gpkg', 'htm', 'html', 'csv', 'txt', 'asc', 'c', 'cc', 'h', 'srt', 'tiff', 'pdf', 'png', 'zip', 'jpg', 'jpeg', 'jpe', 'xlsx'] // size: '512mb' - public constructor(validationOptions?: FileRuleValidationOptions, options?: FieldOptions, validations?: Validation<any>[]) { + // public constructor(validationOptions?: FileRuleValidationOptions, options?: FieldOptions, validations?: Validation<any>[]) { + public constructor(validationOptions?: FileRuleValidationOptions, options?: FieldOptions) { // super(options, validations); super(options, [isMultipartFile(validationOptions || {})]); this.validationOptions = validationOptions; } public clone(): any { - return new VineMultipartFile(this.validationOptions, this.cloneOptions(), this.cloneValidations()); + // return new VineMultipartFile(this.validationOptions, this.cloneOptions(), this.cloneValidations()); + return new VineMultipartFile(this.validationOptions, this.cloneOptions()); } } @@ -152,10 +169,12 @@ export default class VinejsProvider { * The validate method can be used to validate the request * data for the current request using VineJS validators */ - Request.macro('validateUsing', function (...args) { - return new RequestValidator(this.ctx).validateUsing(...args); - }); - + Request.macro('validateUsing', function (this: Request, ...args) { + if (!this.ctx) { + throw new Error('HttpContext is not available'); + } + return new RequestValidator(this.ctx).validateUsing(...args); + }); } /** diff --git a/public/android-chrome-192x192.png b/public/android-chrome-192x192.png new file mode 100644 index 0000000..4d9cf1b Binary files /dev/null and b/public/android-chrome-192x192.png differ diff --git a/public/android-chrome-512x512.png b/public/android-chrome-512x512.png new file mode 100644 index 0000000..7929356 Binary files /dev/null and b/public/android-chrome-512x512.png differ diff --git a/public/apple-touch-icon.png b/public/apple-touch-icon.png new file mode 100644 index 0000000..50558dc Binary files /dev/null and b/public/apple-touch-icon.png differ diff --git a/public/assets/entrypoints.json b/public/assets/entrypoints.json deleted file mode 100644 index 21084b2..0000000 --- a/public/assets/entrypoints.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "entrypoints": { - "app": { - "css": [ - "http://localhost:8080/assets/app.css" - ], - "js": [ - "http://localhost:8080/assets/app.js" - ] - } - } -} \ No newline at end of file diff --git a/public/assets/manifest.json b/public/assets/manifest.json deleted file mode 100644 index 9062764..0000000 --- a/public/assets/manifest.json +++ /dev/null @@ -1,103 +0,0 @@ -{ - "assets/app.css": "http://localhost:8080/assets/app.css", - "assets/app.js": "http://localhost:8080/assets/app.js", - "assets/resources_js_apps_settings_l18n_de_js.js": "http://localhost:8080/assets/resources_js_apps_settings_l18n_de_js.js", - "assets/resources_js_apps_settings_l18n_en_js.js": "http://localhost:8080/assets/resources_js_apps_settings_l18n_en_js.js", - "assets/resources_js_Pages_Admin_License_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_License_Index_vue.js", - "assets/resources_js_Pages_Admin_Mimetype_Create_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Mimetype_Create_vue.js", - "assets/resources_js_Pages_Admin_Mimetype_Delete_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Mimetype_Delete_vue.js", - "assets/resources_js_Pages_Admin_Mimetype_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Mimetype_Index_vue.js", - "assets/resources_js_Pages_Admin_Permission_Create_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Permission_Create_vue.js", - "assets/resources_js_Pages_Admin_Permission_Edit_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Permission_Edit_vue.js", - "assets/resources_js_Pages_Admin_Permission_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Permission_Index_vue.js", - "assets/resources_js_Pages_Admin_Permission_Show_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Permission_Show_vue.js", - "assets/resources_js_Pages_Admin_Role_Create_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Role_Create_vue.js", - "assets/resources_js_Pages_Admin_Role_Edit_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Role_Edit_vue.js", - "assets/resources_js_Pages_Admin_Role_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Role_Index_vue.js", - "assets/resources_js_Pages_Admin_Role_Show_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Role_Show_vue.js", - "assets/resources_js_Pages_Admin_Settings_vue-resources_js_utils_toast_css.css": "http://localhost:8080/assets/resources_js_Pages_Admin_Settings_vue-resources_js_utils_toast_css.css", - "assets/resources_js_Pages_Admin_Settings_vue-resources_js_utils_toast_css.js": "http://localhost:8080/assets/resources_js_Pages_Admin_Settings_vue-resources_js_utils_toast_css.js", - "assets/resources_js_Pages_Admin_User_Create_vue-resources_js_Components_SimplePasswordMeter_password-f3312a.css": "http://localhost:8080/assets/resources_js_Pages_Admin_User_Create_vue-resources_js_Components_SimplePasswordMeter_password-f3312a.css", - "assets/resources_js_Pages_Admin_User_Create_vue-resources_js_Components_SimplePasswordMeter_password-f3312a.js": "http://localhost:8080/assets/resources_js_Pages_Admin_User_Create_vue-resources_js_Components_SimplePasswordMeter_password-f3312a.js", - "assets/resources_js_Pages_Admin_User_Edit_vue-resources_js_Components_SimplePasswordMeter_password-m-6dc207.css": "http://localhost:8080/assets/resources_js_Pages_Admin_User_Edit_vue-resources_js_Components_SimplePasswordMeter_password-m-6dc207.css", - "assets/resources_js_Pages_Admin_User_Edit_vue-resources_js_Components_SimplePasswordMeter_password-m-6dc207.js": "http://localhost:8080/assets/resources_js_Pages_Admin_User_Edit_vue-resources_js_Components_SimplePasswordMeter_password-m-6dc207.js", - "assets/resources_js_Pages_Admin_User_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_User_Index_vue.js", - "assets/resources_js_Pages_Admin_User_Show_vue.js": "http://localhost:8080/assets/resources_js_Pages_Admin_User_Show_vue.js", - "assets/resources_js_Pages_App_vue.js": "http://localhost:8080/assets/resources_js_Pages_App_vue.js", - "assets/resources_js_Pages_Auth_AccountInfo_vue-resources_js_utils_toast_css-resources_js_Components_-06c7b5.css": "http://localhost:8080/assets/resources_js_Pages_Auth_AccountInfo_vue-resources_js_utils_toast_css-resources_js_Components_-06c7b5.css", - "assets/resources_js_Pages_Auth_AccountInfo_vue-resources_js_utils_toast_css-resources_js_Components_-06c7b5.js": "http://localhost:8080/assets/resources_js_Pages_Auth_AccountInfo_vue-resources_js_utils_toast_css-resources_js_Components_-06c7b5.js", - "assets/resources_js_Pages_Auth_Login_vue.js": "http://localhost:8080/assets/resources_js_Pages_Auth_Login_vue.js", - "assets/resources_js_Pages_Auth_Register_vue.js": "http://localhost:8080/assets/resources_js_Pages_Auth_Register_vue.js", - "assets/resources_js_Pages_Dashboard_vue.js": "http://localhost:8080/assets/resources_js_Pages_Dashboard_vue.js", - "assets/resources_js_Pages_Editor_Dataset_Approve_vue.js": "http://localhost:8080/assets/resources_js_Pages_Editor_Dataset_Approve_vue.js", - "assets/resources_js_Pages_Editor_Dataset_Doi_vue.js": "http://localhost:8080/assets/resources_js_Pages_Editor_Dataset_Doi_vue.js", - "assets/resources_js_Pages_Editor_Dataset_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Editor_Dataset_Index_vue.js", - "assets/resources_js_Pages_Editor_Dataset_Publish_vue.js": "http://localhost:8080/assets/resources_js_Pages_Editor_Dataset_Publish_vue.js", - "assets/resources_js_Pages_Editor_Dataset_Receive_vue.js": "http://localhost:8080/assets/resources_js_Pages_Editor_Dataset_Receive_vue.js", - "assets/resources_js_Pages_Editor_Dataset_Reject_vue.js": "http://localhost:8080/assets/resources_js_Pages_Editor_Dataset_Reject_vue.js", - "assets/resources_js_Pages_Error_vue.js": "http://localhost:8080/assets/resources_js_Pages_Error_vue.js", - "assets/resources_js_Pages_Errors_ServerError_vue.js": "http://localhost:8080/assets/resources_js_Pages_Errors_ServerError_vue.js", - "assets/resources_js_Pages_Errors_not_found_vue.js": "http://localhost:8080/assets/resources_js_Pages_Errors_not_found_vue.js", - "assets/resources_js_Pages_Map_vue-resources_js_Components_Map_draw_component_vue-resources_js_Compon-b0925c.css": "http://localhost:8080/assets/resources_js_Pages_Map_vue-resources_js_Components_Map_draw_component_vue-resources_js_Compon-b0925c.css", - "assets/resources_js_Pages_Map_vue-resources_js_Components_Map_draw_component_vue-resources_js_Compon-b0925c.js": "http://localhost:8080/assets/resources_js_Pages_Map_vue-resources_js_Components_Map_draw_component_vue-resources_js_Compon-b0925c.js", - "assets/resources_js_Pages_ProfileView_vue.js": "http://localhost:8080/assets/resources_js_Pages_ProfileView_vue.js", - "assets/resources_js_Pages_Reviewer_Dataset_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Reviewer_Dataset_Index_vue.js", - "assets/resources_js_Pages_Reviewer_Dataset_Reject_vue.js": "http://localhost:8080/assets/resources_js_Pages_Reviewer_Dataset_Reject_vue.js", - "assets/resources_js_Pages_Reviewer_Dataset_Review_vue.js": "http://localhost:8080/assets/resources_js_Pages_Reviewer_Dataset_Review_vue.js", - "assets/resources_js_Pages_Submitter_Dataset_Category_vue.css": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Category_vue.css", - "assets/resources_js_Pages_Submitter_Dataset_Category_vue.js": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Category_vue.js", - "assets/resources_js_Pages_Submitter_Dataset_Create_vue-resources_js_utils_toast_css-resources_js_Com-03a898.css": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Create_vue-resources_js_utils_toast_css-resources_js_Com-03a898.css", - "assets/resources_js_Pages_Submitter_Dataset_Create_vue-resources_js_utils_toast_css-resources_js_Com-03a898.js": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Create_vue-resources_js_utils_toast_css-resources_js_Com-03a898.js", - "assets/resources_js_Pages_Submitter_Dataset_Delete_vue.js": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Delete_vue.js", - "assets/resources_js_Pages_Submitter_Dataset_Edit_vue-resources_js_utils_toast_css-resources_js_Compo-a37b65.css": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Edit_vue-resources_js_utils_toast_css-resources_js_Compo-a37b65.css", - "assets/resources_js_Pages_Submitter_Dataset_Edit_vue-resources_js_utils_toast_css-resources_js_Compo-a37b65.js": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Edit_vue-resources_js_utils_toast_css-resources_js_Compo-a37b65.js", - "assets/resources_js_Pages_Submitter_Dataset_Index_vue.css": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Index_vue.css", - "assets/resources_js_Pages_Submitter_Dataset_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Index_vue.js", - "assets/resources_js_Pages_Submitter_Dataset_Release_vue.js": "http://localhost:8080/assets/resources_js_Pages_Submitter_Dataset_Release_vue.js", - "assets/resources_js_Pages_Submitter_Person_Index_vue.js": "http://localhost:8080/assets/resources_js_Pages_Submitter_Person_Index_vue.js", - "assets/resources_js_Pages_register-view_register-view-component_vue.js": "http://localhost:8080/assets/resources_js_Pages_register-view_register-view-component_vue.js", - "assets/vendors-node_modules_mdi_js_mdi_js-node_modules_vue-loader_dist_exportHelper_js.js": "http://localhost:8080/assets/vendors-node_modules_mdi_js_mdi_js-node_modules_vue-loader_dist_exportHelper_js.js", - "assets/vendors-node_modules_focus-trap_dist_focus-trap_esm_js-node_modules_notiwind_dist_index_esm_js.js": "http://localhost:8080/assets/vendors-node_modules_focus-trap_dist_focus-trap_esm_js-node_modules_notiwind_dist_index_esm_js.js", - "assets/vendors-node_modules_vue-facing-decorator_dist_esm_utils_js.js": "http://localhost:8080/assets/vendors-node_modules_vue-facing-decorator_dist_esm_utils_js.js", - "assets/vendors-node_modules_toastify-js_src_toastify_js.js": "http://localhost:8080/assets/vendors-node_modules_toastify-js_src_toastify_js.js", - "assets/vendors-node_modules_leaflet_dist_leaflet-src_js-node_modules_leaflet_src_control_Control_Att-adabdc.js": "http://localhost:8080/assets/vendors-node_modules_leaflet_dist_leaflet-src_js-node_modules_leaflet_src_control_Control_Att-adabdc.js", - "assets/vendors-node_modules_buffer_index_js-node_modules_vuedraggable_dist_vuedraggable_umd_js.js": "http://localhost:8080/assets/vendors-node_modules_buffer_index_js-node_modules_vuedraggable_dist_vuedraggable_umd_js.js", - "assets/vendors-node_modules_mime_dist_src_index_js.js": "http://localhost:8080/assets/vendors-node_modules_mime_dist_src_index_js.js", - "assets/vendors-node_modules_numeral_numeral_js-node_modules_chart_js_dist_chart_js.js": "http://localhost:8080/assets/vendors-node_modules_numeral_numeral_js-node_modules_chart_js_dist_chart_js.js", - "assets/resources_js_Components_BaseButton_vue.js": "http://localhost:8080/assets/resources_js_Components_BaseButton_vue.js", - "assets/resources_js_Stores_main_ts-resources_js_Components_BaseDivider_vue-resources_js_Components_C-b45805.js": "http://localhost:8080/assets/resources_js_Stores_main_ts-resources_js_Components_BaseDivider_vue-resources_js_Components_C-b45805.js", - "assets/resources_js_Layouts_LayoutAuthenticated_vue.css": "http://localhost:8080/assets/resources_js_Layouts_LayoutAuthenticated_vue.css", - "assets/resources_js_Layouts_LayoutAuthenticated_vue.js": "http://localhost:8080/assets/resources_js_Layouts_LayoutAuthenticated_vue.js", - "assets/resources_js_Components_BaseButtons_vue-resources_js_Components_FormControl_vue-resources_js_-d830d6.js": "http://localhost:8080/assets/resources_js_Components_BaseButtons_vue-resources_js_Components_FormControl_vue-resources_js_-d830d6.js", - "assets/resources_js_Components_Admin_Pagination_vue-resources_js_Components_BaseButtons_vue-resource-6f3a70.js": "http://localhost:8080/assets/resources_js_Components_Admin_Pagination_vue-resources_js_Components_BaseButtons_vue-resource-6f3a70.js", - "assets/resources_js_utils_toast_ts-resources_js_Components_NotificationBar_vue.js": "http://localhost:8080/assets/resources_js_utils_toast_ts-resources_js_Components_NotificationBar_vue.js", - "assets/resources_js_Components_Map_draw_component_vue-resources_js_Components_Map_zoom_component_vue-058bcc.js": "http://localhost:8080/assets/resources_js_Components_Map_draw_component_vue-resources_js_Components_Map_zoom_component_vue-058bcc.js", - "assets/resources_js_Components_SectionMain_vue-resources_js_Components_SectionTitleLineWithButton_vu-764dfe.js": "http://localhost:8080/assets/resources_js_Components_SectionMain_vue-resources_js_Components_SectionTitleLineWithButton_vu-764dfe.js", - "assets/resources_js_Components_BaseButtons_vue-resources_js_Components_NotificationBar_vue-resources-7e06d8.js": "http://localhost:8080/assets/resources_js_Components_BaseButtons_vue-resources_js_Components_NotificationBar_vue-resources-7e06d8.js", - "assets/resources_js_Components_Admin_Sort_vue-resources_js_Components_SectionTitleLineWithButton_vue.js": "http://localhost:8080/assets/resources_js_Components_Admin_Sort_vue-resources_js_Components_SectionTitleLineWithButton_vue.js", - "assets/resources_js_Components_CardBoxModal_vue.js": "http://localhost:8080/assets/resources_js_Components_CardBoxModal_vue.js", - "assets/resources_js_Components_FileUpload_vue-resources_js_Components_FormCheckRadioGroup_vue-resour-25e686.js": "http://localhost:8080/assets/resources_js_Components_FileUpload_vue-resources_js_Components_FormCheckRadioGroup_vue-resour-25e686.js", - "assets/fonts/inter-latin-ext-400-normal.woff": "http://localhost:8080/assets/fonts/inter-latin-ext-400-normal.1c20f7dc.woff", - "assets/fonts/inter-latin-400-normal.woff": "http://localhost:8080/assets/fonts/inter-latin-400-normal.b0c8fe9d.woff", - "assets/fonts/inter-latin-ext-400-normal.woff2": "http://localhost:8080/assets/fonts/inter-latin-ext-400-normal.3d10c85f.woff2", - "assets/fonts/inter-latin-400-normal.woff2": "http://localhost:8080/assets/fonts/inter-latin-400-normal.9698cc7d.woff2", - "assets/fonts/archivo-black-latin-400-normal.woff2": "http://localhost:8080/assets/fonts/archivo-black-latin-400-normal.fc847a1f.woff2", - "assets/fonts/archivo-black-latin-ext-400-normal.woff2": "http://localhost:8080/assets/fonts/archivo-black-latin-ext-400-normal.21761451.woff2", - "assets/fonts/inter-cyrillic-ext-400-normal.woff": "http://localhost:8080/assets/fonts/inter-cyrillic-ext-400-normal.e8945162.woff", - "assets/fonts/archivo-black-latin-400-normal.woff": "http://localhost:8080/assets/fonts/archivo-black-latin-400-normal.58a301a6.woff", - "assets/fonts/inter-cyrillic-ext-400-normal.woff2": "http://localhost:8080/assets/fonts/inter-cyrillic-ext-400-normal.fd1478dc.woff2", - "assets/fonts/inter-cyrillic-400-normal.woff": "http://localhost:8080/assets/fonts/inter-cyrillic-400-normal.e2841352.woff", - "assets/fonts/inter-greek-400-normal.woff": "http://localhost:8080/assets/fonts/inter-greek-400-normal.a42da273.woff", - "assets/fonts/archivo-black-latin-ext-400-normal.woff": "http://localhost:8080/assets/fonts/archivo-black-latin-ext-400-normal.5ab5ba92.woff", - "assets/fonts/inter-greek-400-normal.woff2": "http://localhost:8080/assets/fonts/inter-greek-400-normal.a8de720a.woff2", - "assets/fonts/inter-cyrillic-400-normal.woff2": "http://localhost:8080/assets/fonts/inter-cyrillic-400-normal.cb04b2ee.woff2", - "assets/fonts/inter-greek-ext-400-normal.woff": "http://localhost:8080/assets/fonts/inter-greek-ext-400-normal.b9e1e894.woff", - "assets/fonts/inter-vietnamese-400-normal.woff": "http://localhost:8080/assets/fonts/inter-vietnamese-400-normal.96f8adc7.woff", - "assets/fonts/inter-greek-ext-400-normal.woff2": "http://localhost:8080/assets/fonts/inter-greek-ext-400-normal.f2fa0d9e.woff2", - "assets/fonts/inter-vietnamese-400-normal.woff2": "http://localhost:8080/assets/fonts/inter-vietnamese-400-normal.44c9df13.woff2", - "assets/images/marker-icon.png": "http://localhost:8080/assets/images/marker-icon.2b3e1faf.png", - "assets/images/layers-2x.png": "http://localhost:8080/assets/images/layers-2x.8f2c4d11.png", - "assets/images/layers.png": "http://localhost:8080/assets/images/layers.416d9136.png", - "assets/images/Close.svg": "http://localhost:8080/assets/images/Close.e4887675.svg", - "assets/vendors-node_modules_vue-facing-decorator_dist_esm_index_js-node_modules_vue-facing-decorator-818045.js": "http://localhost:8080/assets/vendors-node_modules_vue-facing-decorator_dist_esm_index_js-node_modules_vue-facing-decorator-818045.js" -} \ No newline at end of file diff --git a/public/favicon-16x16.png b/public/favicon-16x16.png new file mode 100644 index 0000000..c614e7d Binary files /dev/null and b/public/favicon-16x16.png differ diff --git a/public/favicon-32x32.png b/public/favicon-32x32.png new file mode 100644 index 0000000..6886e7b Binary files /dev/null and b/public/favicon-32x32.png differ diff --git a/public/favicon-32x32.png:Zone.Identifier b/public/favicon-32x32.png:Zone.Identifier new file mode 100644 index 0000000..8bcdfdc --- /dev/null +++ b/public/favicon-32x32.png:Zone.Identifier @@ -0,0 +1,3 @@ +[ZoneTransfer] +ZoneId=3 +HostUrl=https://sea1.geoinformation.dev/favicon-32x32.png diff --git a/public/favicon.ico b/public/favicon.ico deleted file mode 100644 index ce3fc0a..0000000 Binary files a/public/favicon.ico and /dev/null differ diff --git a/public/favicon.png b/public/favicon.png deleted file mode 100644 index d3d3eb5..0000000 Binary files a/public/favicon.png and /dev/null differ diff --git a/public/favicon.svg b/public/favicon.svg new file mode 100644 index 0000000..eb81bca --- /dev/null +++ b/public/favicon.svg @@ -0,0 +1,9 @@ +<svg width="1435" height="1597" viewBox="0 0 1435 1597" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> +<rect width="1435" height="1597" fill="url(#pattern0_321_38)"/> +<defs> +<pattern id="pattern0_321_38" patternContentUnits="objectBoundingBox" width="1" height="1"> +<use xlink:href="#image0_321_38" transform="matrix(0.000696864 0 0 0.000626174 0 0.0225423)"/> +</pattern> +<image id="image0_321_38" width="4096" height="1684" preserveAspectRatio="none" xlink:href=""/> +</defs> +</svg> diff --git a/public/site.webmanifest b/public/site.webmanifest new file mode 100644 index 0000000..45dc8a2 --- /dev/null +++ b/public/site.webmanifest @@ -0,0 +1 @@ +{"name":"","short_name":"","icons":[{"src":"/android-chrome-192x192.png","sizes":"192x192","type":"image/png"},{"src":"/android-chrome-512x512.png","sizes":"512x512","type":"image/png"}],"theme_color":"#ffffff","background_color":"#ffffff","display":"standalone"} \ No newline at end of file diff --git a/resources/css/app.css b/resources/css/app.css index 1347a65..0301a3a 100644 --- a/resources/css/app.css +++ b/resources/css/app.css @@ -1,19 +1,20 @@ /* @import url('https://fonts.googleapis.com/css2?family=Poppins:wght@400;500&display=swap'); */ /* @import url('https://fonts.googleapis.com/css?family=Roboto:400,400i,600,700'); */ -@tailwind base; -@tailwind components; -@tailwind utilities; - -@import '_checkbox-radio-switch.css'; +/* @import '_checkbox-radio-switch.css'; */ @import '_progress.css'; @import '_scrollbars.css'; @import '_table.css'; -@import '~leaflet/dist/leaflet.css'; +/* @import '~leaflet/dist/leaflet.css'; */ +@import '~/leaflet/dist/leaflet.css'; @import '@fontsource/inter/index.css'; @import '@fontsource/archivo-black/index.css'; + +@tailwind base; +@tailwind components; +@tailwind utilities; :root { --color-main-background: #ffffff; --color-main-background-rgb: 255,255,255; @@ -108,6 +109,9 @@ --radius: 15; --pi: 3.14159265358979; } +.leaflet-container .leaflet-pane { + z-index: 30!important; +} /* @layer base { html, diff --git a/resources/js/Components/.FormField.vue.swo b/resources/js/Components/.FormField.vue.swo new file mode 100644 index 0000000..b00c9b9 Binary files /dev/null and b/resources/js/Components/.FormField.vue.swo differ diff --git a/resources/js/Components/AsideMenuItem.vue b/resources/js/Components/AsideMenuItem.vue index 4e52d40..52e4bbc 100644 --- a/resources/js/Components/AsideMenuItem.vue +++ b/resources/js/Components/AsideMenuItem.vue @@ -1,162 +1,143 @@ <script lang="ts" setup> -import { computed, ComputedRef } from 'vue'; +import { computed } from 'vue'; import { Link, usePage } from '@inertiajs/vue3'; -// import { Link } from '@inertiajs/inertia-vue3'; - import { StyleService } from '@/Stores/style.service'; import { mdiMinus, mdiPlus } from '@mdi/js'; import { getButtonColor } from '@/colors'; import BaseIcon from '@/Components/BaseIcon.vue'; -// import AsideMenuList from '@/Components/AsideMenuList.vue'; import { stardust } from '@eidellev/adonis-stardust/client'; import type { User } from '@/Dataset'; +import { MenuItem } from '@headlessui/vue'; -const props = defineProps({ - item: { - type: Object, - required: true, - }, - parentItem: { - type: Object, - required: false, - }, - // isDropdownList: Boolean, -}); +interface MenuItem { + href?: string; + route?: string; + icon?: string; + label: string; + target?: string; + color?: string; + children?: MenuItem[]; + isOpen?: boolean; + roles?: string[]; +} -const user: ComputedRef<User> = computed(() => { - return usePage().props.authUser as User; +const props = defineProps<{ + item: MenuItem; + parentItem?: MenuItem; + // isDropdownList?: boolean; +}>(); +const emit = defineEmits<{ + (e: 'menu-click', event: Event, item: MenuItem): void; +}>(); + +// Retrieve authenticated user from page props +const user = computed<User>(() => usePage().props.authUser as User); + +// Check if the menu item has children +const hasChildren = computed(() => { + return Array.isArray(props.item?.children) && props.item.children.length > 0; }); const itemRoute = computed(() => (props.item && props.item.route ? stardust.route(props.item.route) : '')); // const isCurrentRoute = computed(() => (props.item && props.item.route ? stardust.isCurrent(props.item.route): false)); -// const itemHref = computed(() => (props.item && props.item.href ? props.item.href : '')); -const emit = defineEmits(['menu-click']); +// Determine which element to render based on 'href' or 'route' +const isComponent = computed(() => { + if (props.item.href) { + return 'a'; + } + if (props.item.route) { + return Link; + } + return 'div'; +}); + +// Check if any child route is active +const isChildActive = computed(() => { + if (props.item.children && props.item.children.length > 0) { + return props.item.children.some(child => child.route && stardust.isCurrent(child.route)); + } + return false; +}); + +// Automatically use prop item.isOpen if set from the parent, +// or if one of its children is active then force open state. +const isOpen = computed(() => { + return props.item.isOpen || isChildActive.value; +}); + const styleService = StyleService(); const hasColor = computed(() => props.item && props.item.color); -// const isDropdownOpen = ref(false); -// const isChildSelected = computed(() => { -// if (props.item.children && props.item.children.length > 0) { -// return children.value.some(childItem => stardust.isCurrent(childItem.route)); -// } -// return false; + +// const children = computed(() => { +// return props.item.children || []; // }); - -const hasChildren = computed(() => { - // props.item.children?.length > 0 - if (props.item.children && props.item.children.length > 0) { - return true; - } - return false; -}); -const children = computed(() => { - return props.item.children || []; -}); - const componentClass = computed(() => [ hasChildren ? 'py-3 px-6 text-sm font-semibold' : 'py-3 px-6', hasColor.value ? getButtonColor(props.item.color, false, true) : styleService.asideMenuItemStyle, ]); - -// const toggleDropdown = () => { -// // emit('menu-click', event, props.item); -// // console.log(props.item); -// if (hasChildren.value) { -// isDropdownOpen.value = !isDropdownOpen.value; -// } -// // if (props.parentItem?.hasDropdown.value) { -// // props.parentItem.isDropdownActive.value = true; -// // } -// }; - -const menuClick = (event) => { +const menuClick = (event: Event) => { emit('menu-click', event, props.item); - if (hasChildren.value) { - // if (isChildSelected.value == false) { - // isDropdownOpen.value = !isDropdownOpen.value; - props.item.isOpen = !props.item.isOpen; - // } + // Toggle open state if the menu has children + props.item.isOpen = !props.item.isOpen; } - }; -// const handleChildSelected = () => { -// isChildSelected.value = true; -// }; - - -const activeInactiveStyle = computed(() => { +const activeStyle = computed(() => { if (props.item.route && stardust.isCurrent(props.item.route)) { // console.log(props.item.route); - return styleService.asideMenuItemActiveStyle; + return 'text-sky-600 font-bold'; } else { return null; } }); -const is = computed(() => { - if (props.item.href) { - return 'a'; - } - if (props.item.route) { - return Link; - } - - return 'div'; -}); - const hasRoles = computed(() => { if (props.item.roles) { - return user.value.roles.some(role => props.item.roles.includes(role.name)); + return user.value.roles.some(role => props.item.roles?.includes(role.name)); // return test; } return true }); -// props.routeName && stardust.isCurrent(props.routeName) ? props.activeColor : null </script> <!-- :target="props.item.target ?? null" --> <template> <li v-if="hasRoles"> - <!-- <component :is="itemHref ? 'div' : Link" :href="itemHref ? itemHref : itemRoute" --> - <component :is="is" :href="itemRoute ? stardust.route(props.item.route) : props.item.href" - class="flex cursor-pointer dark:text-slate-300 dark:hover:text-white menu-item-wrapper" :class="componentClass" - @click="menuClick" v-bind:target="props.item.target ?? null"> - <BaseIcon v-if="item.icon" :path="item.icon" class="flex-none menu-item-icon" :class="activeInactiveStyle" - w="w-16" :size="18" /> + <component :is="isComponent" :href="props.item.href ? props.item.href : itemRoute" + class="flex cursor-pointer dark:text-slate-300 dark:hover:text-white menu-item-wrapper" + :class="componentClass" @click="menuClick" :target="props.item.target || null"> + <BaseIcon v-if="props.item.icon" :path="props.item.icon" class="flex-none menu-item-icon" + :class="activeStyle" w="w-16" :size="18" /> <div class="menu-item-label"> - <span class="grow text-ellipsis line-clamp-1" :class="activeInactiveStyle"> - {{ item.label }} + <span class="grow text-ellipsis line-clamp-1" :class="[activeStyle]"> + {{ props.item.label }} </span> </div> - <!-- plus icon for expanding sub menu --> - <BaseIcon v-if="hasChildren" :path="props.item.isOpen ? mdiMinus : mdiPlus" class="flex-none" - :class="[activeInactiveStyle]" w="w-12" /> + <!-- Display plus or minus icon if there are child items --> + <BaseIcon v-if="hasChildren" :path="isOpen ? mdiMinus : mdiPlus" class="flex-none" + :class="[activeStyle]" w="w-12" /> </component> + <!-- Render dropdown --> <div class="menu-item-dropdown" - :class="[styleService.asideMenuDropdownStyle, props.item.isOpen ? 'block dark:bg-slate-800/50' : 'hidden']" - v-if="hasChildren"> + :class="[styleService.asideMenuDropdownStyle, isOpen ? 'block dark:bg-slate-800/50' : 'hidden']" + v-if="props.item.children && props.item.children.length > 0"> <ul> - <!-- <li v-for="( child, index ) in children " :key="index"> - <AsideMenuItem :item="child" :key="index"> </AsideMenuItem> - </li> --> - <AsideMenuItem v-for="(childItem, index) in children" :key="index" :item="childItem" /> + + <AsideMenuItem v-for="(childItem, index) in (props.item.children as any[])" :key="index" :item="childItem" + @menu-click="$emit('menu-click', $event, childItem)" /> </ul> - </div> - <!-- <AsideMenuList v-if="hasChildren" :items="item.children" - :class="[styleService.asideMenuDropdownStyle, isDropdownOpen ? 'block dark:bg-slate-800/50' : 'hidden']" /> --> - - - </li> + </div> + </li> </template> <style> @@ -167,17 +148,12 @@ const hasRoles = computed(() => { } .menu-item-icon { - font-size: 2.5rem; - /* margin-right: 10px; */ + font-size: 2.5rem; + /* margin-right: 10px; */ } -/* .menu-item-label { - font-size: 1.2rem; - font-weight: bold; -} */ - .menu-item-dropdown { - /* margin-left: 10px; */ - padding-left: 0.75rem; + /* margin-left: 10px; */ + padding-left: 0.75rem; } </style> diff --git a/resources/js/Components/AsideMenuLayer.vue b/resources/js/Components/AsideMenuLayer.vue index 7617136..6e6c5b0 100644 --- a/resources/js/Components/AsideMenuLayer.vue +++ b/resources/js/Components/AsideMenuLayer.vue @@ -36,13 +36,24 @@ const logoutItemClick = async () => { await router.post(stardust.route('logout')); }; -const menuClick = (event, item) => { +interface MenuItem { + name: string; + label: string; + icon: string; + color: string; + link: string; +} + +const menuClick = (event: Event, item: MenuItem) => { emit('menu-click', event, item); }; </script> <template> - <aside id="aside" class="lg:py-2 lg:pl-2 w-60 fixed flex z-40 top-0 h-screen transition-position overflow-hidden"> + <aside + id="aside" + class="lg:pb-2 lg:pl-2 w-60 fixed flex z-40 top-0 lg:top-16 h-screen lg:h-[calc(100vh-64px)] transition-position overflow-hidden" + > <div :class="styleStore.asideStyle" class="lg:rounded-xl flex-1 flex flex-col overflow-hidden dark:bg-slate-900"> <div :class="styleStore.asideBrandStyle" class="flex flex-row h-14 items-center justify-between dark:bg-slate-900"> <div class="text-center flex-1 lg:text-left lg:pl-6 xl:text-center xl:pl-0"> diff --git a/resources/js/Components/CardBox.vue b/resources/js/Components/CardBox.vue index e00ec72..c2d200e 100644 --- a/resources/js/Components/CardBox.vue +++ b/resources/js/Components/CardBox.vue @@ -12,6 +12,10 @@ const props = defineProps({ type: String, default: null, }, + showHeaderIcon: { + type: Boolean, + default: true, + }, headerIcon: { type: String, default: null, @@ -63,7 +67,7 @@ const submit = (e) => { <BaseIcon v-if="icon" :path="icon" class="mr-3" /> {{ title }} </div> - <button class="flex items-center py-3 px-4 justify-center ring-blue-700 focus:ring" @click="headerIconClick"> + <button v-if="showHeaderIcon" class="flex items-center py-3 px-4 justify-center ring-blue-700 focus:ring" @click="headerIconClick"> <BaseIcon :path="computedHeaderIcon" /> </button> </header> diff --git a/resources/js/Components/CardBoxClient.vue b/resources/js/Components/CardBoxClient.vue index bf17b32..cd9bf57 100644 --- a/resources/js/Components/CardBoxClient.vue +++ b/resources/js/Components/CardBoxClient.vue @@ -1,6 +1,6 @@ -<script setup> +<script lang="ts" setup> import { computed } from 'vue'; -import { mdiTrendingDown, mdiTrendingUp, mdiTrendingNeutral } from '@mdi/js'; +// import { mdiTrendingDown, mdiTrendingUp, mdiTrendingNeutral } from '@mdi/js'; import CardBox from '@/Components/CardBox.vue'; import BaseLevel from '@/Components/BaseLevel.vue'; import PillTag from '@/Components/PillTag.vue'; @@ -27,6 +27,10 @@ const props = defineProps({ type: Number, default: 0, }, + count: { + type: Number, + default: 0, + }, text: { type: String, default: null, @@ -35,6 +39,10 @@ const props = defineProps({ type: String, default: null, }, + allowEmailContact: { + type: Boolean, + default: false, + } }); const pillType = computed(() => { @@ -42,11 +50,11 @@ const pillType = computed(() => { return props.type; } - if (props.progress) { - if (props.progress >= 60) { + if (props.count) { + if (props.count >= 20) { return 'success'; } - if (props.progress >= 40) { + if (props.count >= 5) { return 'warning'; } @@ -56,17 +64,17 @@ const pillType = computed(() => { return 'info'; }); -const pillIcon = computed(() => { - return { - success: mdiTrendingUp, - warning: mdiTrendingNeutral, - danger: mdiTrendingDown, - info: mdiTrendingNeutral, - }[pillType.value]; -}); +// const pillIcon = computed(() => { +// return { +// success: mdiTrendingUp, +// warning: mdiTrendingNeutral, +// danger: mdiTrendingDown, +// info: mdiTrendingNeutral, +// }[pillType.value]; +// }); -const pillText = computed(() => props.text ?? `${props.progress}%`); -</script> +// const pillText = computed(() => props.text ?? `${props.progress}%`); +// </script> <template> <CardBox class="mb-6 last:mb-0" hoverable> @@ -77,13 +85,22 @@ const pillText = computed(() => props.text ?? `${props.progress}%`); <h4 class="text-xl text-ellipsis"> {{ name }} </h4> - <p class="text-gray-500 dark:text-slate-400"> - <!-- {{ date }} @ {{ login }} --> - {{ email }} + <p class="text-gray-500 dark:text-slate-400"> + <div v-if="props.allowEmailContact"> {{ email }}</div> </p> </div> </BaseLevel> - <PillTag :type="pillType" :text="pillText" small :icon="pillIcon" /> + <!-- <PillTag :type="pillType" :text="text" small :icon="pillIcon" /> --> + + <div class="text-center md:text-right space-y-2"> + <p class="text-sm text-gray-500"> + Count + </p> + <div> + <PillTag :type="pillType" :text="String(count)" small /> + </div> + </div> + </BaseLevel> </CardBox> </template> diff --git a/resources/js/Components/CardBoxDataset.vue b/resources/js/Components/CardBoxDataset.vue new file mode 100644 index 0000000..72d3f4d --- /dev/null +++ b/resources/js/Components/CardBoxDataset.vue @@ -0,0 +1,107 @@ +<script lang="ts" setup> +import { computed, PropType } from 'vue'; +import { mdiChartTimelineVariant, mdiFileDocumentOutline, mdiFileOutline, mdiDatabase } from '@mdi/js'; +import CardBox from '@/Components/CardBox.vue'; +import PillTag from '@/Components/PillTag.vue'; +import IconRounded from '@/Components/IconRounded.vue'; +import dayjs from 'dayjs'; +import relativeTime from 'dayjs/plugin/relativeTime'; + +// Extend dayjs to support relative times +dayjs.extend(relativeTime); + +interface Dataset { + account_id: number; + created_at: string; + creating_corporation: string; + editor_id: number; + embargo_date: string | null; + id: number; + language: string; + main_abstract: string | null; + main_title: string | null; + preferred_reviewer: string | null; + preferred_reviewer_email: string | null; + project_id: number | null; + publish_id: number; + publisher_name: string; + reject_editor_note: string | null; + reject_reviewer_note: string | null; + remaining_time: number; + reviewer_id: number; + server_date_modified: string; + server_date_published: string; + server_state: string; + type: string; + doi_identifier: string; +} + +const props = defineProps({ + dataset: { + type: Object as PropType<Dataset>, + required: true + } +}); + +const icon = computed(() => { + switch (props.dataset.type) { + case 'analysisdata': + return { icon: mdiChartTimelineVariant, type: 'success' }; + case 'measurementdata': + return { icon: mdiFileDocumentOutline, type: 'warning' }; + case 'monitoring': + return { icon: mdiFileOutline, type: 'info' }; + case 'remotesensing': + return { icon: mdiDatabase, type: 'primary' }; + case 'gis': + return { icon: mdiDatabase, type: 'info' }; + case 'models': + return { icon: mdiChartTimelineVariant, type: 'success' }; + case 'mixedtype': + return { icon: mdiFileDocumentOutline, type: 'warning' }; + case 'vocabulary': + return { icon: mdiFileOutline, type: 'info' }; + default: + return { icon: mdiDatabase, type: 'secondary' }; + } +}); + +const displayTitle = computed(() => props.dataset.main_title || 'Untitled Dataset'); + +const doiLink = computed(() => { + return `https://doi.tethys.at/10.24341/tethys.${props.dataset.publish_id}`; +}); + +const relativeDate = computed(() => { + const publishedDate = dayjs(props.dataset.server_date_published); + if (publishedDate.isValid()) { + return publishedDate.fromNow(); + } + return props.dataset.server_date_published; +}); + +// const displayBusiness = computed(() => props.dataset.publisher_name); +</script> + +<template> + <CardBox class="mb-6 last:mb-0" hoverable> + <div class="flex items-start justify-between"> + <IconRounded :icon="icon.icon" :type="icon.type" class="mr-6" /> + <div class="flex-grow space-y-1 text-left" style="width: 70%;"> + <h4 class="text-lg truncate" > + {{ displayTitle }} + </h4> + <p class="text-gray-500 dark:text-slate-400"> + <b> + <a :href="doiLink" target="_blank">View Publication</a> + </b> + • {{ relativeDate }} + </p> + </div> + <div class="flex flex-col items-end gap-2"> + <p class="text-sm text-gray-500">{{ props.dataset.type }}</p> + <PillTag :type="icon.type" :text="props.dataset.type" small class="inline-flex" /> + </div> + </div> + </CardBox> +</template> diff --git a/resources/js/Components/CardBoxModal.vue b/resources/js/Components/CardBoxModal.vue index 6e7f9aa..b0fdf01 100644 --- a/resources/js/Components/CardBoxModal.vue +++ b/resources/js/Components/CardBoxModal.vue @@ -61,10 +61,10 @@ const cancel = () => confirmCancel('cancel'); <CardBox v-show="value" :title="title" - class="shadow-lg max-h-modal w-11/12 md:w-3/5 lg:w-2/5 xl:w-4/12 z-50" + class="p-4 shadow-lg max-h-modal w-11/12 md:w-3/5 lg:w-2/5 xl:w-4/12 z-50" :header-icon="mdiClose" modal - @header-icon-click="cancel" + @header-icon-click="cancel" > <div class="space-y-3"> <h1 v-if="largeTitle" class="text-2xl"> diff --git a/resources/js/Components/CardBoxSimple.vue b/resources/js/Components/CardBoxSimple.vue new file mode 100644 index 0000000..ef6a624 --- /dev/null +++ b/resources/js/Components/CardBoxSimple.vue @@ -0,0 +1,75 @@ +<script setup lang="ts"> + +import { computed, useSlots } from 'vue'; + + +const props = defineProps({ + title: { + type: String, + default: null, + }, + icon: { + type: String, + default: null, + }, + showHeaderIcon: { + type: Boolean, + default: true, + }, + headerIcon: { + type: String, + default: null, + }, + rounded: { + type: String, + default: 'rounded-xl', + }, + hasFormData: Boolean, + empty: Boolean, + form: Boolean, + hoverable: Boolean, + modal: Boolean, +}); + +const emit = defineEmits(['header-icon-click', 'submit']); + +const is = computed(() => (props.form ? 'form' : 'div')); + +const slots = useSlots(); + +// const footer = computed(() => slots.footer && !!slots.footer()); + +const componentClass = computed(() => { + const base = [props.rounded, props.modal ? 'dark:bg-slate-900' : 'dark:bg-slate-900/70']; + + if (props.hoverable) { + base.push('hover:shadow-lg transition-shadow duration-500'); + } + + return base; +}); + + + +// const headerIconClick = () => { +// emit('header-icon-click'); +// }; + +// const submit = (e) => { +// emit('submit', e); +// }; +</script> + +<template> + <component :is="is" :class="componentClass" class="bg-white flex flex-col border border-gray-100 dark:border-slate-800 mb-4"> + + <div v-if="empty" class="text-center py-24 text-gray-500 dark:text-slate-400"> + <p>Nothing's here…</p> + </div> + + <div v-else class="flex-1" :class="[!hasFormData && 'p-6']"> + <slot /> + </div> + + </component> +</template> diff --git a/resources/js/Components/CardBoxWidget.vue b/resources/js/Components/CardBoxWidget.vue index 58b3e3c..fe10723 100644 --- a/resources/js/Components/CardBoxWidget.vue +++ b/resources/js/Components/CardBoxWidget.vue @@ -1,4 +1,4 @@ -<script setup> +<script lang="ts" setup> import { mdiCog } from '@mdi/js'; import CardBox from '@/Components/CardBox.vue'; import NumberDynamic from '@/Components/NumberDynamic.vue'; @@ -49,6 +49,9 @@ defineProps({ <PillTagTrend :trend="trend" :trend-type="trendType" small /> <BaseButton :icon="mdiCog" icon-w="w-4" icon-h="h-4" color="white" small /> </BaseLevel> + <BaseLevel v-else class="mb-3" mobile> + <BaseIcon v-if="icon" :path="icon" size="48" w="w-4" h="h-4" :class="color" /> + </BaseLevel> <BaseLevel mobile> <div> <h3 class="text-lg leading-tight text-gray-500 dark:text-slate-400"> diff --git a/resources/js/Components/FileUpload.vue b/resources/js/Components/FileUpload.vue index 3b4ba35..c5ee2ce 100644 --- a/resources/js/Components/FileUpload.vue +++ b/resources/js/Components/FileUpload.vue @@ -17,6 +17,15 @@ <p class="text-lg text-blue-700">Drop files to upload</p> </div> + <!-- Loading Spinner when processing big files --> + <div v-if="isLoading" class="absolute inset-0 z-60 flex items-center justify-center bg-gray-500 bg-opacity-50"> + <svg class="animate-spin h-12 w-12 text-white" xmlns="http://www.w3.org/2000/svg" fill="none" + viewBox="0 0 24 24"> + <circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle> + <path class="opacity-75" fill="currentColor" d="M12 2a10 10 0 0110 10h-4a6 6 0 00-6-6V2z"></path> + </svg> + </div> + <!-- scroll area --> <div class="h-full p-8 w-full h-full flex flex-col"> <header class="flex items-center justify-center w-full"> @@ -32,9 +41,9 @@ <p class="mb-2 text-sm text-gray-500 dark:text-gray-400"> <span class="font-semibold">Click to upload</span> or drag and drop </p> - <!-- <p class="text-xs text-gray-500 dark:text-gray-400">SVG, PNG, JPG or GIF (MAX. 800x400px)</p> --> </div> - <input id="dropzone-file" type="file" class="hidden" @change="onChangeFile" multiple="true" /> + <input id="dropzone-file" type="file" class="hidden" @click="showSpinner" @change="onChangeFile" + @cancel="cancelSpinner" multiple="true" /> </label> </header> @@ -182,7 +191,7 @@ <!-- sticky footer --> <footer class="flex justify-end px-8 pb-8 pt-4"> - <button id="cancel" + <button v-if="showClearButton" id="cancel" class="ml-3 rounded-sm px-3 py-1 hover:bg-gray-300 focus:shadow-outline focus:outline-none" @click="clearAllFiles"> Clear @@ -198,7 +207,7 @@ import DeleteIcon from '@/Components/Icons/Delete.vue'; import RefreshIcon from '@/Components/Icons/Refresh.vue'; // import { Page, PageProps, Errors, ErrorBag } from '@inertiajs/inertia'; import Draggable from 'vuedraggable'; -import { Buffer } from 'buffer'; +// import { Buffer } from 'buffer'; import { TethysFile } from '@/Dataset'; // lastModified: 1691759507591 @@ -241,6 +250,8 @@ class FileUploadComponent extends Vue { @Ref('overlay') overlay: HTMLDivElement; + + public isLoading: boolean = false; private counter: number = 0; // @Prop() files: Array<TestFile>; @@ -257,6 +268,12 @@ class FileUploadComponent extends Vue { }) filesToDelete: Array<TethysFile>; + @Prop({ + type: Boolean, + default: true, + }) + showClearButton: boolean; + // // deletetFiles: Array<TethysFile> = []; get deletetFiles(): Array<TethysFile> { return this.filesToDelete; @@ -264,7 +281,7 @@ class FileUploadComponent extends Vue { set deletetFiles(values: Array<TethysFile>) { // this.modelValue = value; this.filesToDelete.length = 0; - this.filesToDelete.push(...values); + this.filesToDelete.push(...values); } get items(): Array<TethysFile | File> { @@ -342,10 +359,10 @@ class FileUploadComponent extends Vue { } // reset counter and append file to gallery when file is dropped - public dropHandler(event: DragEvent): void { event.preventDefault(); const dataTransfer = event.dataTransfer; + // let bigFileFound = false; if (dataTransfer) { for (const file of event.dataTransfer?.files) { // let fileName = String(file.name.replace(/\.[^/.]+$/, '')); @@ -353,28 +370,73 @@ class FileUploadComponent extends Vue { // if (file.type.match('image.*')) { // this.generateURL(file); // } + // if (file.size > 62914560) { // 60 MB in bytes + // bigFileFound = true; + // } this._addFile(file); } this.overlay.classList.remove('draggedover'); this.counter = 0; } + // if (bigFileFound) { + // this.isLoading = true; + // // Assume file processing delay; adjust timeout as needed or rely on async processing completion. + // setTimeout(() => { + // this.isLoading = false; + // }, 1500); + // } + } + + public showSpinner() { + // event.preventDefault(); + this.isLoading = true; + } + + public cancelSpinner() { + // const target = event.target as HTMLInputElement; + // // If no files were selected, remove spinner + // if (!target.files || target.files.length === 0) { + // this.isLoading = false; + // } + this.isLoading = false; } public onChangeFile(event: Event) { event.preventDefault(); + let target = event.target as HTMLInputElement; // let uploadedFile = event.target.files[0]; // let fileName = String(event.target.files[0].name.replace(/\.[^/.]+$/, '')); - for (const file of event.target.files) { - // let fileName = String(event.target.files[0].name.replace(/\.[^/.]+$/, '')); - // file.label = fileName; - // if (file.type.match('image.*')) { - // this.generateURL(file); - // } - this._addFile(file); + if (target && target.files) { + for (const file of event.target.files) { + // let fileName = String(event.target.files[0].name.replace(/\.[^/.]+$/, '')); + // file.label = fileName; + // if (file.type.match('image.*')) { + // this.generateURL(file); + // } + // Immediately set spinner if any file is large (over 100 MB) + // for (const file of target.files) { + // if (file.size > 62914560) { // 100 MB + // bigFileFound = true; + // break; + // } + // } + // if (bigFileFound) { + // this.isLoading = true; + // } + this._addFile(file); + + } } + // if (bigFileFound) { + // this.isLoading = true; + // setTimeout(() => { + // this.isLoading = false; + // }, 1500); + // } // this.overlay.classList.remove('draggedover'); this.counter = 0; + this.isLoading = false; } get errors(): IDictionary { @@ -396,7 +458,9 @@ class FileUploadComponent extends Vue { public clearAllFiles(event: Event) { event.preventDefault(); - this.items.splice(0); + if (this.showClearButton == true) { + this.items.splice(0); + } } public removeFile(key: number) { @@ -445,18 +509,19 @@ class FileUploadComponent extends Vue { let localUrl: string = ''; if (file instanceof File) { localUrl = URL.createObjectURL(file as Blob); - } else if (file.fileData) { - // const blob = new Blob([file.fileData]); - // localUrl = URL.createObjectURL(blob); - const parsed = JSON.parse(file.fileData); - file.fileData = ''; - // retrieve the original buffer of data - const buff = Buffer.from(parsed.blob, 'base64'); - const blob = new Blob([buff], { type: 'application/octet-stream' }); - // file.blob = blob; - localUrl = URL.createObjectURL(blob); - file.fileSrc = localUrl; } + // else if (file.fileData) { + // // const blob = new Blob([file.fileData]); + // // localUrl = URL.createObjectURL(blob); + // const parsed = JSON.parse(file.fileData); + // file.fileData = ''; + // // retrieve the original buffer of data + // const buff = Buffer.from(parsed.blob, 'base64'); + // const blob = new Blob([buff], { type: 'application/octet-stream' }); + // // file.blob = blob; + // localUrl = URL.createObjectURL(blob); + // file.fileSrc = localUrl; + // } // setTimeout(() => { // URL.revokeObjectURL(localUrl); @@ -464,17 +529,6 @@ class FileUploadComponent extends Vue { return localUrl; } - // private async downloadFile(id: number): Promise<string> { - // const response = await axios.get<Blob>(`/api/download/${id}`, { - // responseType: 'blob', - // }); - // const url = URL.createObjectURL(response.data); - // setTimeout(() => { - // URL.revokeObjectURL(url); - // }, 1000); - // return url; - // } - public getFileSize(file: File) { if (file.size > 1024) { if (file.size > 1048576) { @@ -487,17 +541,6 @@ class FileUploadComponent extends Vue { } } - // private _addFile(file) { - // // const isImage = file.type.match('image.*'); - // // const objectURL = URL.createObjectURL(file); - - // // this.files[objectURL] = file; - // // let test: TethysFile = { upload: file, label: "dfdsfs", sorting: 0 }; - // // file.sorting = this.files.length; - // file.sort_order = (this.items.length + 1), - // this.files.push(file); - // } - private _addFile(file: File) { // const reader = new FileReader(); // reader.onload = (event) => { @@ -529,14 +572,11 @@ class FileUploadComponent extends Vue { // this.items.push(test); this.items[this.items.length] = test; } else { + file.sort_order = this.items.length + 1; this.items.push(file); } } - // use to check if a file is being dragged - // private _hasFiles({ types = [] as Array<string> }) { - // return types.indexOf('Files') > -1; - // } private _hasFiles(dataTransfer: DataTransfer | null): boolean { return dataTransfer ? dataTransfer.items.length > 0 : false; } diff --git a/resources/js/Components/FooterBar.vue b/resources/js/Components/FooterBar.vue index edf8ecf..4408cf0 100644 --- a/resources/js/Components/FooterBar.vue +++ b/resources/js/Components/FooterBar.vue @@ -15,9 +15,10 @@ const year = computed(() => new Date().getFullYear()); <!-- Get more with <a href="https://tailwind-vue.justboil.me/" target="_blank" class="text-blue-600">Premium version</a> --> </div> - <div class="md:py-3"> + <div class="md:py-1"> <a href="https://www.tethys.at" target="_blank"> - <JustboilLogo class="w-auto h-8 md:h-6" /> + <!-- <JustboilLogo class="w-auto h-8 md:h-6" /> --> + <JustboilLogo class="w-auto h-12 md:h-10 dark:invert" /> </a> </div> </BaseLevel> diff --git a/resources/js/Components/FormCheckRadio.vue b/resources/js/Components/FormCheckRadio.vue index 1f991e7..5d22212 100644 --- a/resources/js/Components/FormCheckRadio.vue +++ b/resources/js/Components/FormCheckRadio.vue @@ -1,43 +1,59 @@ -<script setup> +<script setup lang="ts"> import { computed } from 'vue'; -const props = defineProps({ - name: { - type: String, - required: true, - }, - type: { - type: String, - default: 'checkbox', - validator: (value) => ['checkbox', 'radio', 'switch'].includes(value), - }, - label: { - type: String, - default: null, - }, - modelValue: { - type: [Array, String, Number, Boolean], - default: null, - }, - inputValue: { - type: [String, Number, Boolean], - required: true, - }, -}); -const emit = defineEmits(['update:modelValue']); +interface Props { + name: string; + type?: 'checkbox' | 'radio' | 'switch'; + label?: string | null; + modelValue: Array<any> | string | number | boolean | null; + inputValue: string | number | boolean; +} + +const props = defineProps<Props>(); + +const emit = defineEmits<{ (e: 'update:modelValue', value: Props['modelValue']): void }>(); + const computedValue = computed({ get: () => props.modelValue, set: (value) => { - emit('update:modelValue', value); + emit('update:modelValue', props.type === 'radio' ? [value] : value); }, }); + const inputType = computed(() => (props.type === 'radio' ? 'radio' : 'checkbox')); + +// Define isChecked for radio inputs: it's true when the current modelValue equals the inputValue +const isChecked = computed(() => { + if (Array.isArray(computedValue.value) && computedValue.value.length > 0) { + return props.type === 'radio' + ? computedValue.value[0] === props.inputValue + : computedValue.value.includes(props.inputValue); + } + return computedValue.value === props.inputValue; +}); </script> <template> - <label :class="type" class="mr-6 mb-3 last:mr-0"> - <input v-model="computedValue" :type="inputType" :name="name" :value="inputValue" /> - <span class="check" /> - <span class="pl-2">{{ label }}</span> + <label v-if="type === 'radio'" :class="[type]" + class="mr-6 mb-3 last:mr-0 inline-flex items-center cursor-pointer relative"> + <input v-model="computedValue" :type="inputType" :name="name" :value="inputValue" + class="absolute left-0 opacity-0 -z-1 focus:outline-none focus:ring-0" + :checked="isChecked" /> + <span class="check border transition-colors duration-200 dark:bg-slate-800 block w-5 h-5 rounded-full" :class="{ + 'border-gray-700': !isChecked, + 'bg-radio-checked bg-no-repeat bg-center bg-lime-600 border-lime-600 border-4': isChecked + }" /> + <span class="pl-2 control-label">{{ label }}</span> + </label> + + <label v-else-if="type === 'checkbox'" :class="[type]" + class="mr-6 mb-3 last:mr-0 inline-flex items-center cursor-pointer relative"> + <input v-model="computedValue" :type="inputType" :name="name" :value="inputValue" + class="absolute left-0 opacity-0 -z-1 focus:outline-none focus:ring-0" /> + <span class="check border transition-colors duration-200 dark:bg-slate-800 block w-5 h-5 rounded" :class="{ + 'border-gray-700': !isChecked, + 'bg-checkbox-checked bg-no-repeat bg-center bg-lime-600 border-lime-600 border-4': isChecked + }" /> + <span class="pl-2 control-label">{{ label }}</span> </label> </template> diff --git a/resources/js/Components/FormCheckRadioGroup.vue b/resources/js/Components/FormCheckRadioGroup.vue index 0ceb4aa..c995cab 100644 --- a/resources/js/Components/FormCheckRadioGroup.vue +++ b/resources/js/Components/FormCheckRadioGroup.vue @@ -1,17 +1,29 @@ <script setup lang="ts"> -import { computed } from 'vue'; +import { computed, ref, PropType } from 'vue'; import FormCheckRadio from '@/Components/FormCheckRadio.vue'; +// import BaseButton from '@/Components/BaseButton.vue'; +// import FormControl from '@/Components/FormControl.vue'; + const props = defineProps({ options: { type: Object, - default: () => {}, + default: () => { }, + }, + allowManualAdding: { + type: Boolean, + default: false, + }, + manualAddingPlaceholder: { + type: String, + default: 'Add manually', + required: false, }, name: { type: String, required: true, }, type: { - type: String, + type: String as PropType<'checkbox' | 'radio' | 'switch'>, default: 'checkbox', validator: (value: string) => ['checkbox', 'radio', 'switch'].includes(value), }, @@ -35,7 +47,7 @@ const computedValue = computed({ if (props.modelValue.every((item) => typeof item === 'number')) { return props.modelValue; } else if (props.modelValue.every((item) => hasIdAttribute(item))) { - const ids = props.modelValue.map((obj) => obj.id.toString()); + const ids = props.modelValue.map((obj) => obj.id); return ids; } return props.modelValue; @@ -55,6 +67,29 @@ const computedValue = computed({ const hasIdAttribute = (obj: any): obj is { id: any } => { return typeof obj === 'object' && 'id' in obj; }; + +const newOption = ref<string>(''); +const addOption = () => { + if (newOption.value && !props.options[newOption.value]) { + props.options[newOption.value] = newOption.value; + newOption.value = ''; + } +}; + +const inputElClass = computed(() => { + const base = [ + 'px-3 py-2 max-w-full border-gray-700 rounded w-full', + 'dark:placeholder-gray-400', + 'h-12', + 'border', + 'bg-transparent' + // props.isReadOnly ? 'bg-gray-50 dark:bg-slate-600' : 'bg-white dark:bg-slate-800', + ]; + // if (props.icon) { + // base.push('pl-10'); + // } + return base; +}); </script> <template> @@ -63,15 +98,17 @@ const hasIdAttribute = (obj: any): obj is { id: any } => { <!-- :label="value" --> <!-- :input-value="value.id" :label="value.name" --> - <FormCheckRadio - v-for="(value, key) in options" - :key="key" - v-model="computedValue" - :type="type" - :name="name" - :input-value="key" - :label="value" - :class="componentClass" - /> + <div v-if="allowManualAdding && type === 'checkbox'" class="flex items-center mt-2 mb-2 relative"> + <input v-model="newOption" :placeholder="manualAddingPlaceholder" :class="inputElClass" + @keydown.prevent.enter="addOption" /> + <svg v-show="newOption.length >= 2" @click.prevent="addOption" xmlns="http://www.w3.org/2000/svg" + class="w-6 h-6 absolute right-2 top-1/2 transform -translate-y-1/2 cursor-pointer text-gray-500" + viewBox="0 0 24 24" fill="currentColor"> + <path + d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 17h-2v-6H5v-2h6V5h2v6h6v2h-6v6z" /> + </svg> + </div> + <FormCheckRadio v-for="(value, key) in options" :key="key" v-model="computedValue" :type="type" + :name="name" :input-value="Number(key)" :label="value" :class="componentClass" /> </div> </template> diff --git a/resources/js/Components/FormControl.vue b/resources/js/Components/FormControl.vue index 32085b7..f66a2b3 100644 --- a/resources/js/Components/FormControl.vue +++ b/resources/js/Components/FormControl.vue @@ -67,15 +67,28 @@ const computedValue = computed({ emit('update:modelValue', value); }, }); + +// focus:ring focus:outline-none border-gray-700 const inputElClass = computed(() => { const base = [ - 'px-3 py-2 max-w-full focus:ring focus:outline-none border-gray-700 rounded w-full', + 'px-3 py-2 max-w-full rounded w-full', 'dark:placeholder-gray-400', props.extraHigh ? 'h-80' : (computedType.value === 'textarea' ? 'h-44' : 'h-12'), props.borderless ? 'border-0' : 'border', - // props.transparent && !props.isReadOnly ? 'bg-transparent' : 'bg-white dark:bg-slate-800', - props.isReadOnly ? 'bg-gray-50 dark:bg-slate-600' : 'bg-white dark:bg-slate-800', + // // props.transparent && !props.isReadOnly ? 'bg-transparent' : 'bg-white dark:bg-slate-800', + // props.isReadOnly ? 'bg-gray-50 dark:bg-slate-600' : 'bg-white dark:bg-slate-800', ]; + + // Apply styles based on read-only state. + if (props.isReadOnly) { + // Read-only: no focus ring, grayed-out text and border, and disabled cursor. + base.push('bg-gray-50', 'dark:bg-slate-600', 'border', 'border-gray-300', 'dark:border-slate-600', 'text-gray-500', 'cursor-not-allowed', 'focus:outline-none' ,'focus:ring-0', 'focus:border-gray-300'); + } else { + // Actionable field: focus ring, white/dark background, and darker border. + base.push('bg-white dark:bg-slate-800', 'focus:ring focus:outline-none', 'border', 'border-gray-700'); + } + + if (props.icon) { base.push('pl-10', 'pr-10'); } @@ -118,6 +131,9 @@ if (props.ctrlKFocus) { mainService.isFieldFocusRegistered = false; }); } +const focus = () => { + inputEl?.value.focus(); +}; </script> <template> @@ -130,7 +146,7 @@ if (props.ctrlKFocus) { </option> </select> <textarea v-else-if="computedType === 'textarea'" :id="id" v-model="computedValue" :class="inputElClass" - :name="name" :placeholder="placeholder" :required="required" /> + :name="name" :placeholder="placeholder" :required="required" :readonly="isReadOnly"/> <input v-else :id="id" ref="inputEl" v-model="computedValue" :name="name" :inputmode="inputmode" :autocomplete="autocomplete" :required="required" :placeholder="placeholder" :type="computedType" :class="inputElClass" :readonly="isReadOnly" /> diff --git a/resources/js/Components/FormField.vue b/resources/js/Components/FormField.vue index 9aa22f6..bcd806d 100644 --- a/resources/js/Components/FormField.vue +++ b/resources/js/Components/FormField.vue @@ -2,7 +2,7 @@ import { computed, useSlots } from 'vue'; -defineProps({ +const props = defineProps({ label: { type: String, default: null, @@ -15,6 +15,10 @@ defineProps({ type: String, default: null, }, + // class: { + // type: Object, + // default: {}, + // }, }); const slots = useSlots(); @@ -36,7 +40,7 @@ const wrapperClass = computed(() => { </script> <template> - <div class="mb-6 last:mb-0"> + <div :class="['last:mb-0', 'mb-6']"> <!-- <label v-if="label" :for="labelFor" class="block font-bold mb-2">{{ label }}</label> --> <label v-if="label" :for="labelFor" class="font-bold h-6 mt-3 text-xs leading-8 uppercase">{{ label }}</label> <div v-bind:class="wrapperClass"> diff --git a/resources/js/Components/Icons/IconSvg.vue b/resources/js/Components/Icons/IconSvg.vue new file mode 100644 index 0000000..b83e742 --- /dev/null +++ b/resources/js/Components/Icons/IconSvg.vue @@ -0,0 +1,74 @@ +<script setup lang="ts"> +import { computed } from 'vue'; +import { mdiLicense } from '@mdi/js'; + +const props = defineProps({ + path: { + type: String, + required: true + }, + size: { + type: Number, + default: 24 + }, + viewBox: { + type: String, + default: '0 0 24 24' + }, + color: { + type: String, + default: 'currentColor' + }, + className: { + type: String, + default: '' + } +}); + +// Define all the SVG paths we need +const svgPaths = { + // Document/File icons + document: 'M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z', + documentPlus: 'M9 13h6m-3-3v6m5 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z', + + // Communication icons + email: 'M2.003 5.884L10 9.882l7.997-3.998A2 2 0 0016 4H4a2 2 0 00-1.997 1.884z M18 8.118l-8 4-8-4V14a2 2 0 002 2h12a2 2 0 002-2V8.118z', + + // Identity/User icons + idCard: '10 2a1 1 0 00-1 1v1a1 1 0 002 0V3a1 1 0 00-1-1zM4 4h3a3 3 0 006 0h3a2 2 0 012 2v9a2 2 0 01-2 2H4a2 2 0 01-2-2V6a2 2 0 012-2zm2.5 7a1.5 1.5 0 100-3 1.5 1.5 0 000 3zm2.45 4a2.5 2.5 0 10-4.9 0h4.9zM12 9a1 1 0 100 2h3a1 1 0 100-2h-3zm-1 4a1 1 0 011-1h2a1 1 0 110 2h-2a1 1 0 01-1-1z', + + // Language/Translation icons + // language: 'M7 2a1 1 0 011 1v1h3a1 1 0 110 2H9.578a18.87 18.87 0 01-1.724 4.78c.29.354.596.696.914 1.026a1 1 0 11-1.44 1.389c-.188-.196-.373-.396-.554-.6a19.098 19.098 0 01-3.107 3.567 1 1 0 01-1.334-1.49 17.087 17.087 0 003.13-3.733 18.992 18.992 0 01-1.487-2.494 1 1 0 111.79-.89c.234.47.489.928.764 1.372.417-.934.752-1.913.997-2.927H3a1 1 0 110-2h3V3a1 1 0 011-1zm6 6a1 1 0 01.894.553l2.991 5.982a.869.869 0 01.02.037l.99 1.98a1 1 0 11-1.79.895L15.383 16h-4.764l-.724 1.447a1 1 0 11-1.788-.894l.99-1.98.019-.038 2.99-5.982A1 1 0 0113 8zm-1.382 6h2.764L13 11.236 11.618 14z', + language: 'M12 2a10 10 0 1 0 0 20a10 10 0 1 0 0-20zm0 0c2.5 0 4.5 4.5 4.5 10s-2 10-4.5 10-4.5-4.5-4.5-10 2-10 4.5-10zm0 0a10 10 0 0 1 0 20a10 10 0 0 1 0-20z', + // License/Legal icons + // license: 'M10 2a1 1 0 00-1 1v1.323l-3.954 1.582A1 1 0 004 6.32V16a1 1 0 001.555.832l3-1.2a1 1 0 01.8 0l3 1.2a1 1 0 001.555-.832V6.32a1 1 0 00-1.046-.894L9 4.877V3a1 1 0 00-1-1zm0 14.5a.5.5 0 01-.5-.5v-4a.5.5 0 011 0v4a.5.5 0 01-.5.5zm1.5-10.5a.5.5 0 11-1 0 .5.5 0 011 0z', + license: mdiLicense, + + // Building/Organization icons + building: 'M4 4a2 2 0 012-2h8a2 2 0 012 2v12a1 1 0 110 2h-3a1 1 0 01-1-1v-2a1 1 0 00-1-1H9a1 1 0 00-1 1v2a1 1 0 01-1 1H4a1 1 0 110-2V4zm3 1h2v2H7V5zm2 4H7v2h2V9zm2-4h2v2h-2V5zm2 4h-2v2h2V9z', + + // Book/Publication icons + book: 'M9 4.804A7.968 7.968 0 005.5 4c-1.255 0-2.443.29-3.5.804v10A7.969 7.969 0 015.5 14c1.669 0 3.218.51 4.5 1.385A7.962 7.962 0 0114.5 14c1.255 0 2.443.29 3.5.804v-10A7.968 7.968 0 0014.5 4c-1.255 0-2.443.29-3.5.804V12a1 1 0 11-2 0V4.804z', + + // Download icon + download: 'M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-4l-4 4m0 0l-4-4m4 4V4' +}; + +const pathData = computed(() => { + return svgPaths[props.path] || props.path; +}); + +const sizeStyle = computed(() => { + return { + width: `${props.size}px`, + height: `${props.size}px` + }; +}); +</script> + +<template> + <svg :style="sizeStyle" :class="className" :viewBox="viewBox" xmlns="http://www.w3.org/2000/svg" fill="none" + :stroke="color" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> + <path :d="pathData" /> + </svg> +</template> \ No newline at end of file diff --git a/resources/js/Components/JustboilLogo.vue b/resources/js/Components/JustboilLogo.vue index 21e4193..d295d63 100644 --- a/resources/js/Components/JustboilLogo.vue +++ b/resources/js/Components/JustboilLogo.vue @@ -1,145 +1,11 @@ <template> - <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" - y="0px" viewBox="0 0 626.4 224.7" style="enable-background:new 0 0 626.4 224.7;" xml:space="preserve" fill="currentColor"> - <g> - <g> - <path class="st0" - d="M215.3,69.1l-4.4,19.2h-47.1l-4.2,18h43.7l-4.5,19.2h-43.6l-5.9,25.4h47L192,170h-73.1l23.3-100.9H215.3z" /> - <path class="st0" d="M311.6,88.6h-31.4L261.4,170h-26.1L254,88.6h-31.5l4.5-19.5h89L311.6,88.6z" /> - <path class="st0" d="M422.1,69.1L398.9,170h-26.1l10.3-44.6h-38.3L334.4,170h-26.1l23.3-100.9h26.1l-8.6,37.1h38.3l8.6-37.1H422.1 - z" /> - <path class="st0" d="M536.7,69.1l-51.4,61.9l-9.1,39h-26.2l8.8-37.8l-23.2-63.1h27.9l13,39.6l30.2-39.6H536.7z" /> - <path class="st0" d="M563.8,171.8c-8.6,0-16.3-0.7-23.1-2.2c-6.8-1.5-12.6-3.3-17.5-5.6l5.2-24.2h2.8c4.7,4.2,10.1,7.5,16.4,9.9 - c6.3,2.3,13,3.5,20.3,3.5c7.4,0,12.9-1,16.3-3c3.5-2,5.2-4.9,5.2-8.6c0-1.4-0.3-2.6-0.8-3.6c-0.6-1-1.7-2-3.3-3 - c-1.6-1-3.9-2-6.7-2.9c-2.8-1-6.5-2.1-10.9-3.4c-4.9-1.4-9.3-2.8-13.2-4.4c-3.9-1.6-7.2-3.5-10-5.6c-2.8-2.2-4.8-4.8-6.3-7.7 - c-1.4-2.9-2.1-6.3-2.1-10.4c0-10.1,4.5-18.2,13.4-24.2c8.9-6,21.1-9,36.6-9c7.5,0,14.5,0.6,21.1,1.8c6.6,1.2,12.2,2.9,16.8,4.9 - l-4.9,23.2h-2.8c-3.5-3.2-8.2-5.9-13.9-8.1c-5.8-2.2-12.1-3.3-18.9-3.3c-6.7,0-11.9,1-15.5,2.9c-3.7,1.9-5.5,4.6-5.5,8 - c0,1.6,0.3,2.9,0.8,3.9c0.5,1,1.6,2.1,3.3,3c1.4,0.9,3.7,1.9,6.8,2.9c3.1,1.1,6.8,2.2,11,3.3c11.3,3,19.4,6.6,24.2,10.6 - s7.2,9.6,7.2,16.5c0,5.8-1.3,11-3.9,15.4c-2.6,4.4-6.2,8-10.8,10.8c-4.8,2.9-10.3,5.1-16.5,6.4 - C578.5,171.1,571.6,171.8,563.8,171.8z" /> - </g> - <path class="st1" d="M539.2,37.1" /> - <path class="st1" d="M539.2,37.1" /> - <polygon class="st2" points="24,213.8 57.7,213.8 98.2,44.6 12.5,45.1 3.7,81 56.7,80.6 " /> - <polygon class="st0" points="72.1,170.5 112.4,2.8 199.8,2.8 192.4,39.2 137.1,39.2 105.8,170.1 " /> - <polygon class="st0" points="207,2.6 225.9,2.6 218.6,39 199.9,39 " /> - <polygon class="st3" points="233.2,2.9 252.1,2.9 244.7,39.4 226.1,39.4 " /> - <polygon class="st4" points="259.2,2.9 278.1,2.9 270.8,39.4 252.1,39.4 " /> - <g> - <path class="st5" d="M97.4,213.7h-4.7l-8-12.1h-5.8l-2.8,12.1H72l7-30.2h8.3c1.9,0,3.4,0.1,4.5,0.4c1.2,0.2,2.2,0.7,3,1.3 - c0.8,0.6,1.4,1.2,1.8,2c0.4,0.8,0.7,1.8,0.7,2.9c0,2.5-0.8,4.7-2.4,6.6c-1.6,1.9-3.7,3.2-6.2,4L97.4,213.7z M92.9,191 - c0-0.7-0.1-1.3-0.3-1.8c-0.2-0.5-0.6-0.9-1-1.3c-0.5-0.4-1.2-0.7-1.9-0.9c-0.7-0.2-1.7-0.2-2.8-0.2h-4.6l-2.7,11.6h4.3 - c1.3,0,2.5-0.1,3.5-0.4c1-0.2,1.9-0.7,2.7-1.3c0.9-0.7,1.6-1.5,2.1-2.5C92.6,193.2,92.9,192.1,92.9,191z" /> - <path class="st5" d="M111.5,214.2c-3.2,0-5.6-0.7-7.4-2.2c-1.8-1.5-2.7-3.6-2.7-6.5c0-4.2,1.3-7.7,4-10.7c2.6-3,5.9-4.5,9.8-4.5 - c2.6,0,4.6,0.6,5.9,1.9c1.4,1.3,2.1,3.1,2.1,5.4c0,0.4-0.1,1-0.2,1.9c-0.1,0.9-0.3,1.9-0.6,3.1h-16.8c-0.1,0.4-0.1,0.8-0.2,1.2 - c0,0.4-0.1,0.7-0.1,1.1c0,1.9,0.6,3.4,1.8,4.5c1.2,1.1,2.8,1.6,5,1.6c1.5,0,3-0.3,4.6-0.9s2.9-1.2,4-2h0.2l-0.8,4.1 - c-0.7,0.2-1.3,0.5-1.8,0.7c-0.5,0.2-1.2,0.4-2.1,0.6c-0.8,0.2-1.6,0.4-2.2,0.5C113.3,214.2,112.5,214.2,111.5,214.2z M119.2,199.9 - c0.1-0.4,0.1-0.7,0.1-1c0-0.3,0-0.6,0-0.9c0-1.4-0.4-2.6-1.2-3.4c-0.8-0.8-2.1-1.2-3.8-1.2c-1.9,0-3.6,0.6-5.1,1.8 - c-1.5,1.2-2.5,2.8-3,4.7H119.2z" /> - <path class="st5" d="M132.8,214.3c-1.7,0-3.2-0.2-4.6-0.6s-2.5-0.9-3.4-1.4l0.9-4.1h0.2c0.3,0.2,0.7,0.5,1.1,0.9 - c0.5,0.3,1,0.7,1.7,1c0.6,0.3,1.4,0.6,2.2,0.8c0.8,0.2,1.7,0.3,2.6,0.3c1.9,0,3.4-0.4,4.5-1.1c1.1-0.7,1.6-1.7,1.6-3.1 - c0-0.7-0.3-1.3-0.8-1.7c-0.6-0.4-1.4-0.7-2.4-0.9c-0.5-0.1-1.2-0.3-1.9-0.4c-0.7-0.1-1.5-0.3-2.3-0.5c-1.6-0.4-2.8-1.1-3.5-1.9 - c-0.8-0.8-1.1-1.9-1.1-3.1c0-1.1,0.2-2,0.7-3c0.5-0.9,1.1-1.8,2.1-2.5c0.9-0.7,2-1.3,3.3-1.8c1.3-0.5,2.8-0.7,4.5-0.7 - c1.4,0,2.7,0.2,4.1,0.5c1.4,0.3,2.5,0.8,3.3,1.3l-0.8,3.9h-0.2c-0.2-0.2-0.5-0.4-1-0.7c-0.4-0.3-1-0.6-1.7-0.9 - c-0.6-0.3-1.3-0.5-2.1-0.7c-0.8-0.2-1.6-0.3-2.4-0.3c-1.7,0-3.1,0.4-4.1,1.1s-1.6,1.7-1.6,2.9c0,0.7,0.3,1.2,0.8,1.7 - c0.5,0.5,1.3,0.8,2.4,1.1c0.7,0.2,1.4,0.3,2.1,0.5c0.7,0.1,1.4,0.3,2.1,0.5c1.6,0.4,2.8,1,3.6,1.8s1.2,1.9,1.2,3.1 - c0,1-0.2,2.1-0.7,3.1c-0.5,1-1.2,1.9-2.2,2.6c-1,0.8-2.1,1.4-3.5,1.8C136,214,134.5,214.3,132.8,214.3z" /> - <path class="st5" d="M158.2,214.2c-3.2,0-5.6-0.7-7.4-2.2c-1.8-1.5-2.7-3.6-2.7-6.5c0-4.2,1.3-7.7,4-10.7c2.6-3,5.9-4.5,9.8-4.5 - c2.6,0,4.6,0.6,5.9,1.9c1.4,1.3,2.1,3.1,2.1,5.4c0,0.4-0.1,1-0.2,1.9c-0.1,0.9-0.3,1.9-0.6,3.1h-16.8c-0.1,0.4-0.1,0.8-0.2,1.2 - c0,0.4-0.1,0.7-0.1,1.1c0,1.9,0.6,3.4,1.8,4.5c1.2,1.1,2.8,1.6,5,1.6c1.5,0,3-0.3,4.6-0.9c1.6-0.6,2.9-1.2,4-2h0.2l-0.8,4.1 - c-0.7,0.2-1.3,0.5-1.8,0.7c-0.5,0.2-1.2,0.4-2.1,0.6c-0.8,0.2-1.6,0.4-2.2,0.5C160,214.2,159.2,214.2,158.2,214.2z M165.9,199.9 - c0.1-0.4,0.1-0.7,0.1-1c0-0.3,0-0.6,0-0.9c0-1.4-0.4-2.6-1.2-3.4c-0.8-0.8-2.1-1.2-3.8-1.2c-1.9,0-3.6,0.6-5.1,1.8 - c-1.5,1.2-2.5,2.8-3,4.7H165.9z" /> - <path class="st5" - d="M186.8,211.3c-0.4,0.2-0.9,0.5-1.5,0.9c-0.6,0.4-1.3,0.7-1.9,1c-0.7,0.3-1.5,0.6-2.3,0.8 - c-0.9,0.2-1.8,0.3-3,0.3c-1.8,0-3.3-0.5-4.4-1.6c-1.1-1-1.7-2.4-1.7-4.1c0-1.8,0.4-3.3,1.2-4.6c0.8-1.2,2-2.2,3.5-3 - c1.5-0.8,3.4-1.3,5.6-1.7c2.2-0.3,4.6-0.6,7.4-0.6c0.1-0.4,0.2-0.7,0.2-1c0.1-0.3,0.1-0.6,0.1-0.9c0-0.6-0.1-1.2-0.4-1.6 - c-0.3-0.4-0.6-0.7-1.1-1c-0.5-0.2-1-0.4-1.7-0.5c-0.6-0.1-1.3-0.1-2.1-0.1c-1.2,0-2.5,0.2-4,0.6c-1.5,0.4-2.7,0.7-3.6,1.1H177 - l0.8-3.8c0.8-0.2,1.9-0.4,3.4-0.7c1.5-0.3,2.9-0.4,4.3-0.4c2.8,0,5,0.4,6.4,1.3c1.4,0.9,2.1,2.3,2.1,4.2c0,0.4,0,0.8-0.1,1.2 - s-0.1,0.8-0.2,1.2l-3.6,15.4h-3.8L186.8,211.3z M189.1,201.8c-2.1,0.1-4,0.2-5.6,0.5c-1.6,0.2-3,0.6-4,1c-1.1,0.4-1.9,1.1-2.5,1.8 - c-0.6,0.8-0.9,1.7-0.9,2.9c0,1,0.3,1.7,1,2.2c0.7,0.5,1.7,0.8,3.1,0.8c1.2,0,2.5-0.3,3.8-0.8c1.3-0.5,2.5-1.2,3.6-2L189.1,201.8z" /> - <path class="st5" d="M215.9,195.1h-0.2c-0.5-0.1-1-0.2-1.5-0.3c-0.5-0.1-1-0.1-1.7-0.1c-1.3,0-2.5,0.3-3.8,0.9 - c-1.3,0.6-2.5,1.3-3.6,2.1l-3.7,16.1h-3.9l5.2-22.7h3.9l-0.8,3.3c1.8-1.3,3.3-2.1,4.6-2.6c1.3-0.5,2.5-0.7,3.6-0.7 - c0.7,0,1.2,0,1.4,0.1c0.3,0,0.7,0.1,1.3,0.2L215.9,195.1z" /> - <path class="st5" d="M225.2,214.2c-1.4,0-2.7-0.2-3.8-0.5c-1.1-0.3-2.1-0.9-2.9-1.6c-0.8-0.7-1.4-1.6-1.9-2.7 - c-0.4-1.1-0.7-2.3-0.7-3.7c0-2.1,0.3-4.1,1-5.9c0.7-1.8,1.6-3.5,2.8-4.9c1.2-1.4,2.6-2.4,4.4-3.2c1.7-0.8,3.6-1.2,5.6-1.2 - c1.3,0,2.6,0.2,3.8,0.5c1.2,0.4,2.2,0.8,3.1,1.3l-0.8,4.1h-0.2c-0.3-0.2-0.6-0.5-1-0.8c-0.4-0.3-0.9-0.6-1.4-0.9 - c-0.6-0.3-1.2-0.5-1.9-0.7c-0.7-0.2-1.5-0.3-2.3-0.3c-2.6,0-4.8,1.1-6.5,3.3c-1.7,2.2-2.5,4.9-2.5,8.1c0,1.9,0.5,3.4,1.5,4.4 - c1,1,2.4,1.5,4.2,1.5c0.9,0,1.7-0.1,2.6-0.4c0.9-0.2,1.6-0.5,2.2-0.8c0.7-0.3,1.3-0.6,1.9-1c0.6-0.3,1-0.6,1.2-0.8h0.2l-0.8,4.2 - c-1.2,0.5-2.4,1-3.8,1.4C227.9,214,226.5,214.2,225.2,214.2z" /> - <path class="st5" d="M259.9,196.2c0,0.3,0,0.8-0.1,1.3c-0.1,0.6-0.1,1-0.3,1.5l-3.4,14.7h-3.8l3-12.9c0.2-0.7,0.3-1.3,0.4-1.9 - c0.1-0.5,0.1-1.1,0.1-1.6c0-1.1-0.3-2-0.9-2.6c-0.6-0.6-1.6-0.9-3.1-0.9c-1,0-2.2,0.3-3.4,0.9c-1.2,0.6-2.5,1.2-3.7,2l-3.9,16.9 - h-3.8l7.3-31.6h3.8l-2.7,11.4c1.5-1,2.8-1.8,4.1-2.3c1.3-0.5,2.6-0.8,3.9-0.8c2,0,3.5,0.5,4.6,1.5 - C259.4,192.9,259.9,194.3,259.9,196.2z" /> - <path class="st5" d="M308.1,195.1c0,3.3-0.8,6.4-2.5,9.4c-1.7,3-4,5.2-6.9,6.8c-1.8,1-3.6,1.6-5.5,1.9c-1.9,0.3-4,0.5-6.4,0.5 - h-8.2l7-30.2h7.1c2.2,0,4.2,0.2,6,0.5c1.8,0.3,3.5,1,5,2c1.4,1,2.5,2.2,3.3,3.8C307.7,191.2,308.1,193,308.1,195.1z M303.8,195.4 - c0-1.5-0.3-2.9-0.8-4c-0.6-1.1-1.4-2-2.5-2.8c-1.1-0.7-2.3-1.2-3.5-1.5c-1.3-0.2-2.8-0.4-4.8-0.4h-3.4l-5.5,23.6h4.2 - c1.9,0,3.7-0.2,5.2-0.5c1.5-0.3,3-0.9,4.3-1.6c2.2-1.2,3.9-3,5-5.4C303.3,200.6,303.8,198.1,303.8,195.4z" /> - <path class="st5" - d="M324.7,211.3c-0.4,0.2-0.9,0.5-1.5,0.9c-0.6,0.4-1.3,0.7-1.9,1c-0.7,0.3-1.5,0.6-2.3,0.8 - c-0.9,0.2-1.8,0.3-3,0.3c-1.8,0-3.3-0.5-4.4-1.6c-1.1-1-1.7-2.4-1.7-4.1c0-1.8,0.4-3.3,1.2-4.6c0.8-1.2,2-2.2,3.5-3 - c1.5-0.8,3.4-1.3,5.6-1.7c2.2-0.3,4.6-0.6,7.4-0.6c0.1-0.4,0.2-0.7,0.2-1c0.1-0.3,0.1-0.6,0.1-0.9c0-0.6-0.1-1.2-0.4-1.6 - c-0.3-0.4-0.6-0.7-1.1-1c-0.5-0.2-1-0.4-1.7-0.5c-0.6-0.1-1.3-0.1-2.1-0.1c-1.2,0-2.5,0.2-4,0.6c-1.5,0.4-2.7,0.7-3.6,1.1h-0.2 - l0.8-3.8c0.8-0.2,1.9-0.4,3.4-0.7c1.5-0.3,2.9-0.4,4.3-0.4c2.8,0,5,0.4,6.4,1.3c1.4,0.9,2.1,2.3,2.1,4.2c0,0.4,0,0.8-0.1,1.2 - s-0.1,0.8-0.2,1.2l-3.6,15.4h-3.8L324.7,211.3z M327,201.8c-2.1,0.1-4,0.2-5.6,0.5c-1.6,0.2-3,0.6-4,1c-1.1,0.4-1.9,1.1-2.5,1.8 - c-0.6,0.8-0.9,1.7-0.9,2.9c0,1,0.3,1.7,1,2.2c0.7,0.5,1.7,0.8,3.1,0.8c1.2,0,2.5-0.3,3.8-0.8c1.3-0.5,2.5-1.2,3.6-2L327,201.8z" /> - <path class="st5" d="M352.5,191l-0.7,3.1h-7.9l-2.4,10.5c-0.1,0.5-0.3,1.1-0.4,1.8c-0.1,0.7-0.2,1.2-0.2,1.6c0,1,0.3,1.7,0.8,2.2 - c0.5,0.5,1.4,0.7,2.8,0.7c0.6,0,1.2-0.1,2-0.3c0.8-0.2,1.3-0.3,1.6-0.4h0.2l-0.7,3.3c-0.8,0.2-1.6,0.3-2.4,0.5 - c-0.9,0.1-1.6,0.2-2.3,0.2c-1.9,0-3.3-0.4-4.4-1.2c-1-0.8-1.5-2.1-1.5-3.9c0-0.4,0-0.9,0.1-1.3c0.1-0.4,0.1-0.9,0.3-1.5l2.8-12.2 - h-2.6l0.7-3.1h2.6l1.5-6.5h3.9l-1.5,6.5H352.5z" /> - <path class="st5" - d="M366.3,211.3c-0.4,0.2-0.9,0.5-1.5,0.9c-0.6,0.4-1.3,0.7-1.9,1c-0.7,0.3-1.5,0.6-2.3,0.8 - c-0.9,0.2-1.8,0.3-3,0.3c-1.8,0-3.3-0.5-4.4-1.6c-1.1-1-1.7-2.4-1.7-4.1c0-1.8,0.4-3.3,1.2-4.6c0.8-1.2,2-2.2,3.5-3 - c1.5-0.8,3.4-1.3,5.6-1.7c2.2-0.3,4.6-0.6,7.4-0.6c0.1-0.4,0.2-0.7,0.2-1c0.1-0.3,0.1-0.6,0.1-0.9c0-0.6-0.1-1.2-0.4-1.6 - c-0.3-0.4-0.6-0.7-1.1-1c-0.5-0.2-1-0.4-1.7-0.5c-0.6-0.1-1.3-0.1-2.1-0.1c-1.2,0-2.5,0.2-4,0.6c-1.5,0.4-2.7,0.7-3.6,1.1h-0.2 - l0.8-3.8c0.8-0.2,1.9-0.4,3.4-0.7c1.5-0.3,2.9-0.4,4.3-0.4c2.8,0,5,0.4,6.4,1.3c1.4,0.9,2.1,2.3,2.1,4.2c0,0.4,0,0.8-0.1,1.2 - s-0.1,0.8-0.2,1.2l-3.6,15.4h-3.8L366.3,211.3z M368.5,201.8c-2.1,0.1-4,0.2-5.6,0.5c-1.6,0.2-3,0.6-4,1c-1.1,0.4-1.9,1.1-2.5,1.8 - c-0.6,0.8-0.9,1.7-0.9,2.9c0,1,0.3,1.7,1,2.2c0.7,0.5,1.7,0.8,3.1,0.8c1.2,0,2.5-0.3,3.8-0.8c1.3-0.5,2.5-1.2,3.6-2L368.5,201.8z" /> - <path class="st5" d="M417.4,213.7h-4.7l-8-12.1h-5.8l-2.8,12.1H392l7-30.2h8.3c1.9,0,3.4,0.1,4.5,0.4c1.2,0.2,2.2,0.7,3,1.3 - c0.8,0.6,1.4,1.2,1.8,2c0.4,0.8,0.7,1.8,0.7,2.9c0,2.5-0.8,4.7-2.4,6.6c-1.6,1.9-3.7,3.2-6.2,4L417.4,213.7z M412.8,191 - c0-0.7-0.1-1.3-0.3-1.8c-0.2-0.5-0.6-0.9-1-1.3c-0.5-0.4-1.2-0.7-1.9-0.9c-0.7-0.2-1.7-0.2-2.8-0.2h-4.6l-2.7,11.6h4.3 - c1.3,0,2.5-0.1,3.5-0.4c1-0.2,1.9-0.7,2.7-1.3c0.9-0.7,1.6-1.5,2.1-2.5C412.6,193.2,412.8,192.1,412.8,191z" /> - <path class="st5" d="M431.5,214.2c-3.2,0-5.6-0.7-7.4-2.2c-1.8-1.5-2.7-3.6-2.7-6.5c0-4.2,1.3-7.7,4-10.7s5.9-4.5,9.8-4.5 - c2.6,0,4.6,0.6,5.9,1.9c1.4,1.3,2.1,3.1,2.1,5.4c0,0.4-0.1,1-0.2,1.9c-0.1,0.9-0.3,1.9-0.6,3.1h-16.8c-0.1,0.4-0.1,0.8-0.2,1.2 - c0,0.4-0.1,0.7-0.1,1.1c0,1.9,0.6,3.4,1.8,4.5c1.2,1.1,2.8,1.6,5,1.6c1.5,0,3-0.3,4.6-0.9c1.6-0.6,2.9-1.2,4-2h0.2l-0.8,4.1 - c-0.7,0.2-1.3,0.5-1.8,0.7s-1.2,0.4-2.1,0.6c-0.8,0.2-1.6,0.4-2.2,0.5C433.3,214.2,432.5,214.2,431.5,214.2z M439.2,199.9 - c0.1-0.4,0.1-0.7,0.1-1c0-0.3,0-0.6,0-0.9c0-1.4-0.4-2.6-1.2-3.4c-0.8-0.8-2.1-1.2-3.8-1.2c-1.9,0-3.6,0.6-5.1,1.8 - c-1.5,1.2-2.5,2.8-3,4.7H439.2z" /> - <path class="st5" d="M468.8,198.2c0,2.2-0.4,4.4-1.1,6.3c-0.7,2-1.6,3.7-2.8,5c-1.2,1.4-2.5,2.5-4.1,3.4c-1.6,0.8-3.2,1.2-5,1.2 - c-1.2,0-2.4-0.1-3.4-0.4s-2-0.7-2.8-1.2l-2.2,9.5h-3.8l7.2-31h3.8l-0.6,2.4c1.3-0.9,2.5-1.6,3.7-2.2c1.2-0.6,2.6-0.8,4.1-0.8 - c2.2,0,3.9,0.7,5.1,2.1C468.2,193.8,468.8,195.7,468.8,198.2z M464.8,198.9c0-1.6-0.4-2.9-1.1-3.7c-0.7-0.9-1.8-1.3-3.4-1.3 - c-1.1,0-2.3,0.3-3.5,0.8c-1.2,0.6-2.3,1.2-3.4,1.9l-3,12.9c0.9,0.5,1.7,0.8,2.5,1.1c0.8,0.2,1.8,0.3,2.9,0.3c1.4,0,2.7-0.3,3.8-1 - s2.1-1.6,2.8-2.6c0.8-1.1,1.4-2.4,1.7-3.8C464.6,202,464.8,200.5,464.8,198.9z" /> - <path class="st5" d="M494.1,199.1c0,2-0.3,4-0.9,5.8c-0.6,1.9-1.5,3.5-2.7,4.9c-1.2,1.4-2.6,2.6-4.1,3.4c-1.6,0.8-3.4,1.2-5.4,1.2 - c-2.7,0-4.8-0.8-6.3-2.3c-1.5-1.5-2.3-3.7-2.3-6.4c0-2,0.3-4,0.9-5.8c0.6-1.8,1.5-3.5,2.7-4.9c1.1-1.4,2.5-2.5,4.2-3.3 - c1.6-0.8,3.4-1.2,5.4-1.2c2.6,0,4.7,0.7,6.3,2.2S494.1,196.3,494.1,199.1z M487.8,207.6c0.7-1.1,1.3-2.3,1.7-3.8s0.6-2.9,0.6-4.5 - c0-1.9-0.5-3.3-1.4-4.3c-0.9-1-2.2-1.5-3.9-1.5c-1.3,0-2.5,0.3-3.6,0.9c-1,0.6-2,1.5-2.7,2.6c-0.7,1.1-1.3,2.3-1.7,3.8 - c-0.4,1.4-0.6,2.9-0.6,4.5c0,1.9,0.5,3.3,1.4,4.3c0.9,1,2.2,1.5,3.9,1.5c1.3,0,2.5-0.3,3.6-0.9 - C486.2,209.6,487.1,208.7,487.8,207.6z" /> - <path class="st5" d="M504.2,214.3c-1.7,0-3.2-0.2-4.6-0.6c-1.3-0.4-2.5-0.9-3.4-1.4l0.9-4.1h0.2c0.3,0.2,0.7,0.5,1.1,0.9 - c0.5,0.3,1,0.7,1.7,1c0.6,0.3,1.4,0.6,2.2,0.8c0.8,0.2,1.7,0.3,2.6,0.3c1.9,0,3.4-0.4,4.5-1.1s1.6-1.7,1.6-3.1 - c0-0.7-0.3-1.3-0.8-1.7c-0.6-0.4-1.4-0.7-2.4-0.9c-0.5-0.1-1.2-0.3-1.9-0.4c-0.7-0.1-1.5-0.3-2.3-0.5c-1.6-0.4-2.8-1.1-3.5-1.9 - c-0.8-0.8-1.1-1.9-1.1-3.1c0-1.1,0.2-2,0.7-3c0.5-0.9,1.1-1.8,2.1-2.5c0.9-0.7,2-1.3,3.3-1.8c1.3-0.5,2.8-0.7,4.5-0.7 - c1.4,0,2.7,0.2,4.1,0.5c1.4,0.3,2.5,0.8,3.3,1.3l-0.8,3.9H516c-0.2-0.2-0.5-0.4-1-0.7c-0.4-0.3-1-0.6-1.7-0.9 - c-0.6-0.3-1.3-0.5-2.1-0.7c-0.8-0.2-1.6-0.3-2.4-0.3c-1.7,0-3.1,0.4-4.1,1.1c-1.1,0.7-1.6,1.7-1.6,2.9c0,0.7,0.3,1.2,0.8,1.7 - c0.5,0.5,1.3,0.8,2.4,1.1c0.7,0.2,1.4,0.3,2.1,0.5c0.7,0.1,1.4,0.3,2.1,0.5c1.6,0.4,2.8,1,3.6,1.8s1.2,1.9,1.2,3.1 - c0,1-0.2,2.1-0.7,3.1c-0.5,1-1.2,1.9-2.2,2.6c-1,0.8-2.1,1.4-3.5,1.8C507.4,214,505.9,214.3,504.2,214.3z" /> - <path class="st5" d="M528,191l-5.3,22.7h-3.8l5.3-22.7H528z M530.1,183.3l-0.9,4h-4.3l0.9-4H530.1z" /> - <path class="st5" d="M547.4,191l-0.7,3.1h-7.9l-2.4,10.5c-0.1,0.5-0.3,1.1-0.4,1.8c-0.1,0.7-0.2,1.2-0.2,1.6c0,1,0.3,1.7,0.8,2.2 - c0.5,0.5,1.4,0.7,2.8,0.7c0.6,0,1.2-0.1,2-0.3c0.8-0.2,1.3-0.3,1.6-0.4h0.2l-0.7,3.3c-0.8,0.2-1.6,0.3-2.4,0.5 - c-0.9,0.1-1.6,0.2-2.3,0.2c-1.9,0-3.3-0.4-4.4-1.2c-1-0.8-1.5-2.1-1.5-3.9c0-0.4,0-0.9,0.1-1.3c0.1-0.4,0.1-0.9,0.3-1.5l2.8-12.2 - h-2.6l0.7-3.1h2.6l1.5-6.5h3.9l-1.5,6.5H547.4z" /> - <path class="st5" d="M569.2,199.1c0,2-0.3,4-0.9,5.8c-0.6,1.9-1.5,3.5-2.7,4.9c-1.2,1.4-2.6,2.6-4.1,3.4c-1.5,0.8-3.4,1.2-5.4,1.2 - c-2.7,0-4.8-0.8-6.3-2.3c-1.5-1.5-2.3-3.7-2.3-6.4c0-2,0.3-4,0.9-5.8c0.6-1.8,1.5-3.5,2.7-4.9c1.1-1.4,2.5-2.5,4.2-3.3 - c1.6-0.8,3.4-1.2,5.4-1.2c2.6,0,4.7,0.7,6.3,2.2C568.4,194.1,569.2,196.3,569.2,199.1z M563,207.6c0.7-1.1,1.3-2.3,1.7-3.8 - c0.4-1.4,0.6-2.9,0.6-4.5c0-1.9-0.5-3.3-1.4-4.3c-0.9-1-2.2-1.5-3.9-1.5c-1.3,0-2.5,0.3-3.6,0.9c-1,0.6-2,1.5-2.7,2.6 - c-0.7,1.1-1.3,2.3-1.7,3.8c-0.4,1.4-0.6,2.9-0.6,4.5c0,1.9,0.5,3.3,1.4,4.3c0.9,1,2.2,1.5,3.9,1.5c1.3,0,2.5-0.3,3.6-0.9 - C561.3,209.6,562.2,208.7,563,207.6z" /> - <path class="st5" d="M590.6,195.1h-0.2c-0.5-0.1-1.1-0.2-1.5-0.3c-0.5-0.1-1-0.1-1.7-0.1c-1.3,0-2.5,0.3-3.8,0.9 - c-1.3,0.6-2.5,1.3-3.6,2.1l-3.7,16.1h-3.9l5.2-22.7h3.9l-0.8,3.3c1.8-1.3,3.3-2.1,4.6-2.6c1.3-0.5,2.5-0.7,3.6-0.7 - c0.7,0,1.2,0,1.4,0.1c0.3,0,0.7,0.1,1.3,0.2L590.6,195.1z" /> - <path class="st5" d="M594.1,222.1h-4.2l6.6-9.7l-4.1-21.3h4l3.2,16.9l10.9-16.9h4.2L594.1,222.1z" /> - </g> - </g> - </svg> + <svg width="3420" height="756" viewBox="0 0 3420 756" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> +<rect width="3420" height="756" fill="url(#pattern0_280_776)"/> +<defs> +<pattern id="pattern0_280_776" patternContentUnits="objectBoundingBox" width="1" height="1"> +<use xlink:href="#image0_280_776" transform="scale(0.000292398 0.00132275)"/> +</pattern> +<image id="image0_280_776" width="3420" height="756" preserveAspectRatio="none" xlink:href=""/> +</defs> +</svg> </template> diff --git a/resources/js/Components/Map/MapComponentView.vue b/resources/js/Components/Map/MapComponentView.vue new file mode 100644 index 0000000..0dfd0aa --- /dev/null +++ b/resources/js/Components/Map/MapComponentView.vue @@ -0,0 +1,124 @@ +<script setup> +import { onMounted, ref, watch } from 'vue'; +import L from 'leaflet'; +import 'leaflet/dist/leaflet.css'; + +const DEFAULT_BASE_LAYER_NAME = 'BaseLayer'; +const DEFAULT_BASE_LAYER_ATTRIBUTION = '© <a target="_blank" href="http://osm.org/copyright">OpenStreetMap</a> contributors'; + +const props = defineProps({ + coverage: { + type: Object, + required: true + }, + height: { + type: String, + default: '250px' + }, + mapId: { + type: String, + default: 'view-map' + } +}); + +const map = ref(null); +const mapContainer = ref(null); + +onMounted(() => { + initializeMap(); +}); + +watch(() => props.coverage, (newCoverage) => { + if (map.value && newCoverage) { + updateBounds(); + } +}, { deep: true }); + +const initializeMap = () => { + // Create the map with minimal controls + map.value = L.map(mapContainer.value, { + zoomControl: false, + attributionControl: false, + dragging: false, + scrollWheelZoom: false, + doubleClickZoom: false, + boxZoom: false, + tap: false, + keyboard: false, + touchZoom: false + }); + + // // Add a simple tile layer (OpenStreetMap) + let osmGgray = new L.TileLayer.WMS('https://ows.terrestris.de/osm-gray/service', { + format: 'image/png', + attribution: DEFAULT_BASE_LAYER_ATTRIBUTION, + layers: 'OSM-WMS', + }); + let layerOptions = { + label: DEFAULT_BASE_LAYER_NAME, + visible: true, + layer: osmGgray, + }; + layerOptions.layer.addTo(map.value); + + // Add a light-colored rectangle for the coverage area + updateBounds(); +}; + +const updateBounds = () => { + if (!props.coverage || !map.value) return; + + // Clear any existing layers except the base tile layer + map.value.eachLayer(layer => { + if (layer instanceof L.Rectangle) { + map.value.removeLayer(layer); + } + }); + + // Create bounds from the coverage coordinates + const bounds = L.latLngBounds( + [props.coverage.y_min, props.coverage.x_min], + [props.coverage.y_max, props.coverage.x_max] + ); + + // Add a rectangle with emerald styling + L.rectangle(bounds, { + color: '#10b981', // emerald-500 + weight: 2, + fillColor: '#d1fae5', // emerald-100 + fillOpacity: 0.5 + }).addTo(map.value); + + // Fit the map to the bounds with some padding + map.value.fitBounds(bounds, { + padding: [20, 20] + }); +}; +</script> + +<template> + <div class="map-container bg-emerald-50 dark:bg-emerald-900/30 + rounded-lg shadow-sm overflow-hidden"> + <div :id="mapId" ref="mapContainer" :style="{ height: height }" class="w-full"></div> + </div> +</template> + +<style scoped> +/* Ensure the Leaflet container has proper styling */ +:deep(.leaflet-container) { + background-color: #f0fdf4; + /* emerald-50 */ +} + +/* Dark mode adjustments */ +@media (prefers-color-scheme: dark) { + :deep(.leaflet-container) { + background-color: rgba(6, 78, 59, 0.3); + /* emerald-900/30 */ + } + + :deep(.leaflet-tile) { + filter: brightness(0.8) contrast(1.2) grayscale(0.3); + } +} +</style> \ No newline at end of file diff --git a/resources/js/Components/Map/SearchMap.vue b/resources/js/Components/Map/SearchMap.vue index c846ed8..aac036e 100644 --- a/resources/js/Components/Map/SearchMap.vue +++ b/resources/js/Components/Map/SearchMap.vue @@ -282,7 +282,7 @@ const handleDrawEventCreated = async (event) => { </template> -<style lang="css"> +<style scoped lang="css"> /* .leaflet-container { height: 600px; width: 100%; diff --git a/resources/js/Components/Map/draw.component.vue b/resources/js/Components/Map/draw.component.vue index 95a3c56..1bfb9af 100644 --- a/resources/js/Components/Map/draw.component.vue +++ b/resources/js/Components/Map/draw.component.vue @@ -268,7 +268,7 @@ export default class DrawControlComponent extends Vue { position: absolute; left: 10px; top: 100px; - z-index: 999; + z-index: 40; } .btn-group-vertical button { diff --git a/resources/js/Components/Map/map.component.vue b/resources/js/Components/Map/map.component.vue index 53e2f07..d6d6f29 100644 --- a/resources/js/Components/Map/map.component.vue +++ b/resources/js/Components/Map/map.component.vue @@ -372,6 +372,9 @@ export default class MapComponent extends Vue { margin-top: 0.5em; } +.leaflet-container .leaflet-pane { + z-index: 30!important; +} /* .leaflet-pane { z-index: 30; } */ diff --git a/resources/js/Components/Map/zoom.component.vue b/resources/js/Components/Map/zoom.component.vue index e6cc5c8..bd7fc4a 100644 --- a/resources/js/Components/Map/zoom.component.vue +++ b/resources/js/Components/Map/zoom.component.vue @@ -100,7 +100,7 @@ export default class ZoomControlComponent extends Vue { position: absolute; left: 10px; top: 10px; - z-index: 999; + z-index: 40; } .btn-group-vertical button { diff --git a/resources/js/Components/MimetypeInput.vue b/resources/js/Components/MimetypeInput.vue new file mode 100644 index 0000000..f612a34 --- /dev/null +++ b/resources/js/Components/MimetypeInput.vue @@ -0,0 +1,138 @@ +<template> + <div class="relative mb-4"> + <!-- <label for="mimetype-input" class="block text-sm font-medium text-gray-700">Search for Mimetypes</label> --> + <input id="mimetype-input" v-model="newExtension" type="text" placeholder="Enter Mimetype Name" + class="block w-full border border-gray-300 rounded-lg px-4 py-2 focus:ring-blue-500 focus:border-blue-500" + :class="inputElClass" @input="handleInputChange" @keydown.down="onArrowDown" @keydown.up="onArrowUp" + @keydown.prevent.enter="onEnter"> + </input> + <svg class="w-4 h-4 absolute left-2.5 top-3.5" v-show="newExtension.length < 2" + xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor"> + <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" + d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z" /> + </svg> + <svg class="w-4 h-4 absolute left-2.5 top-3.5" v-show="newExtension.length >= 2" + xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" + @click="clearInput"> + <path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12" /> + </svg> + + <ul v-if="showDropdown && filteredMimetypes.length" + class="bg-white dark:bg-slate-800 w-full mt-2 max-h-28 overflow-y-auto scroll-smooth"> + <li v-for="(mimeType, index) in filteredMimetypes" :key="index" @click="selectResult(mimeType)" + class="pl-8 pr-2 py-1 border-b-2 border-gray-100 relative cursor-pointer hover:bg-blue-500 hover:text-white" + :class="{ + 'bg-blue-500 text-white': selectedIndex === index, + 'bg-white text-gray-900': selectedIndex !== index + }" :ref="(el) => { + if (ul) { + ul[index] = el as HTMLLIElement; + } + }"> + <svg class="absolute w-4 h-4 left-2 top-2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" + fill="currentColor"> + <path fill-rule="evenodd" + d="M12.293 5.293a1 1 0 011.414 0l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414-1.414L14.586 11H3a1 1 0 110-2h11.586l-2.293-2.293a1 1 0 010-1.414z" + clip-rule="evenodd" /> + </svg> + <span>{{ mimeType }}</span> + </li> + </ul> + </div> +</template> + +<script lang="ts" setup> +import { ref, Ref, watch, computed } from 'vue'; +// import mime from 'mime'; + +const emit = defineEmits(['onSelectResult', 'onClearInput']) + +const props = defineProps({ + borderless: Boolean, + transparent: Boolean, + mimeTypes: { + type: Array as () => string[], + required: true + }, + // form: Object, + // isValidMimeType: Function, +}); + +const newExtension = ref(''); +const showDropdown = ref(false); +const filteredMimetypes = ref<string[]>([]); +const selectedIndex: Ref<number> = ref(0); +const ul: Ref<HTMLLIElement[] | null> = ref<HTMLLIElement[]>([]); + +watch(selectedIndex, (selectedIndex: number) => { + if (selectedIndex != null && ul.value != null) { + const currentElement: HTMLLIElement = ul.value[selectedIndex]; + currentElement && + currentElement?.scrollIntoView({ + behavior: 'smooth', + block: 'nearest', + inline: 'start', + }); + } +}); + +const inputElClass = computed(() => { + const base = [ + 'block p-2.5 w-full z-20 text-sm text-gray-900 bg-gray-50 rounded-r-lg', + 'dark:placeholder-gray-400 dark:text-white dark:focus:border-blue-500', + 'h-12', + props.borderless ? 'border-0' : 'border', + props.transparent ? 'bg-transparent' : 'bg-white dark:bg-slate-800', + ]; + base.push('pl-10'); + return base; +}); + +const handleInputChange = (e: Event) => { + const target = <HTMLInputElement>e.target; + newExtension.value = target.value; + + if (newExtension.value.length >= 2) { + showDropdown.value = true; + filteredMimetypes.value = props.mimeTypes.filter(mimeType => + mimeType.toLowerCase().includes(newExtension.value.toLowerCase()) + ); + } else { + showDropdown.value = false; + } +}; + +const selectResult = (mimeType: string) => { + showDropdown.value = false; + newExtension.value = ''; + selectedIndex.value = -1; + emit('onSelectResult', mimeType); +}; + +const clearInput = () => { + newExtension.value = ''; + showDropdown.value = false; + // props.form.name = ''; + // props.resetFileExtensions(); + emit('onClearInput'); +}; + +const onArrowDown = () => { + if (filteredMimetypes.value.length > 0) { + selectedIndex.value = selectedIndex.value === filteredMimetypes.value.length - 1 ? 0 : selectedIndex.value + 1; + } +}; + +const onArrowUp = () => { + if (filteredMimetypes.value.length > 0) { + selectedIndex.value = selectedIndex.value == 0 || selectedIndex.value == -1 ? filteredMimetypes.value.length - 1 : selectedIndex.value - 1; + } +}; + +const onEnter = () => { + if (Array.isArray(filteredMimetypes.value) && filteredMimetypes.value.length && selectedIndex.value !== -1 && selectedIndex.value < filteredMimetypes.value.length) { + const mimeType = filteredMimetypes.value[selectedIndex.value]; + selectResult(mimeType); + } +}; +</script> diff --git a/resources/js/Components/NavBar.vue b/resources/js/Components/NavBar.vue index 2580c93..1df8ccb 100644 --- a/resources/js/Components/NavBar.vue +++ b/resources/js/Components/NavBar.vue @@ -21,12 +21,11 @@ import { mdiFormatListGroup, mdiFormatListNumbered, mdiLogout, - mdiGithub, mdiThemeLightDark, mdiViewDashboard, - mdiMapSearch, mdiInformationVariant, mdiGlasses, + mdiXml } from '@mdi/js'; import NavBarItem from '@/Components/NavBarItem.vue'; import NavBarItemLabel from '@/Components/NavBarItemLabel.vue'; @@ -72,7 +71,7 @@ const menuNavBarToggle = () => { const menuOpenLg = () => { layoutStore.isAsideLgActive = true; }; -const userHasRoles = (roleNames): boolean => { +const userHasRoles = (roleNames: Array<string>): boolean => { return user.value.roles.some(role => roleNames.includes(role.name)); }; @@ -95,12 +94,13 @@ const showAbout = async () => { </script> <template> - <nav class="text-base top-0 left-0 right-0 fixed bg-gray-50 h-14 z-40 w-screen transition-position lg:w-auto dark:bg-slate-800" + <nav class="text-base top-0 left-0 right-0 fixed bg-lime h-14 z-50 w-screen transition-position lg:w-auto dark:bg-slate-800" :class="{ 'xl:pl-60': props.showBurger == true }"> <FirstrunWizard ref="about"></FirstrunWizard> <div class="flex lg:items-stretch" :class="containerMaxW"> <div class="flex-1 items-stretch flex h-14"> - <NavBarItem type="flex lg:hidden" @click.prevent="layoutStore.asideMobileToggle()" v-if="props.showBurger"> + <NavBarItem type="flex lg:hidden" @click.prevent="layoutStore.asideMobileToggle()" + v-if="props.showBurger"> <BaseIcon :path="layoutStore.isAsideMobileExpanded ? mdiBackburger : mdiForwardburger" size="24" /> </NavBarItem> <NavBarItem type="hidden lg:flex xl:hidden" @click.prevent="menuOpenLg" v-if="props.showBurger"> @@ -110,9 +110,9 @@ const showAbout = async () => { <NavBarItemLabel :icon="mdiViewDashboard" label="Dashboard" size="22" is-hover-label-only route-name="apps.dashboard" /> </NavBarItem> - <NavBarItem route-name="apps.map"> + <!-- <NavBarItem route-name="apps.map"> <NavBarItemLabel :icon="mdiMapSearch" label="Map" size="22" is-hover-label-only route-name="apps.map" /> - </NavBarItem> + </NavBarItem> --> <!-- <NavBarItem> <NavBarSearch /> </NavBarItem> --> @@ -122,10 +122,10 @@ const showAbout = async () => { <BaseIcon :path="isMenuNavBarActive ? mdiClose : mdiDotsVertical" size="24" /> </NavBarItem> </div> - <div class="absolute w-screen top-14 left-0 bg-gray-50 shadow lg:w-auto lg:items-stretch lg:flex lg:grow lg:static lg:border-b-0 lg:overflow-visible lg:shadow-none dark:bg-slate-800" + <div class="fixed w-screen top-14 left-0 shadow lg:w-auto lg:items-stretch lg:flex lg:grow lg:static lg:border-b-0 lg:overflow-visible lg:shadow-none dark:bg-slate-800" :class="[isMenuNavBarActive ? 'block' : 'hidden']"> <div - class="max-h-screen-menu overflow-y-auto lg:overflow-visible lg:flex lg:items-stretch lg:justify-end lg:ml-auto"> + class="bg-white lg:bg-lime dark:bg-transparent max-h-screen-menu overflow-y-auto lg:overflow-visible lg:flex lg:items-stretch lg:justify-end lg:ml-auto"> <!-- help menu --> <NavBarMenu> @@ -150,7 +150,7 @@ const showAbout = async () => { <!-- personal menu --> <NavBarMenu> <NavBarItemLabel v-bind:label="`hello ${user.login}`"> - <UserAvatarCurrentUser class="w-6 h-6 mr-3 inline-flex" /> + <UserAvatarCurrentUser :user="user" class="w-6 h-6 mr-3 inline-flex" /> </NavBarItemLabel> <template #dropdown> <!-- <NavBarItem> --> @@ -169,13 +169,10 @@ const showAbout = async () => { </NavBarItem> <NavBarItem v-if="userHasRoles(['reviewer'])" :route-name="'reviewer.dataset.list'"> <NavBarItemLabel :icon="mdiGlasses" label="Reviewer Menu" /> - </NavBarItem> - <!-- <NavBarItem> - <NavBarItemLabel :icon="mdiEmail" label="Messages" /> - </NavBarItem> --> - <NavBarItem @click="showAbout"> + </NavBarItem> + <!-- <NavBarItem @click="showAbout"> <NavBarItemLabel :icon="mdiInformationVariant" label="About" /> - </NavBarItem> + </NavBarItem> --> <BaseDivider nav-bar /> <NavBarItem @click="logout"> <NavBarItemLabel :icon="mdiLogout" label="Log Out" /> @@ -186,12 +183,15 @@ const showAbout = async () => { <NavBarItem is-desktop-icon-only @click.prevent="toggleLightDark"> <NavBarItemLabel v-bind:icon="mdiThemeLightDark" label="Light/Dark" is-desktop-icon-only /> </NavBarItem> - <NavBarItem href="https://gitea.geologie.ac.at/geolba/tethys" target="_blank" is-desktop-icon-only> + <!-- <NavBarItem href="" target="_blank" is-desktop-icon-only> <NavBarItemLabel v-bind:icon="mdiGithub" label="GitHub" is-desktop-icon-only /> + </NavBarItem> --> + <NavBarItem href="/oai" target="_blank" is-desktop-icon-only> + <NavBarItemLabel v-bind:icon="mdiXml" label="OAI Interface" is-desktop-icon-only /> </NavBarItem> - <NavBarItem is-desktop-icon-only @click="showAbout"> + <!-- <NavBarItem is-desktop-icon-only @click="showAbout"> <NavBarItemLabel v-bind:icon="mdiInformationVariant" label="About" is-desktop-icon-only /> - </NavBarItem> + </NavBarItem> --> <NavBarItem is-desktop-icon-only @click="logout"> <NavBarItemLabel v-bind:icon="mdiLogout" label="Log out" is-desktop-icon-only /> </NavBarItem> diff --git a/resources/js/Components/NavBarMenu.vue b/resources/js/Components/NavBarMenu.vue index 10a96ab..d37fb08 100644 --- a/resources/js/Components/NavBarMenu.vue +++ b/resources/js/Components/NavBarMenu.vue @@ -1,4 +1,4 @@ -<script setup> +<script lang="ts" setup> import { StyleService } from '@/Stores/style.service'; import { computed, ref, onMounted, onBeforeUnmount } from 'vue'; import { mdiChevronUp, mdiChevronDown } from '@mdi/js'; @@ -15,10 +15,10 @@ const toggle = () => { isDropdownActive.value = !isDropdownActive.value; }; -const root = ref(null); +const root = ref(NavBarItem); -const forceClose = (event) => { - if (!root.value.$el.contains(event.target)) { +const forceClose = (event: MouseEvent) => { + if (!root.value?.$el.contains(event.target)) { isDropdownActive.value = false; } }; diff --git a/resources/js/Components/SearchAutocomplete.vue b/resources/js/Components/SearchAutocomplete.vue index 553c7b5..a3b3b09 100644 --- a/resources/js/Components/SearchAutocomplete.vue +++ b/resources/js/Components/SearchAutocomplete.vue @@ -28,7 +28,7 @@ autocomplete="off" @keydown.down="onArrowDown" @keydown.up="onArrowUp" - @keydown.enter="onEnter" + @keydown.enter.prevent="onEnter" /> <svg class="w-4 h-4 absolute left-2.5 top-3.5" diff --git a/resources/js/Components/SearchCategoryAutocomplete.vue b/resources/js/Components/SearchCategoryAutocomplete.vue index 3a73641..9441055 100644 --- a/resources/js/Components/SearchCategoryAutocomplete.vue +++ b/resources/js/Components/SearchCategoryAutocomplete.vue @@ -5,7 +5,7 @@ <div class="relative" data-te-dropdown-ref> <button id="states-button" data-dropdown-toggle="dropdown-states" class="whitespace-nowrap h-12 z-10 inline-flex items-center py-2.5 px-4 text-sm font-medium text-center text-gray-500 bg-gray-100 border border-gray-300 rounded-l-lg hover:bg-gray-200 focus:ring-4 focus:outline-none focus:ring-gray-100 dark:bg-gray-700 dark:hover:bg-gray-600 dark:focus:ring-gray-700 dark:text-white dark:border-gray-600" - type="button" @click.prevent="showStates"> + type="button" :disabled="isReadOnly" @click.prevent="showStates"> <!-- <svg aria-hidden="true" class="h-3 mr-2" viewBox="0 0 15 12" fill="none" xmlns="http://www.w3.org/2000/svg"> <rect x="0.5" width="14" height="12" rx="2" fill="white" /> <mask id="mask0_12694_49953" style="mask-type: alpha" maskUnits="userSpaceOnUse" x="0" y="0" width="15" height="12"> @@ -65,7 +65,7 @@ </svg> --> <!-- eng --> {{ language }} - <svg aria-hidden="true" class="w-4 h-4 ml-1" fill="currentColor" viewBox="0 0 20 20" + <svg aria-hidden="true" class="w-4 h-4 ml-1" fill="currentColor" viewBox="0 0 20 20" v-if="!isReadOnly" xmlns="http://www.w3.org/2000/svg"> <path fill-rule="evenodd" d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z" @@ -93,7 +93,7 @@ <!-- :class="inputElClass" --> <!-- class="block p-2.5 w-full z-20 text-sm text-gray-900 bg-gray-50 rounded-r-lg border-l-gray-50 border-l-2 border border-gray-300 focus:ring-blue-500 focus:border-blue-500 dark:bg-gray-700 dark:border-l-gray-700 dark:border-gray-600 dark:placeholder-gray-400 dark:text-white dark:focus:border-blue-500" --> <input v-model="computedValue" type="text" :name="props.name" autocomplete="off" :class="inputElClass" - placeholder="Search Keywords..." required @input="handleInput" /> + placeholder="Search Keywords..." required @input="handleInput" :readonly="isReadOnly" /> <!-- v-model="data.search" --> <svg class="w-4 h-4 absolute left-2.5 top-3.5" v-show="computedValue.length < 2" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor"> @@ -101,12 +101,12 @@ d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z" /> </svg> - <svg class="w-4 h-4 absolute left-2.5 top-3.5" v-show="computedValue.length >= 2" + <svg class="w-4 h-4 absolute left-2.5 top-3.5" v-show="computedValue.length >= 2 && !isReadOnly" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" @click="() => { computedValue = ''; data.isOpen = false; } - "> + "> <path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12" /> </svg> </div> @@ -166,6 +166,10 @@ let props = defineProps({ type: String, default: '', }, + isReadOnly: { + type: Boolean, + default: false, + }, required: Boolean, borderless: Boolean, transparent: Boolean, @@ -190,11 +194,18 @@ const inputElClass = computed(() => { 'dark:placeholder-gray-400 dark:text-white dark:focus:border-blue-500', 'h-12', props.borderless ? 'border-0' : 'border', - props.transparent ? 'bg-transparent' : 'bg-white dark:bg-slate-800', + props.transparent && 'bg-transparent', // props.isReadOnly ? 'bg-gray-50 dark:bg-slate-600' : 'bg-white dark:bg-slate-800', ]; // if (props.icon) { base.push('pl-10'); + if (props.isReadOnly) { + // Read-only: no focus ring, grayed-out text and border, and disabled cursor. + base.push('bg-gray-50', 'dark:bg-slate-600', 'border', 'border-gray-300', 'dark:border-slate-600', 'text-gray-500', 'cursor-not-allowed', 'focus:outline-none', 'focus:ring-0', 'focus:border-gray-300'); + } else { + // Actionable field: focus ring, white/dark background, and darker border. + base.push('bg-white dark:bg-slate-800', 'focus:ring focus:outline-none', 'border', 'border-gray-700'); + } // } return base; }); diff --git a/resources/js/Components/SectionBannerStarOnGitea.vue b/resources/js/Components/SectionBannerStarOnGitea.vue index b2ac77f..f2d57e5 100644 --- a/resources/js/Components/SectionBannerStarOnGitea.vue +++ b/resources/js/Components/SectionBannerStarOnGitea.vue @@ -5,9 +5,9 @@ import SectionBanner from '@/Components/SectionBanner.vue'; </script> <template> <SectionBanner bg="greenBlue"> - <h1 class="text-3xl text-white mb-6">Like the project? Please star on <b>Gitea</b>!</h1> + <h1 class="text-3xl text-white mb-6">Like the project? Please star on <b>GeoSphere Git Repository</b>!</h1> <div> - <BaseButton href="https://gitea.geologie.ac.at/geolba/tethys" :icon="mdiGithub" label="Gitea" target="_blank" rounded-full /> + <BaseButton href="https://gitea.geosphere.at/geolba/tethys.backend" :icon="mdiGithub" label="Forgejo" target="_blank" rounded-full /> </div> </SectionBanner> </template> diff --git a/resources/js/Components/SectionTitleLineWithButton.vue b/resources/js/Components/SectionTitleLineWithButton.vue index ca6fa30..e243946 100644 --- a/resources/js/Components/SectionTitleLineWithButton.vue +++ b/resources/js/Components/SectionTitleLineWithButton.vue @@ -15,6 +15,10 @@ defineProps({ required: true, }, main: Boolean, + showCogButton: { + type: Boolean, + default: false, + } }); const hasSlot = computed(() => useSlots().default); @@ -30,6 +34,6 @@ const hasSlot = computed(() => useSlots().default); </h1> </div> <slot v-if="hasSlot" /> - <BaseButton v-else :icon="mdiCog" small /> + <BaseButton v-else-if="showCogButton" :icon="mdiCog" small /> </section> </template> diff --git a/resources/js/Components/SimplePasswordMeter/password-meter.vue b/resources/js/Components/SimplePasswordMeter/password-meter.vue index 45eca5a..e2dffd1 100644 --- a/resources/js/Components/SimplePasswordMeter/password-meter.vue +++ b/resources/js/Components/SimplePasswordMeter/password-meter.vue @@ -1,23 +1,50 @@ <script lang="ts" setup> -import { computed, ref, watch } from 'vue'; +import { computed } from 'vue'; import { checkStrength } from './logic/index'; import { mdiFormTextboxPassword } from '@mdi/js'; import FormField from '@/Components/FormField.vue'; import FormControl from '@/Components/FormControl.vue'; // Define props -const props = defineProps<{ - password: string; - errors: Partial<Record<"new_password" | "old_password" | "confirm_password", string>>; -}>(); +// const props = defineProps<{ +// modelValue: string, +// errors: Partial<Record<"new_password" | "old_password" | "confirm_password", string>>, +// showRequiredMessage: boolean, +// }>(); -const emit = defineEmits(['update:password', 'score']); +const props = defineProps({ + modelValue: { + type: String, + }, + errors: { + type: Object, + default: () => ({} as Partial<Record<"new_password" | "old_password" | "confirm_password", string>>), + }, + showRequiredMessage: { + type: Boolean, + default:true, + }, + fieldLabel: { + type: String, + default: 'New password', + } +}); -// A local reactive variable for password input -const localPassword = ref(props.password); -// Watch localPassword and emit changes back to the parent -watch(localPassword, (newValue) => { - emit('update:password', newValue); +const emit = defineEmits(['update:modelValue', 'score']); + +// // A local reactive variable for password input +// const localPassword = ref(props.modelValue); +// // Watch localPassword and emit changes back to the parent +// watch(localPassword, (newValue) => { +// emit('update:modelValue', newValue); +// }); +const localPassword = computed({ + get: () => props.modelValue, + set: (value) => { + emit('update:modelValue', value); + // const { score } = checkStrength(localPassword.value); + // emit('score', score); + }, }); type PasswordMetrics = { @@ -53,8 +80,8 @@ const passwordMetrics = computed<PasswordMetrics>(() => { <template> <!-- Password input Form --> - <FormField label="New password" help="Required. New password" :class="{ 'text-red-400': errors.new_password }"> - <FormControl v-model="localPassword" :icon="mdiFormTextboxPassword" name="new_password" type="password" required + <FormField :label="fieldLabel" :help="showRequiredMessage ? 'Required. New password' : ''" :class="{'text-red-400': errors.new_password }"> + <FormControl v-model="localPassword" :icon="mdiFormTextboxPassword" name="new_password" type="password" :required="showRequiredMessage" :error="errors.new_password"> <!-- Secure Icon --> <template #right> @@ -72,10 +99,14 @@ const passwordMetrics = computed<PasswordMetrics>(() => { {{ errors.new_password }} </div> </FormControl> - </FormField> + <!-- Score Display --> + <div class="text-gray-700 text-sm"> + {{ passwordMetrics.score }} / 6 points max + </div> +</FormField> <!-- Password Strength Bar --> - <div class="po-password-strength-bar w-full h-2 rounded transition-all duration-200 mb-4" + <div v-if="passwordMetrics.score > 0"class="po-password-strength-bar w-full h-2 rounded transition-all duration-200 mb-4" :class="passwordMetrics.scoreLabel" :style="{ width: `${(passwordMetrics.score / 6) * 100}%` }" role="progressbar" :aria-valuenow="passwordMetrics.score" aria-valuemin="0" aria-valuemax="6" :aria-label="`Password strength: ${passwordMetrics.scoreLabel || 'unknown'}`"> @@ -93,9 +124,9 @@ const passwordMetrics = computed<PasswordMetrics>(() => { </ul> </div> <!-- Score Display --> - <div class="text-gray-700 text-sm"> + <!-- <div class="text-gray-700 text-sm"> {{ passwordMetrics.score }} / 6 points max - </div> + </div> --> </template> <style lang="css" scoped> diff --git a/resources/js/Components/TableKeywords.vue b/resources/js/Components/TableKeywords.vue index 835dd51..fbe979c 100644 --- a/resources/js/Components/TableKeywords.vue +++ b/resources/js/Components/TableKeywords.vue @@ -12,6 +12,7 @@ import { Subject } from '@/Dataset'; // import FormField from '@/Components/FormField.vue'; import FormControl from '@/Components/FormControl.vue'; import SearchCategoryAutocomplete from '@/Components/SearchCategoryAutocomplete.vue'; +import { mdiRefresh } from '@mdi/js'; const props = defineProps({ checkable: Boolean, @@ -27,6 +28,22 @@ const props = defineProps({ type: Object, default: () => ({}), }, + subjectsToDelete: { + type: Array<Subject>, + default: [], + } +}); + +const emit = defineEmits(['update:subjectsToDelete']); + +// Create a computed property for subjectsToDelete with getter and setter +const deletetSubjects = computed({ + get: () => props.subjectsToDelete, + set: (values: Array<Subject>) => { + props.subjectsToDelete.length = 0; + props.subjectsToDelete.push(...values); + emit('update:subjectsToDelete', values); + } }); const styleService = StyleService(); @@ -58,21 +75,45 @@ const pagesList = computed(() => { }); const removeItem = (key: number) => { + // items.value.splice(key, 1); + const item = items.value[key]; + + // If the item has an ID, add it to the delete list + if (item.id) { + addToDeleteList(item); + } + + // Remove from the visible list items.value.splice(key, 1); }; +// Helper function to add a subject to the delete list +const addToDeleteList = (subject: Subject) => { + if (subject.id) { + const newList = [...props.subjectsToDelete, subject]; + deletetSubjects.value = newList; + } +}; + + +// Helper function to reactivate a subject (remove from delete list) +const reactivateSubject = (index: number) => { + const newList = [...props.subjectsToDelete]; + const removedSubject = newList.splice(index, 1)[0]; + deletetSubjects.value = newList; + + // Add the subject back to the keywords list if it's not already there + if (removedSubject && !props.keywords.some(k => k.id === removedSubject.id)) { + props.keywords.push(removedSubject); + } +}; + +const isKeywordReadOnly = (item: Subject) => { + return (item.dataset_count ?? 0) > 1 || item.type !== 'uncontrolled'; +}; </script> <template> - <!-- <CardBoxModal v-model="isModalActive" title="Sample modal"> - <p>Lorem ipsum dolor sit amet <b>adipiscing elit</b></p> - <p>This is sample modal</p> - </CardBoxModal> - - <CardBoxModal v-model="isModalDangerActive" large-title="Please confirm" button="danger" has-cancel> - <p>Lorem ipsum dolor sit amet <b>adipiscing elit</b></p> - <p>This is sample modal</p> - </CardBoxModal> --> <!-- <div v-if="checkedRows.length" class="p-3 bg-gray-100/50 dark:bg-slate-800"> <span v-for="checkedRow in checkedRows" :key="checkedRow.id" @@ -87,17 +128,34 @@ const removeItem = (key: number) => { <!-- <th v-if="checkable" /> --> <!-- <th class="hidden lg:table-cell"></th> --> <th scope="col">Type</th> - <th scope="col">Value</th> + <th scope="col" class="relative"> + Value + <div class="inline-block relative ml-1 group"> + <button + class="w-4 h-4 rounded-full bg-gray-200 text-gray-600 text-xs flex items-center justify-center focus:outline-none hover:bg-gray-300"> + i + </button> + <div + class="absolute left-0 top-full mt-1 w-64 bg-white shadow-lg rounded-md p-3 text-xs text-left z-50 transform scale-0 origin-top-left transition-transform duration-100 group-hover:scale-100"> + <p class="text-gray-700"> + Keywords are only editable if they are used by a single dataset (Usage Count = 1)". + </p> + <div class="absolute -top-1 left-1 w-2 h-2 bg-white transform rotate-45"></div> + </div> + </div> + </th> <th scope="col">Language</th> - + <th scope="col">Usage Count</th> <th scope="col" /> </tr> </thead> <tbody> <tr v-for="(item, index) in itemsPaginated" :key="index"> - + <td data-label="Type" scope="row"> - <FormControl required v-model="item.type" @update:modelValue="() => {item.external_key = undefined; item.value= '';}" :type="'select'" placeholder="[Enter Language]" :options="props.subjectTypes"> + <FormControl required v-model="item.type" + @update:modelValue="() => { item.value = ''; }" :type="'select'" + placeholder="[Enter Language]" :options="props.subjectTypes"> <div class="text-red-400 text-sm" v-if="errors[`subjects.${index}.type`]"> {{ errors[`subjects.${index}.type`].join(', ') }} </div> @@ -105,22 +163,19 @@ const removeItem = (key: number) => { </td> <td data-label="Value" scope="row"> - <SearchCategoryAutocomplete - v-if="item.type !== 'uncontrolled'" - v-model="item.value" - @subject=" - (result) => { - item.language = result.language; - item.external_key = result.uri; - } - " - > + <SearchCategoryAutocomplete v-if="item.type !== 'uncontrolled'" v-model="item.value" @subject=" + (result) => { + item.language = result.language; + item.external_key = result.uri; + } + " :is-read-only="item.dataset_count > 1"> <div class="text-red-400 text-sm" v-if="errors[`subjects.${index}.value`]"> {{ errors[`subjects.${index}.value`].join(', ') }} </div> </SearchCategoryAutocomplete> - <FormControl v-else required v-model="item.value" type="text" placeholder="[enter keyword value]" :borderless="true"> + <FormControl v-else required v-model="item.value" type="text" placeholder="[enter keyword value]" + :borderless="true" :is-read-only="item.dataset_count > 1"> <div class="text-red-400 text-sm" v-if="errors[`subjects.${index}.value`]"> {{ errors[`subjects.${index}.value`].join(', ') }} </div> @@ -128,23 +183,24 @@ const removeItem = (key: number) => { </td> <td data-label="Language" scope="row"> - <FormControl - required - v-model="item.language" - :type="'select'" - placeholder="[Enter Lang]" - :options="{ de: 'de', en: 'en' }" - :is-read-only="item.type != 'uncontrolled'" - > + <FormControl required v-model="item.language" :type="'select'" placeholder="[Enter Lang]" + :options="{ de: 'de', en: 'en' }" :is-read-only="isKeywordReadOnly(item)"> <div class="text-red-400 text-sm" v-if="errors[`subjects.${index}.language`]"> {{ errors[`subjects.${index}.language`].join(', ') }} </div> </FormControl> </td> + + <td data-label="Usage Count" scope="row"> + <div class="text-center"> + {{ item.dataset_count || 1 }} + </div> + </td> + <td class="before:hidden lg:w-1 whitespace-nowrap" scope="row"> <BaseButtons type="justify-start lg:justify-end" no-wrap> <!-- <BaseButton color="info" :icon="mdiEye" small @click="isModalActive = true" /> --> - <BaseButton v-if="index > 2" color="danger" :icon="mdiTrashCan" small @click.prevent="removeItem(index)" /> + <BaseButton color="danger" :icon="mdiTrashCan" small @click.prevent="removeItem(index)" /> </BaseButtons> </td> </tr> @@ -155,15 +211,8 @@ const removeItem = (key: number) => { <div class="p-3 lg:px-6 border-t border-gray-100 dark:border-slate-800"> <BaseLevel> <BaseButtons> - <BaseButton - v-for="page in pagesList" - :key="page" - :active="page === currentPage" - :label="page + 1" - small - :outline="styleService.darkMode" - @click="currentPage = page" - /> + <BaseButton v-for="page in pagesList" :key="page" :active="page === currentPage" :label="page + 1" small + :outline="styleService.darkMode" @click="currentPage = page" /> </BaseButtons> <small>Page {{ currentPageHuman }} of {{ numPages }}</small> </BaseLevel> @@ -172,6 +221,47 @@ const removeItem = (key: number) => { <div class="text-red-400 text-sm" v-if="errors.subjects && Array.isArray(errors.subjects)"> {{ errors.subjects.join(', ') }} </div> + + <!-- Subjects to delete section --> + <div v-if="deletetSubjects.length > 0" class="mt-8"> + <h1 class="pt-8 pb-3 font-semibold sm:text-lg text-gray-900">Keywords To Delete</h1> + <ul id="deleteSubjects" tag="ul" class="flex flex-1 flex-wrap -m-1"> + <li v-for="(element, index) in deletetSubjects" :key="index" + class="block p-1 w-1/2 sm:w-1/3 md:w-1/4 lg:w-1/6 xl:w-1/8 h-32"> + <article tabindex="0" + class="bg-red-100 group w-full h-full rounded-md cursor-pointer relative shadow-sm overflow-hidden"> + <section + class="flex flex-col rounded-md text-xs break-words w-full h-full z-20 absolute top-0 py-2 px-3"> + <h1 class="flex-1 text-gray-700 group-hover:text-blue-800 font-medium text-sm mb-1">{{ + element.value }}</h1> + <div class="flex items-center justify-between mt-auto"> + <div class="flex flex-col"> + <p class="p-1 size text-xs text-gray-700"> + <span class="font-semibold">Type:</span> {{ element.type }} + </p> + <p class="p-1 size text-xs text-gray-700" v-if="element.dataset_count"> + <span class="font-semibold">Used by:</span> + <span + class="inline-flex items-center justify-center bg-gray-200 text-gray-800 rounded-full w-5 h-5 text-xs"> + {{ element.dataset_count }} + </span> datasets + </p> + </div> + <button + class="delete ml-auto focus:outline-none hover:bg-gray-300 p-1 rounded-md text-gray-800" + @click.prevent="reactivateSubject(index)"> + <svg viewBox="0 0 24 24" class="w-5 h-5"> + <path fill="currentColor" :d="mdiRefresh"></path> + </svg> + </button> + </div> + </section> + </article> + </li> + </ul> + </div> + + </template> <style scoped> diff --git a/resources/js/Components/TableSampleClients.vue b/resources/js/Components/TableSampleClients.vue index f2a062a..9fb1c01 100644 --- a/resources/js/Components/TableSampleClients.vue +++ b/resources/js/Components/TableSampleClients.vue @@ -1,17 +1,19 @@ -<script setup> -import { computed, ref } from 'vue'; +<script lang="ts" setup> +import { computed, ref, Ref } from 'vue'; import { MainService } from '@/Stores/main'; import { StyleService } from '@/Stores/style.service'; -import { mdiEye, mdiTrashCan } from '@mdi/js'; +import { mdiEye } from '@mdi/js'; import CardBoxModal from '@/Components/CardBoxModal.vue'; import TableCheckboxCell from '@/Components/TableCheckboxCell.vue'; import BaseLevel from '@/Components/BaseLevel.vue'; import BaseButtons from '@/Components/BaseButtons.vue'; import BaseButton from '@/Components/BaseButton.vue'; import UserAvatar from '@/Components/UserAvatar.vue'; +import dayjs from 'dayjs'; +import { User } from '@/Stores/main'; defineProps({ - checkable: Boolean, + checkable: Boolean, }); const styleService = StyleService(); @@ -19,128 +21,124 @@ const mainService = MainService(); const items = computed(() => mainService.clients); const isModalActive = ref(false); -const isModalDangerActive = ref(false); +// const isModalDangerActive = ref(false); const perPage = ref(5); const currentPage = ref(0); const checkedRows = ref([]); +const currentClient: Ref<User | null> = ref(null); const itemsPaginated = computed(() => items.value.slice(perPage.value * currentPage.value, perPage.value * (currentPage.value + 1))); const numPages = computed(() => Math.ceil(items.value.length / perPage.value)); - const currentPageHuman = computed(() => currentPage.value + 1); const pagesList = computed(() => { - const pagesList = []; - - for (let i = 0; i < numPages.value; i++) { - pagesList.push(i); - } - - return pagesList; + const pagesList = []; + for (let i = 0; i < numPages.value; i++) { + pagesList.push(i); + } + return pagesList; }); const remove = (arr, cb) => { - const newArr = []; - - arr.forEach((item) => { - if (!cb(item)) { - newArr.push(item); - } - }); - - return newArr; + const newArr = []; + arr.forEach((item) => { + if (!cb(item)) { + newArr.push(item); + } + }); + return newArr; }; const checked = (isChecked, client) => { - if (isChecked) { - checkedRows.value.push(client); - } else { - checkedRows.value = remove(checkedRows.value, (row) => row.id === client.id); - } + if (isChecked) { + checkedRows.value.push(client); + } else { + checkedRows.value = remove(checkedRows.value, (row) => row.id === client.id); + } +}; + +const showModal = (client: User) => { + currentClient.value = client; + isModalActive.value = true; }; </script> <template> - <CardBoxModal v-model="isModalActive" title="Sample modal"> - <p>Lorem ipsum dolor sit amet <b>adipiscing elit</b></p> - <p>This is sample modal</p> - </CardBoxModal> - - <CardBoxModal v-model="isModalDangerActive" large-title="Please confirm" button="danger" has-cancel> - <p>Lorem ipsum dolor sit amet <b>adipiscing elit</b></p> - <p>This is sample modal</p> - </CardBoxModal> - - <div v-if="checkedRows.length" class="p-3 bg-gray-100/50 dark:bg-slate-800"> - <span - v-for="checkedRow in checkedRows" - :key="checkedRow.id" - class="inline-block px-2 py-1 rounded-sm mr-2 text-sm bg-gray-100 dark:bg-slate-700" - > - {{ checkedRow.name }} - </span> + <CardBoxModal v-model="isModalActive" :title="currentClient ? currentClient.login : ''"> + <div v-if="currentClient"> + <p>Login: {{ currentClient.login }}</p> + <p>Email: {{ currentClient.email }}</p> + <p>Created: {{ currentClient?.created_at ? dayjs(currentClient.created_at).format('MMM D, YYYY h:mm A') : 'N/A' }} + </p> </div> + </CardBoxModal> + <!-- <CardBoxModal v-model="isModalDangerActive" large-title="Please confirm" button="danger" has-cancel> + <p>Lorem ipsum dolor sit amet <b>adipiscing elit</b></p> + <p>This is sample modal</p> + </CardBoxModal> --> - <table> - <thead> - <tr> - <th v-if="checkable" /> - <th /> - <th>Name</th> - <th>Email</th> - <th>City</th> - <th>Progress</th> - <th>Created</th> - <th /> - </tr> - </thead> - <tbody> - <tr v-for="client in itemsPaginated" :key="client.id"> - <TableCheckboxCell v-if="checkable" @checked="checked($event, client)" /> - <td class="border-b-0 lg:w-6 before:hidden"> - <UserAvatar :username="client.name" class="w-24 h-24 mx-auto lg:w-6 lg:h-6" /> - </td> - <td data-label="Name"> - {{ client.name }} - </td> - <td data-label="Email"> - {{ client.email }} - </td> - <td data-label="City"> - {{ client.city }} - </td> - <td data-label="Progress" class="lg:w-32"> - <progress class="flex w-2/5 self-center lg:w-full" max="100" v-bind:value="client.progress"> - {{ client.progress }} - </progress> - </td> - <td data-label="Created" class="lg:w-1 whitespace-nowrap"> - <small class="text-gray-500 dark:text-slate-400" :title="client.created">{{ client.created }}</small> - </td> - <td class="before:hidden lg:w-1 whitespace-nowrap"> - <BaseButtons type="justify-start lg:justify-end" no-wrap> - <BaseButton color="info" :icon="mdiEye" small @click="isModalActive = true" /> - <BaseButton color="danger" :icon="mdiTrashCan" small @click="isModalDangerActive = true" /> - </BaseButtons> - </td> - </tr> - </tbody> - </table> - <div class="p-3 lg:px-6 border-t border-gray-100 dark:border-slate-800"> - <BaseLevel> - <BaseButtons> - <BaseButton - v-for="page in pagesList" - :key="page" - :active="page === currentPage" - :label="page + 1" - small - :outline="styleService.darkMode" - @click="currentPage = page" - /> - </BaseButtons> - <small>Page {{ currentPageHuman }} of {{ numPages }}</small> - </BaseLevel> - </div> -</template> \ No newline at end of file + <div v-if="checkedRows.length" class="p-3 bg-gray-100/50 dark:bg-slate-800"> + <span v-for="checkedRow in checkedRows" :key="checkedRow.id" + class="inline-block px-2 py-1 rounded-sm mr-2 text-sm bg-gray-100 dark:bg-slate-700"> + {{ checkedRow.login }} + </span> + </div> + + <table> + <thead> + <tr> + <th v-if="checkable" /> + <th /> + <th>Login</th> + <th>Email</th> + <th>Created</th> + <th /> + </tr> + </thead> + <tbody> + <tr v-for="client in itemsPaginated" :key="client.id"> + <TableCheckboxCell v-if="checkable" @checked="checked($event, client)" /> + <td class="border-b-0 lg:w-6 before:hidden"> + <!-- <UserAvatar :username="client.login" :avatar="client.avatar" class="w-24 h-24 mx-auto lg:w-6 lg:h-6" /> --> + <div v-if="client.avatar"> + <UserAvatar :default-url="client.avatar ? '/public' + client.avatar : ''" + :username="client.first_name + ' ' + client.last_name" class="w-24 h-24 mx-auto lg:w-6 lg:h-6" /> + </div> + + + <div v-else> + <UserAvatar :username="client.first_name + ' ' + client.last_name" class="w-24 h-24 mx-auto lg:w-6 lg:h-6" /> + </div> + </td> + <td data-label="Login"> + {{ client.login }} + </td> + <td data-label="Email"> + {{ client.email }} + </td> + <td data-label="Created"> + <small class="text-gray-500 dark:text-slate-400" + :title="client.created_at ? dayjs(client.created_at).format('MMM D, YYYY h:mm A') : 'N/A'"> + {{ client.created_at ? dayjs(client.created_at).format('MMM D, YYYY h:mm A') : 'N/A' }} + </small> + </td> + <td class="before:hidden lg:w-1 whitespace-nowrap"> + <BaseButtons type="justify-start lg:justify-end" no-wrap> + <BaseButton color="info" :icon="mdiEye" small @click="showModal(client)" /> + <!-- <BaseButton color="danger" :icon="mdiTrashCan" small @click="isModalDangerActive = true" /> --> + </BaseButtons> + </td> + </tr> + </tbody> + </table> + <div class="p-3 lg:px-6 border-t border-gray-100 dark:border-slate-800"> + <BaseLevel> + <BaseButtons> + <BaseButton v-for="page in pagesList" :key="page" :active="page === currentPage" :label="page + 1" small + :outline="styleService.darkMode" @click="currentPage = page" /> + </BaseButtons> + <small>Page {{ currentPageHuman }} of {{ numPages }}</small> + </BaseLevel> + </div> +</template> diff --git a/resources/js/Components/UserAvatar.vue b/resources/js/Components/UserAvatar.vue index ace3a03..6484d83 100644 --- a/resources/js/Components/UserAvatar.vue +++ b/resources/js/Components/UserAvatar.vue @@ -1,4 +1,4 @@ -<script setup> +<script lang="ts" setup> import { computed } from 'vue'; const props = defineProps({ @@ -6,9 +6,9 @@ const props = defineProps({ type: String, required: true, }, - avatar: { + defaultUrl: { type: String, - default: null, + required: false }, api: { type: String, @@ -16,93 +16,63 @@ const props = defineProps({ }, }); -const avatar = computed( - // () => props.avatar ?? `https://avatars.dicebear.com/api/${props.api}/${props.username?.replace(/[^a-z0-9]+/i, '-')}.svg` - - // () => props.avatar ?? `https://avatars.dicebear.com/api/initials/${props.username}.svg`, - - () => { - const initials = props.username - .split(' ') - .map((part) => part.charAt(0).toUpperCase()) - .join(''); - - return props.avatar ?? generateAvatarUrl(props.username); - }, -); +const avatar = computed(() => { + return props.defaultUrl ?? generateAvatarUrl(props.username); +}); const username = computed(() => props.username); -const darkenColor = (color) => { - // Convert hex to RGB - const r = parseInt(color.slice(0, 2), 16); - const g = parseInt(color.slice(2, 4), 16); - const b = parseInt(color.slice(4, 6), 16); +// const darkenColor = (color: string) => { +// const r = parseInt(color.slice(0, 2), 16); +// const g = parseInt(color.slice(2, 4), 16); +// const b = parseInt(color.slice(4, 6), 16); - // Calculate darker color by reducing 20% of each RGB component - const darkerR = Math.round(r * 0.6); - const darkerG = Math.round(g * 0.6); - const darkerB = Math.round(b * 0.6); +// const darkerR = Math.round(r * 0.6); +// const darkerG = Math.round(g * 0.6); +// const darkerB = Math.round(b * 0.6); - // Convert back to hex - const darkerColor = ((darkerR << 16) + (darkerG << 8) + darkerB).toString(16); +// const darkerColor = ((darkerR << 16) + (darkerG << 8) + darkerB).toString(16); - return darkerColor.padStart(6, '0'); // Ensure it's 6 digits -}; +// return darkerColor.padStart(6, '0'); +// }; -const getRandomColor = () => { - return Math.floor(Math.random() * 16777215).toString(16); -}; +// const getColorFromName = (name: string): string => { +// let hash = 0; +// for (let i = 0; i < name.length; i++) { +// hash = name.charCodeAt(i) + ((hash << 5) - hash); +// } +// let color = '#'; +// for (let i = 0; i < 3; i++) { +// const value = (hash >> (i * 8)) & 0xff; +// color += ('00' + value.toString(16)).substr(-2); +// } +// return color.replace('#', ''); +// }; -const adjustOpacity = (hexColor, opacity) => { - // Remove # if present - hexColor = hexColor.replace('#', ''); - // Convert hex to RGB - // const r = parseInt(hexColor.slice(0, 2), 16); - // const g = parseInt(hexColor.slice(2, 4), 16); - // const b = parseInt(hexColor.slice(4, 6), 16); +// const lightenColor = (hexColor: string, percent: number): string => { +// let r = parseInt(hexColor.substring(0, 2), 16); +// let g = parseInt(hexColor.substring(2, 4), 16); +// let b = parseInt(hexColor.substring(4, 6), 16); - // const r = parseInt(hexColor.slice(1, 3), 16); - // const g = parseInt(hexColor.slice(3, 5), 16); - // const b = parseInt(hexColor.slice(5, 7), 16); - const [r, g, b] = hexColor.match(/\w\w/g).map(x => parseInt(x, 16)); +// r = Math.floor(r * (100 + percent) / 100); +// g = Math.floor(g * (100 + percent) / 100); +// b = Math.floor(b * (100 + percent) / 100); - return `rgba(${r},${g},${b},${opacity})`; -}; +// r = (r < 255) ? r : 255; +// g = (g < 255) ? g : 255; +// b = (b < 255) ? b : 255; -const lightenColor = (hexColor, percent) => { - let r = parseInt(hexColor.substring(0, 2), 16); - let g = parseInt(hexColor.substring(2, 4), 16); - let b = parseInt(hexColor.substring(4, 6), 16); +// const lighterHex = ((r << 16) | (g << 8) | b).toString(16); - r = Math.floor(r * (100 + percent) / 100); - g = Math.floor(g * (100 + percent) / 100); - b = Math.floor(b * (100 + percent) / 100); +// return lighterHex.padStart(6, '0'); +// }; - r = (r < 255) ? r : 255; - g = (g < 255) ? g : 255; - b = (b < 255) ? b : 255; +const generateAvatarUrl = (name: string): string => { + // const originalColor = getColorFromName(name); + // const backgroundColor = lightenColor(originalColor, 60); + // const textColor = darkenColor(originalColor); - const lighterHex = ((r << 16) | (g << 8) | b).toString(16); - - return lighterHex.padStart(6, '0'); -}; - -// backgroundColor = '7F9CF5', -const generateAvatarUrl = (name) => { - const initials = name - .split(' ') - .map((part) => part.charAt(0).toUpperCase()) - .join(''); - - const originalColor = getRandomColor(); - - const backgroundColor = lightenColor(originalColor, 60); // Lighten by 20% - - const textColor = darkenColor(originalColor); - - // const avatarUrl = `https://ui-avatars.com/api/?name=${initials}&size=50&background=${backgroundColor}&color=${textColor}`; - const avatarUrl = `/api/avatar?name=${name}&size=50&background=${backgroundColor}&textColor=${textColor}`; + const avatarUrl = `/api/avatar?name=${name}&size=50`; return avatarUrl; }; </script> diff --git a/resources/js/Components/UserAvatarCurrentUser.vue b/resources/js/Components/UserAvatarCurrentUser.vue index 3669138..a065078 100644 --- a/resources/js/Components/UserAvatarCurrentUser.vue +++ b/resources/js/Components/UserAvatarCurrentUser.vue @@ -1,12 +1,29 @@ <script setup> -import { computed } from 'vue'; -// import { usePage } from '@inertiajs/vue3' -import { usePage } from '@inertiajs/vue3'; +// import { computed } from 'vue'; +// import { usePage } from '@inertiajs/vue3'; import UserAvatar from '@/Components/UserAvatar.vue'; -const userName = computed(() => usePage().props.auth?.user.name); +defineProps({ + user: { + type: Object, + required: true, + }, +}); </script> <template> - <UserAvatar v-bind:username="'userName'" api="initials" /> + + <div v-if="user.avatar"> + <UserAvatar :default-url="user.avatar ? '/public' + user.avatar : ''" + :username="user.first_name + ' ' + user.last_name" class="w-24 h-24 mx-auto lg:w-6 lg:h-6" /> + </div> + + + <div v-else> + <UserAvatar :username="user.first_name + ' ' + user.last_name" class="w-24 h-24 mx-auto lg:w-6 lg:h-6" /> + </div> </template> +<!-- <template v-else> + <UserAvatar :username="user.first_name + ' ' + user.last_name" class="w-24 h-24 mx-auto lg:w-6 lg:h-6" /> + +</template> --> diff --git a/resources/js/Components/action-message.vue b/resources/js/Components/action-message.vue new file mode 100644 index 0000000..34cd4da --- /dev/null +++ b/resources/js/Components/action-message.vue @@ -0,0 +1,30 @@ +<script lang="ts" setup> +import { computed } from 'vue'; +import { colorsBgLight, colorsOutline } from '@/colors'; + +const props = defineProps({ + on: Boolean, + icon: { + type: String, + default: null, + }, + outline: Boolean, + color: { + type: String, + required: false, + default: 'info', + }, +}); +const componentClass = computed(() => (props.outline ? colorsOutline[props.color] : colorsBgLight[props.color])); +</script> + +<template> + <div > + <transition leave-active-class="transition ease-in duration-1000" leave-from-class="opacity-100" + leave-to-class="opacity-0" :class="componentClass" class="px-3 py-2 last:mb-0 border rounded transition-colors duration-150 text-sm text-gray-600"> + <div v-show="on"> + <slot /> + </div> + </transition> + </div> +</template> \ No newline at end of file diff --git a/resources/js/Components/avatar-input.vue b/resources/js/Components/avatar-input.vue new file mode 100644 index 0000000..fcfa14d --- /dev/null +++ b/resources/js/Components/avatar-input.vue @@ -0,0 +1,79 @@ +<template> + <div class="relative inline-block overflow-hidden rounded-full"> + <input type="file" ref="avatarInput" @change="onChangeFile" class="hidden" accept="image/*"> + + <img :src="avatarUrl" alt="Avatar" class="h-full w-full object-cover"> + <div class="absolute top-0 h-full w-full bg-black bg-opacity-25 flex items-center justify-center"> + <button @click.prevent="browse" + class="rounded-full hover:bg-white hover:bg-opacity-25 p-2 focus:outline-none text-white transition-colors duration-300"> + <IconRounded :icon="mdiCameraEnhanceOutline" class="bg-transparent h-6 w-6" /> + </button> + <button v-if="file" @click.prevent="reset" + class="rounded-full hover:bg-white hover:bg-opacity-25 p-2 focus:outline-none text-white transition-colors duration-300"> + <IconRounded :icon="mdiAlphaXCircleOutline " class="bg-transparent h-6 w-6" /> + </button> + </div> + </div> +</template> + +<script setup lang="ts"> +import { ref, defineProps, defineEmits } from 'vue'; +import { mdiCameraEnhanceOutline, mdiAlphaXCircleOutline } from '@mdi/js'; +import IconRounded from './IconRounded.vue'; + +const props = defineProps({ + + modelValue: File, + defaultSrc: { + type: String, + required: true, + }, +}); + +// vue data properties +const file = ref<File | null>(null); +const avatarUrl = ref<string>(props.defaultSrc); +const avatarInput = ref<HTMLInputElement | null>(null); + + +// const avatarUrl = computed({ +// get: () => props.modelValue ? props.modelValue : props.defaultSrc, +// set: (value: string) => { +// emit('update:modelValue', value); +// }, +// }); + + +const emit = defineEmits<{ (e: 'update:modelValue', file: File | null): void; (e: 'input', file: File | null): void }>(); + +// const avatarInput = ref<HTMLInputElement | null>(null); + +const browse = () => { + avatarInput.value?.click(); +}; + +const reset = () => { + file.value = null; + avatarUrl.value = props.defaultSrc; + emit('input', file.value); +}; + +const onChangeFile = (e: Event) => { + // const target = (<HTMLInputElement>e.target) + const target = e.target as HTMLInputElement; + if (target.files && target.files[0]) { + file.value = target.files[0]; + } + if (file.value) { + emit('input', file.value); + emit('update:modelValue', file.value); + let reader = new FileReader(); + reader.readAsDataURL(file.value); + reader.onload = (e) => { + avatarUrl.value = e.target?.result as string; + }; + + } + +}; +</script> diff --git a/resources/js/Components/unused/UserCard.vue b/resources/js/Components/unused/UserCard.vue index 373300d..3d9098d 100644 --- a/resources/js/Components/unused/UserCard.vue +++ b/resources/js/Components/unused/UserCard.vue @@ -1,43 +1,42 @@ -<script setup> -import { computed, ref } from 'vue' -import { MainService } from '@/Stores/main' -import { mdiCheckDecagram } from '@mdi/js' -import BaseLevel from '@/Components/BaseLevel.vue' -import UserAvatarCurrentUser from '@/Components/UserAvatarCurrentUser.vue' -import CardBox from '@/Components/CardBox.vue' -import FormCheckRadioGroup from '@/Components/FormCheckRadioGroup.vue' -import PillTag from '@/Components/PillTag.vue' +<script lang="ts" setup> +import { computed, ref } from 'vue'; +import { MainService } from '@/Stores/main'; +import { mdiCheckDecagram } from '@mdi/js'; +import BaseLevel from '@/Components/BaseLevel.vue'; +import UserAvatarCurrentUser from '@/Components/UserAvatarCurrentUser.vue'; +import CardBox from '@/Components/CardBox.vue'; +import FormCheckRadioGroup from '@/Components/FormCheckRadioGroup.vue'; +import PillTag from '@/Components/PillTag.vue'; +import { usePage } from '@inertiajs/vue3'; +import type { User } from '@/Dataset'; +import type { ComputedRef } from 'vue'; -const mainService = MainService() +const mainService = MainService(); -const userName = computed(() => mainService.userName) +const userName = computed(() => mainService.userName); -const userSwitchVal = ref([]) +const user: ComputedRef<User> = computed(() => { + return usePage().props.authUser as User; +}); + +const userSwitchVal = ref([]); </script> <template> <CardBox> <BaseLevel type="justify-around lg:justify-center"> - <UserAvatarCurrentUser class="lg:mx-12" /> + <UserAvatarCurrentUser :user="user" class="lg:mx-12" /> <div class="space-y-3 text-center md:text-left lg:mx-12"> <div class="flex justify-center md:block"> - <FormCheckRadioGroup - v-model="userSwitchVal" - name="sample-switch" - type="switch" - :options="{ one: 'Notifications' }" - /> + <FormCheckRadioGroup v-model="userSwitchVal" name="sample-switch" type="switch" + :options="{ one: 'Notifications' }" /> </div> <h1 class="text-2xl"> Howdy, <b>{{ userName }}</b>! </h1> <p>Last login <b>12 mins ago</b> from <b>127.0.0.1</b></p> <div class="flex justify-center md:block"> - <PillTag - text="Verified" - type="info" - :icon="mdiCheckDecagram" - /> + <PillTag text="Verified" type="info" :icon="mdiCheckDecagram" /> </div> </div> </BaseLevel> diff --git a/resources/js/Pages/Admin/License/Index.vue b/resources/js/Pages/Admin/License/Index.vue index 0d2f70e..73192dc 100644 --- a/resources/js/Pages/Admin/License/Index.vue +++ b/resources/js/Pages/Admin/License/Index.vue @@ -1,5 +1,5 @@ <script lang="ts" setup> -import { Head, usePage } from '@inertiajs/vue3'; +import { usePage } from '@inertiajs/vue3'; import { mdiAccountKey, mdiSquareEditOutline, mdiAlertBoxOutline } from '@mdi/js'; import { computed, ComputedRef } from 'vue'; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; diff --git a/resources/js/Pages/Admin/Mimetype/Create.vue b/resources/js/Pages/Admin/Mimetype/Create.vue index 60ce1a9..86592c2 100644 --- a/resources/js/Pages/Admin/Mimetype/Create.vue +++ b/resources/js/Pages/Admin/Mimetype/Create.vue @@ -1,7 +1,7 @@ <script lang="ts" setup> -import { ref, watch, computed, Ref, reactive } from 'vue'; +import { ref, reactive } from 'vue'; import { Head, useForm } from '@inertiajs/vue3'; -import { mdiAccountKey, mdiArrowLeftBoldOutline } from '@mdi/js'; +import { mdiAccountKey, mdiArrowLeftBoldOutline, mdiTrashCan, mdiImageText, mdiPlus } from '@mdi/js'; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; import SectionMain from '@/Components/SectionMain.vue'; import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; @@ -10,115 +10,97 @@ import BaseDivider from '@/Components/BaseDivider.vue'; import BaseButton from '@/Components/BaseButton.vue'; import BaseButtons from '@/Components/BaseButtons.vue'; import { stardust } from '@eidellev/adonis-stardust/client'; -import mime from 'mime'; import FormField from '@/Components/FormField.vue'; import FormControl from '@/Components/FormControl.vue'; import standardTypes from 'mime/types/standard.js'; import otherTypes from 'mime/types/other.js'; import FormCheckRadioGroup from '@/Components/FormCheckRadioGroup.vue'; +import MimetypeInput from '@/Components/MimetypeInput.vue'; -const props = defineProps({ - permissions: { - type: Object, - default: () => ({}), - }, +defineProps({ borderless: Boolean, transparent: Boolean, ctrlKFocus: Boolean, }); +const customTypes: { [key: string]: string[] } = { + 'application/vnd.opengeospatial.geopackage+sqlite3': ['gpkg'], + 'text/plain': ['txt', 'asc', 'c', 'cc', 'h', 'srt'], +}; + const isReadOnly: boolean = true; -const standardMimeTypes = Object.keys(standardTypes); -const otherMimeTypes = Object.keys(otherTypes); -const mimeTypes = [...standardMimeTypes, ...otherMimeTypes]; +// const standardMimeTypes = Object.keys(standardTypes); +// const otherMimeTypes = Object.keys(otherTypes); +// const customMimeTypes = Object.keys(customTypes); +const mimeTypesMap = new Map<string, string[]>(); + +Object.entries(standardTypes).forEach(([mimeType, extensions]) => { + mimeTypesMap.set(mimeType, extensions); +}); + +Object.entries(otherTypes).forEach(([mimeType, extensions]) => { + mimeTypesMap.set(mimeType, extensions); +}); + +Object.entries(customTypes).forEach(([mimeType, extensions]) => { + mimeTypesMap.set(mimeType, extensions); +}); + +const mimeTypes = Array.from(mimeTypesMap.keys()); const file_extensions = reactive<Record<string, string>>({}); interface FormData { name: string; file_extension: string[]; + alternate_mimetype: string[]; enabled: boolean; + [key: string]: string | string[] | boolean; } const form = useForm<FormData>({ name: '', file_extension: [], + alternate_mimetype: [], enabled: true, }); -const filteredMimetypes = ref<string[]>([]); // Stores the filtered MIME types for the dropdown -const showDropdown = ref(false); // Controls the visibility of the autocomplete dropdown -const selectedIndex: Ref<number> = ref(0); // Track selected MIME type in the dropdown -const ul: Ref<HTMLLIElement[] | null> = ref<HTMLLIElement[]>([]); -const newExtension: Ref = ref(''); //reactive([] as Array<string>); const mimetypeError = ref<string | null>(null); -watch(selectedIndex, (selectedIndex: number) => { - if (selectedIndex != null && ul.value != null) { - const currentElement: HTMLLIElement = ul.value[selectedIndex]; - currentElement && - currentElement?.scrollIntoView({ - behavior: 'smooth', - block: 'nearest', - inline: 'start', - }); - } -}); +const addAlternateMimetype = () => { + form.alternate_mimetype.push(""); +}; +const removeAliasMimetype = (index: number) => { + form.alternate_mimetype.splice(index, 1); +}; -// Function to reset the object function resetFileExtensions() { - // Reset to an empty object Object.keys(file_extensions).forEach(key => { delete file_extensions[key]; }); } -const inputElClass = computed(() => { - const base = [ - 'block p-2.5 w-full z-20 text-sm text-gray-900 bg-gray-50 rounded-r-lg', - 'dark:placeholder-gray-400 dark:text-white dark:focus:border-blue-500', - 'h-12', - props.borderless ? 'border-0' : 'border', - props.transparent ? 'bg-transparent' : 'bg-white dark:bg-slate-800', - ]; - // if (props.icon) { - base.push('pl-10'); - // } - return base; -}); - -// Check if the MIME type is valid const isValidMimeType = (mimeType: string): boolean => { - let extensions = mime.getExtension(mimeType) + let extensions = mimeTypesMap.get(mimeType) || null; return extensions !== null; }; -async function handleInputChange(e: Event) { - const target = <HTMLInputElement>e.target; - newExtension.value = target.value; +const clearInput = () => { + // newExtension.value = ''; + // showDropdown.value = false; + form.name = ''; + resetFileExtensions(); +}; - if (newExtension.value.length >= 2) { - showDropdown.value = true; - filteredMimetypes.value = mimeTypes.filter(mimeType => - mimeType.toLowerCase().includes(newExtension.value.toLowerCase()) - ); - } else { - // data.results = []; - showDropdown.value = false; - } -} - -// Handle MIME type selection from the dropdown const selectResult = (mimeType: string) => { form.name = mimeType; - // file_extensions.values = []; resetFileExtensions(); - showDropdown.value = false; - newExtension.value = ''; // Reset the input - selectedIndex.value = -1; + // showDropdown.value = false; + // newExtension.value = ''; + // selectedIndex.value = -1; if (form.name && isValidMimeType(form.name)) { - const extensions = mime.getAllExtensions(form.name) as Set<string>; - // Iterate over each extension and set both key and value to the extension - Array.from(extensions).forEach(extension => { + // const extensions = mime.getAllExtensions(form.name) as Set<string>; + const extensions = mimeTypesMap.get(mimeType); + extensions?.forEach(extension => { file_extensions[extension] = extension; }); } else { @@ -126,61 +108,14 @@ const selectResult = (mimeType: string) => { } }; -function onArrowDown() { - if (filteredMimetypes.value.length > 0) { - selectedIndex.value = selectedIndex.value === filteredMimetypes.value.length - 1 ? 0 : selectedIndex.value + 1; - // const currentElement: HTMLLIElement = ul.value[selectedIndex.value]; - } -} - -function onArrowUp() { - if (filteredMimetypes.value.length > 0) { - selectedIndex.value = selectedIndex.value == 0 || selectedIndex.value == -1 ? filteredMimetypes.value.length - 1 : selectedIndex.value - 1; - } -} - -function onEnter() { - if (Array.isArray(filteredMimetypes.value) && filteredMimetypes.value.length && selectedIndex.value !== -1 && selectedIndex.value < filteredMimetypes.value.length) { - const mimeType = filteredMimetypes.value[selectedIndex.value]; - // this.$emit('person', person); - form.name = mimeType; - // reset form file extensions - // file_extensions.values = []; - resetFileExtensions(); - showDropdown.value = false; - newExtension.value = ''; // Reset the input - selectedIndex.value = -1; - if (form.name) { - // clear all loaded file extensions - // file_extensions.values = []; - resetFileExtensions(); - if (isValidMimeType(form.name)) { - let extensions = mime.getAllExtensions(form.name) as Set<string>; - // Convert the Set to an array of objects - // Convert the Set to an object - Array.from(extensions).forEach(extension => { - file_extensions[extension] = extension; - }); - - // file_extensions.push(...formattedExtensions); - } else { - mimetypeError.value = 'Invalid MIME type.'; - } - } - } -} - -// Handle form submission const submit = async () => { if (isValidForm()) { await form.post(stardust.route('settings.mimetype.store'), { preserveScroll: true, }); - } }; -// Form validation before submission const isValidForm = (): boolean => { if (!form.name) { form.errors.name = 'Name is required.'; @@ -190,6 +125,7 @@ const isValidForm = (): boolean => { } if (!form.file_extension.length) { form.errors.file_extension = 'At least one file extension is required.'; + return false; } return true; @@ -205,59 +141,15 @@ const isValidForm = (): boolean => { <BaseButton :route-name="stardust.route('settings.mimetype.index')" :icon="mdiArrowLeftBoldOutline" label="Back" color="white" rounded-full small /> </SectionTitleLineWithButton> - <!-- <CardBox form @submit.prevent="form.post(stardust.route('role.store'))"> --> - <CardBox form @submit.prevent="submit()"> - - <!-- MIME Type Input Field with Autocomplete --> - <div class="relative mb-4"> - - <input v-model="newExtension" type="text" placeholder="Enter Mimetype Name" - class="block w-full border border-gray-300 rounded-lg px-4 py-2 focus:ring-blue-500 focus:border-blue-500" - :class="inputElClass" @input="handleInputChange" @keydown.down="onArrowDown" - @keydown.up="onArrowUp" @keydown.prevent.enter="onEnter"> - </input> - <svg class="w-4 h-4 absolute left-2.5 top-3.5" v-show="newExtension.length < 2" - xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor"> - <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" - d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z" /> - </svg> - <svg class="w-4 h-4 absolute left-2.5 top-3.5" v-show="newExtension.length >= 2" - xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke="currentColor" @click="() => { - newExtension = ''; - showDropdown = false; - form.name = ''; - resetFileExtensions(); - } - "> - <path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12" /> - </svg> - - <ul v-if="showDropdown && filteredMimetypes.length" - class="bg-white dark:bg-slate-800 w-full mt-2 max-h-28 overflow-y-auto scroll-smooth"> - <li v-for="(mimeType, index) in filteredMimetypes" :key="index" @click="selectResult(mimeType)" - class="pl-8 pr-2 py-1 border-b-2 border-gray-100 relative cursor-pointer hover:bg-blue-500 hover:text-white" - :class="{ - 'bg-blue-500 text-white': selectedIndex === index, - 'bg-white text-gray-900': selectedIndex !== index - }" :ref="(el) => { - if (ul) { - ul[index] = el as HTMLLIElement; - } - }"> - <svg class="absolute w-4 h-4 left-2 top-2" xmlns="http://www.w3.org/2000/svg" - viewBox="0 0 20 20" fill="currentColor"> - <path fill-rule="evenodd" - d="M12.293 5.293a1 1 0 011.414 0l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414-1.414L14.586 11H3a1 1 0 110-2h11.586l-2.293-2.293a1 1 0 010-1.414z" - clip-rule="evenodd" /> - </svg> - <span>{{ mimeType }}</span> - </li> - </ul> - </div> + <CardBox form> + <MimetypeInput @on-select-result="selectResult" @on-clear-input="clearInput" :transparent="transparent" + :borderless="borderless" :mimeTypes="mimeTypes" :isValidMimeType="isValidMimeType" /> <div v-if="mimetypeError" class="text-red-400 text-sm mt-1"> {{ mimetypeError }} </div> + <BaseDivider v-if="form.name" /> + <FormField v-if="form.name" label="Mimetype Name" :class="{ 'text-red-400': form.errors.name }"> <FormControl v-model="form.name" name="display_name" :error="form.errors.name" :is-read-only=isReadOnly> @@ -266,34 +158,79 @@ const isValidForm = (): boolean => { </div> </FormControl> </FormField> - - - <FormField v-if="form.name" help="Activate mimetype immediately?" wrap-body + <!-- <FormField v-if="form.name" help="Activate mimetype immediately?" wrap-body class="mt-8 w-full mx-2 flex-1 flex-col"> <label for="rights" class="checkbox mr-6 mb-3 last:mr-0"> <input type="checkbox" id="rights" required v-model="form.enabled" /> <span class="check" /> <a class="pl-2" target="_blank">Enable mimetype immediately </a> - </label> - </FormField> - - <FormField label="Extensions" wrap-body> + </FormField> --> + <FormField v-if="Object.keys(file_extensions).length > 0" label="File Extensions" wrap-body> + <!-- <div class="flex items-center mt-2"> + <FormControl v-model="newExtension" placeholder="Enter file extension" class="mr-2" /> + <BaseButton color="info" @click="addFileExtension" label="Add" /> + </div> --> <FormCheckRadioGroup v-model="form.file_extension" :options="file_extensions" name="file_extensions" - is-column /> + is-column allow-manual-adding manual-adding-placeholder="Enter additional file extension manually" /> + </FormField> - <div class="text-red-400 text-sm" - v-if="form.errors.file_extension && Array.isArray(form.errors.file_extension)"> - {{ form.errors.file_extension.join(', ') }} + <div class="text-red-400 text-sm" v-if="form.errors.file_extension"> + {{ form.errors.file_extension }} </div> - <BaseDivider /> + <BaseDivider v-if="Object.keys(file_extensions).length > 0" /> + <CardBox v-if="form.name" class="mb-6 shadow" has-table :icon="mdiImageText" title="Alternate Mimetypes" + :header-icon="mdiPlus" @header-icon-click="addAlternateMimetype"> + <div v-if="form.alternate_mimetype.length === 0" class="text-center py-4"> + <p class="text-gray-600">No alternate mimetypes added yet.</p> + <p class="text-gray-400"> + Click the plus icon above to add a new alternate mimetype. + <br> + An alternate mimetype is needed to ensure compatibility across different systems and + software. + For example, the GeoPackage standard mimetype is + 'application/vnd.opengeospatial.geopackage+sqlite3', but most software stores it as + 'application/x-sqlite3'. Therefore, 'application/x-sqlite3' must be added as an alternate + mimetype. + </p> + </div> + <table class="table-fixed border-green-900" v-if="form.alternate_mimetype.length > 0"> + <thead> + <tr> + <th class="w-10/12">Alias Mimetype</th> + <th class="w-2/12"></th> + </tr> + </thead> + <tbody> + <tr v-for="(alias, index) in form.alternate_mimetype" :key="index"> + <td> + <FormControl required v-model="form.alternate_mimetype[index]" + placeholder="[alternate mimetype]"> + <div class="text-red-400 text-sm" + v-if="Array.isArray(form.errors[`alternate_mimetype.${index}`])"> + {{ form.errors[`alternate_mimetype.${index}`]?.join(', ') }} + </div> + </FormControl> + </td> + <td class="before:hidden lg:w-1 whitespace-nowrap"> + <BaseButton color="danger" :icon="mdiTrashCan" small + @click.prevent="removeAliasMimetype(index)" /> + </td> + </tr> + </tbody> + </table> + <div class="text-red-400 text-sm" + v-if="form.errors.alternate_mimetype && Array.isArray(form.errors.alternate_mimetype)"> + {{ form.errors.alternate_mimetype.join(', ') }} + </div> + </CardBox> <template #footer> <BaseButtons> <BaseButton type="submit" color="info" label="Create" :class="{ 'opacity-25': form.processing }" - :disabled="form.processing" /> + :disabled="form.processing" @click.prevent="submit()" /> </BaseButtons> </template> </CardBox> diff --git a/resources/js/Pages/Admin/Mimetype/Delete.vue b/resources/js/Pages/Admin/Mimetype/Delete.vue index 833fac9..0b6f158 100644 --- a/resources/js/Pages/Admin/Mimetype/Delete.vue +++ b/resources/js/Pages/Admin/Mimetype/Delete.vue @@ -32,7 +32,7 @@ const form = useForm({ // isPreferationRequired: false, }); -const handleSubmit = async (e) => { +const handleSubmit = async (e: Event) => { e.preventDefault(); await form.delete(stardust.route('settings.mimetype.deleteStore', [props.mimetype.id])); }; diff --git a/resources/js/Pages/Admin/Mimetype/Index.vue b/resources/js/Pages/Admin/Mimetype/Index.vue index 6a50ce5..aaf0aa0 100644 --- a/resources/js/Pages/Admin/Mimetype/Index.vue +++ b/resources/js/Pages/Admin/Mimetype/Index.vue @@ -1,6 +1,6 @@ <script lang="ts" setup> import { Head, usePage } from '@inertiajs/vue3'; -import { mdiAccountKey, mdiSquareEditOutline, mdiAlertBoxOutline, mdiPlus, mdiTrashCan } from '@mdi/js'; +import { mdiAccountKey, mdiSquareEditOutline, mdiAlertBoxOutline, mdiPlus, mdiTrashCan, mdiCheckCircle, mdiCloseCircle } from '@mdi/js'; import { computed, ComputedRef } from 'vue'; import type { PropType } from "vue"; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; @@ -16,6 +16,7 @@ interface MimeType { id: number; name: string; file_extension: string; + alternate_mimetype: string; enabled: boolean; } @@ -54,24 +55,41 @@ const flash: ComputedRef<any> = computed(() => { {{ flash.message }} </NotificationBar> <CardBox class="mb-6" has-table> - <table> + <table class="min-w-full divide-y divide-gray-200"> <thead> <tr> - <th>Name</th> - <th>Status</th> - <th v-if="can.edit">Actions</th> + <th class="px-4 py-2 text-left text-sm font-medium uppercase tracking-wider">Mimetype</th> + <th class="px-4 py-2 text-left text-sm font-medium uppercase tracking-wider">Alternate Mime Types</th> + <th v-if="can.edit" class="px-4 py-2 text-left text-sm font-medium uppercase tracking-wider">Actions</th> </tr> </thead> - <tbody> + <tbody class="divide-y divide-gray-200"> <tr v-for="mimetype in mimetypes" :key="mimetype.id"> - <td data-label="Name"> - {{ mimetype.name }} ({{ mimetype.file_extension }}) + <td class="px-4 py-2 whitespace-nowrap text-left text-sm" data-label="Name"> + <span class="flex items-center"> + <svg viewBox="0 0 24 24" v-if="mimetype.enabled" :class="{'text-green-500': mimetype.enabled}" class="w-4 h-4 mr-2"> + <path fill="currentColor" :d="mdiCheckCircle" /> + </svg> + <svg v-else viewBox="0 0 24 24" :class="{'text-red-500': !mimetype.enabled}" class="w-4 h-4 mr-2"> + <path fill="currentColor" :d="mdiCloseCircle" /> + </svg> + <br> + <span class="truncate block max-w-xs">{{ mimetype.name }}</span> + </span> + <ul class="list-none pl-0"> + <li v-for="ext in mimetype.file_extension?.split('|')" :key="ext" class="flex items-center truncate block max-w-xs">- .{{ ext }}</li> + </ul> </td> - <td data-label="Status"> + <td class="px-4 py-2 whitespace-nowrap text-left text-sm" data-label="Alternate Mime Types"> + <ul class="list-none pl-0"> + <li v-for="alt in mimetype.alternate_mimetype?.split('|')" :key="alt" class="flex items-center truncate block max-w-xs">- {{ alt }}</li> + </ul> + </td> + <!-- <td class="px-4 py-2 whitespace-nowrap" data-label="Status"> <template v-if="mimetype.enabled">Active</template> <template v-else>Inactive</template> - </td> - <td v-if="can.edit" class="before:hidden lg:w-1 whitespace-nowrap"> + </td> --> + <td v-if="can.edit" class="before:hidden lg:w-1 whitespace-nowrap px-4 py-2 whitespace-nowrap text-left text-sm"> <BaseButtons type="justify-start lg:justify-end" no-wrap> <BaseButton v-if="mimetype.enabled" :route-name="stardust.route('settings.mimetype.down', [mimetype.id])" diff --git a/resources/js/Pages/Admin/Role/Create.vue b/resources/js/Pages/Admin/Role/Create.vue index 513d06c..e0820c5 100644 --- a/resources/js/Pages/Admin/Role/Create.vue +++ b/resources/js/Pages/Admin/Role/Create.vue @@ -28,7 +28,7 @@ const form = useForm({ }); const submit = async () => { - await form.post(stardust.route('settings.role.store'), form); + await form.post(stardust.route('settings.role.store')); }; </script> diff --git a/resources/js/Pages/Admin/Role/Edit.vue b/resources/js/Pages/Admin/Role/Edit.vue index 29e88d4..495d55b 100644 --- a/resources/js/Pages/Admin/Role/Edit.vue +++ b/resources/js/Pages/Admin/Role/Edit.vue @@ -1,5 +1,5 @@ -<script setup> -import { Head, Link, useForm } from '@inertiajs/vue3'; +<script setup lang="ts"> +import { Head, useForm } from '@inertiajs/vue3'; import { mdiAccountKey, mdiArrowLeftBoldOutline, mdiFormTextarea } from '@mdi/js'; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; import SectionMain from '@/Components/SectionMain.vue'; @@ -29,15 +29,15 @@ const props = defineProps({ }); const form = useForm({ - _method: 'put', name: props.role.name, description: props.role.description, permissions: props.roleHasPermissions, }); const submit = async () => { - // await Inertia.post(stardust.route('user.store'), form); - await form.put(stardust.route('settings.role.update', [props.role.id]), form); + // await Inertia.post(stardust.route('user.store'), form); old + await form.put(stardust.route('settings.role.update', [props.role.id])); + // await router.put(stardust.route('settings.role.update', [props.role.id]), form); }; </script> @@ -58,7 +58,7 @@ const submit = async () => { <!-- <CardBox form @submit.prevent="form.put(stardust.route('role.update', [props.role.id]))"> --> <CardBox form @submit.prevent="submit()"> <FormField label="Name" :class="{ 'text-red-400': form.errors.name }"> - <FormControl v-model="form.name" type="text" placeholder="Enter Name" required :error="form.errors.name" is-read-only="true"> + <FormControl v-model="form.name" type="text" placeholder="Enter Name" required :error="form.errors.name" :is-read-only=true> <div class="text-red-400 text-sm" v-if="form.errors.name"> {{ form.errors.name }} </div> diff --git a/resources/js/Pages/Admin/User/Create.vue b/resources/js/Pages/Admin/User/Create.vue index ae29e84..17e525b 100644 --- a/resources/js/Pages/Admin/User/Create.vue +++ b/resources/js/Pages/Admin/User/Create.vue @@ -41,13 +41,28 @@ const form = useForm({ first_name: '', last_name: '', email: '', - password: '', + new_password: '', password_confirmation: '', roles: [], }); const submit = async () => { - await router.post(stardust.route('settings.user.store'), form); + // await router.post(stardust.route('settings.user.store'), form); + await form.post(stardust.route('settings.user.store'), { + preserveScroll: true, + onSuccess: () => { + form.reset(); + }, + onError: () => { + if (form.errors.new_password) { + form.reset('new_password'); + enabled.value = false; + // newPasswordInput.value.focus(); + // newPasswordInput.value?.focus(); + } + + }, + }); }; </script> @@ -109,7 +124,7 @@ const submit = async () => { </FormControl> </FormField> <password-meter :password="form.password" @score="handleScore" /> --> - <PasswordMeter v-model:password="form.password" :errors="form.errors" @score="handleScore" /> + <PasswordMeter v-model="form.new_password" :errors="form.errors" @score="handleScore" /> <FormField label="Password Confirmation" :class="{ 'text-red-400': errors.password_confirmation }"> <FormControl diff --git a/resources/js/Pages/Admin/User/Edit.vue b/resources/js/Pages/Admin/User/Edit.vue index 288af1c..ee7078a 100644 --- a/resources/js/Pages/Admin/User/Edit.vue +++ b/resources/js/Pages/Admin/User/Edit.vue @@ -42,14 +42,29 @@ const form = useForm({ first_name: props.user.first_name, last_name: props.user.last_name, email: props.user.email, - password: '', + new_password: '', password_confirmation: '', roles: props.userHasRoles, // fill actual user roles from db }); const submit = async () => { // await Inertia.post(stardust.route('user.store'), form); - await router.put(stardust.route('settings.user.update', [props.user.id]), form); + // await router.put(stardust.route('settings.user.update', [props.user.id]), form); + await form.put(stardust.route('settings.user.update', [props.user.id]), { + preserveScroll: true, + onSuccess: () => { + form.reset(); + }, + onError: () => { + if (form.errors.new_password) { + form.reset('new_password'); + enabled.value = false; + // newPasswordInput.value.focus(); + // newPasswordInput.value?.focus(); + } + + }, + }); }; const handleScore = (score: number) => { if (score >= 4){ @@ -108,15 +123,16 @@ const handleScore = (score: number) => { </FormControl> </FormField> - <FormField label="Password" :class="{ 'text-red-400': errors.password }"> + <!-- <FormField label="Password" :class="{ 'text-red-400': errors.password }"> <FormControl v-model="form.password" type="password" placeholder="Enter Password" :errors="errors.password"> <div class="text-red-400 text-sm" v-if="errors.password"> {{ errors.password }} </div> </FormControl> - </FormField> + </FormField> --> + + <PasswordMeter field-label="Reset User Password" :show-required-message="false" ref="newPasswordInput" v-model="form.new_password" :errors="form.errors" @score="handleScore" /> - <PasswordMeter v-model:password="form.password" :errors="form.errors" @score="handleScore" /> <FormField label="Password Confirmation" :class="{ 'text-red-400': errors.password_confirmation }"> <FormControl @@ -151,7 +167,7 @@ const handleScore = (score: number) => { color="info" label="Submit" :class="{ 'opacity-25': form.processing }" - :disabled="form.processing == true|| (form.password != '' && enabled == false)" + :disabled="form.processing == true|| (form.new_password != '' && enabled == false)" /> </BaseButtons> </template> diff --git a/resources/js/Pages/Auth/AccountInfo.vue b/resources/js/Pages/Auth/AccountInfo.vue index 9647a25..d5f698d 100644 --- a/resources/js/Pages/Auth/AccountInfo.vue +++ b/resources/js/Pages/Auth/AccountInfo.vue @@ -1,7 +1,7 @@ <script setup lang="ts"> // import { Head, Link, useForm } from '@inertiajs/inertia-vue3'; import { ref } from 'vue'; -import { useForm } from '@inertiajs/vue3'; +import { useForm, Head } from '@inertiajs/vue3'; // import { ref } from 'vue'; // import { reactive } from 'vue'; import { @@ -12,7 +12,7 @@ import { mdiAsterisk, mdiFormTextboxPassword, mdiArrowLeftBoldOutline, - mdiAlertBoxOutline, + mdiAlertBoxOutline, } from '@mdi/js'; import SectionMain from '@/Components/SectionMain.vue'; import CardBox from '@/Components/CardBox.vue'; @@ -36,16 +36,16 @@ import PasswordMeter from '@/Components/SimplePasswordMeter/password-meter.vue'; const emit = defineEmits(['confirm', 'update:confirmation']) -const enabled = ref(false); -const handleScore = (score: number) => { - if (score >= 4){ - enabled.value = true; - } else { - enabled.value = false; - } -// strengthLabel.value = scoreLabel; -// score.value = scoreValue; -}; +// const enabled = ref(false); +// const handleScore = (score: number) => { +// if (score >= 4) { +// enabled.value = true; +// } else { +// enabled.value = false; +// } +// // strengthLabel.value = scoreLabel; +// // score.value = scoreValue; +// }; defineProps({ // user will be returned from controller action @@ -82,20 +82,20 @@ defineProps({ // }; -const passwordForm = useForm({ - old_password: '', - new_password: '', - confirm_password: '', -}); -const passwordSubmit = async () => { - await passwordForm.post(stardust.route('account.password.store'), { - preserveScroll: true, - onSuccess: () => { - // console.log(resp); - passwordForm.reset(); - }, - }); -}; +// const passwordForm = useForm({ +// old_password: '', +// new_password: '', +// confirm_password: '', +// }); +// const passwordSubmit = async () => { +// await passwordForm.post(stardust.route('account.password.store'), { +// preserveScroll: true, +// onSuccess: () => { +// // console.log(resp); +// passwordForm.reset(); +// }, +// }); +// }; const flash: Ref<any> = computed(() => { return usePage().props.flash; @@ -126,6 +126,7 @@ const flash: Ref<any> = computed(() => { <template> <LayoutAuthenticated> + <Head title="Profile Security"></Head> <SectionMain> <SectionTitleLineWithButton :icon="mdiAccount" title="Profile" main> <BaseButton :route-name="stardust.route('dashboard')" :icon="mdiArrowLeftBoldOutline" label="Back" @@ -139,40 +140,10 @@ const flash: Ref<any> = computed(() => { {{ $page.props.flash.message }} </NotificationBar> --> - <div class="grid grid-cols-1 lg:grid-cols-2 gap-6"> - <!-- <div class="grid grid-cols-1 lg:grid-cols-1 gap-6"> --> + <div class="grid grid-cols-1 gap-6"> - <!-- password form --> - <!-- <CardBox title="Edit Profile" :icon="mdiAccountCircle" form @submit.prevent="profileForm.post(route('admin.account.info.store'))"> --> - <!-- <CardBox title="Edit Profile" :icon="mdiAccountCircle" form @submit.prevent="profileSubmit()"> - <FormField label="Login" help="Required. Your login name" :class="{ 'text-red-400': errors.login }"> - <FormControl v-model="profileForm.login" v-bind:icon="mdiAccount" name="login" required :error="errors.login"> - <div class="text-red-400 text-sm" v-if="errors.login"> - {{ errors.login }} - </div> - </FormControl> - </FormField> - <FormField label="Email" help="Required. Your e-mail" :class="{ 'text-red-400': errors.email }"> - <FormControl v-model="profileForm.email" :icon="mdiMail" type="email" name="email" required :error="errors.email"> - <div class="text-red-400 text-sm" v-if="errors.email"> - {{ errors.email }} - </div> - </FormControl> - </FormField> - - <template #footer> - <BaseButtons> - <BaseButton color="info" type="submit" label="Submit" /> - </BaseButtons> - </template> - </CardBox> --> - - <!-- password form --> - <!-- <CardBox title="Change Password" :icon="mdiLock" form @submit.prevent="passwordForm.post(route('admin.account.password.store'), { - preserveScroll: true, - onSuccess: () => passwordForm.reset(), - }) "> --> - <CardBox id="passwordForm" title="Change Password" :icon="mdiLock" form @submit.prevent="passwordSubmit()"> + <!-- <CardBox id="passwordForm" title="Change Password" :icon="mdiLock" form + @submit.prevent="passwordSubmit()"> <FormValidationErrors v-bind:errors="errors" /> <FormField label="Current password" help="Required. Your current password" @@ -186,22 +157,15 @@ const flash: Ref<any> = computed(() => { </FormField> <BaseDivider /> - <!-- <FormField label="New password" help="Required. New password" - :class="{ 'text-red-400': passwordForm.errors.new_password }"> - <FormControl v-model="passwordForm.new_password" :icon="mdiFormTextboxPassword" name="new_password" - type="password" required :error="passwordForm.errors.new_password"> - <div class="text-red-400 text-sm" v-if="passwordForm.errors.new_password"> - {{ passwordForm.errors.new_password }} - </div> - </FormControl> - </FormField> --> - <PasswordMeter v-model:password="passwordForm.new_password" :errors="passwordForm.errors" @score="handleScore" /> + <PasswordMeter v-model="passwordForm.new_password" :errors="passwordForm.errors" + @score="handleScore" /> <FormField label="Confirm password" help="Required. New password one more time" :class="{ 'text-red-400': passwordForm.errors.confirm_password }"> <FormControl v-model="passwordForm.confirm_password" :icon="mdiFormTextboxPassword" - name="confirm_password" type="password" required :error="passwordForm.errors.confirm_password"> + name="confirm_password" type="password" required + :error="passwordForm.errors.confirm_password"> <div class="text-red-400 text-sm" v-if="passwordForm.errors.confirm_password"> {{ passwordForm.errors.confirm_password }} </div> @@ -219,16 +183,17 @@ const flash: Ref<any> = computed(() => { <template #footer> <BaseButtons> - <BaseButton type="submit" color="info" label="Change password" :disabled="passwordForm.processing == true || enabled == false" /> + <BaseButton type="submit" color="info" label="Change password" + :disabled="passwordForm.processing == true || enabled == false" /> </BaseButtons> </template> - </CardBox> + </CardBox> --> - <PersonalTotpSettings :twoFactorEnabled="twoFactorEnabled" :backupState="backupState"> - </PersonalTotpSettings> - <!-- <PersonalSettings :state="backupState"/> --> + <PersonalTotpSettings :twoFactorEnabled="twoFactorEnabled" :backupState="backupState"> + </PersonalTotpSettings> + <!-- <PersonalSettings :state="backupState"/> --> <!-- <CardBox v-if="!props.twoFactorEnabled" title="Two-Factor Authentication" :icon="mdiInformation" form @submit.prevent="enableTwoFactorAuthentication()"> @@ -248,7 +213,7 @@ const flash: Ref<any> = computed(() => { </template> </CardBox> --> - + </div> </SectionMain> diff --git a/resources/js/Pages/Auth/Login.vue b/resources/js/Pages/Auth/Login.vue index b1321c4..b903c8a 100644 --- a/resources/js/Pages/Auth/Login.vue +++ b/resources/js/Pages/Auth/Login.vue @@ -6,7 +6,7 @@ <!-- <SectionFullScreen v-slot="{ cardClass }" :bg="'greenBlue'"> --> <SectionFullScreen v-slot="{ cardClass }"> <a class="text-2xl font-semibold flex justify-center items-center mb-8 lg:mb-10"> - <img src="/logo.svg" class="h-10 mr-4" alt="Windster Logo" /> + <img src="../../logo.svg" class="h-10 mr-4 dark:invert" alt="Windster Logo" /> <!-- <span class="self-center text-2xl font-bold whitespace-nowrap">Tethys</span> --> </a> diff --git a/resources/js/Pages/Auth/Register.vue b/resources/js/Pages/Auth/Register.vue index 8a0775d..49032a6 100644 --- a/resources/js/Pages/Auth/Register.vue +++ b/resources/js/Pages/Auth/Register.vue @@ -20,7 +20,7 @@ import AuthLayout from '@/Layouts/Auth.vue'; import { reactive } from 'vue'; import { useForm } from '@inertiajs/vue3'; -import { Inertia } from '@inertiajs/inertia'; +// import { Inertia } from '@inertiajs/inertia'; // import { NButton, NInput } from 'naive-ui'; // import { useForm } from '@inertiajs/inertia-vue3' import FormInput from '@/Components/FormInput.vue'; @@ -45,7 +45,7 @@ export default { }); const submit = async () => { - await Inertia.post('/app/register', form); + // await Inertia.post('/app/register', form); }; return { form, submit }; diff --git a/resources/js/Pages/Dashboard.vue b/resources/js/Pages/Dashboard.vue index d466ebc..73af81c 100644 --- a/resources/js/Pages/Dashboard.vue +++ b/resources/js/Pages/Dashboard.vue @@ -1,8 +1,7 @@ <script setup lang="ts"> -import { Head } from '@inertiajs/vue3'; -import { computed, onMounted } from 'vue'; +import { Head, usePage } from '@inertiajs/vue3'; +import { computed } from 'vue'; import { MainService } from '@/Stores/main'; -// import { Inertia } from '@inertiajs/inertia'; import { mdiAccountMultiple, mdiDatabaseOutline, @@ -10,73 +9,71 @@ import { mdiFinance, mdiMonitorCellphone, mdiReload, - mdiGithub, mdiChartPie, } from '@mdi/js'; -// import { containerMaxW } from '@/config.js'; // "xl:max-w-6xl xl:mx-auto" -// import * as chartConfig from '@/Components/Charts/chart.config.js'; import LineChart from '@/Components/Charts/LineChart.vue'; import SectionMain from '@/Components/SectionMain.vue'; import CardBoxWidget from '@/Components/CardBoxWidget.vue'; import CardBox from '@/Components/CardBox.vue'; import TableSampleClients from '@/Components/TableSampleClients.vue'; -import NotificationBar from '@/Components/NotificationBar.vue'; -import BaseButton from '@/Components/BaseButton.vue'; -import CardBoxTransaction from '@/Components/CardBoxTransaction.vue'; +// import NotificationBar from '@/Components/NotificationBar.vue'; import CardBoxClient from '@/Components/CardBoxClient.vue'; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; -import SectionBannerStarOnGitHub from '@/Components/SectionBannerStarOnGitea.vue'; +// import SectionBannerStarOnGitHub from '@/Components/SectionBannerStarOnGitea.vue'; +import CardBoxDataset from '@/Components/CardBoxDataset.vue'; +import type { User } from '@/Dataset'; const mainService = MainService() // const chartData = ref(); const fillChartData = async () => { - await mainService.fetchChartData("2022"); + await mainService.fetchChartData(); // chartData.value = chartConfig.sampleChartData(); // chartData.value = mainService.graphData; }; const chartData = computed(() => mainService.graphData); -onMounted(async () => { - await mainService.fetchChartData("2022"); -}); -; -/* Fetch sample data */ -mainService.fetch('clients'); -mainService.fetch('history'); +// onMounted(async () => { +// await mainService.fetchChartData("2022"); +// }); -mainService.fetchApi('authors'); -mainService.fetchApi('datasets'); +// mainService.fetch('clients'); +// mainService.fetch('history'); + +// mainService.fetchApi('authors'); +// mainService.fetchApi('datasets'); // const clientBarItems = computed(() => mainService.clients.slice(0, 4)); -const transactionBarItems = computed(() => mainService.history); +// const transactionBarItems = computed(() => mainService.history); -const authorBarItems = computed(() => mainService.authors.slice(0, 4)); +const authorBarItems = computed(() => mainService.authors.slice(0, 5)); const authors = computed(() => mainService.authors); const datasets = computed(() => mainService.datasets); -// const props = defineProps({ -// user: { -// type: Object, -// default: () => ({}), -// } -// }); +const datasetBarItems = computed(() => mainService.datasets.slice(0, 5)); +const submitters = computed(() => mainService.clients); +const user = computed(() => { + return usePage().props.authUser as User; +}); + +const userHasRoles = (roleNames: Array<string>): boolean => { + return user.value.roles.some(role => roleNames.includes(role.name)); +}; </script> <template> <LayoutAuthenticated :showAsideMenu="false"> <Head title="Dashboard" /> - <!-- <section class="p-6" v-bind:class="containerMaxW"> --> <SectionMain> <SectionTitleLineWithButton v-bind:icon="mdiChartTimelineVariant" title="Overview" main> - <BaseButton - href="https://gitea.geologie.ac.at/geolba/tethys" + <!-- <BaseButton + href="" target="_blank" :icon="mdiGithub" - label="Star on Gitea" + label="Star on GeoSphere Forgejo" color="contrast" rounded-full small - /> + /> --> </SectionTitleLineWithButton> <div class="grid grid-cols-1 gap-6 lg:grid-cols-3 mb-6"> @@ -87,89 +84,59 @@ const datasets = computed(() => mainService.datasets); :icon="mdiAccountMultiple" :number="authors.length" label="Authors" - /> - <CardBoxWidget - trend="193" + /> + <CardBoxWidget trend-type="info" color="text-blue-500" :icon="mdiDatabaseOutline" :number="datasets.length" label="Publications" - /> - <!-- <CardBoxWidget trend="193" trend-type="info" color="text-blue-500" :icon="mdiCartOutline" :number="datasets.length" - prefix="$" label="Publications" /> --> - <CardBoxWidget - trend="Overflow" - trend-type="alert" - color="text-red-500" + /> + <CardBoxWidget + trend-type="up" + color="text-purple-500" :icon="mdiChartTimelineVariant" - :number="256" - suffix="%" - label="Performance" + :number="submitters.length" + label="Submitters" /> </div> <div class="grid grid-cols-1 lg:grid-cols-2 gap-6 mb-6"> - <div class="flex flex-col justify-between"> + <!-- <div class="flex flex-col justify-between"> <CardBoxClient v-for="client in authorBarItems" :key="client.id" :name="client.name" :email="client.email" :date="client.created_at" - :text="client.datasetCount" + :text="client.identifier_orcid" + :count="client.dataset_count" + /> - </div> + </div> --> <div class="flex flex-col justify-between"> - <CardBoxTransaction - v-for="(transaction, index) in transactionBarItems" + <CardBoxDataset + v-for="(dataset, index) in datasetBarItems" :key="index" - :amount="transaction.amount" - :date="transaction.date" - :business="transaction.business" - :type="transaction.type" - :name="transaction.name" - :account="transaction.account" + :dataset="dataset" /> </div> </div> - <SectionBannerStarOnGitHub /> + <!-- <SectionBannerStarOnGitHub /> --> - <SectionTitleLineWithButton :icon="mdiChartPie" title="Trends overview: Publications per month" /> + <SectionTitleLineWithButton :icon="mdiChartPie" title="Trends overview: Publications per month" ></SectionTitleLineWithButton> <CardBox title="Performance" :icon="mdiFinance" :header-icon="mdiReload" class="mb-6" @header-icon-click="fillChartData"> <div v-if="chartData"> <line-chart :data="chartData" class="h-96" /> </div> </CardBox> - <SectionTitleLineWithButton :icon="mdiAccountMultiple" title="Submitters (to do)" /> - - <NotificationBar color="info" :icon="mdiMonitorCellphone"> <b>Responsive table.</b> Collapses on mobile </NotificationBar> - - <CardBox :icon="mdiMonitorCellphone" title="Responsive table" has-table> + <SectionTitleLineWithButton v-if="userHasRoles(['administrator'])" :icon="mdiAccountMultiple" title="Submitters" /> + <!-- <NotificationBar color="info" :icon="mdiMonitorCellphone"> <b>Responsive table.</b> Collapses on mobile </NotificationBar> --> + <CardBox v-if="userHasRoles(['administrator'])" :icon="mdiMonitorCellphone" title="Responsive table" has-table> <TableSampleClients /> </CardBox> - - <!-- <CardBox> - <p class="mb-3 text-gray-500 dark:text-gray-400"> - Discover the power of Tethys, the cutting-edge web backend solution that revolutionizes the way you handle research - data. At the heart of Tethys lies our meticulously developed research data repository, which leverages state-of-the-art - CI/CD techniques to deliver a seamless and efficient experience. - </p> - <p class="mb-3 text-gray-500 dark:text-gray-400"> - CI/CD, or Continuous Integration and Continuous Deployment, is a modern software development approach that ensures your - code undergoes automated testing, continuous integration, and frequent deployment. By embracing CI/CD techniques, we - ensure that every code change in our research data repository is thoroughly validated, enhancing reliability and - accelerating development cycles. - </p> - <p class="mb-3 text-gray-500 dark:text-gray-400"> - With Tethys, you can say goodbye to the complexities of manual deployments and embrace a streamlined process that - eliminates errors and minimizes downtime. Our CI/CD pipeline automatically verifies each code commit, runs comprehensive - tests, and deploys the repository seamlessly, ensuring that your research data is always up-to-date and accessible. - </p> - </CardBox> --> </SectionMain> - <!-- </section> --> </LayoutAuthenticated> </template> diff --git a/resources/js/Pages/Editor/Dataset/Category.vue b/resources/js/Pages/Editor/Dataset/Category.vue new file mode 100644 index 0000000..ab40981 --- /dev/null +++ b/resources/js/Pages/Editor/Dataset/Category.vue @@ -0,0 +1,371 @@ +<template> + <LayoutAuthenticated> + <Head title="Classify"></Head> + <SectionMain> + <SectionTitleLineWithButton :icon="mdiLibraryShelves" title="Library Classification" main> + <div class="bg-lime-100 shadow rounded-lg p-6 mb-6 flex items-center justify-between"> + <div> + <label for="role-select" class="block text-lg font-medium text-gray-700 mb-1"> + Select Classification Role <span class="text-red-500">*</span> + </label> + <select id="role-select" v-model="selectedCollectionRole" + class="w-full border border-gray-300 rounded-md p-2 text-gray-700 focus:ring-2 focus:ring-indigo-500" + required> + <!-- <option value="" disabled selected>Please select a role</option> --> + <option v-for="collRole in collectionRoles" :key="collRole.id" :value="collRole"> + {{ collRole.name }} + </option> + </select> + </div> + <div class="ml-4 hidden md:block"> + <span class="text-sm text-gray-600 italic">* required</span> + </div> + </div> + </SectionTitleLineWithButton> + + + <!-- Available TopLevel Collections --> + <CardBox class="mb-4 rounded-lg p-4"> + <h2 class="text-lg font-bold text-gray-800 dark:text-slate-400 mb-2">Available Toplevel-Collections + <span v-if="selectedCollectionRole && !selectedToplevelCollection" + class="text-sm text-red-500 italic">(click to + select)</span> + </h2> + <ul class="flex flex-wrap gap-2"> + <li v-for="col in collections" :key="col.id" :class="{ + 'cursor-pointer p-2 border border-gray-200 rounded hover:bg-sky-50 text-sky-700 text-sm': true, + 'bg-sky-100 border-sky-500': selectedToplevelCollection && selectedToplevelCollection.id === col.id + }" @click="onToplevelCollectionSelected(col)"> + {{ `${col.name} (${col.number})` }} + </li> + <li v-if="collections.length === 0" class="text-gray-800 dark:text-slate-400"> + No collections available. + </li> + </ul> + </CardBox> + + <!-- Collections Listing --> + <div class="grid grid-cols-1 md:grid-cols-3 gap-4 mb-4"> + + <!-- Broader Collection (Parent) --> + <CardBox v-if="selectedCollection" class="rounded-lg p-4" has-form-data> + <h2 class="text-lg font-bold text-gray-800 dark:text-slate-400 mb-2">Broader Collection</h2> + <draggable v-if="broaderCollections.length > 0" v-model="broaderCollections" + :group="{ name: 'collections' }" tag="ul" class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <template #item="{ element: parent }"> + <li :key="parent.id" :draggable="!parent.inUse" :class="getChildClasses(parent)" + @click="onCollectionSelected(parent)"> + {{ `${parent.name} (${parent.number})` }} + </li> + </template> + </draggable> + <ul v-else class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <li class="text-gray-500 text-sm"> + No broader collections available. + </li> + </ul> + </CardBox> + + <!-- Selected Collection Details --> + <CardBox v-if="selectedCollection" class="rounded-lg p-4" has-form-data> + <h3 class="text-xl font-bold text-gray-800 dark:text-slate-400 mb-2">Selected Collection</h3> + <!-- <p :class="[ + 'cursor-pointer p-2 border border-gray-200 rounded text-sm', + selectedCollection.inUse ? 'bg-gray-200 text-gray-500 drag-none' : 'bg-green-50 text-green-700 hover:bg-green-100 hover:underline cursor-move' + ]"></p> --> + <draggable v-model="selectedCollectionArray" :group="{ name: 'collections', pull: 'clone', put: false }" tag="ul" class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <template #item="{ element }"> + <li :key="element.id" :class="[ + 'p-2 border border-gray-200 rounded text-sm', + element.inUse ? 'bg-gray-200 text-gray-500 drag-none' : 'bg-green-50 text-green-700 hover:bg-green-100 hover:underline cursor-move' + ]"> + {{ `${element.name} (${element.number})` }} + </li> + </template> + </draggable> + </CardBox> + <!-- Narrower Collections (Children) --> + <CardBox v-if="selectedCollection" class="rounded-lg p-4" has-form-data> + <h2 class="text-lg font-bold text-gray-800 dark:text-slate-400 mb-2">Narrower Collections</h2> + <draggable v-if="narrowerCollections.length > 0" v-model="narrowerCollections" + :group="{ name: 'collections' }" tag="ul" class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <template #item="{ element: child }"> + <li :key="child.id" :draggable="!child.inUse" :class="getChildClasses(child)" + @click="onCollectionSelected(child)"> + {{ `${child.name} (${child.number})` }} + </li> + </template> + </draggable> + <ul v-else class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <li class="text-gray-500 text-sm"> + No sub-collections available. + </li> + </ul> + </CardBox> + + </div> + + <div class="mb-4 rounded-lg"> + <div v-if="selectedCollection || selectedCollectionList.length > 0" class="bg-gray-100 shadow rounded-lg p-6 mb-6" :class="{ 'opacity-50': selectedCollection && selectedCollectionList.length === 0 }"> + <p class="mb-4 text-gray-700">Please drag your collections here to classify your previously created + dataset + according to library classification standards.</p> + <draggable v-model="selectedCollectionList" :group="{ name: 'collections' }" + class="min-h-36 border-dashed border-2 border-gray-400 p-4 text-sm flex flex-wrap gap-2 max-h-60 overflow-y-auto" + tag="ul" + :disabled="selectedCollection === null && selectedCollectionList.length > 0" + :style="{ opacity: (selectedCollection === null && selectedCollectionList.length > 0) ? 0.5 : 1, pointerEvents: (selectedCollection === null && selectedCollectionList.length > 0) ? 'none' : 'auto' }"> + <template #item="{ element }"> + <div :key="element.id" + class="p-2 m-1 bg-sky-200 text-sky-800 rounded flex items-center gap-2 h-7"> + <span>{{ element.name }} ({{ element.number }})</span> + <button + @click="selectedCollectionList = selectedCollectionList.filter(item => item.id !== element.id)" + class="hover:text-sky-600 flex items-center"> + <svg xmlns="http://www.w3.org/2000/svg" class="h-4 w-4" viewBox="0 0 20 20" + fill="currentColor"> + <path fill-rule="evenodd" + d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" + clip-rule="evenodd" /> + </svg> + </button> + </div> + </template> + </draggable> + </div> + </div> + <div class="p-6 border-t border-gray-100 dark:border-slate-800"> + <BaseButtons> + <BaseButton @click.stop="syncDatasetCollections" label="Save" color="info" small + :disabled="isSaveDisabled" :style="{ opacity: isSaveDisabled ? 0.5 : 1 }"> + </BaseButton> + </BaseButtons> + </div> + </SectionMain> + </LayoutAuthenticated> +</template> + +<script setup lang="ts"> +import { ref, Ref, watch, computed } from 'vue'; +import { useForm } from '@inertiajs/vue3'; +import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; +import SectionMain from '@/Components/SectionMain.vue'; +import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; +import axios from 'axios'; +import { mdiLibraryShelves } from '@mdi/js'; +import draggable from 'vuedraggable'; +import BaseButton from '@/Components/BaseButton.vue'; +import BaseButtons from '@/Components/BaseButtons.vue'; +import CardBox from '@/Components/CardBox.vue'; +import { stardust } from '@eidellev/adonis-stardust/client'; +import { CollectionRole, Collection } from '@/types/models'; + +// import CollectionRoleSelector from '@/Components/Collection/CollectionRoleSelector.vue'; +// import ToplevelCollections from '@/Components/Collection/ToplevelCollections.vue'; +// import CollectionHierarchy from '@/Components/Collection/CollectionHierarchy.vue'; +// import CollectionDropZone from '@/Components/Collection/CollectionDropZone.vue'; + + + +/* -------------------------------------------------------------------------- + Props & Reactive State +-------------------------------------------------------------------------- */ +const props = defineProps({ + collectionRoles: { + type: Array, + required: true, + default: () => [] + }, + dataset: { + type: Object, + default: () => ({}), + }, + relatedCollections: { + type: Array as () => Collection[], + default: () => [] as const + } +}); + +const collectionRoles: Ref<CollectionRole[]> = ref(props.collectionRoles as CollectionRole[]); +const collections: Ref<Collection[]> = ref<Collection[]>([]); +const selectedCollectionRole = ref<CollectionRole | null>(null); +const selectedToplevelCollection = ref<Collection | null>(null); +const selectedCollection = ref<Collection | null>(null); +const narrowerCollections = ref<Collection[]>([]); +const broaderCollections = ref<Collection[]>([]); +// Reactive list that holds collections dropped by the user +const selectedCollectionList: Ref<Collection[]> = ref<Collection[]>([]); + + +// Wrap selectedCollection in an array for draggable (always expects an array) +const selectedCollectionArray = computed({ + get: () => (selectedCollection.value ? [selectedCollection.value] : []), + set: (value: Collection[]) => { + selectedCollection.value = value.length ? value[0] : null + } +}) + + +const form = useForm({ + collections: [] as number[], +}); + +// Watch for changes in dropCollections +watch( + () => selectedCollectionList.value, + () => { + if (selectedCollection.value) { + fetchCollections(selectedCollection.value.id); + } + }, + { deep: true } +); + + +/* -------------------------------------------------------------------------- + Watchers and Initial Setup +-------------------------------------------------------------------------- */ +// If the collectionRoles prop might load asynchronously (or change), you can watch for it: +watch( + () => props.collectionRoles as CollectionRole[], + (newCollectionRoles: CollectionRole[]) => { + collectionRoles.value = newCollectionRoles; + // Preselect the role with name "ccs" if it exists + const found: CollectionRole | undefined = collectionRoles.value.find( + role => role.name.toLowerCase() === 'ccs' + ); + if (found?.name === 'ccs') { + selectedCollectionRole.value = found; + } + }, + { immediate: true } +); + +// When collection role changes, update available collections and clear dependent state. +watch( + () => selectedCollectionRole.value as CollectionRole, + (newSelectedCollectionRole: CollectionRole | null) => { + if (newSelectedCollectionRole != null) { + collections.value = newSelectedCollectionRole.collections || [] + } else { + selectedToplevelCollection.value = null; + selectedCollection.value = null; + collections.value = [] + } + // Reset dependent variables when the role changes + selectedCollection.value = null + narrowerCollections.value = [] + broaderCollections.value = [] + }, + { immediate: true } +); + +/* -------------------------------------------------------------------------- + Methods +-------------------------------------------------------------------------- */ +const onToplevelCollectionSelected = (collection: Collection) => { + selectedToplevelCollection.value = collection; + selectedCollection.value = collection; + // call the API endpoint to get both. + fetchCollections(collection.id); +}; + +const onCollectionSelected = (collection: Collection) => { + selectedCollection.value = collection; + // call the API endpoint to get both. + fetchCollections(collection.id); +}; + +/** + * fetchCollections: Retrieves broader and narrower collections. + * Marks any narrower collection as inUse if it appears in selectedCollectionList. + */ +const fetchCollections = async (collectionId: number) => { + try { + const response = await axios.get(`/api/collections/${collectionId}`); + const data = response.data; + // Map each returned narrower collection + narrowerCollections.value = data.narrowerCollections.map((collection: Collection) => { + // If found, mark it as inUse. + const alreadyDropped = selectedCollectionList.value.find(dc => dc.id === collection.id); + return alreadyDropped ? { ...collection, inUse: true } : { ...collection, inUse: false }; + }); + broaderCollections.value = data.broaderCollection.map((collection: Collection) => { + const alreadyDropped = selectedCollectionList.value.find(dc => dc.id === collection.id); + return alreadyDropped ? { ...collection, inUse: true } : { ...collection, inUse: false }; + }); + // Check if selected collection is in the selected list + if (selectedCollection.value && selectedCollectionList.value.find(dc => dc.id === selectedCollection.value.id)) { + selectedCollection.value = { ...selectedCollection.value, inUse: true }; + } else if (selectedCollection.value) { + selectedCollection.value = { ...selectedCollection.value, inUse: false }; + } + } catch (error) { + console.error('Error in fetchCollections:', error); + } +}; + +const syncDatasetCollections = async () => { + // Extract the ids from the dropCollections list + form.collections = selectedCollectionList.value.map((item: Collection) => item.id); + form.put(stardust.route('editor.dataset.categorizeUpdate', [props.dataset.id]), { + preserveState: true, + onSuccess: () => { + console.log('Dataset collections synced successfully'); + }, + onError: (errors) => { + console.error('Error syncing dataset collections:', errors); + }, + }); +}; + +/** + * getChildClasses returns the Tailwind CSS classes to apply to each collection list item. + */ +const getChildClasses = (child: Collection) => { + return child.inUse + ? 'p-2 border border-gray-200 rounded bg-gray-200 text-gray-500 cursor-pointer drag-none' + : 'p-2 border border-gray-200 rounded bg-green-50 text-green-700 cursor-move hover:bg-green-100 hover:underline' +} + +// If there are related collections passed in, fill dropCollections with these. +if (props.relatedCollections && props.relatedCollections.length > 0) { + selectedCollectionList.value = props.relatedCollections; +} + +// Add a computed property for the disabled state based on dropCollections length +const isSaveDisabled = computed(() => selectedCollectionList.value.length === 0); +</script> + +<style scoped> +.btn-primary { + background-color: #4f46e5; + color: white; + border-radius: 0.25rem; +} + +.btn-primary:hover { + background-color: #4338ca; +} + +.btn-primary:focus { + outline: none; + box-shadow: 0 0 0 2px #fff, 0 0 0 4px #4f46e5; +} + +.btn-secondary { + background-color: white; + color: #374151; + border: 1px solid #d1d5db; + border-radius: 0.25rem; +} + +.btn-secondary:hover { + background-color: #f9fafb; +} + +.btn-secondary:focus { + outline: none; + box-shadow: 0 0 0 2px #fff, 0 0 0 4px #6366f1; +} +</style> diff --git a/resources/js/Pages/Editor/Dataset/Doi.vue b/resources/js/Pages/Editor/Dataset/Doi.vue index 412622b..096afcb 100644 --- a/resources/js/Pages/Editor/Dataset/Doi.vue +++ b/resources/js/Pages/Editor/Dataset/Doi.vue @@ -26,17 +26,6 @@ const errors: Ref<any> = computed(() => { return usePage().props.errors; }); -// const form = useForm({ -// preferred_reviewer: '', -// preferred_reviewer_email: '', -// preferation: 'yes_preferation', - -// // preferation: '', -// // isPreferationRequired: false, -// }); - -// const isPreferationRequired = computed(() => form.preferation === 'yes_preferation'); - const handleSubmit = async (e) => { e.preventDefault(); // Notification.showInfo(`doi implementation is in developement. Create DOI for dataset ${props.dataset.publish_id} later on`); diff --git a/resources/js/Pages/Editor/Dataset/Edit.vue b/resources/js/Pages/Editor/Dataset/Edit.vue new file mode 100644 index 0000000..e33e5dd --- /dev/null +++ b/resources/js/Pages/Editor/Dataset/Edit.vue @@ -0,0 +1,865 @@ +<template> + <LayoutAuthenticated> + + <Head title="Edit dataset" /> + <SectionMain> + <SectionTitleLineWithButton :icon="mdiImageText" title="Update dataset" main> + <BaseButton :route-name="stardust.route('editor.dataset.list')" :icon="mdiArrowLeftBoldOutline" + label="Back" color="white" rounded-full small /> + </SectionTitleLineWithButton> + + <NotificationBar v-if="flash.message" color="success" :icon="mdiAlertBoxOutline"> + {{ flash.message }} + </NotificationBar> + <FormValidationErrors v-bind:errors="errors" /> + + <CardBox :form="true"> + <!-- <FormValidationErrors v-bind:errors="errors" /> --> + <div class="mb-4"> + <div class="flex flex-col md:flex-row"> + <!-- (1) language field --> + <FormField label="Language *" help="required: select dataset main language" + :class="{ 'text-red-400': form.errors.language }" class="w-full flex-1"> + <FormControl required v-model="form.language" :type="'select'" + placeholder="[Enter Language]" :errors="form.errors.language" + :options="{ de: 'de', en: 'en' }"> + <div class="text-red-400 text-sm" v-if="form.errors.language"> + {{ form.errors.language.join(', ') }} + </div> + </FormControl> + </FormField> + </div> + + <!-- (2) licenses --> + <FormField label="Licenses" wrap-body :class="{ 'text-red-400': form.errors.licenses }" + class="mt-8 w-full mx-2 flex-1"> + <FormCheckRadioGroup type="radio" v-model="form.licenses" name="licenses" is-column + :options="licenses" /> + </FormField> + + <div class="flex flex-col md:flex-row"> + <!-- (3) dataset_type --> + <FormField label="Dataset Type *" help="required: dataset type" + :class="{ 'text-red-400': form.errors.type }" class="w-full mx-2 flex-1"> + <FormControl required v-model="form.type" :type="'select'" placeholder="-- select type --" + :errors="form.errors.type" :options="doctypes"> + <div class="text-red-400 text-sm" + v-if="form.errors.type && Array.isArray(form.errors.type)"> + {{ form.errors.type.join(', ') }} + </div> + </FormControl> + </FormField> + <!-- (4) creating_corporation --> + <FormField label="Creating Corporation *" + :class="{ 'text-red-400': form.errors.creating_corporation }" class="w-full mx-2 flex-1"> + <FormControl required v-model="form.creating_corporation" type="text" + placeholder="[enter creating corporation]" :is-read-only="true"> + <div class="text-red-400 text-sm" + v-if="form.errors.creating_corporation && Array.isArray(form.errors.creating_corporation)"> + {{ form.errors.creating_corporation.join(', ') }} + </div> + </FormControl> + </FormField> + </div> + + <BaseDivider /> + + <!-- (5) titles --> + <CardBox class="mb-6 shadow" :has-form-data="false" title="Titles" :icon="mdiFinance" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addTitle()"> + <div class="flex flex-col md:flex-row"> + <FormField label="Main Title *" help="required: main title" + :class="{ 'text-red-400': form.errors['titles.0.value'] }" class="w-full mr-1 flex-1"> + <FormControl required v-model="form.titles[0].value" type="text" + placeholder="[enter main title]" :show-char-count="true" :max-input-length="255"> + <div class="text-red-400 text-sm" + v-if="form.errors['titles.0.value'] && Array.isArray(form.errors['titles.0.value'])"> + {{ form.errors['titles.0.value'].join(', ') }} + </div> + </FormControl> + </FormField> + <FormField label="Main Title Language*" help="required: main title language" + :class="{ 'text-red-400': form.errors['titles.0.language'] }" + class="w-full ml-1 flex-1"> + <FormControl required v-model="form.titles[0].language" type="text" + :is-read-only="true"> + <div class="text-red-400 text-sm" + v-if="form.errors['titles.0.language'] && Array.isArray(form.errors['titles.0.language'])"> + {{ form.errors['titles.0.language'].join(', ') }} + </div> + </FormControl> + </FormField> + </div> + + <label v-if="form.titles.length > 1">additional titles </label> + <!-- <BaseButton :icon="mdiPlusCircle" @click.prevent="addTitle()" color="modern" rounded-full small /> --> + <table> + <thead> + <tr> + <!-- <th v-if="checkable" /> --> + <th>Title Value</th> + <th>Title Type</th> + <th>Title Language</th> + <th /> + </tr> + </thead> + <tbody> + <template v-for="(title, index) in form.titles" :key="index"> + <tr v-if="title.type != 'Main'"> + <!-- <td scope="row">{{ index + 1 }}</td> --> + <td data-label="Title Value"> + <FormControl required v-model="form.titles[index].value" type="text" + placeholder="[enter main title]"> + <div class="text-red-400 text-sm" + v-if="form.errors[`titles.${index}.value`]"> + {{ form.errors[`titles.${index}.value`].join(', ') }} + </div> + </FormControl> + </td> + <td data-label="Title Type"> + <FormControl required v-model="form.titles[index].type" type="select" + :options="titletypes" placeholder="[select title type]"> + <div class="text-red-400 text-sm" + v-if="Array.isArray(form.errors[`titles.${index}.type`])"> + {{ form.errors[`titles.${index}.type`].join(', ') }} + </div> + </FormControl> + </td> + <td data-label="Title Language"> + <FormControl required v-model="form.titles[index].language" type="select" + :options="{ de: 'de', en: 'en' }" placeholder="[select title language]"> + <div class="text-red-400 text-sm" + v-if="form.errors[`titles.${index}.language`]"> + {{ form.errors[`titles.${index}.language`].join(', ') }} + </div> + </FormControl> + </td> + <td class="before:hidden lg:w-1 whitespace-nowrap"> + <BaseButtons type="justify-start lg:justify-end" no-wrap> + <!-- <BaseButton color="info" :icon="mdiEye" small @click="isModalActive = true" /> --> + <BaseButton color="danger" :icon="mdiTrashCan" small + v-if="title.id == undefined" @click.prevent="removeTitle(index)" /> + </BaseButtons> + </td> + </tr> + </template> + </tbody> + </table> + </CardBox> + + <!-- (6) descriptions --> + <CardBox class="mb-6 shadow" :has-form-data="false" title="Descriptions" :icon="mdiFinance" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addDescription()"> + <div class="flex flex-col md:flex-row"> + <FormField label="Main Abstract *" help="required: main abstract" + :class="{ 'text-red-400': form.errors['descriptions.0.value'] }" + class="w-full mr-1 flex-1"> + <FormControl required v-model="form.descriptions[0].value" type="textarea" + placeholder="[enter main abstract]" :show-char-count="true" + :max-input-length="2500"> + <div class="text-red-400 text-sm" + v-if="form.errors['descriptions.0.value'] && Array.isArray(form.errors['descriptions.0.value'])"> + {{ form.errors['descriptions.0.value'].join(', ') }} + </div> + </FormControl> + </FormField> + <FormField label="Main Title Language*" help="required: main abstract language" + :class="{ 'text-red-400': form.errors['descriptions.0.language'] }" + class="w-full ml-1 flex-1"> + <FormControl required v-model="form.descriptions[0].language" type="text" + :is-read-only="true"> + <div class="text-red-400 text-sm" v-if="form.errors['descriptions.0.value'] && Array.isArray(form.errors['descriptions.0.language']) + "> + {{ form.errors['descriptions.0.language'].join(', ') }} + </div> + </FormControl> + </FormField> + </div> + <table> + <thead> + <tr> + <!-- <th v-if="checkable" /> --> + <th>Title Value</th> + <th>Title Type</th> + <th>Title Language</th> + <th /> + </tr> + </thead> + <tbody> + <template v-for="(item, index) in form.descriptions" :key="index"> + <tr v-if="item.type != 'Abstract'"> + <!-- <td scope="row">{{ index + 1 }}</td> --> + <td data-label="Description Value"> + <FormControl required v-model="form.descriptions[index].value" type="text" + placeholder="[enter main title]"> + <div class="text-red-400 text-sm" + v-if="form.errors[`descriptions.${index}.value`]"> + {{ form.errors[`descriptions.${index}.value`].join(', ') }} + </div> + </FormControl> + </td> + <td data-label="Description Type"> + <FormControl required v-model="form.descriptions[index].type" type="select" + :options="descriptiontypes" placeholder="[select title type]"> + <div class="text-red-400 text-sm" + v-if="Array.isArray(form.errors[`descriptions.${index}.type`])"> + {{ form.errors[`descriptions.${index}.type`].join(', ') }} + </div> + </FormControl> + </td> + <td data-label="Description Language"> + <FormControl required v-model="form.descriptions[index].language" + type="select" :options="{ de: 'de', en: 'en' }" + placeholder="[select title language]"> + <div class="text-red-400 text-sm" + v-if="form.errors[`descriptions.${index}.language`]"> + {{ form.errors[`descriptions.${index}.language`].join(', ') }} + </div> + </FormControl> + </td> + <td class="before:hidden lg:w-1 whitespace-nowrap"> + <BaseButtons type="justify-start lg:justify-end" no-wrap> + <!-- <BaseButton color="info" :icon="mdiEye" small @click="isModalActive = true" /> --> + <BaseButton color="danger" :icon="mdiTrashCan" small + v-if="item.id == undefined" + @click.prevent="removeDescription(index)" /> + </BaseButtons> + </td> + </tr> + </template> + </tbody> + </table> + </CardBox> + + <!-- (7) authors --> + <CardBox class="mb-6 shadow" has-table title="Creators" :icon="mdiBookOpenPageVariant" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addNewAuthor()"> + <SearchAutocomplete source="/api/persons" :response-property="'first_name'" + placeholder="search in person table...." v-on:person="onAddAuthor"></SearchAutocomplete> + + <TablePersons :persons="form.authors" v-if="form.authors.length > 0" :relation="'authors'" /> + <div class="text-red-400 text-sm" + v-if="form.errors.authors && Array.isArray(form.errors.authors)"> + {{ form.errors.authors.join(', ') }} + </div> + </CardBox> + + + <!-- (8) contributors --> + <CardBox class="mb-6 shadow" has-table title="Contributors" :icon="mdiBookOpenPageVariant" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addNewContributor()"> + <SearchAutocomplete source="/api/persons" :response-property="'first_name'" + placeholder="search in person table...." v-on:person="onAddContributor"> + </SearchAutocomplete> + + <TablePersons :persons="form.contributors" v-if="form.contributors.length > 0" + :contributortypes="contributorTypes" :errors="form.errors" :relation="'contributors'" /> + <div class="text-red-400 text-sm" + v-if="form.errors.contributors && Array.isArray(form.errors.contributors)"> + {{ form.errors.contributors.join(', ') }} + </div> + </CardBox> + + <div class="flex flex-col md:flex-row"> + <!-- (9) project_id --> + <FormField label="Project.." help="project is optional" + :class="{ 'text-red-400': form.errors.project_id }" class="w-full mx-2 flex-1"> + <FormControl required v-model="form.project_id" :type="'select'" + placeholder="[Select Project]" :errors="form.errors.project_id" :options="projects"> + <div class="text-red-400 text-sm" v-if="form.errors.project_id"> + {{ form.errors.project_id.join(', ') }} + </div> + </FormControl> + </FormField> + <!-- (10) embargo_date --> + <FormField label="Embargo Date.." help="embargo date is optional" + :class="{ 'text-red-400': form.errors.embargo_date }" class="w-full mx-2 flex-1"> + <FormControl v-model="form.embargo_date" :type="'date'" placeholder="date('y-m-d')" + :errors="form.errors.embargo_date"> + <div class="text-red-400 text-sm" v-if="form.errors.embargo_date"> + {{ form.errors.embargo_date.join(', ') }} + </div> + </FormControl> + </FormField> + </div> + + <BaseDivider /> + + <MapComponent v-if="form.coverage" :mapOptions="mapOptions" :baseMaps="baseMaps" + :fitBounds="fitBounds" :coverage="form.coverage" :mapId="mapId" + v-bind-event:onMapInitializedEvent="onMapInitialized"> + </MapComponent> + <div class="flex flex-col md:flex-row"> + <!-- x min and max --> + <FormField label="Coverage X Min" :class="{ 'text-red-400': form.errors['coverage.x_min'] }" + class="w-full mx-2 flex-1"> + <FormControl required v-model="form.coverage.x_min" type="text" placeholder="[enter x_min]"> + <div class="text-red-400 text-sm" + v-if="form.errors['coverage.x_min'] && Array.isArray(form.errors['coverage.x_min'])"> + {{ form.errors['coverage.x_min'].join(', ') }} + </div> + </FormControl> + </FormField> + <FormField label="Coverage X Max" :class="{ 'text-red-400': form.errors['coverage.x_max'] }" + class="w-full mx-2 flex-1"> + <FormControl required v-model="form.coverage.x_max" type="text" placeholder="[enter x_max]"> + <div class="text-red-400 text-sm" + v-if="form.errors['coverage.x_max'] && Array.isArray(form.errors['coverage.x_max'])"> + {{ form.errors['coverage.x_max'].join(', ') }} + </div> + </FormControl> + </FormField> + <!-- y min and max --> + <FormField label="Coverage Y Min" :class="{ 'text-red-400': form.errors['coverage.y_min'] }" + class="w-full mx-2 flex-1"> + <FormControl required v-model="form.coverage.y_min" type="text" placeholder="[enter y_min]"> + <div class="text-red-400 text-sm" + v-if="form.errors['coverage.y_min'] && Array.isArray(form.errors['coverage.y_min'])"> + {{ form.errors['coverage.y_min'].join(', ') }} + </div> + </FormControl> + </FormField> + <FormField label="Coverage Y Max" :class="{ 'text-red-400': form.errors['coverage.y_max'] }" + class="w-full mx-2 flex-1"> + <FormControl required v-model="form.coverage.y_max" type="text" placeholder="[enter y_max]"> + <div class="text-red-400 text-sm" + v-if="form.errors['coverage.y_max'] && Array.isArray(form.errors['coverage.y_max'])"> + {{ form.errors['coverage.y_max'].join(', ') }} + </div> + </FormControl> + </FormField> + </div> + + <CardBox class="mb-6 shadow" has-table title="Dataset References" :icon="mdiEarthPlus" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addReference()"> + <!-- Message when no references exist --> + <div v-if="form.references.length === 0" class="text-center py-4"> + <p class="text-gray-600">No references added yet.</p> + <p class="text-gray-400">Click the plus icon above to add a new reference.</p> + </div> + <!-- Reference form --> + <table class="table-fixed border-green-900" v-if="form.references.length"> + <thead> + <tr> + <th class="w-4/12">Value</th> + <th class="w-2/12">Type</th> + <th class="w-3/12">Relation</th> + <th class="w-2/12">Label</th> + <th class="w-1/12"></th> + </tr> + </thead> + <tbody> + <tr v-for="(item, index) in form.references"> + <td data-label="Reference Value"> + <!-- <input name="Reference Value" class="form-control" + placeholder="[VALUE]" v-model="item.value" /> --> + <FormControl required v-model="item.value" :type="'text'" placeholder="[VALUE]" + :errors="form.errors.embargo_date"> + <div class="text-red-400 text-sm" + v-if="form.errors[`references.${index}.value`] && Array.isArray(form.errors[`references.${index}.value`])"> + {{ form.errors[`references.${index}.value`].join(', ') }} + </div> + </FormControl> + + </td> + <td> + <FormControl required v-model="form.references[index].type" type="select" + :options="referenceIdentifierTypes" placeholder="[type]"> + <div class="text-red-400 text-sm" + v-if="Array.isArray(form.errors[`references.${index}.type`])"> + {{ form.errors[`references.${index}.type`].join(', ') }} + </div> + </FormControl> + </td> + + <td> + <!-- {!! Form::select('Reference[Relation]', $relationTypes, null, + ['placeholder' => '[relationType]', 'v-model' => 'item.relation', + 'data-vv-scope' => 'step-2']) + !!} --> + <FormControl required v-model="form.references[index].relation" type="select" + :options="relationTypes" placeholder="[relation type]"> + <div class="text-red-400 text-sm" + v-if="Array.isArray(form.errors[`references.${index}.relation`])"> + {{ form.errors[`references.${index}.relation`].join(', ') }} + </div> + </FormControl> + </td> + <td data-label="Reference Label"> + <!-- <input name="Reference Label" class="form-control" v-model="item.label" /> --> + <FormControl required v-model="form.references[index].label" type="text" + placeholder="[reference label]"> + <div class="text-red-400 text-sm" + v-if="form.errors[`references.${index}.label`]"> + {{ form.errors[`references.${index}.label`].join(', ') }} + + </div> + </FormControl> + </td> + <td class="before:hidden lg:w-1 whitespace-nowrap"> + <!-- <BaseButton color="info" :icon="mdiEye" small @click="isModalActive = true" /> --> + <BaseButton color="danger" :icon="mdiTrashCan" small + @click.prevent="removeReference(index)" /> + </td> + </tr> + </tbody> + </table> + <!-- References to delete section --> + <div v-if="form.referencesToDelete && form.referencesToDelete.length > 0" class="mt-8"> + <h1 class="pt-8 pb-3 font-semibold sm:text-lg text-gray-900">References To Delete</h1> + <ul class="flex flex-1 flex-wrap -m-1"> + <li v-for="(element, index) in form.referencesToDelete" :key="index" + class="block p-1 w-1/2 sm:w-1/3 md:w-1/4 lg:w-1/6 xl:w-1/8 h-40"> + <article tabindex="0" + class="bg-red-100 group w-full h-full rounded-md cursor-pointer relative shadow-sm overflow-hidden"> + <section + class="flex flex-col rounded-md text-xs break-words w-full h-full z-20 absolute top-0 py-2 px-3"> + <h1 class="flex-1 text-gray-700 group-hover:text-blue-800 font-medium text-sm mb-1 truncate overflow-hidden whitespace-nowrap"> + {{ element.value }} + </h1> + <div class="flex flex-col mt-auto"> + <p class="p-1 size text-xs text-gray-700"> + <span class="font-semibold">Type:</span> {{ element.type }} + </p> + <p class="p-1 size text-xs text-gray-700"> + <span class="font-semibold">Relation:</span> {{ element.relation }} + </p> + <div class="flex justify-end mt-1"> + <button + class="restore ml-auto focus:outline-none hover:bg-gray-300 p-1 rounded-md text-gray-800" + @click.prevent="restoreReference(index)"> + <svg viewBox="0 0 24 24" class="w-5 h-5"> + <path fill="currentColor" :d="mdiRestore"></path> + </svg> + </button> + </div> + </div> + </section> + </article> + </li> + </ul> + </div> + </CardBox> + + <BaseDivider /> + + <CardBox class="mb-6 shadow" has-table title="Dataset Keywords" :icon="mdiEarthPlus" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addKeyword()"> + <!-- <ul> + <li v-for="(subject, index) in form.subjects" :key="index"> + {{ subject.value }} <BaseButton color="danger" :icon="mdiTrashCan" small @click.prevent="removeKeyword(index)" /> + </li> + </ul> --> + <TableKeywords :keywords="form.subjects" :errors="form.errors" :subjectTypes="subjectTypes" v-model:subjects-to-delete="form.subjectsToDelete" + v-if="form.subjects.length > 0" /> + </CardBox> + + </div> + + <!-- download file list --> + <div class="mb-6"> + <h3 class="text-lg font-medium text-gray-700 mb-2 flex items-center"> + Files + <div class="inline-block relative ml-2 group"> + <button + class="w-5 h-5 rounded-full bg-gray-200 text-gray-600 text-xs flex items-center justify-center focus:outline-none hover:bg-gray-300"> + i + </button> + <div + class="absolute left-0 top-full mt-1 w-64 bg-white shadow-lg rounded-md p-3 text-xs text-left z-50 transform scale-0 origin-top-left transition-transform duration-100 group-hover:scale-100"> + <p class="text-gray-700"> + As a research data repository editor, you can only download the submitted files. + Files cannot be + edited or replaced at this stage. + </p> + <div class="absolute -top-1 left-1 w-2 h-2 bg-white transform rotate-45"></div> + </div> + </div> + </h3> + + <div v-if="form.files && form.files.length > 0" class="bg-white rounded-lg shadow overflow-hidden"> + <ul class="divide-y divide-gray-200"> + <li v-for="file in form.files" :key="file.id" + class="px-4 py-3 flex items-center justify-between hover:bg-gray-50"> + <div class="flex items-center space-x-3 flex-1"> + <div class="flex-shrink-0"> + <svg class="h-6 w-6 text-gray-400" xmlns="http://www.w3.org/2000/svg" + fill="none" viewBox="0 0 24 24" stroke="currentColor"> + <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" + d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" /> + </svg> + </div> + <div class="min-w-0 flex-1"> + <p class="text-sm font-medium text-gray-900 truncate"> + {{ file.label }} + </p> + <p class="text-xs text-gray-500 truncate"> + {{ getFileSize(file) }} + </p> + </div> + </div> + <div class="ml-2 flex-shrink-0 flex space-x-2"> + <a v-if="file.id != undefined" + :href="stardust.route('editor.file.download', [file.id])" + class="inline-flex items-center px-2.5 py-1.5 border border-transparent text-xs font-medium rounded text-blue-700 bg-blue-100 hover:bg-blue-200"> + Download + </a> + </div> + + </li> + </ul> + </div> + </div> + + <template #footer> + <BaseButtons> + <BaseButton @click.stop="submit" :disabled="form.processing" label="Save" color="info" + :class="{ 'opacity-25': form.processing }" small> + </BaseButton> + <!-- <button :disabled="form.processing" :class="{ 'opacity-25': form.processing }" + class="text-base hover:scale-110 focus:outline-none flex justify-center px-4 py-2 rounded font-bold cursor-pointer hover:bg-teal-200 bg-teal-100 text-teal-700 border duration-200 ease-in-out border-teal-600 transition" + @click.stop="submit"> + Save + </button> --> + </BaseButtons> + </template> + </CardBox> + <!-- Loading Spinner --> + <div v-if="form.processing" + class="fixed inset-0 flex items-center justify-center bg-gray-500 bg-opacity-50 z-50"> + <svg class="animate-spin h-12 w-12 text-white" xmlns="http://www.w3.org/2000/svg" fill="none" + viewBox="0 0 24 24"> + <circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle> + <path class="opacity-75" fill="currentColor" d="M12 2a10 10 0 0110 10h-4a6 6 0 00-6-6V2z"></path> + </svg> + </div> + </SectionMain> + </LayoutAuthenticated> +</template> + +<script setup lang="ts"> +// import EditComponent from "./../EditComponent"; +// export default EditComponent; + +// import { Component, Vue, Prop, Setup, toNative } from 'vue-facing-decorator'; +// import AuthLayout from '@/Layouts/Auth.vue'; +import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; +import { useForm, Head, usePage } from '@inertiajs/vue3'; +import { computed, ComputedRef } from 'vue'; +import { Dataset, Title, Subject, Person, License } from '@/Dataset'; +import { stardust } from '@eidellev/adonis-stardust/client'; + +import FormField from '@/Components/FormField.vue'; +import FormControl from '@/Components/FormControl.vue'; +import SectionMain from '@/Components/SectionMain.vue'; +import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; +import FormCheckRadioGroup from '@/Components/FormCheckRadioGroup.vue'; +import BaseButton from '@/Components/BaseButton.vue'; +import BaseButtons from '@/Components/BaseButtons.vue'; +import BaseDivider from '@/Components/BaseDivider.vue'; +import CardBox from '@/Components/CardBox.vue'; +import MapComponent from '@/Components/Map/map.component.vue'; +import SearchAutocomplete from '@/Components/SearchAutocomplete.vue'; +import TablePersons from '@/Components/TablePersons.vue'; +import TableKeywords from '@/Components/TableKeywords.vue'; +import FormValidationErrors from '@/Components/FormValidationErrors.vue'; +import { MapOptions } from '@/Components/Map/MapOptions'; +import { LatLngBoundsExpression } from 'leaflet'; +import { LayerOptions } from '@/Components/Map/LayerOptions'; +import { + mdiImageText, + mdiArrowLeftBoldOutline, + mdiPlusCircle, + mdiFinance, + mdiTrashCan, + mdiBookOpenPageVariant, + mdiEarthPlus, + mdiAlertBoxOutline, + mdiRestore +} from '@mdi/js'; +import { notify } from '@/notiwind'; +import NotificationBar from '@/Components/NotificationBar.vue'; + +const props = defineProps({ + // errors: { + // type: Object, + // default: () => ({}), + // }, + licenses: { + type: Object, + default: () => ({}), + }, + languages: { + type: Object, + default: () => ({}), + }, + doctypes: { + type: Object, + default: () => ({}), + }, + titletypes: { + type: Object, + default: () => ({}), + }, + projects: { + type: Object, + default: () => ({}), + }, + descriptiontypes: { + type: Object, + default: () => ({}), + }, + contributorTypes: { + type: Object, + default: () => ({}), + }, + subjectTypes: { + type: Object, + default: () => ({}), + }, + referenceIdentifierTypes: { + type: Object, + default: () => ({}), + }, + relationTypes: { + type: Object, + default: () => ({}), + }, + dataset: { + type: Object, + default: () => ({}), + }, + + + +}); +const flash: ComputedRef<any> = computed(() => { + return usePage().props.flash; +}); +const errors: ComputedRef<any> = computed(() => { + return usePage().props.errors; +}); + +const mapOptions: MapOptions = { + center: [48.208174, 16.373819], + zoom: 3, + zoomControl: false, + attributionControl: false, +}; +const baseMaps: Map<string, LayerOptions> = new Map<string, LayerOptions>(); +const fitBounds: LatLngBoundsExpression = [ + [46.4318173285, 9.47996951665], + [49.0390742051, 16.9796667823], +]; +const mapId = 'test'; + +// const downloadFile = async (id: string): Promise<string> => { +// const response = await axios.get<Blob>(`/api/download/${id}`, { +// responseType: 'blob', +// }); +// const url = URL.createObjectURL(response.data); +// setTimeout(() => { +// URL.revokeObjectURL(url); +// }, 1000); +// return url; +// }; + +// for (const file of props.dataset.files) { +// // console.log(`${file.name} path is ${file.filePath} here.`); +// file.fileSrc = ref(""); +// // downloadFile(file.id).then((value: string) => { +// // file.fileSrc = ref(value); +// // form = useForm<Dataset>(props.dataset as Dataset); +// // }); +// } + +// props.dataset.filesToDelete = []; +props.dataset.subjectsToDelete = []; +props.dataset.referencesToDelete = []; +let form = useForm<Dataset>(props.dataset as Dataset); + +// const mainService = MainService(); +// mainService.fetchfiles(props.dataset); + + +const submit = async (): Promise<void> => { + let route = stardust.route('editor.dataset.update', [props.dataset.id]); + // await Inertia.post('/app/register', this.form); + // await router.post('/app/register', this.form); + + + let licenses = form.licenses.map((obj) => { + if (hasIdAttribute(obj)) { + return obj.id.toString() + } else { + return obj; + } + }); + + await form + .transform((data) => ({ + ...data, + licenses: licenses, + rights: 'true', + })) + // .put(route); + .put(route, { + onSuccess: () => { + // console.log(form.data()); + // mainService.setDataset(form.data()); + // formStep.value++; + // form.filesToDelete = []; + // Clear the array using splice + form.subjectsToDelete?.splice(0, form.subjectsToDelete.length); + form.referencesToDelete?.splice(0, form.referencesToDelete.length); + }, + }); +}; + +const hasIdAttribute = (obj: License | number): obj is License => { + return typeof obj === 'object' && 'id' in obj; +}; + +const addTitle = () => { + let newTitle: Title = { value: '', language: '', type: '' }; + form.titles.push(newTitle); +}; +const removeTitle = (key: any) => { + form.titles.splice(key, 1); +}; + +const addDescription = () => { + let newDescription = { value: '', language: '', type: '' }; + form.descriptions.push(newDescription); +}; +const removeDescription = (key: any) => { + form.descriptions.splice(key, 1); +}; + +const addNewAuthor = () => { + let newAuthor = { status: false, first_name: '', last_name: '', email: '', academic_title: '', identifier_orcid: '', name_type: 'Personal' }; + form.authors.push(newAuthor); +}; + +const onAddAuthor = (person: Person) => { + if (form.authors.filter((e) => e.id === person.id).length > 0) { + notify({ type: 'warning', title: 'Warning', text: 'person is already defined as author' }, 4000); + } else if (form.contributors.filter((e) => e.id === person.id).length > 0) { + notify({ type: 'warning', title: 'Warning', text: 'person is already defined as contributor' }); + } else { + form.authors.push(person); + notify({ type: 'info', text: 'person has been successfully added as author' }); + } +}; + +const addNewContributor = () => { + let newContributor = { status: false, first_name: '', last_name: '', email: '', academic_title: '', identifier_orcid: '', name_type: 'Personal', pivot: { contributor_type: '' } }; + form.contributors.push(newContributor); +}; + +const onAddContributor = (person: Person) => { + if (form.contributors.filter((e) => e.id === person.id).length > 0) { + notify({ type: 'warning', title: 'Warning', text: 'person is already defined as contributor' }, 4000); + } else if (form.authors.filter((e) => e.id === person.id).length > 0) { + notify({ type: 'warning', title: 'Warning', text: 'person is already defined as author' }, 4000); + } else { + // person.pivot = { contributor_type: '' }; + // // person.pivot = { name_type: '', contributor_type: '' }; + form.contributors.push(person); + notify({ type: 'info', text: 'person has been successfully added as contributor' }, 4000); + } +}; + +const addKeyword = () => { + let newSubject: Subject = { value: '', language: '', type: 'uncontrolled' }; + //this.dataset.files.push(uploadedFiles[i]); + form.subjects.push(newSubject); +}; + +const addReference = () => { + let newReference = { value: '', label: '', relation: '', type: '' }; + //this.dataset.files.push(uploadedFiles[i]); + form.references.push(newReference); +}; + + +const removeReference = (key: any) => { + const reference = form.references[key]; + + // If the reference has an ID, it exists in the database + // and should be added to referencesToDelete + if (reference.id) { + // Initialize referencesToDelete array if it doesn't exist + if (!form.referencesToDelete) { + form.referencesToDelete = []; + } + + // Add to referencesToDelete + form.referencesToDelete.push(reference); + } + + // Remove from form.references array + form.references.splice(key, 1); +}; + +const restoreReference = (index: number) => { + // Get the reference from referencesToDelete + const reference = form.referencesToDelete[index]; + + // Add it back to form.references + form.references.push(reference); + + // Remove it from referencesToDelete + form.referencesToDelete.splice(index, 1); +}; + +const onMapInitialized = (newItem: any) => { + console.log(newItem); +}; + +const getFileSize = (file: File) => { + if (file.size > 1024) { + if (file.size > 1048576) { + return Math.round(file.size / 1048576) + 'mb'; + } else { + return Math.round(file.size / 1024) + 'kb'; + } + } else { + return file.size + 'b'; + } +} +</script> + +<style scoped> +.max-w-2xl { + max-width: 2xl; +} + +.text-2xl { + font-size: 2xl; +} + +.font-bold { + font-weight: bold; +} + +.mb-4 { + margin-bottom: 1rem; +} + +.block { + display: block; +} + +.text-gray-700 { + color: #4b5563; +} + +.shadow { + box-shadow: + 0 0 0 1px rgba(66, 72, 78, 0.05), + 0 1px 2px 0 rgba(66, 72, 78, 0.08), + 0 2px 4px 0 rgba(66, 72, 78, 0.12), + 0 4px 8px 0 rgba(66, 72, 78, 0.16); +} +</style> \ No newline at end of file diff --git a/resources/js/Pages/Editor/Dataset/Index.vue b/resources/js/Pages/Editor/Dataset/Index.vue index ef6156b..e154d48 100644 --- a/resources/js/Pages/Editor/Dataset/Index.vue +++ b/resources/js/Pages/Editor/Dataset/Index.vue @@ -2,7 +2,7 @@ // import { Head, Link, useForm, usePage } from '@inertiajs/inertia-vue3'; import { Head, usePage } from '@inertiajs/vue3'; import { ComputedRef } from 'vue'; -import { mdiSquareEditOutline, mdiAlertBoxOutline, mdiShareVariant, mdiBookEdit, mdiUndo } from '@mdi/js'; +import { mdiSquareEditOutline, mdiAlertBoxOutline, mdiShareVariant, mdiBookEdit, mdiUndo, mdiLibraryShelves } from '@mdi/js'; import { computed } from 'vue'; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; import SectionMain from '@/Components/SectionMain.vue'; @@ -96,8 +96,8 @@ const formatServerState = (state: string) => { <Head title="Editor Datasets" /> <SectionMain> - - + + <NotificationBar v-if="flash.message" color="success" :icon="mdiAlertBoxOutline"> {{ flash.message }} </NotificationBar> @@ -108,31 +108,30 @@ const formatServerState = (state: string) => { {{ flash.error }} </NotificationBar> + <!-- table --> <CardBox class="mb-6" has-table> <div v-if="props.datasets.data.length > 0"> <table> <thead> <tr> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> <!-- <Sort label="Dataset Title" attribute="title" :search="form.search" /> --> Title </th> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> Submitter </th> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> - <!-- <Sort label="Email" attribute="email" :search="form.search" /> --> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> State </th> - - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> Editor </th> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> Date of last modification </th> - <th scope="col" class="relative px-6 py-3" v-if="can.edit || can.delete"> + <th scope="col" class="relative px-6 py-3 dark:text-white" v-if="can.edit || can.delete"> <span class="sr-only">Actions</span> </th> </tr> @@ -141,70 +140,110 @@ const formatServerState = (state: string) => { <tbody class="bg-white divide-y divide-gray-200"> <tr v-for="dataset in props.datasets.data" :key="dataset.id" :class="[getRowClass(dataset)]"> - <td data-label="Login" class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> - <!-- <Link v-bind:href="stardust.route('user.show', [user.id])" + <td data-label="Login" + class="py-4 whitespace-nowrap text-gray-700 table-title"> + <!-- <Link v-bind:href="stardust.route('settings.user.show', [user.id])" class="no-underline hover:underline text-cyan-600 dark:text-cyan-400"> {{ user.login }} </Link> --> - <div class="text-sm font-medium">{{ dataset.main_title }}</div> + <!-- {{ user.id }} --> + {{ dataset.main_title }} </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td class="py-4 whitespace-nowrap text-gray-700"> <div class="text-sm">{{ dataset.user.login }}</div> </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td class="py-4 whitespace-nowrap text-gray-700"> <div class="text-sm"> {{ formatServerState(dataset.server_state) }}</div> + <div v-if="dataset.server_state === 'rejected_reviewer' && dataset.reject_reviewer_note" + class="inline-block relative ml-2 group"> + <button + class="w-5 h-5 rounded-full bg-gray-200 text-gray-600 text-xs flex items-center justify-center focus:outline-none hover:bg-gray-300"> + i + </button> + <div + class="absolute left-0 top-full mt-1 w-64 bg-white shadow-lg rounded-md p-3 text-xs text-left z-50 transform scale-0 origin-top-left transition-transform duration-100 group-hover:scale-100"> + <p class="text-gray-700 max-h-40 overflow-y-auto overflow-x-hidden whitespace-normal break-words"> + {{ dataset.reject_reviewer_note }} + </p> + <div class="absolute -top-1 left-1 w-2 h-2 bg-white transform rotate-45"> + </div> + </div> + </div> </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white" + <td class="py-4 whitespace-nowrap text-gray-700" v-if="dataset.server_state === 'released'"> <div class="text-sm" :title="dataset.server_date_modified"> Preferred reviewer: {{ dataset.preferred_reviewer }} </div> </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white" + <td class="py-4 whitespace-nowrap text-gray-700" v-else-if="dataset.server_state === 'editor_accepted' || dataset.server_state === 'rejected_reviewer'"> <div class="text-sm" :title="dataset.server_date_modified"> In approval by: {{ dataset.editor?.login }} </div> </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white" v-else> + <td class="py-4 whitespace-nowrap text-gray-700" v-else> <div class="text-sm">{{ dataset.editor?.login }}</div> </td> - <td data-label="modified" class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td data-label="modified" class="py-4 whitespace-nowrap text-gray-700"> <div class="text-sm" :title="dataset.server_date_modified"> {{ dataset.server_date_modified }} </div> </td> <td class="py-4 whitespace-nowrap text-right text-sm font-medium text-gray-700 dark:text-white"> - <BaseButtons type="justify-start lg:justify-end" no-wrap> + <div type="justify-start lg:justify-end" class="grid grid-cols-2 gap-x-2 gap-y-2" + no-wrap> + <BaseButton v-if="can.receive && (dataset.server_state == 'released')" :route-name="stardust.route('editor.dataset.receive', [dataset.id])" - color="info" :icon="mdiSquareEditOutline" :label="'Receive edit task'" - small /> + color="info" :icon="mdiSquareEditOutline" :label="'Receive edit task'" small + class="col-span-1" /> <BaseButton v-if="can.approve && (dataset.server_state == 'editor_accepted' || dataset.server_state == 'rejected_reviewer')" :route-name="stardust.route('editor.dataset.approve', [dataset.id])" - color="info" :icon="mdiShareVariant" :label="'Approve'" small /> + color="info" :icon="mdiShareVariant" :label="'Approve'" small + class="col-span-1" /> <BaseButton v-if="can.approve && (dataset.server_state == 'editor_accepted' || dataset.server_state == 'rejected_reviewer')" - :route-name="stardust.route('editor.dataset.reject', [dataset.id])" - color="info" :icon="mdiUndo" label="Reject" small> + :route-name="stardust.route('editor.dataset.reject', [dataset.id])" + color="info" :icon="mdiUndo" label="Reject" small class="col-span-1"> </BaseButton> + <BaseButton + v-if="can.edit && (dataset.server_state == 'editor_accepted' || dataset.server_state == 'rejected_reviewer')" + :route-name="stardust.route('editor.dataset.edit', [Number(dataset.id)])" + color="info" :icon="mdiSquareEditOutline" :label="'Edit'" small + class="col-span-1"> + </BaseButton> + + <BaseButton + v-if="can.edit && (dataset.server_state == 'editor_accepted' || dataset.server_state == 'rejected_reviewer')" + :route-name="stardust.route('editor.dataset.categorize', [dataset.id])" + color="info" :icon="mdiLibraryShelves" :label="'Classify'" small + class="col-span-1"> + </BaseButton> + + <BaseButton v-if="can.publish && (dataset.server_state == 'reviewed')" + :route-name="stardust.route('editor.dataset.rejectToReviewer', [dataset.id])" + color="info" :icon="mdiUndo" :label="'Reject To Reviewer'" small + class="col-span-1" /> <BaseButton v-if="can.publish && (dataset.server_state == 'reviewed')" :route-name="stardust.route('editor.dataset.publish', [dataset.id])" - color="info" :icon="mdiBookEdit" :label="'Publish'" small /> + color="info" :icon="mdiBookEdit" :label="'Publish'" small + class="col-span-1" /> <BaseButton v-if="can.publish && (dataset.server_state == 'published' && !dataset.identifier)" :route-name="stardust.route('editor.dataset.doi', [dataset.id])" - color="info" :icon="mdiBookEdit" :label="'Mint DOI'" small /> + color="info" :icon="mdiBookEdit" :label="'Mint DOI'" small + class="col-span-1 last-in-row" /> - </BaseButtons> + </div> </td> </tr> </tbody> @@ -231,3 +270,17 @@ const formatServerState = (state: string) => { </SectionMain> </LayoutAuthenticated> </template> + + +<style scoped lang="css"> +.table-title { + max-width: 200px; + /* set a maximum width */ + overflow: hidden; + /* hide overflow */ + text-overflow: ellipsis; + /* show ellipsis for overflowed text */ + white-space: nowrap; + /* prevent wrapping */ +} +</style> \ No newline at end of file diff --git a/resources/js/Pages/Editor/Dataset/Publish.vue b/resources/js/Pages/Editor/Dataset/Publish.vue index d6929d2..5df3958 100644 --- a/resources/js/Pages/Editor/Dataset/Publish.vue +++ b/resources/js/Pages/Editor/Dataset/Publish.vue @@ -10,7 +10,7 @@ import FormControl from '@/Components/FormControl.vue'; import BaseButton from '@/Components/BaseButton.vue'; import BaseButtons from '@/Components/BaseButtons.vue'; import { stardust } from '@eidellev/adonis-stardust/client'; -import { mdiArrowLeftBoldOutline, mdiReiterate } from '@mdi/js'; +import { mdiArrowLeftBoldOutline, mdiReiterate, mdiBookEdit, mdiUndo } from '@mdi/js'; import FormValidationErrors from '@/Components/FormValidationErrors.vue'; const props = defineProps({ @@ -18,6 +18,10 @@ const props = defineProps({ type: Object, default: () => ({}), }, + can: { + type: Object, + default: () => ({}), + }, }); const flash: Ref<any> = computed(() => { @@ -93,7 +97,11 @@ const handleSubmit = async (e) => { </p> <BaseButtons> <BaseButton type="submit" color="info" label="Set published" - :class="{ 'opacity-25': form.processing }" :disabled="form.processing" /> + :class="{ 'opacity-25': form.processing }" :disabled="form.processing" :icon="mdiBookEdit" small /> + <BaseButton v-if="can.reject && (dataset.server_state == 'reviewed')" + :route-name="stardust.route('editor.dataset.rejectToReviewer', [dataset.id])" + color="info" :icon="mdiUndo" :label="'Reject To Reviewer'" small + class="col-span-1" /> </BaseButtons> </template> </CardBox> diff --git a/resources/js/Pages/Editor/Dataset/RejectToReviewer.vue b/resources/js/Pages/Editor/Dataset/RejectToReviewer.vue new file mode 100644 index 0000000..7889c2c --- /dev/null +++ b/resources/js/Pages/Editor/Dataset/RejectToReviewer.vue @@ -0,0 +1,117 @@ +<script setup lang="ts"> +import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; +import SectionMain from '@/Components/SectionMain.vue'; +import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; +import { useForm, Head, usePage } from '@inertiajs/vue3'; +import { computed, Ref } from 'vue'; +import CardBox from '@/Components/CardBox.vue'; +import FormField from '@/Components/FormField.vue'; +import FormControl from '@/Components/FormControl.vue'; +import BaseButton from '@/Components/BaseButton.vue'; +import BaseButtons from '@/Components/BaseButtons.vue'; +import { stardust } from '@eidellev/adonis-stardust/client'; +import { mdiArrowLeftBoldOutline, mdiReiterate } from '@mdi/js'; +import FormValidationErrors from '@/Components/FormValidationErrors.vue'; + +const props = defineProps({ + dataset: { + type: Object, + default: () => ({}), + }, +}); + +// Define the computed property for the label +const computedLabel = computed(() => { + return `Reject to reviewer: ${props.dataset.reviewer?.login || 'Unknown User'}`; +}); +const computedEmailLabel = computed(() => { + return props.dataset.reviewer?.email || ''; +}); + + +const flash: Ref<any> = computed(() => { + return usePage().props.flash; +}); +const errors: Ref<any> = computed(() => { + return usePage().props.errors; +}); + +const form = useForm({ + server_state: 'rejected_to_reviewer', + reject_editor_note: '', + send_email: false, +}); + +const handleSubmit = async (e: SubmitEvent) => { + e.preventDefault(); + await form.put(stardust.route('editor.dataset.rejectToReviewerUpdate', [props.dataset.id])); + // await form.put(stardust.route('editor.dataset.update', [props.dataset.id])); +}; +</script> + +<template> + <LayoutAuthenticated> + + <Head title="Reject reviewed dataset" /> + <SectionMain> + <SectionTitleLineWithButton :icon="mdiReiterate" title="Reject reviewed dataset to reviewer" main> + <BaseButton :route-name="stardust.route('editor.dataset.list')" :icon="mdiArrowLeftBoldOutline" + label="Back" color="white" rounded-full small /> + </SectionTitleLineWithButton> + + <CardBox form @submit.prevent="handleSubmit"> + <FormValidationErrors v-bind:errors="errors" /> + + + <FormField label="server state" :class="{ 'text-red-400': form.errors.server_state }"> + <FormControl v-model="form.server_state" type="text" placeholder="-- server state --" + :is-read-only="true" :error="form.errors.server_state"> + <div class="text-red-400 text-sm" v-if="form.errors.server_state"> + {{ form.errors.server_state }} + </div> + </FormControl> + </FormField> + <FormField label="reject note" :class="{ 'text-red-400': form.errors.reject_editor_note }"> + <FormControl v-model="form.reject_editor_note" type="textarea" + placeholder="-- reject note for reviewer --" :error="form.errors.reject_editor_note"> + <div class="text-red-400 text-sm" v-if="form.errors.reject_editor_note"> + {{ form.errors.reject_editor_note }} + </div> + </FormControl> + </FormField> + + <!-- <FormControl + type="checkbox" + v-model="form.send_email" + :error="form.errors.send_email"> + </FormControl> --> + + <FormField label="Email Notification"> + <label for="send_email" class="flex items-center mr-6 mb-3"> + <input type="checkbox" id="send_email" v-model="form.send_email" class="mr-2" /> + <span class="check"></span> + <a class="pl-2 " target="_blank">send email to reviewer + <span class="text-blue-600 hover:underline"> + {{ computedEmailLabel }} + </span> + </a> + </label> + </FormField> + + <div v-if="flash && flash.warning" class="flex flex-col mt-6 animate-fade-in"> + <div class="bg-yellow-500 border-l-4 border-orange-400 text-white p-4" role="alert"> + <p class="font-bold">Be Warned</p> + <p>{{ flash.warning }}</p> + </div> + </div> + + <template #footer> + <BaseButtons> + <BaseButton type="submit" color="info" :label="computedLabel" + :class="{ 'opacity-25': form.processing }" :disabled="form.processing" /> + </BaseButtons> + </template> + </CardBox> + </SectionMain> + </LayoutAuthenticated> +</template> diff --git a/resources/js/Pages/Errors/ServerError.vue b/resources/js/Pages/Errors/ServerError.vue index 5d97ede..74bcdc0 100644 --- a/resources/js/Pages/Errors/ServerError.vue +++ b/resources/js/Pages/Errors/ServerError.vue @@ -20,11 +20,13 @@ import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton. import BaseButton from '@/Components/BaseButton.vue'; import { mdiLightbulbAlert, mdiArrowLeftBoldOutline } from '@mdi/js'; import { stardust } from '@eidellev/adonis-stardust/client'; - -@Component({ - // options: { - // layout: DefaultLayout, - // }, +import LayoutGuest from '@/Layouts/LayoutGuest.vue'; + + + @Component({ + options: { + layout: LayoutGuest, + }, name: 'AppComponent', components: { diff --git a/resources/js/Pages/Errors/postgres_error.vue b/resources/js/Pages/Errors/postgres_error.vue new file mode 100644 index 0000000..ed5dd59 --- /dev/null +++ b/resources/js/Pages/Errors/postgres_error.vue @@ -0,0 +1,71 @@ +<template> + <div class="min-h-screen flex items-center justify-center bg-gray-100"> + <div class="max-w-md w-full p-6 bg-white rounded-md shadow-md"> + <h1 class="text-2xl font-bold text-red-500 mb-4">{{ status }}</h1> + <p class="text-gray-700 mb-4">{{ message }}</p> + <div class="text-sm text-gray-500 mb-4"> + <p>Error Code: {{ details.code }}</p> + <p>Type: {{ details.type }}</p> + <div v-for="(port, index) in details.ports" :key="index"> + <p>Connection attempt {{ index + 1 }}: {{ port.address }}:{{ port.port }}</p> + </div> + </div> + <SectionTitleLineWithButton :icon="mdiLightbulbAlert" :title="'Database Error'" :main="true"> + <BaseButton @click.prevent="handleAction" :icon="mdiArrowLeftBoldOutline" label="Dashboard" + color="white" rounded-full small /> + </SectionTitleLineWithButton> + </div> + </div> +</template> + +<script lang="ts"> +import { Component, Vue, Prop } from 'vue-facing-decorator'; +import { Link, router } from '@inertiajs/vue3'; +import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; +import BaseButton from '@/Components/BaseButton.vue'; +import { mdiLightbulbAlert, mdiArrowLeftBoldOutline } from '@mdi/js'; +import { stardust } from '@eidellev/adonis-stardust/client'; +import LayoutGuest from '@/Layouts/LayoutGuest.vue'; + +@Component({ + options: { + layout: LayoutGuest, + }, + name: 'PostgresError', + components: { + Link, + BaseButton, + SectionTitleLineWithButton, + }, +}) +export default class AppComponent extends Vue { + @Prop({ + type: String, + default: '', + }) + status: string; + + @Prop({ + type: String, + default: '', + }) + message: string; + + @Prop({ + type: Object, + default: () => ({}), + }) + details: { + code: string; + type: string; + ports: Array<{ port: number; address: string }>; + }; + + mdiLightbulbAlert = mdiLightbulbAlert; + mdiArrowLeftBoldOutline = mdiArrowLeftBoldOutline; + + public async handleAction() { + await router.get(stardust.route('dashboard')); + } +} +</script> diff --git a/resources/js/Pages/Map.vue b/resources/js/Pages/Map.vue index 15d40e3..4c19625 100644 --- a/resources/js/Pages/Map.vue +++ b/resources/js/Pages/Map.vue @@ -1,13 +1,13 @@ <script setup lang="ts"> import { Head } from '@inertiajs/vue3'; import { ref, Ref } from 'vue'; -import { mdiChartTimelineVariant } from '@mdi/js'; +import { mdiChartTimelineVariant, mdiGithub } from '@mdi/js'; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; import SectionMain from '@/Components/SectionMain.vue'; import BaseButton from '@/Components/BaseButton.vue'; import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; import { MapOptions } from '@/Components/Map/MapOptions'; -import { stardust } from '@eidellev/adonis-stardust/client'; +// import { stardust } from '@eidellev/adonis-stardust/client'; import SearchMap from '@/Components/Map/SearchMap.vue'; import { OpensearchDocument } from '@/Dataset'; @@ -48,14 +48,15 @@ const mapOptions: MapOptions = { <template> <LayoutAuthenticated :showAsideMenu="false"> + <Head title="Map" /> <!-- <section class="p-6" v-bind:class="containerMaxW"> --> <SectionMain> <SectionTitleLineWithButton v-bind:icon="mdiChartTimelineVariant" title="Tethys Map" main> - <!-- <BaseButton href="https://gitea.geologie.ac.at/geolba/tethys" target="_blank" :icon="mdiGithub" - label="Star on Gitea" color="contrast" rounded-full small /> --> - <BaseButton :route-name="stardust.route('app.login.show')" label="Login" color="white" rounded-full small /> + <BaseButton href="https://gitea.geosphere.at/geolba/tethys" target="_blank" :icon="mdiGithub" + label="Star on GeoSPhere Forgejo" color="contrast" rounded-full small /> + <!-- <BaseButton :route-name="stardust.route('app.login.show')" label="Login" color="white" rounded-full small /> --> </SectionTitleLineWithButton> <!-- <SectionBannerStarOnGitea /> --> @@ -80,19 +81,20 @@ const mapOptions: MapOptions = { <div v-for="author in dataset.author" :key="author" class="mb-1">{{ author }}</div> </div> <div class="mt-4"> - <span class="inline-block bg-gray-200 rounded-full px-3 py-1 text-sm font-semibold text-gray-700 mr-2 mb-2"> + <span + class="inline-block bg-gray-200 rounded-full px-3 py-1 text-sm font-semibold text-gray-700 mr-2 mb-2"> {{ dataset.year }} </span> - <span class="inline-block bg-gray-200 rounded-full px-3 py-1 text-sm font-semibold text-gray-700 mr-2 mb-2"> + <span + class="inline-block bg-gray-200 rounded-full px-3 py-1 text-sm font-semibold text-gray-700 mr-2 mb-2"> {{ dataset.language }} </span> </div> <p> <span class="label"><i class="fas fa-file"></i> {{ dataset.doctype }}</span> <!-- <span>Licence: {{ document.licence }}</span> --> - <span v-if="openAccessLicences.includes(dataset.licence)" class="label titlecase" - ><i class="fas fa-lock-open"></i> Open Access</span - > + <span v-if="openAccessLicences.includes(dataset.licence)" class="label titlecase"><i + class="fas fa-lock-open"></i> Open Access</span> </p> </div> </div> diff --git a/resources/js/Pages/Reviewer/Dataset/Index.vue b/resources/js/Pages/Reviewer/Dataset/Index.vue index 84d56e7..36d2351 100644 --- a/resources/js/Pages/Reviewer/Dataset/Index.vue +++ b/resources/js/Pages/Reviewer/Dataset/Index.vue @@ -51,6 +51,8 @@ const getRowClass = (dataset) => { rowclass = 'bg-released'; } else if (dataset.server_state == 'published') { rowclass = 'bg-published'; + } else if (dataset.server_state == 'rejected_to_reviewer') { + rowclass = 'bg-rejected-reviewer'; } else { rowclass = ''; } @@ -96,14 +98,14 @@ const formatServerState = (state: string) => { <table> <thead> <tr> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> <!-- <Sort label="Dataset Title" attribute="title" :search="form.search" /> --> Title </th> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> ID </th> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> <!-- <Sort label="Email" attribute="email" :search="form.search" /> --> State </th> @@ -111,10 +113,10 @@ const formatServerState = (state: string) => { <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> Editor </th> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> Remaining Time </th> - <th scope="col" class="relative px-6 py-3" v-if="can.edit || can.delete"> + <th scope="col" class="relative px-6 py-3 dark:text-white" v-if="can.edit || can.delete"> <span class="sr-only">Actions</span> </th> </tr> @@ -122,39 +124,52 @@ const formatServerState = (state: string) => { <tbody> <tr v-for="dataset in props.datasets.data" :key="dataset.id" :class="[getRowClass(dataset)]"> - <td data-label="Login" class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> - <!-- <Link v-bind:href="stardust.route('user.show', [user.id])" - class="no-underline hover:underline text-cyan-600 dark:text-cyan-400"> - {{ user.login }} - </Link> --> - <div class="text-sm font-medium">{{ dataset.main_title }}</div> + <td data-label="Login" + class="py-4 whitespace-nowrap text-gray-700"> + <div class="text-sm table-title">{{ dataset.main_title }}</div> </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td class="py-4 whitespace-nowrap text-gray-700"> <div class="text-sm">{{ dataset.id }}</div> </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td class="py-4 whitespace-nowrap text-gray-700"> <div class="text-sm">{{ formatServerState(dataset.server_state) }}</div> + <div v-if="dataset.server_state === 'rejected_to_reviewer' && dataset.reject_editor_note" + class="inline-block relative ml-2 group"> + <button + class="w-5 h-5 rounded-full bg-gray-200 text-gray-600 text-xs flex items-center justify-center focus:outline-none hover:bg-gray-300"> + i + </button> + <div + class="absolute left-0 top-full mt-1 w-64 bg-white shadow-lg rounded-md p-3 text-xs text-left z-50 transform scale-0 origin-top-left transition-transform duration-100 group-hover:scale-100"> + <p class="text-gray-700 max-h-40 overflow-y-auto overflow-x-hidden whitespace-normal break-words"> + {{ dataset.reject_editor_note }} + </p> + <div class="absolute -top-1 left-1 w-2 h-2 bg-white transform rotate-45"> + </div> + </div> + </div> </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td class="py-4 whitespace-nowrap text-gray-700"> <div class="text-sm">{{ dataset.editor?.login }}</div> </td> - <td data-label="modified" class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td data-label="modified" class="py-4 whitespace-nowrap text-gray-700"> <div class="text-sm" :title="dataset.remaining_time"> {{ dataset.remaining_time + ' days' }} </div> </td> <td - class="py-4 whitespace-nowrap text-right text-sm font-medium text-gray-700 dark:text-white"> + class="py-4 whitespace-nowrap text-right text-sm font-medium text-gray-700"> <BaseButtons type="justify-start lg:justify-end" no-wrap> - <BaseButton v-if="can.review && (dataset.server_state == 'approved')" + <BaseButton + v-if="can.reject && (dataset.server_state == 'approved' || dataset.server_state == 'rejected_to_reviewer')" :route-name="stardust.route('reviewer.dataset.review', [dataset.id])" - color="info" :icon="mdiGlasses" :label="'Review'" small /> + color="info" :icon="mdiGlasses" :label="'View'" small /> <BaseButton - v-if="can.reject && (dataset.server_state == 'approved')" + v-if="can.reject && (dataset.server_state == 'approved' || dataset.server_state == 'rejected_to_reviewer')" :route-name="stardust.route('reviewer.dataset.reject', [dataset.id])" color="info" :icon="mdiReiterate" :label="'Reject'" small /> </BaseButtons> @@ -185,3 +200,16 @@ const formatServerState = (state: string) => { </LayoutAuthenticated> </template> +<style scoped lang="css"> +.table-title { + max-width: 200px; + /* set a maximum width */ + overflow: hidden; + /* hide overflow */ + text-overflow: ellipsis; + /* show ellipsis for overflowed text */ + white-space: nowrap; + /* prevent wrapping */ +} + +</style> \ No newline at end of file diff --git a/resources/js/Pages/Reviewer/Dataset/Review.vue b/resources/js/Pages/Reviewer/Dataset/Review.vue index 4213bc2..b33500b 100644 --- a/resources/js/Pages/Reviewer/Dataset/Review.vue +++ b/resources/js/Pages/Reviewer/Dataset/Review.vue @@ -10,15 +10,23 @@ import BaseButtons from '@/Components/BaseButtons.vue'; import { stardust } from '@eidellev/adonis-stardust/client'; import { mdiArrowLeftBoldOutline, mdiGlasses } from '@mdi/js'; import FormValidationErrors from '@/Components/FormValidationErrors.vue'; +import { mdiReiterate, mdiBookOpenPageVariant, mdiFinance } from '@mdi/js'; +import MapComponentView from '@/Components/Map/MapComponentView.vue'; +import IconSvg from '@/Components/Icons/IconSvg.vue'; +import CardBoxSimple from '@/Components/CardBoxSimple.vue'; const props = defineProps({ dataset: { type: Object, default: () => ({}), }, - fields: { + // fields: { + // type: Object, + // required: true, + // }, + can: { type: Object, - required: true, + default: () => ({}), }, }); @@ -29,18 +37,6 @@ const errors: Ref<any> = computed(() => { return usePage().props.errors; }); -// const form = useForm({ -// preferred_reviewer: '', -// preferred_reviewer_email: '', -// preferation: 'yes_preferation', - -// // preferation: '', -// // isPreferationRequired: false, -// }); - - -// const isPreferationRequired = computed(() => form.preferation === 'yes_preferation'); - const handleSubmit = async (e) => { e.preventDefault(); @@ -48,6 +44,19 @@ const handleSubmit = async (e) => { // await form.put(stardust.route('dataset.releaseUpdate', [props.dataset.id])); // // await form.put(stardust.route('editor.dataset.update', [props.dataset.id])); }; + + +const getFileSize = (file: File) => { + if (file.size > 1024) { + if (file.size > 1048576) { + return Math.round(file.size / 1048576) + 'mb'; + } else { + return Math.round(file.size / 1024) + 'kb'; + } + } else { + return file.size + 'b'; + } +} </script> <template> @@ -56,10 +65,10 @@ const handleSubmit = async (e) => { <Head title="Review dataset" /> <SectionMain> <SectionTitleLineWithButton :icon="mdiGlasses" title="Review approved dataset" main> - <BaseButton :route-name="stardust.route('reviewer.dataset.list')" :icon="mdiArrowLeftBoldOutline" label="Back" - color="white" rounded-full small /> + <BaseButton :route-name="stardust.route('reviewer.dataset.list')" :icon="mdiArrowLeftBoldOutline" + label="Back" color="white" rounded-full small /> </SectionTitleLineWithButton> - <CardBox form @submit.prevent="handleSubmit"> + <component is="form" form @submit.prevent="handleSubmit"> <FormValidationErrors v-bind:errors="errors" /> <div v-if="flash && flash.warning" class="flex flex-col mt-6 animate-fade-in"> @@ -70,26 +79,545 @@ const handleSubmit = async (e) => { </div> - <div class="flex flex-col"> - <div class="flex flex-row items-center justify-between dark:bg-slate-900 bg-gray-200 p-2 mb-2" - v-for="(fieldValue, field) in fields" :key="field"> + <!-- <div class="mb-4"> --> - <label :for="field" class="font-bold h-6 mt-3 text-xs leading-8 uppercase">{{ field }}</label> - <span class="text-sm text-gray-600" v-html="fieldValue"></span> + <CardBoxSimple> + <div class="mb-6"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">Language</h4> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm"> + <div class="flex items-center"> + <IconSvg path="language" :size="20" + className="mr-2 text-emerald-600 dark:text-emerald-400" /> + <span class="text-emerald-700 dark:text-emerald-300 font-medium"> + {{ dataset.language || 'Not specified' }} + </span> + </div> + </div> </div> - </div> + + <!-- Licenses --> + <div class="mb-6"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Licenses</h4> + <div v-if="dataset.licenses && dataset.licenses.length > 0" class="space-y-2"> + <div v-for="license in dataset.licenses" :key="license.id" class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm hover:bg-emerald-100 dark:hover:bg-emerald-900/50 transition-colors"> + <div class="flex items-center"> + <IconSvg path="license" :size="20" + className="mr-2 text-emerald-600 dark:text-emerald-400" /> + <span class="text-emerald-700 dark:text-emerald-300"> + {{ license.name }} + </span> + </div> + </div> + </div> + <div v-else class="text-center py-4"> + <p class="text-gray-500 dark:text-gray-400 italic">No licenses specified</p> + </div> + <div v-if="dataset.licenses && dataset.licenses.length > 0" + class="mt-3 text-xs text-gray-500 dark:text-gray-400"> + Total licenses: {{ dataset.licenses.length }} + </div> + </div> + + <div class="flex flex-col md:flex-row gap-4 mb-6"> + <!-- (3) dataset_type --> + <div class="w-full mx-2 flex-1"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">Dataset Type</h4> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm"> + <div class="flex items-center"> + <IconSvg path="book" :size="20" + className="mr-2 text-emerald-600 dark:text-emerald-400" /> + <span class="text-emerald-700 dark:text-emerald-300 font-medium"> + {{ dataset.type || 'Not specified' }} + </span> + </div> + </div> + </div> + + <!-- (4) creating_corporation --> + <div class="w-full mx-2 flex-1"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">Creating + Corporation + </h4> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm"> + <div class="flex items-center"> + <IconSvg path="building" :size="20" + className="mr-2 text-emerald-600 dark:text-emerald-400" /> + <span class="text-emerald-700 dark:text-emerald-300 font-medium"> + {{ dataset.creating_corporation || 'Not specified' }} + </span> + </div> + </div> + </div> + </div> + + <div class="flex flex-col md:flex-row gap-4 mb-6"> + <!-- (9) project_id --> + <div class="w-full mx-2 flex-1"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">Project</h4> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm"> + <span class="text-emerald-700 dark:text-emerald-300"> + {{ dataset.project?.label || 'Not specified' }} + </span> + </div> + </div> + <!-- (10) embargo_date --> + <div class="w-full mx-2 flex-1"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">Embargo Date</h4> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm"> + <span v-if="dataset.embargo_date" class="text-emerald-700 dark:text-emerald-300"> + {{ dataset.embargo_date }} + </span> + <span v-else class="text-gray-500 dark:text-gray-400 italic"> + No embargo date set + </span> + </div> + </div> + </div> + + </CardBoxSimple> + + <!-- (5) titles --> + <CardBox class="mb-6 shadow" :has-form-data="false" title="Titles" :icon="mdiFinance" + :show-header-icon="false"> + <div class="p-4"> + <!-- Main Title (highlighted) --> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Main Title</h4> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border-l-4 border border-emerald-300 dark:border-emerald-700 + rounded-lg p-4 shadow-sm mb-4"> + <div class="flex justify-between items-start mb-2"> + <span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium bg-emerald-200 + dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200"> + {{ dataset.titles[0].language }} + </span> + <span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium bg-emerald-200 + dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200"> + Main + </span> + </div> + <h3 + class="text-lg font-medium text-emerald-800 dark:text-emerald-200 whitespace-pre-line break-words overflow-wrap-anywhere"> + {{ dataset.titles[0].value }} + </h3> + </div> + + <!-- Additional titles --> + <div v-if="dataset.titles.length > 1"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Additional Titles + </h4> + <div class="grid gap-3"> + <template v-for="(title, index) in dataset.titles" :key="index"> + <div v-if="title.type != 'Main'" + class="bg-emerald-50/70 dark:bg-emerald-900/20 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm hover:bg-emerald-50 dark:hover:bg-emerald-900/30 transition-colors"> + <div class="flex justify-between items-start mb-2"> + <span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium + bg-emerald-100 dark:bg-emerald-800/70 text-emerald-700 dark:text-emerald-300"> + {{ title.language }} + </span> + <span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium + bg-emerald-100 dark:bg-emerald-800/70 text-emerald-700 dark:text-emerald-300"> + {{ title.type }} + </span> + </div> + <p + class="text-emerald-700 dark:text-emerald-300 font-medium whitespace-pre-line break-words overflow-wrap-anywhere"> + {{ title.value }} + </p> + </div> + </template> + </div> + </div> + <div class="mt-3 text-xs text-gray-500 dark:text-gray-400"> + Total titles: {{ dataset.titles.length }} + </div> + </div> + </CardBox> + + <!-- (6) descriptions --> + <CardBox class="mb-6 shadow" :has-form-data="false" title="Descriptions" :icon="mdiFinance" + :show-header-icon="false"> + <!-- Main Abstract (highlighted) --> + <div class="p-4"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Main Abstract</h4> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border-l-4 border border-emerald-300 dark:border-emerald-700 + rounded-lg p-4 shadow-sm mb-4"> + <div class="flex justify-between items-start mb-2"> + <span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium bg-emerald-200 + dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200"> + {{ dataset.descriptions[0].language }} + </span> + <span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium bg-emerald-200 + dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200"> + Abstract + </span> + </div> + <div class="prose prose-emerald dark:prose-invert max-w-none"> + <p + class="text-emerald-800 dark:text-emerald-200 whitespace-pre-line break-words overflow-wrap-anywhere"> + {{ dataset.descriptions[0].value }} + </p> + </div> + </div> + <!-- Additional descriptions --> + <div v-if="dataset.descriptions.length > 1"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Additional + Descriptions</h4> + <div class="grid gap-3"> + <div v-for="(item, index) in dataset.descriptions" :key="index"> + <div v-if="item.type != 'Abstract'" class="bg-emerald-50/70 dark:bg-emerald-900/20 border border-emerald-200 + dark:border-emerald-800 + rounded-lg p-3 shadow-sm hover:bg-emerald-50 dark:hover:bg-emerald-900/30 + transition-colors"> + <div class="flex justify-between items-start mb-2"> + <span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium + bg-emerald-100 dark:bg-emerald-800/70 text-emerald-700 dark:text-emerald-300"> + {{ item.language }} + </span> + <span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium + bg-emerald-100 dark:bg-emerald-800/70 text-emerald-700 dark:text-emerald-300"> + {{ item.type }} + </span> + </div> + <p + class="text-emerald-700 dark:text-emerald-300 text-sm whitespace-pre-line break-words overflow-wrap-anywhere"> + {{ item.value }} + </p> + </div> + </div> + </div> + </div> + <div class="mt-3 text-xs text-gray-500 dark:text-gray-400"> + Total descriptions: {{ dataset.descriptions.length }} + </div> + </div> + </CardBox> - + <!-- (7) authors --> + <CardBox class="mb-6 shadow" has-table title="Creators" :icon="mdiBookOpenPageVariant" + :show-header-icon="false"> + <div v-if="dataset.authors.length === 0" class="text-center py-6"> + <p class="text-gray-500 dark:text-gray-400 italic">No authors defined</p> + </div> + <div v-else class="p-4"> + <!-- <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Dataset Authors:</h4> --> + <div class="grid gap-3"> + <div v-for="(author, index) in dataset.authors" :key="index" class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm hover:bg-emerald-100 dark:hover:bg-emerald-900/50 transition-colors"> + <div class="flex flex-col md:flex-row md:items-center"> + <div class="flex-1"> + <p class="font-medium text-emerald-700 dark:text-emerald-300"> + {{ author.academic_title }} {{ author.first_name }} {{ author.last_name + }} + </p> + <div class="mt-2 grid grid-cols-1 md:grid-cols-2 gap-2"> + <div v-if="author.email" class="flex items-center text-sm"> + <IconSvg path="email" :size="16" + className="mr-1 text-emerald-600 dark:text-emerald-400" /> + <span class="text-emerald-600 dark:text-emerald-400">{{ author.email + }}</span> + </div> + <div v-if="author.identifier_orcid" class="flex items-center text-sm"> + <IconSvg path="idCard" :size="16" + className="mr-1 text-emerald-600 dark:text-emerald-400"> + </IconSvg> + <span class="text-emerald-600 dark:text-emerald-400">ORCID: {{ + author.identifier_orcid + }}</span> + </div> + </div> + </div> + <div v-if="author.academic_title" class="mt-2 md:mt-0 md:ml-4"> + <span class="bg-emerald-200 dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200 + text-xs px-2 py-1 rounded-full"> + {{ author.academic_title }} + </span> + </div> + </div> + </div> + </div> + <div v-if="dataset.authors.length > 0" class="mt-3 text-xs text-gray-500 dark:text-gray-400"> + Total authors: {{ dataset.authors.length }} + </div> + </div> + </CardBox> - <template #footer> - <BaseButtons> - <!-- <BaseButton type="submit" color="info" label="Receive" + <!-- (8) contributors --> + <CardBox class="mb-6 shadow" has-table title="Contributors" :icon="mdiBookOpenPageVariant" + :show-header-icon="false"> + <div v-if="dataset.contributors.length === 0" class="text-center py-6"> + <p class="text-gray-500 dark:text-gray-400 italic">No contributors defined</p> + </div> + <div v-else class="p-4"> + <!-- <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Dataset Contributors: + </h4> --> + <div class="grid gap-3"> + <div v-for="(contributor, index) in dataset.contributors" :key="index" class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm hover:bg-emerald-100 dark:hover:bg-emerald-900/50 transition-colors"> + <div class="flex flex-col md:flex-row md:items-center"> + <div class="flex-1"> + <div class="flex flex-wrap items-center gap-2"> + <p class="font-medium text-emerald-700 dark:text-emerald-300"> + {{ contributor.academic_title }} {{ contributor.first_name }} {{ + contributor.last_name + }} + </p> + <span v-if="contributor.pivot_contributor_type" class="bg-emerald-200 dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200 + text-xs px-2 py-1 rounded-full"> + {{ contributor.pivot_contributor_type }} + </span> + </div> + <div class="mt-2 grid grid-cols-1 md:grid-cols-2 gap-2"> + <div v-if="contributor.email" class="flex items-center text-sm"> + <IconSvg path="email" :size="16" + className="mr-1 text-emerald-600 dark:text-emerald-400" /> + <span class="text-emerald-600 dark:text-emerald-400">{{ + contributor.email }}</span> + </div> + <div v-if="contributor.identifier_orcid" class="flex items-center text-sm"> + <IconSvg path="idCard" :size="16" + className="mr-1 text-emerald-600 dark:text-emerald-400"> + </IconSvg> + <span class="text-emerald-600 dark:text-emerald-400">ORCID: {{ + contributor.identifier_orcid }}</span> + </div> + </div> + </div> + <div v-if="contributor.academic_title" class="mt-2 md:mt-0 md:ml-4"> + <span class="bg-emerald-200 dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200 + text-xs px-2 py-1 rounded-full"> + {{ contributor.academic_title }} + </span> + </div> + </div> + </div> + </div> + <div v-if="dataset.contributors.length > 0" + class="mt-3 text-xs text-gray-500 dark:text-gray-400"> + Total contributors: {{ dataset.contributors.length }} + </div> + </div> + </CardBox> + + + <!-- Map component --> + <CardBoxSimple> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Geographic Coverage</h4> + <!-- Map container with emerald styling --> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg shadow-sm overflow-hidden"> + <!-- The actual map component --> + <div class="h-64 rounded-md overflow-hidden"> + <!-- Use the simplified map component --> + <MapComponentView v-if="dataset.coverage" :coverage="dataset.coverage" height="250px" + :mapId="'dataset-review-map'" /> + </div> + <!-- Optional: Add a caption or description --> + <div class="mt-2 text-xs text-emerald-600 dark:text-emerald-400 text-center"> + Geographic extent of the dataset + </div> + </div> + <!-- Coordinates display below the map --> + <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-3 mt-3"> + <!-- x min --> + <div> + <label class="block text-xs font-medium text-gray-600 dark:text-gray-400 mb-1">X Min + (Longitude)</label> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-2.5 shadow-sm"> + <span class="text-emerald-700 dark:text-emerald-300"> + {{ dataset.coverage.x_min }} + </span> + </div> + </div> + + <!-- x max --> + <div> + <label class="block text-xs font-medium text-gray-600 dark:text-gray-400 mb-1">X Max + (Longitude)</label> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-2.5 shadow-sm"> + <span class="text-emerald-700 dark:text-emerald-300"> + {{ dataset.coverage.x_max }} + </span> + </div> + </div> + + <!-- y min --> + <div> + <label class="block text-xs font-medium text-gray-600 dark:text-gray-400 mb-1">Y Min + (Latitude)</label> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-2.5 shadow-sm"> + <span class="text-emerald-700 dark:text-emerald-300"> + {{ dataset.coverage.y_min }} + </span> + </div> + </div> + + <!-- y max --> + <div> + <label class="block text-xs font-medium text-gray-600 dark:text-gray-400 mb-1">Y Max + (Latitude)</label> + <div class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-2.5 shadow-sm"> + <span class="text-emerald-700 dark:text-emerald-300"> + {{ dataset.coverage.y_max }} + </span> + </div> + </div> + </div> + </CardBoxSimple> + + <!-- References --> + <CardBoxSimple> + <div v-if="dataset.references.length === 0" class="text-center py-6"> + <p class="text-gray-500 dark:text-gray-400 italic">No references added.</p> + </div> + <div v-else class="p-4"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Dataset References: + </h4> + <div class="grid gap-3"> + <div v-for="(item, index) in dataset.references" :key="index" class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm hover:bg-emerald-100 dark:hover:bg-emerald-900/50 transition-colors"> + <div class="flex flex-col md:flex-row md:items-center md:justify-between"> + <div class="flex-1"> + <p class="font-medium text-emerald-700 dark:text-emerald-300">{{ item.value + }}</p> + <p class="text-sm text-emerald-600 dark:text-emerald-400 mt-1">{{ item.label + }}</p> + </div> + <div class="flex mt-2 md:mt-0 space-x-2"> + <span class="bg-emerald-200 dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200 + text-xs px-2 py-1 rounded-full"> + {{ item.type }} + </span> + <span class="bg-emerald-200 dark:bg-emerald-800 text-emerald-800 dark:text-emerald-200 + text-xs px-2 py-1 rounded-full"> + {{ item.relation }} + </span> + </div> + </div> + </div> + </div> + <div v-if="dataset.references.length > 0" class="mt-3 text-xs text-gray-500 dark:text-gray-400"> + Total references: {{ dataset.references.length }} + </div> + </div> + </CardBoxSimple> + + + + <!-- Keywords --> + <CardBoxSimple> + <div v-if="dataset.subjects.length === 0" class="text-center py-6"> + <p class="text-gray-500 dark:text-gray-400 italic">No keywords added.</p> + </div> + <div v-else class="p-4"> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Keywords/Subjects: + </h4> + <div class="flex flex-wrap gap-2"> + <div v-for="(subject, index) in dataset.subjects" :key="index" class="bg-emerald-50 dark:bg-emerald-900/30 text-emerald-700 dark:text-emerald-300 + px-3 py-1.5 rounded-full text-sm font-medium border border-emerald-200 + dark:border-emerald-800 shadow-sm hover:bg-emerald-100 + dark:hover:bg-emerald-900/50 transition-colors"> + <span>{{ subject.value }}</span> + <span class="ml-1 text-xs text-emerald-600 dark:text-emerald-400">({{ subject.type + }})</span> + </div> + </div> + <div v-if="dataset.subjects.length > 0" class="mt-3 text-xs text-gray-500 dark:text-gray-400"> + Total keywords: {{ dataset.subjects.length }} + </div> + </div> + </CardBoxSimple> + + + <!-- download file list --> + <CardBoxSimple> + <h4 class="text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">Dataset Files</h4> + <div v-if="dataset.files && dataset.files.length > 0" class="space-y-2"> + <div v-for="file in dataset.files" :key="file.id" class="bg-emerald-50 dark:bg-emerald-900/30 border border-emerald-200 dark:border-emerald-800 + rounded-lg p-3 shadow-sm hover:bg-emerald-100 dark:hover:bg-emerald-900/50 transition-colors"> + <div class="flex items-center justify-between"> + <div class="flex items-center space-x-3 flex-1"> + <div class="flex-shrink-0"> + <svg class="h-6 w-6 text-emerald-600 dark:text-emerald-400" + xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" + stroke="currentColor"> + <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" + d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" /> + </svg> + </div> + <div class="min-w-0 flex-1"> + <p class="text-sm font-medium text-emerald-700 dark:text-emerald-300 truncate"> + {{ file.label }} + </p> + <p class="text-xs text-emerald-600/70 dark:text-emerald-400/70 truncate"> + {{ getFileSize(file) }} + </p> + </div> + </div> + <div class="ml-2 flex-shrink-0"> + <a v-if="file.id != undefined" + :href="stardust.route('reviewer.file.download', [file.id])" class="inline-flex items-center px-3 py-1.5 border border-emerald-300 dark:border-emerald-700 + text-xs font-medium rounded-full text-emerald-700 bg-emerald-100 + dark:text-emerald-200 dark:bg-emerald-800/70 hover:bg-emerald-200 + dark:hover:bg-emerald-800 transition-colors"> + <IconSvg path="download" :size="20" className="mr-1" /> + Download + </a> + </div> + </div> + </div> + </div> + <div v-else + class="text-center py-6 bg-emerald-50/50 dark:bg-emerald-900/10 rounded-lg border border-emerald-100 dark:border-emerald-900/30"> + <svg xmlns="http://www.w3.org/2000/svg" + class="h-12 w-12 mx-auto text-emerald-300 dark:text-emerald-700" fill="none" + viewBox="0 0 24 24" stroke="currentColor"> + <path stroke-linecap="round" stroke-linejoin="round" stroke-width="1.5" + d="M9 13h6m-3-3v6m5 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" /> + </svg> + <p class="mt-2 text-emerald-600 dark:text-emerald-400 italic">No files attached to this dataset + </p> + </div> + <div v-if="dataset.files && dataset.files.length > 0" + class="mt-3 text-xs text-gray-500 dark:text-gray-400"> + Total files: {{ dataset.files.length }} + </div> + </CardBoxSimple> + + + + <BaseButtons> + <!-- <BaseButton type="submit" color="info" label="Receive" :class="{ 'opacity-25': router.processing }" :disabled="form.processing" /> --> - <BaseButton type="submit" color="info" label="Review" /> - </BaseButtons> - </template> - </CardBox> + <BaseButton v-if="can.reject && (dataset.server_state == 'approved' || dataset.server_state == 'rejected_to_reviewer')" type="submit" color="info" label="Accept" small /> + + <BaseButton + v-if="can.reject && (dataset.server_state == 'approved' || dataset.server_state == 'rejected_to_reviewer')" + :route-name="stardust.route('reviewer.dataset.reject', [dataset.id])" color="info" + :icon="mdiReiterate" :label="'Reject'" small /> + </BaseButtons> + + </component> </SectionMain> </LayoutAuthenticated> </template> + +<style scoped> +.break-words { + word-break: break-word; +} + +.overflow-wrap-anywhere { + overflow-wrap: anywhere; +} +</style> \ No newline at end of file diff --git a/resources/js/Pages/Submitter/Dataset/Category.vue b/resources/js/Pages/Submitter/Dataset/Category.vue index f835fe1..029fd7b 100644 --- a/resources/js/Pages/Submitter/Dataset/Category.vue +++ b/resources/js/Pages/Submitter/Dataset/Category.vue @@ -1,91 +1,371 @@ <template> - <div class="flex flex-col h-screen p-4 bg-gray-100"> - <header class="flex justify-between items-center mb-4"> - <h1 class="text-xl font-bold">SKOS Browser</h1> - <div class="flex space-x-2"> - <button @click="updateApp" title="Update the application"> - <img src="/Resources/Images/refresh.png" alt="Update" class="w-4 h-4" /> - </button> - <button @click="showInfo" title="Info"> - <img src="/Resources/Images/info.png" alt="Info" class="w-4 h-4" /> - </button> - </div> - </header> + <LayoutAuthenticated> + <Head title="Classify"></Head> + <SectionMain> + <SectionTitleLineWithButton :icon="mdiLibraryShelves" title="Library Classification" main> + <div class="bg-lime-100 shadow rounded-lg p-6 mb-6 flex items-center justify-between"> + <div> + <label for="role-select" class="block text-lg font-medium text-gray-700 mb-1"> + Select Classification Role <span class="text-red-500">*</span> + </label> + <select id="role-select" v-model="selectedCollectionRole" + class="w-full border border-gray-300 rounded-md p-2 text-gray-700 focus:ring-2 focus:ring-indigo-500" + required> + <!-- <option value="" disabled selected>Please select a role</option> --> + <option v-for="collRole in collectionRoles" :key="collRole.id" :value="collRole"> + {{ collRole.name }} + </option> + </select> + </div> + <div class="ml-4 hidden md:block"> + <span class="text-sm text-gray-600 italic">* required</span> + </div> + </div> + </SectionTitleLineWithButton> - <div class="bg-white shadow-md rounded-lg p-4 mb-6"> - <h2 class="text-lg font-semibold">GBA-Thesaurus</h2> - <label class="block text-sm font-medium">Aktueller Endpoint:</label> - <!-- <TreeView :items="endpoints" @select="onEndpointSelected" /> --> - </div> - <div class="bg-white shadow-md rounded-lg p-4"> - <h2 class="text-lg font-semibold">Konzept-Suche</h2> - <!-- <Autocomplete v-model="selectedConcept" :items="concepts" placeholder="Search for a concept" @change="onConceptSelected" /> --> - <div class="mt-4"> - <h3 class="text-md font-medium">Ausgewähltes Konzept</h3> - <p>{{ selectedConcept.title }}</p> - <a :href="selectedConcept.uri" target="_blank" class="text-blue-500">URI</a> - <textarea - v-model="selectedConcept.description" - class="mt-2 w-full h-24 border rounded" - placeholder="Description" - ></textarea> + <!-- Available TopLevel Collections --> + <CardBox class="mb-4 rounded-lg p-4"> + <h2 class="text-lg font-bold text-gray-800 dark:text-slate-400 mb-2">Available Toplevel-Collections + <span v-if="selectedCollectionRole && !selectedToplevelCollection" + class="text-sm text-red-500 italic">(click to + select)</span> + </h2> + <ul class="flex flex-wrap gap-2"> + <li v-for="col in collections" :key="col.id" :class="{ + 'cursor-pointer p-2 border border-gray-200 rounded hover:bg-sky-50 text-sky-700 text-sm': true, + 'bg-sky-100 border-sky-500': selectedToplevelCollection && selectedToplevelCollection.id === col.id + }" @click="onToplevelCollectionSelected(col)"> + {{ `${col.name} (${col.number})` }} + </li> + <li v-if="collections.length === 0" class="text-gray-800 dark:text-slate-400"> + No collections available. + </li> + </ul> + </CardBox> + + <!-- Collections Listing --> + <div class="grid grid-cols-1 md:grid-cols-3 gap-4 mb-4"> + + <!-- Broader Collection (Parent) --> + <CardBox v-if="selectedCollection" class="rounded-lg p-4" has-form-data> + <h2 class="text-lg font-bold text-gray-800 dark:text-slate-400 mb-2">Broader Collection</h2> + <draggable v-if="broaderCollections.length > 0" v-model="broaderCollections" + :group="{ name: 'collections' }" tag="ul" class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <template #item="{ element: parent }"> + <li :key="parent.id" :draggable="!parent.inUse" :class="getChildClasses(parent)" + @click="onCollectionSelected(parent)"> + {{ `${parent.name} (${parent.number})` }} + </li> + </template> + </draggable> + <ul v-else class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <li class="text-gray-500 text-sm"> + No broader collections available. + </li> + </ul> + </CardBox> + + <!-- Selected Collection Details --> + <CardBox v-if="selectedCollection" class="rounded-lg p-4" has-form-data> + <h3 class="text-xl font-bold text-gray-800 dark:text-slate-400 mb-2">Selected Collection</h3> + <!-- <p :class="[ + 'cursor-pointer p-2 border border-gray-200 rounded text-sm', + selectedCollection.inUse ? 'bg-gray-200 text-gray-500 drag-none' : 'bg-green-50 text-green-700 hover:bg-green-100 hover:underline cursor-move' + ]"></p> --> + <draggable v-model="selectedCollectionArray" :group="{ name: 'collections', pull: 'clone', put: false }" tag="ul" class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <template #item="{ element }"> + <li :key="element.id" :class="[ + 'p-2 border border-gray-200 rounded text-sm', + element.inUse ? 'bg-gray-200 text-gray-500 drag-none' : 'bg-green-50 text-green-700 hover:bg-green-100 hover:underline cursor-move' + ]"> + {{ `${element.name} (${element.number})` }} + </li> + </template> + </draggable> + </CardBox> + <!-- Narrower Collections (Children) --> + <CardBox v-if="selectedCollection" class="rounded-lg p-4" has-form-data> + <h2 class="text-lg font-bold text-gray-800 dark:text-slate-400 mb-2">Narrower Collections</h2> + <draggable v-if="narrowerCollections.length > 0" v-model="narrowerCollections" + :group="{ name: 'collections' }" tag="ul" class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <template #item="{ element: child }"> + <li :key="child.id" :draggable="!child.inUse" :class="getChildClasses(child)" + @click="onCollectionSelected(child)"> + {{ `${child.name} (${child.number})` }} + </li> + </template> + </draggable> + <ul v-else class="flex flex-wrap gap-2 max-h-60 overflow-y-auto"> + <li class="text-gray-500 text-sm"> + No sub-collections available. + </li> + </ul> + </CardBox> + </div> - <div class="mt-4"> - <h3 class="text-md font-medium">Untergeordnete Konzepte</h3> - <!-- <LinkLabelList :items="narrowerConcepts" /> --> + + <div class="mb-4 rounded-lg"> + <div v-if="selectedCollection || selectedCollectionList.length > 0" class="bg-gray-100 shadow rounded-lg p-6 mb-6" :class="{ 'opacity-50': selectedCollection && selectedCollectionList.length === 0 }"> + <p class="mb-4 text-gray-700">Please drag your collections here to classify your previously created + dataset + according to library classification standards.</p> + <draggable v-model="selectedCollectionList" :group="{ name: 'collections' }" + class="min-h-36 border-dashed border-2 border-gray-400 p-4 text-sm flex flex-wrap gap-2 max-h-60 overflow-y-auto" + tag="ul" + :disabled="selectedCollection === null && selectedCollectionList.length > 0" + :style="{ opacity: (selectedCollection === null && selectedCollectionList.length > 0) ? 0.5 : 1, pointerEvents: (selectedCollection === null && selectedCollectionList.length > 0) ? 'none' : 'auto' }"> + <template #item="{ element }"> + <div :key="element.id" + class="p-2 m-1 bg-sky-200 text-sky-800 rounded flex items-center gap-2 h-7"> + <span>{{ element.name }} ({{ element.number }})</span> + <button + @click="selectedCollectionList = selectedCollectionList.filter(item => item.id !== element.id)" + class="hover:text-sky-600 flex items-center"> + <svg xmlns="http://www.w3.org/2000/svg" class="h-4 w-4" viewBox="0 0 20 20" + fill="currentColor"> + <path fill-rule="evenodd" + d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" + clip-rule="evenodd" /> + </svg> + </button> + </div> + </template> + </draggable> + </div> </div> - <div class="mt-4"> - <h3 class="text-md font-medium">Übergeordnete Konzepte</h3> - <!-- <LinkLabelList :items="broaderConcepts" /> --> + <div class="p-6 border-t border-gray-100 dark:border-slate-800"> + <BaseButtons> + <BaseButton @click.stop="syncDatasetCollections" label="Save" color="info" small + :disabled="isSaveDisabled" :style="{ opacity: isSaveDisabled ? 0.5 : 1 }"> + </BaseButton> + </BaseButtons> </div> - <div class="mt-4"> - <h3 class="text-md font-medium">Verwandte Konzepte</h3> - <!-- <LinkLabelList :items="relatedConcepts" /> --> - </div> - </div> - </div> + </SectionMain> + </LayoutAuthenticated> </template> -<script> -// import TreeView from './TreeView.vue'; // Assuming you have a TreeView component -// import Autocomplete from './Autocomplete.vue'; // Assuming you have an Autocomplete component -// import LinkLabelList from './LinkLabelList.vue'; // Assuming you have a LinkLabelList component +<script setup lang="ts"> +import { ref, Ref, watch, computed } from 'vue'; +import { useForm } from '@inertiajs/vue3'; +import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; +import SectionMain from '@/Components/SectionMain.vue'; +import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; +import axios from 'axios'; +import { mdiLibraryShelves } from '@mdi/js'; +import draggable from 'vuedraggable'; +import BaseButton from '@/Components/BaseButton.vue'; +import BaseButtons from '@/Components/BaseButtons.vue'; +import CardBox from '@/Components/CardBox.vue'; +import { stardust } from '@eidellev/adonis-stardust/client'; +import { CollectionRole, Collection } from '@/types/models'; -export default { - components: { - // TreeView, - // Autocomplete, - // LinkLabelList, +// import CollectionRoleSelector from '@/Components/Collection/CollectionRoleSelector.vue'; +// import ToplevelCollections from '@/Components/Collection/ToplevelCollections.vue'; +// import CollectionHierarchy from '@/Components/Collection/CollectionHierarchy.vue'; +// import CollectionDropZone from '@/Components/Collection/CollectionDropZone.vue'; + + + +/* -------------------------------------------------------------------------- + Props & Reactive State +-------------------------------------------------------------------------- */ +const props = defineProps({ + collectionRoles: { + type: Array, + required: true, + default: () => [] }, - data() { - return { - endpoints: [], // This should be populated with your data - concepts: [], // This should be populated with your data - selectedConcept: {}, - narrowerConcepts: [], // Populate with data - broaderConcepts: [], // Populate with data - relatedConcepts: [], // Populate with data - }; + dataset: { + type: Object, + default: () => ({}), }, - methods: { - updateApp() { - // Handle app update logic - }, - showInfo() { - // Handle showing information - }, - onEndpointSelected(endpoint) { - // Handle endpoint selection - }, - onConceptSelected(concept) { - this.selectedConcept = concept; - // Handle concept selection logic, e.g., fetching related concepts - }, + relatedCollections: { + type: Array as () => Collection[], + default: () => [] as const + } +}); + +const collectionRoles: Ref<CollectionRole[]> = ref(props.collectionRoles as CollectionRole[]); +const collections: Ref<Collection[]> = ref<Collection[]>([]); +const selectedCollectionRole = ref<CollectionRole | null>(null); +const selectedToplevelCollection = ref<Collection | null>(null); +const selectedCollection = ref<Collection | null>(null); +const narrowerCollections = ref<Collection[]>([]); +const broaderCollections = ref<Collection[]>([]); +// Reactive list that holds collections dropped by the user +const selectedCollectionList: Ref<Collection[]> = ref<Collection[]>([]); + + +// Wrap selectedCollection in an array for draggable (always expects an array) +const selectedCollectionArray = computed({ + get: () => (selectedCollection.value ? [selectedCollection.value] : []), + set: (value: Collection[]) => { + selectedCollection.value = value.length ? value[0] : null + } +}) + + +const form = useForm({ + collections: [] as number[], +}); + +// Watch for changes in dropCollections +watch( + () => selectedCollectionList.value, + () => { + if (selectedCollection.value) { + fetchCollections(selectedCollection.value.id); + } }, + { deep: true } +); + + +/* -------------------------------------------------------------------------- + Watchers and Initial Setup +-------------------------------------------------------------------------- */ +// If the collectionRoles prop might load asynchronously (or change), you can watch for it: +watch( + () => props.collectionRoles as CollectionRole[], + (newCollectionRoles: CollectionRole[]) => { + collectionRoles.value = newCollectionRoles; + // Preselect the role with name "ccs" if it exists + const found: CollectionRole | undefined = collectionRoles.value.find( + role => role.name.toLowerCase() === 'ccs' + ); + if (found?.name === 'ccs') { + selectedCollectionRole.value = found; + } + }, + { immediate: true } +); + +// When collection role changes, update available collections and clear dependent state. +watch( + () => selectedCollectionRole.value as CollectionRole, + (newSelectedCollectionRole: CollectionRole | null) => { + if (newSelectedCollectionRole != null) { + collections.value = newSelectedCollectionRole.collections || [] + } else { + selectedToplevelCollection.value = null; + selectedCollection.value = null; + collections.value = [] + } + // Reset dependent variables when the role changes + selectedCollection.value = null + narrowerCollections.value = [] + broaderCollections.value = [] + }, + { immediate: true } +); + +/* -------------------------------------------------------------------------- + Methods +-------------------------------------------------------------------------- */ +const onToplevelCollectionSelected = (collection: Collection) => { + selectedToplevelCollection.value = collection; + selectedCollection.value = collection; + // call the API endpoint to get both. + fetchCollections(collection.id); }; + +const onCollectionSelected = (collection: Collection) => { + selectedCollection.value = collection; + // call the API endpoint to get both. + fetchCollections(collection.id); +}; + +/** + * fetchCollections: Retrieves broader and narrower collections. + * Marks any narrower collection as inUse if it appears in selectedCollectionList. + */ +const fetchCollections = async (collectionId: number) => { + try { + const response = await axios.get(`/api/collections/${collectionId}`); + const data = response.data; + // Map each returned narrower collection + narrowerCollections.value = data.narrowerCollections.map((collection: Collection) => { + // If found, mark it as inUse. + const alreadyDropped = selectedCollectionList.value.find(dc => dc.id === collection.id); + return alreadyDropped ? { ...collection, inUse: true } : { ...collection, inUse: false }; + }); + broaderCollections.value = data.broaderCollection.map((collection: Collection) => { + const alreadyDropped = selectedCollectionList.value.find(dc => dc.id === collection.id); + return alreadyDropped ? { ...collection, inUse: true } : { ...collection, inUse: false }; + }); + // Check if selected collection is in the selected list + if (selectedCollection.value && selectedCollectionList.value.find(dc => dc.id === selectedCollection.value?.id)) { + selectedCollection.value = { ...selectedCollection.value, inUse: true }; + } else if (selectedCollection.value) { + selectedCollection.value = { ...selectedCollection.value, inUse: false }; + } + } catch (error) { + console.error('Error in fetchCollections:', error); + } +}; + +const syncDatasetCollections = async () => { + // Extract the ids from the dropCollections list + form.collections = selectedCollectionList.value.map((item: Collection) => item.id); + form.put(stardust.route('dataset.categorizeUpdate', [props.dataset.id]), { + preserveState: true, + onSuccess: () => { + console.log('Dataset collections synced successfully'); + }, + onError: (errors) => { + console.error('Error syncing dataset collections:', errors); + }, + }); +}; + +/** + * getChildClasses returns the Tailwind CSS classes to apply to each collection list item. + */ +const getChildClasses = (child: Collection) => { + return child.inUse + ? 'p-2 border border-gray-200 rounded bg-gray-200 text-gray-500 cursor-pointer drag-none' + : 'p-2 border border-gray-200 rounded bg-green-50 text-green-700 cursor-move hover:bg-green-100 hover:underline' +} + +// If there are related collections passed in, fill dropCollections with these. +if (props.relatedCollections && props.relatedCollections.length > 0) { + selectedCollectionList.value = props.relatedCollections; +} + +// Add a computed property for the disabled state based on dropCollections length +const isSaveDisabled = computed(() => selectedCollectionList.value.length === 0); </script> <style scoped> -/* Add your styles here */ +.btn-primary { + background-color: #4f46e5; + color: white; + border-radius: 0.25rem; +} + +.btn-primary:hover { + background-color: #4338ca; +} + +.btn-primary:focus { + outline: none; + box-shadow: 0 0 0 2px #fff, 0 0 0 4px #4f46e5; +} + +.btn-secondary { + background-color: white; + color: #374151; + border: 1px solid #d1d5db; + border-radius: 0.25rem; +} + +.btn-secondary:hover { + background-color: #f9fafb; +} + +.btn-secondary:focus { + outline: none; + box-shadow: 0 0 0 2px #fff, 0 0 0 4px #6366f1; +} </style> diff --git a/resources/js/Pages/Submitter/Dataset/Create.vue b/resources/js/Pages/Submitter/Dataset/Create.vue index 00c649b..40c25ab 100644 --- a/resources/js/Pages/Submitter/Dataset/Create.vue +++ b/resources/js/Pages/Submitter/Dataset/Create.vue @@ -24,7 +24,6 @@ import FormControl from '@/Components/FormControl.vue'; import FormCheckRadioGroup from '@/Components/FormCheckRadioGroup.vue'; import BaseButton from '@/Components/BaseButton.vue'; import { stardust } from '@eidellev/adonis-stardust/client'; -// import { Inertia } from '@inertiajs/inertia'; import CardBoxModal from '@/Components/CardBoxModal.vue'; import BaseIcon from '@/Components/BaseIcon.vue'; @@ -45,7 +44,7 @@ import { LayerOptions } from '@/Components/Map/LayerOptions'; import TableKeywords from '@/Components/TableKeywords.vue'; import NotificationBar from '@/Components/NotificationBar.vue'; import FileUploadComponent from '@/Components/FileUpload.vue'; -import Person from '#models/person'; +import type Person from '#models/person'; const props = defineProps({ licenses: { @@ -96,6 +95,27 @@ const flash: ComputedRef<any> = computed(() => { return usePage().props.flash; }); +// Computed property to determine the placeholder based on the selected option +const getPlaceholder = computed(() => (type: string) => { + + switch (type) { + case 'DOI': + return 'https://doi.org/10.24341/tethys.236'; + case 'Handle': + return '20.500.12345/67890'; + case 'ISBN': + return '978-3-85316-076-3'; + case 'ISSN': + return '1234-5678'; + case 'URL': + return 'https://example.com'; + case 'URN': + return 'urn:nbn:de:1234-5678'; + default: + return '[VALUE]'; + } +}); + const mainService = MainService(); // let serrors = reactive([]); @@ -200,15 +220,6 @@ if (Object.keys(mainService.dataset).length == 0) { // descriptions: [{ value: '', type: 'Abstract', language: language }], // }); let form = useForm<Dataset>(dataset as Dataset); -// form.defaults(); - -// const emit = defineEmits(['update:modelValue', 'setRef']); -// computed({ -// get: () => form.rights, -// set: (value) => { -// emit('update:modelValue', value); -// }, -// }); watch(language, (currentValue) => { if (currentValue != "") { @@ -299,12 +310,17 @@ const nextStep = async () => { } else if (formStep.value == 3) { route = stardust.route('dataset.third.step'); } - // formStep.value++; + // When posting in steps 1-3, remove any file uploads from the data. await form - .transform((data) => ({ - ...data, - rights: form.rights && form.rights == true ? 'true' : 'false', - })) + .transform((data: Dataset) => { + // Create payload and set rights (transforming to a string if needed) + const payload: any = { ...data, rights: data.rights ? 'true' : 'false' }; + // Remove the files property so that the partial update is done without files + if (payload.files) { + delete payload.files; + } + return payload; + }) .post(route, { onSuccess: () => { // console.log(form.data()); @@ -313,7 +329,6 @@ const nextStep = async () => { }, }); }; - const prevStep = () => { formStep.value--; }; @@ -322,7 +337,7 @@ const submit = async () => { let route = stardust.route('dataset.submit'); const files = form.files.map((obj) => { - return new File([obj.blob], obj.label, { type: obj.type, lastModified: obj.lastModified }); + return new File([obj.blob], obj.label, { type: obj.type, lastModified: obj.lastModified, sort_order: obj.sort_order }); }); // formStep.value++; @@ -419,6 +434,12 @@ const onAddAuthor = (person: Person) => { notify({ type: 'info', text: 'person has been successfully added as author' }); } }; + +const addNewContributor = () => { + let newContributor = { status: false, first_name: '', last_name: '', email: '', academic_title: '', identifier_orcid: '', name_type: 'Personal', pivot: { contributor_type: '' } }; + form.contributors.push(newContributor); +}; + const onAddContributor = (person: Person) => { if (form.contributors.filter((e) => e.id === person.id).length > 0) { notify({ type: 'warning', title: 'Warning', text: 'person is already defined as contributor' }, 4000); @@ -441,7 +462,7 @@ const onMapInitialized = (newItem: any) => { adds a new Keyword */ const addKeyword = () => { - let newSubject: Subject = { value: 'test', language: '', type: 'uncontrolled' }; + let newSubject: Subject = { value: '', language: '', type: 'uncontrolled' }; //this.dataset.files.push(uploadedFiles[i]); form.subjects.push(newSubject); }; @@ -478,16 +499,25 @@ Removes a selected keyword <template> <CardBoxModal v-model="isModalActive" title="Einverständniserklärung *"> - Mit dem Setzen des Hakens bestätige ich hiermit - <ul class="list-decimal"> + <p class="mb-4 text-gray-700"> + Mit dem Setzen des Hakens bestätige ich hiermit folgende Punkte: + </p> + <ul class="list-decimal pl-6 space-y-2 text-sm text-gray-600"> <li> - die Data Policy von Tethys RDR sowie die Terms & Conditions von Tethys gelesen und verstanden zu haben - (<a href="/docs/HandbuchTethys.pdf" target="_blank">siehe hier</a>) + die Data Policy von Tethys RDR sowie die + <a href="/docs/HandbuchTethys.pdf" target="_blank" + class="font-medium text-blue-600 hover:text-blue-800 transition-colors underline"> + Terms & Conditions + </a> + von Tethys gelesen und verstanden zu haben. </li> - <li>das Einverständnis aller Co-Autoren über die bevorstehende Datenpublikation schriftlich eingeholt zu - haben + <li> + das Einverständnis aller Co-Autoren über die bevorstehende Datenpublikation schriftlich eingeholt zu + haben. + </li> + <li> + sowohl mit der Data Policy als auch mit den Terms & Conditions einverstanden zu sein. </li> - <li>sowohl mit der Data Policy als auch mit den Terms & Conditions einverstanden zu sein</li> </ul> </CardBoxModal> @@ -510,15 +540,15 @@ Removes a selected keyword <div class="flex items-center"> <!-- <label>{{ form.titles[0].language }}</label> <label>{{ form.language }}</label> --> - <icon-wizard :is-current="formStep == 1" :is-checked="formStep > 1" :label="'Language'"> + <icon-wizard :is-current="formStep == 1" :is-checked="formStep > 1" :label="'Step 1'"> <icon-language></icon-language> </icon-wizard> - <icon-wizard :is-current="formStep == 2" :is-checked="formStep > 2" :label="'Mandatory'"> + <icon-wizard :is-current="formStep == 2" :is-checked="formStep > 2" :label="'Step 2'"> <icon-mandatory></icon-mandatory> </icon-wizard> - <icon-wizard :is-current="formStep == 3" :is-checked="formStep > 3" :label="'Recommended'"> + <icon-wizard :is-current="formStep == 3" :is-checked="formStep > 3" :label="'Step 3'"> <icon-recommendet></icon-recommendet> </icon-wizard> @@ -546,7 +576,7 @@ Removes a selected keyword <FormField label="Licenses" wrap-body :class="{ 'text-red-400': form.errors.licenses }" class="mt-8 w-full mx-2 flex-1"> - <FormCheckRadioGroup v-model="form.licenses" name="roles" is-column + <FormCheckRadioGroup type="radio" v-model="form.licenses" name="licenses" is-column :options="props.licenses" /> </FormField> @@ -554,8 +584,10 @@ Removes a selected keyword <input class="form-checkbox" name="rights" id="rights" type="checkbox" v-model="dataset.rights" /> terms and conditions </label> --> - <FormField label="Rights" help="You must agree to continue" wrap-body - :class="{ 'text-red-400': form.errors.rights }" class="mt-8 w-full mx-2 flex-1 flex-col"> + <FormField label="Rights" + help="You must agree that you have read the Terms and Conditions. Please click on the 'i' icon to find a read the policy" + wrap-body :class="{ 'text-red-400': form.errors.rights }" + class="mt-8 w-full mx-2 flex-1 flex-col"> <label for="rights" class="checkbox mr-6 mb-3 last:mr-0"> <input type="checkbox" id="rights" required v-model="form.rights" /> <span class="check" /> @@ -641,7 +673,7 @@ Removes a selected keyword <FormField label="Title Value *" :class="{ 'text-red-400': form.errors[`titles.${index}.value`] }" class="w-full mx-2 flex-1"> - <FormControl required v-model="form.titles[index].value" type="text" + <FormControl required v-model="form.titles[index].value" type="textarea" placeholder="[enter main title]"> <div class="text-red-400 text-sm" v-if="form.errors[`titles.${index}.value`]"> @@ -717,7 +749,7 @@ Removes a selected keyword <FormField label="Description Value *" :class="{ 'text-red-400': form.errors[`descriptions.${index}.value`] }" class="w-full mx-2 flex-1"> - <FormControl required v-model="form.descriptions[index].value" type="text" + <FormControl required v-model="form.descriptions[index].value" type="textarea" placeholder="[enter additional description]" :show-char-count="true" :max-input-length="2500"> <div class="text-red-400 text-sm" v-if="form.errors[`descriptions.${index}.value`] && @@ -757,14 +789,14 @@ Removes a selected keyword </CardBox> <!-- authors --> - <CardBox class="mb-6 shadow" has-table title="Creators" :icon="mdiBookOpenPageVariant"> + <CardBox class="mb-6 shadow" has-table title="Creators" :icon="mdiBookOpenPageVariant" :show-header-icon="false"> <SearchAutocomplete source="/api/persons" :response-property="'first_name'" placeholder="search in person table...." v-on:person="onAddAuthor"></SearchAutocomplete> <TablePersons :errors="form.errors" :persons="form.authors" :relation="'authors'" v-if="form.authors.length > 0" /> - <div class="text-red-400 text-sm" v-if="errors.authors && Array.isArray(errors.authors)"> - {{ errors.authors.join(', ') }} + <div class="text-red-400 text-sm" v-if="form.errors.authors && Array.isArray(form.errors.authors)"> + {{ form.errors.authors.join(', ') }} </div> <div class="w-full md:w-1/2"> <label class="block" for="additionalCreators">Add additional creator(s) if creator is @@ -775,7 +807,7 @@ Removes a selected keyword </CardBox> <!-- contributors --> - <CardBox class="mb-6 shadow" has-table title="Contributors" :icon="mdiBookOpenPageVariant"> + <CardBox class="mb-6 shadow" has-table title="Contributors" :icon="mdiBookOpenPageVariant" :show-header-icon="false"> <SearchAutocomplete source="/api/persons" :response-property="'first_name'" placeholder="search in person table...." v-on:person="onAddContributor"> </SearchAutocomplete> @@ -787,6 +819,12 @@ Removes a selected keyword v-if="form.errors.contributors && Array.isArray(form.errors.contributors)"> {{ form.errors.contributors.join(', ') }} </div> + <div class="w-full md:w-1/2"> + <label class="block" for="additionalCreators">Add additional contributor(s) if + contributor is not in database</label> + <button class="bg-blue-500 text-white py-2 px-4 rounded-sm" + @click.prevent="addNewContributor()">+</button> + </div> </CardBox> </div> @@ -813,7 +851,7 @@ Removes a selected keyword </FormControl> </FormField> </div> - <CardBox class="mb-6 shadow" has-table title="Geo Location" :icon="mdiEarthPlus"> + <CardBox class="mb-6 shadow" has-table title="Geo Location" :icon="mdiEarthPlus" :show-header-icon="false"> <!-- @onMapInitialized="onMapInitialized" --> <!-- v-bind-event="{ mapId, name: mapId }" --> <MapComponent :mapOptions="mapOptions" :baseMaps="baseMaps" :fitBounds="fitBounds" @@ -871,9 +909,9 @@ Removes a selected keyword </div> </CardBox> - <CardBox class="mb-6 shadow" has-table title="Coverage Information" :icon="mdiEarthPlus"> + <CardBox class="mb-6 shadow" has-table title="Coverage Information" :icon="mdiEarthPlus" :show-header-icon="false"> <!-- elevation menu --> - <div class="flex flex-col md:flex-row mb-3 space-y-2 md:space-y-0 md:space-x-4"> + <div class="flex flex-col md:flex-row mb-3 space-y-2 md:space-y-0 md:space-x-4"> <label for="elevation-option-one" class="pure-radio mb-2 md:mb-0"> <input id="elevation-option-one" type="radio" v-model="elevation" value="absolut" /> absolut elevation (m) @@ -1050,7 +1088,8 @@ Removes a selected keyword <!-- <input name="Reference Value" class="form-control" placeholder="[VALUE]" v-model="item.value" /> --> <FormControl required v-model="item.value" :type="'text'" - placeholder="[VALUE]" :errors="form.errors.embargo_date"> + :placeholder="getPlaceholder(form.references[index].type)" + :errors="form.errors.embargo_date"> <div class="text-red-400 text-sm" v-if="form.errors[`references.${index}.value`] && Array.isArray(form.errors[`references.${index}.value`])"> {{ form.errors[`references.${index}.value`].join(', ') }} diff --git a/resources/js/Pages/Submitter/Dataset/Edit.vue b/resources/js/Pages/Submitter/Dataset/Edit.vue index 9f3c395..ff885c1 100644 --- a/resources/js/Pages/Submitter/Dataset/Edit.vue +++ b/resources/js/Pages/Submitter/Dataset/Edit.vue @@ -42,7 +42,8 @@ <!-- (2) licenses --> <FormField label="Licenses" wrap-body :class="{ 'text-red-400': form.errors.licenses }" class="mt-8 w-full mx-2 flex-1"> - <FormCheckRadioGroup v-model="form.licenses" name="licenses" is-column :options="licenses" /> + <FormCheckRadioGroup type="radio" v-model="form.licenses" name="licenses" is-column + :options="licenses" /> </FormField> <div class="flex flex-col md:flex-row"> @@ -78,7 +79,7 @@ <div class="flex flex-col md:flex-row"> <FormField label="Main Title *" help="required: main title" :class="{ 'text-red-400': form.errors['titles.0.value'] }" class="w-full mr-1 flex-1"> - <FormControl required v-model="form.titles[0].value" type="text" + <FormControl required v-model="form.titles[0].value" type="textarea" placeholder="[enter main title]" :show-char-count="true" :max-input-length="255"> <div class="text-red-400 text-sm" v-if="form.errors['titles.0.value'] && Array.isArray(form.errors['titles.0.value'])"> @@ -116,7 +117,7 @@ <tr v-if="title.type != 'Main'"> <!-- <td scope="row">{{ index + 1 }}</td> --> <td data-label="Title Value"> - <FormControl required v-model="form.titles[index].value" type="text" + <FormControl required v-model="form.titles[index].value" type="textarea" placeholder="[enter main title]"> <div class="text-red-400 text-sm" v-if="form.errors[`titles.${index}.value`]"> @@ -163,7 +164,8 @@ :class="{ 'text-red-400': form.errors['descriptions.0.value'] }" class="w-full mr-1 flex-1"> <FormControl required v-model="form.descriptions[0].value" type="textarea" - placeholder="[enter main abstract]" :show-char-count="true" :max-input-length="2500"> + placeholder="[enter main abstract]" :show-char-count="true" + :max-input-length="2500"> <div class="text-red-400 text-sm" v-if="form.errors['descriptions.0.value'] && Array.isArray(form.errors['descriptions.0.value'])"> {{ form.errors['descriptions.0.value'].join(', ') }} @@ -176,7 +178,7 @@ <FormControl required v-model="form.descriptions[0].language" type="text" :is-read-only="true"> <div class="text-red-400 text-sm" v-if="form.errors['descriptions.0.value'] && Array.isArray(form.errors['descriptions.0.language']) - "> + "> {{ form.errors['descriptions.0.language'].join(', ') }} </div> </FormControl> @@ -197,7 +199,7 @@ <tr v-if="item.type != 'Abstract'"> <!-- <td scope="row">{{ index + 1 }}</td> --> <td data-label="Description Value"> - <FormControl required v-model="form.descriptions[index].value" type="text" + <FormControl required v-model="form.descriptions[index].value" type="textarea" placeholder="[enter main title]"> <div class="text-red-400 text-sm" v-if="form.errors[`descriptions.${index}.value`]"> @@ -239,19 +241,23 @@ </CardBox> <!-- (7) authors --> - <CardBox class="mb-6 shadow" has-table title="Creators" :icon="mdiBookOpenPageVariant"> + <CardBox class="mb-6 shadow" has-table title="Creators" :icon="mdiBookOpenPageVariant" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addNewAuthor()"> <SearchAutocomplete source="/api/persons" :response-property="'first_name'" placeholder="search in person table...." v-on:person="onAddAuthor"></SearchAutocomplete> - <TablePersons :persons="form.authors" v-if="form.authors.length > 0" :relation="'authors'"/> - <div class="text-red-400 text-sm" v-if="form.errors.authors && Array.isArray(form.errors.authors)"> + <TablePersons :persons="form.authors" v-if="form.authors.length > 0" :errors="form.errors" + :relation="'authors'" /> + <div class="text-red-400 text-sm" + v-if="form.errors.authors && Array.isArray(form.errors.authors)"> {{ form.errors.authors.join(', ') }} </div> </CardBox> <!-- (8) contributors --> - <CardBox class="mb-6 shadow" has-table title="Contributors" :icon="mdiBookOpenPageVariant"> + <CardBox class="mb-6 shadow" has-table title="Contributors" :icon="mdiBookOpenPageVariant" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addNewContributor()"> <SearchAutocomplete source="/api/persons" :response-property="'first_name'" placeholder="search in person table...." v-on:person="onAddContributor"> </SearchAutocomplete> @@ -334,8 +340,8 @@ </FormField> </div> - <CardBox class="mb-6 shadow" has-table title="Dataset References" :icon="mdiEarthPlus" :header-icon="mdiPlusCircle" - v-on:header-icon-click="addReference"> + <CardBox class="mb-6 shadow" has-table title="Dataset References" :icon="mdiEarthPlus" + :header-icon="mdiPlusCircle" v-on:header-icon-click="addReference"> <!-- Message when no references exist --> <div v-if="form.references.length === 0" class="text-center py-4"> <p class="text-gray-600">No references added yet.</p> @@ -408,6 +414,43 @@ </tr> </tbody> </table> + + <!-- References to delete section --> + <div v-if="form.referencesToDelete && form.referencesToDelete.length > 0" class="mt-8"> + <h1 class="pt-8 pb-3 font-semibold sm:text-lg text-gray-900">References To Delete</h1> + <ul class="flex flex-1 flex-wrap -m-1"> + <li v-for="(element, index) in form.referencesToDelete" :key="index" + class="block p-1 w-1/2 sm:w-1/3 md:w-1/4 lg:w-1/6 xl:w-1/8 h-40"> + <article tabindex="0" + class="bg-red-100 group w-full h-full rounded-md cursor-pointer relative shadow-sm overflow-hidden"> + <section + class="flex flex-col rounded-md text-xs break-words w-full h-full z-20 absolute top-0 py-2 px-3"> + <h1 + class="flex-1 text-gray-700 group-hover:text-blue-800 font-medium text-sm mb-1 truncate overflow-hidden whitespace-nowrap"> + {{ element.value }} + </h1> + <div class="flex flex-col mt-auto"> + <p class="p-1 size text-xs text-gray-700"> + <span class="font-semibold">Type:</span> {{ element.type }} + </p> + <p class="p-1 size text-xs text-gray-700"> + <span class="font-semibold">Relation:</span> {{ element.relation }} + </p> + <div class="flex justify-end mt-1"> + <button + class="restore ml-auto focus:outline-none hover:bg-gray-300 p-1 rounded-md text-gray-800" + @click.prevent="restoreReference(index)"> + <svg viewBox="0 0 24 24" class="w-5 h-5"> + <path fill="currentColor" :d="mdiRestore"></path> + </svg> + </button> + </div> + </div> + </section> + </article> + </li> + </ul> + </div> </CardBox> <BaseDivider /> @@ -420,20 +463,11 @@ </li> </ul> --> <TableKeywords :keywords="form.subjects" :errors="form.errors" :subjectTypes="subjectTypes" - v-if="form.subjects.length > 0" /> + v-model:subjects-to-delete="form.subjectsToDelete" v-if="form.subjects.length > 0" /> </CardBox> </div> - <!-- <div class="mb-4"> - <label for="description" class="block text-gray-700 font-bold mb-2">Description:</label> - <textarea id="description" - class="shadow appearance-none border rounded w-full py-2 px-3 text-gray-700 leading-tight focus:outline-none focus:shadow-outline" - v-model="form.type"></textarea> - </div> --> - - - <div class="mb-4"> <!-- <label for="project" class="block text-gray-700 font-bold mb-2">Project:</label> <select @@ -447,36 +481,28 @@ </select> --> </div> - <FileUploadComponent v-model:files="form.files" v-model:filesToDelete="form.filesToDelete"></FileUploadComponent> + <FileUploadComponent v-model:files="form.files" v-model:filesToDelete="form.filesToDelete" + :showClearButton="false"> + </FileUploadComponent> <div class="text-red-400 text-sm" v-if="form.errors['file'] && Array.isArray(form.errors['files'])"> {{ form.errors['files'].join(', ') }} </div> - - <!-- Add more input fields for the other properties of the dataset --> - <!-- <button - type="submit" - class="bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded focus:outline-none focus:shadow-outline" - > - Save - </button> --> - <template #footer> <BaseButtons> - <BaseButton @click.stop="submit" :disabled="form.processing" label="Save" color="info" - :class="{ 'opacity-25': form.processing }" small> + <BaseButton v-if="can.edit" @click.stop="submit" :disabled="form.processing" label="Save" + color="info" :icon="mdiDisc" :class="{ 'opacity-25': form.processing }" small> </BaseButton> - <!-- <button :disabled="form.processing" :class="{ 'opacity-25': form.processing }" - class="text-base hover:scale-110 focus:outline-none flex justify-center px-4 py-2 rounded font-bold cursor-pointer hover:bg-teal-200 bg-teal-100 text-teal-700 border duration-200 ease-in-out border-teal-600 transition" - @click.stop="submit"> - Save - </button> --> + <BaseButton v-if="can.edit" :route-name="stardust.route('dataset.release', [dataset.id])" + color="info" :icon="mdiLockOpen" :label="'Release'" small + :disabled="form.processing" + :class="{ 'opacity-25': form.processing }" /> </BaseButtons> </template> </CardBox> - <!-- Loading Spinner --> - <div v-if="form.processing" + <!-- Loading Spinner --> + <div v-if="form.processing" class="fixed inset-0 flex items-center justify-center bg-gray-500 bg-opacity-50 z-50"> <svg class="animate-spin h-12 w-12 text-white" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"> @@ -492,8 +518,6 @@ // import EditComponent from "./../EditComponent"; // export default EditComponent; -// import { Component, Vue, Prop, Setup, toNative } from 'vue-facing-decorator'; -// import AuthLayout from '@/Layouts/Auth.vue'; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; import { useForm, Head, usePage } from '@inertiajs/vue3'; import { computed, ComputedRef } from 'vue'; @@ -527,6 +551,9 @@ import { mdiBookOpenPageVariant, mdiEarthPlus, mdiAlertBoxOutline, + mdiRestore, + mdiLockOpen, + mdiDisc } from '@mdi/js'; import { notify } from '@/notiwind'; import NotificationBar from '@/Components/NotificationBar.vue'; @@ -580,8 +607,10 @@ const props = defineProps({ type: Object, default: () => ({}), }, - - + can: { + type: Object, + default: () => ({}), + }, }); const flash: ComputedRef<any> = computed(() => { @@ -592,12 +621,6 @@ const flash: ComputedRef<any> = computed(() => { const errors: ComputedRef<any> = computed(() => { return usePage().props.errors; }); -// const errors: ComputedRef<any> = computed(() => { -// return usePage().props.errors; -// }); - -// const projects = reactive([]); -// const licenses = reactive([]); const mapOptions: MapOptions = { center: [48.208174, 16.373819], @@ -612,67 +635,72 @@ const fitBounds: LatLngBoundsExpression = [ ]; const mapId = 'test'; -// const downloadFile = async (id: string): Promise<string> => { -// const response = await axios.get<Blob>(`/api/download/${id}`, { -// responseType: 'blob', -// }); -// const url = URL.createObjectURL(response.data); -// setTimeout(() => { -// URL.revokeObjectURL(url); -// }, 1000); -// return url; -// }; - -// for (const file of props.dataset.files) { -// // console.log(`${file.name} path is ${file.filePath} here.`); -// file.fileSrc = ref(""); -// // downloadFile(file.id).then((value: string) => { -// // file.fileSrc = ref(value); -// // form = useForm<Dataset>(props.dataset as Dataset); -// // }); -// } props.dataset.filesToDelete = []; +props.dataset.subjectsToDelete = []; +props.dataset.referencesToDelete = []; let form = useForm<Dataset>(props.dataset as Dataset); -// const mainService = MainService(); -// mainService.fetchfiles(props.dataset); +// Add this computed property to the script section +const hasUnsavedChanges = computed(() => { + // Check if form is processing + if (form.processing) return true; + // Compare current form state with original dataset + // Check basic properties + if (form.language !== props.dataset.language) return true; + if (form.type !== props.dataset.type) return true; + if (form.project_id !== props.dataset.project_id) return true; + if (form.embargo_date !== props.dataset.embargo_date) return true; + // Check if licenses have changed + const originalLicenses = Array.isArray(props.dataset.licenses) + ? props.dataset.licenses.map(l => typeof l === 'object' ? l.id.toString() : l) + : []; + const currentLicenses = Array.isArray(form.licenses) + ? form.licenses.map(l => typeof l === 'object' ? l.id.toString() : l) + : []; + if (JSON.stringify(currentLicenses) !== JSON.stringify(originalLicenses)) return true; + // Check if titles have changed + if (JSON.stringify(form.titles) !== JSON.stringify(props.dataset.titles)) return true; + // Check if descriptions have changed + if (JSON.stringify(form.descriptions) !== JSON.stringify(props.dataset.descriptions)) return true; + // Check if authors have changed + if (JSON.stringify(form.authors) !== JSON.stringify(props.dataset.authors)) return true; -// const files = computed(() => props.dataset.file); + // Check if contributors have changed + if (JSON.stringify(form.contributors) !== JSON.stringify(props.dataset.contributors)) return true; + // Check if subjects have changed + // if (JSON.stringify(form.subjects) !== JSON.stringify(props.dataset.subjects)) return true; + let test = JSON.stringify(form.subjects); + let test2 = JSON.stringify(props.dataset.subjects); + if (test !== test2) { + return true; + } + // Check if references have changed + if (JSON.stringify(form.references) !== JSON.stringify(props.dataset.references)) return true; -// let form = useForm<Dataset>(props.dataset as Dataset); + // Check if coverage has changed + if (JSON.stringify(form.coverage) !== JSON.stringify(props.dataset.coverage)) return true; -// const form = useForm({ -// _method: 'put', -// login: props.user.login, -// email: props.user.email, -// password: '', -// password_confirmation: '', -// roles: props.userHasRoles, // fill actual user roles from db -// }); - -// async created() { -// // Fetch the list of projects and licenses from the server -// const response = await fetch('/api/datasets/edit/' + this.dataset.id); -// const data = await response.json(); -// this.projects = data.projects; -// this.licenses = data.licenses; -// } + // Check if files have changed + if (form.files?.length !== props.dataset.files?.length) return true; + if (form.filesToDelete?.length > 0) return true; + // Check if there are new files to upload + if (form.files?.some(file => !file.id)) return true; + // No changes detected + return false; +}); const submit = async (): Promise<void> => { let route = stardust.route('dataset.update', [props.dataset.id]); - // await Inertia.post('/app/register', this.form); - // await router.post('/app/register', this.form); - let licenses = form.licenses.map((obj) => { if (hasIdAttribute(obj)) { @@ -682,12 +710,6 @@ const submit = async (): Promise<void> => { } }); - - - // const files = form.files.map((obj) => { - // return new File([obj.blob], obj.label, { type: obj.type, lastModified: obj.lastModified }); - // }); - const [fileUploads, fileInputs] = form.files?.reduce( ([fileUploads, fileInputs], obj) => { if (!obj.id) { @@ -700,11 +722,11 @@ const submit = async (): Promise<void> => { // const file = new File([obj.blob], `${obj.label}?sortOrder=${obj.sort_order}`, options); // const metadata = JSON.stringify({ sort_order: obj.sort_order }); // const metadataBlob = new Blob([metadata + '\n'], { type: 'application/json' }); - const file = new File([obj.blob], `${obj.label}`, options,); - + const file = new File([obj.blob], `${obj.label}?sortorder=${obj.sort_order}`, options,); + // const file = new File([obj.blob], `${obj.label}`, options); - - + + // fileUploads[obj.sort_order] = file; fileUploads.push(file); } else { @@ -744,7 +766,9 @@ const submit = async (): Promise<void> => { // formStep.value++; // form.filesToDelete = []; // Clear the array using splice - form.filesToDelete?.splice(0, form.filesToDelete.length); + form.filesToDelete?.splice(0, form.filesToDelete.length); + form.subjectsToDelete?.splice(0, form.subjectsToDelete.length); + form.referencesToDelete?.splice(0, form.referencesToDelete.length); }, }); }; @@ -769,6 +793,11 @@ const removeDescription = (key: any) => { form.descriptions.splice(key, 1); }; +const addNewAuthor = () => { + let newAuthor = { status: false, first_name: '', last_name: '', email: '', academic_title: '', identifier_orcid: '', name_type: 'Personal' }; + form.authors.push(newAuthor); +}; + const onAddAuthor = (person: Person) => { if (form.authors.filter((e) => e.id === person.id).length > 0) { notify({ type: 'warning', title: 'Warning', text: 'person is already defined as author' }, 4000); @@ -780,6 +809,11 @@ const onAddAuthor = (person: Person) => { } }; +const addNewContributor = () => { + let newContributor = { status: false, first_name: '', last_name: '', email: '', academic_title: '', identifier_orcid: '', name_type: 'Personal', pivot: { contributor_type: '' } }; + form.contributors.push(newContributor); +}; + const onAddContributor = (person: Person) => { if (form.contributors.filter((e) => e.id === person.id).length > 0) { notify({ type: 'warning', title: 'Warning', text: 'person is already defined as contributor' }, 4000); @@ -794,7 +828,7 @@ const onAddContributor = (person: Person) => { }; const addKeyword = () => { - let newSubject: Subject = { value: 'test', language: '', type: 'uncontrolled', dataset_count: 0 }; + let newSubject: Subject = { value: '', language: '', type: 'uncontrolled' }; //this.dataset.files.push(uploadedFiles[i]); form.subjects.push(newSubject); }; @@ -806,9 +840,35 @@ const addReference = () => { }; const removeReference = (key: any) => { + const reference = form.references[key]; + + // If the reference has an ID, it exists in the database + // and should be added to referencesToDelete + if (reference.id) { + // Initialize referencesToDelete array if it doesn't exist + if (!form.referencesToDelete) { + form.referencesToDelete = []; + } + + // Add to referencesToDelete + form.referencesToDelete.push(reference); + } + + // Remove from form.references array form.references.splice(key, 1); }; +const restoreReference = (index: number) => { + // Get the reference from referencesToDelete + const reference = form.referencesToDelete[index]; + + // Add it back to form.references + form.references.push(reference); + + // Remove it from referencesToDelete + form.referencesToDelete.splice(index, 1); +}; + const onMapInitialized = (newItem: any) => { console.log(newItem); }; diff --git a/resources/js/Pages/Submitter/Dataset/Index.vue b/resources/js/Pages/Submitter/Dataset/Index.vue index 0cf8ca5..5b01370 100644 --- a/resources/js/Pages/Submitter/Dataset/Index.vue +++ b/resources/js/Pages/Submitter/Dataset/Index.vue @@ -2,7 +2,7 @@ // import { Head, Link, useForm, usePage } from '@inertiajs/inertia-vue3'; import { Head, usePage } from '@inertiajs/vue3'; import { ComputedRef } from 'vue'; -import { mdiSquareEditOutline, mdiTrashCan, mdiAlertBoxOutline, mdiLockOpen } from '@mdi/js'; +import { mdiSquareEditOutline, mdiTrashCan, mdiAlertBoxOutline, mdiLockOpen, mdiLibraryShelves } from '@mdi/js'; import { computed } from 'vue'; import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; import SectionMain from '@/Components/SectionMain.vue'; @@ -95,18 +95,18 @@ const formatServerState = (state: string) => { <table class="w-full table-fixed"> <thead> <tr> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> <!-- <Sort label="Dataset Title" attribute="title" :search="form.search" /> --> Dataset Title - </th> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + </th> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> <!-- <Sort label="Email" attribute="email" :search="form.search" /> --> Server State </th> - <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider"> + <th scope="col" class="py-3 text-left text-xs font-medium uppercase tracking-wider dark:text-white"> Date of last modification </th> - <th scope="col" class="relative px-6 py-3" v-if="can.edit || can.delete"> + <th scope="col" class="relative px-6 py-3 dark:text-white" v-if="can.edit || can.delete"> <span class="sr-only">Actions</span> </th> </tr> @@ -114,32 +114,54 @@ const formatServerState = (state: string) => { <tbody class="bg-white divide-y divide-gray-200"> <tr v-for="dataset in props.datasets.data" :key="dataset.id" :class="getRowClass(dataset)"> - <td data-label="Login" class="py-4 whitespace-nowrap text-gray-700 dark:text-white table-title"> + <td data-label="Login" + class="py-4 whitespace-nowrap text-gray-700 table-title"> <!-- <Link v-bind:href="stardust.route('settings.user.show', [user.id])" class="no-underline hover:underline text-cyan-600 dark:text-cyan-400"> {{ user.login }} </Link> --> <!-- {{ user.id }} --> - {{ dataset.main_title }} + {{ dataset.main_title }} </td> - <td class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td class="py-4 whitespace-nowrap text-gray-700"> {{ formatServerState(dataset.server_state) }} + <div v-if="dataset.server_state === 'rejected_editor' && dataset.reject_editor_note" + class="inline-block relative ml-2 group"> + <button + class="w-5 h-5 rounded-full bg-gray-200 text-gray-600 text-xs flex items-center justify-center focus:outline-none hover:bg-gray-300"> + i + </button> + <div + class="absolute left-0 top-full mt-1 w-64 bg-white shadow-lg rounded-md p-3 text-xs text-left z-50 transform scale-0 origin-top-left transition-transform duration-100 group-hover:scale-100"> + <p + class="text-gray-700 max-h-40 overflow-y-auto overflow-x-hidden whitespace-normal break-words"> + {{ dataset.reject_editor_note }} + </p> + <div class="absolute -top-1 left-1 w-2 h-2 bg-white transform rotate-45"> + </div> + </div> + </div> </td> - <td data-label="modified" class="py-4 whitespace-nowrap text-gray-700 dark:text-white"> + <td data-label="modified" class="py-4 whitespace-nowrap text-gray-700"> <div class="text-sm" :title="dataset.server_date_modified"> {{ dataset.server_date_modified }} </div> </td> - <td class="py-4 whitespace-nowrap text-right text-sm font-medium text-gray-700 dark:text-white"> + <td + class="py-4 whitespace-nowrap text-right text-sm font-medium text-gray-700"> <BaseButtons v-if="validStates.includes(dataset.server_state)" type="justify-start lg:justify-end" no-wrap> <!-- release created dataset --> <BaseButton v-if="can.edit" :route-name="stardust.route('dataset.release', [dataset.id])" color="info" :icon="mdiLockOpen" :label="'Release'" small /> - <BaseButton v-if="can.edit" :route-name="stardust.route('dataset.edit', [dataset.id])" - color="info" :icon="mdiSquareEditOutline" :label="'Edit'" small /> + <BaseButton v-if="can.edit" + :route-name="stardust.route('dataset.edit', [dataset.id])" color="info" + :icon="mdiSquareEditOutline" :label="'Edit'" small /> + <BaseButton v-if="can.edit" + :route-name="stardust.route('dataset.categorize', [dataset.id])" color="info" + :icon="mdiLibraryShelves" :label="'Classify'" small /> <BaseButton v-if="can.delete" color="danger" :route-name="stardust.route('dataset.delete', [dataset.id])" :icon="mdiTrashCan" small /> @@ -149,7 +171,7 @@ const formatServerState = (state: string) => { </tbody> </table> <div class="py-4"> - <Pagination v-bind:data="datasets.meta" /> + <Pagination v-bind:data="datasets.meta" /> </div> </CardBox> </SectionMain> @@ -157,13 +179,17 @@ const formatServerState = (state: string) => { </template> <style scoped lang="css"> - .table-title { - max-width: 200px; /* set a maximum width */ - overflow: hidden; /* hide overflow */ - text-overflow: ellipsis; /* show ellipsis for overflowed text */ - white-space: nowrap; /* prevent wrapping */ + max-width: 200px; + /* set a maximum width */ + overflow: hidden; + /* hide overflow */ + text-overflow: ellipsis; + /* show ellipsis for overflowed text */ + white-space: nowrap; + /* prevent wrapping */ } + .table-fixed { table-layout: fixed; } @@ -205,6 +231,4 @@ const formatServerState = (state: string) => { color: whitesmoke; }*/ - - -</style> +</style> diff --git a/resources/js/Pages/profile/partials/update-password-form.vue b/resources/js/Pages/profile/partials/update-password-form.vue new file mode 100644 index 0000000..9e0d422 --- /dev/null +++ b/resources/js/Pages/profile/partials/update-password-form.vue @@ -0,0 +1,135 @@ +<script lang="ts" setup> +import FormControl from '@/Components/FormControl.vue'; +import FormField from '@/Components/FormField.vue'; +import ActionMessage from '@/Components/action-message.vue' +import BaseButton from '@/Components/BaseButton.vue'; +import BaseButtons from '@/Components/BaseButtons.vue'; +import { useForm, usePage } from '@inertiajs/vue3'; +import { ref, Ref, computed } from 'vue'; +import { stardust } from '@eidellev/adonis-stardust/client'; +import CardBox from '@/Components/CardBox.vue'; +import { mdiLock } from '@mdi/js'; +import PasswordMeter from '@/Components/SimplePasswordMeter/password-meter.vue'; +// import BaseDivider from '@/Components/BaseDivider.vue'; + +// const errors: Ref<any> = computed(() => { +// return usePage().props.errors; +// }); +const flash: Ref<any> = computed(() => { + return usePage().props.flash; +}); + +const newPasswordInput: Ref<typeof FormControl | null> = ref(null); +const oldPasswordInput: Ref<typeof FormControl | null> = ref(null); + +const enabled = ref(false); +const handleScore = (score: number) => { + if (score >= 4) { + enabled.value = true; + } else { + enabled.value = false; + } + // strengthLabel.value = scoreLabel; + // score.value = scoreValue; +}; + +const form = useForm({ + old_password: '', + new_password: '', + confirm_password: '', +}); +const updatePassword = async () => { + await form.put(stardust.route('settings.password.update'), { + preserveScroll: true, + onSuccess: () => { + form.reset(); + }, + onError: () => { + if (form.errors.new_password) { + form.reset('new_password', 'confirm_password'); + enabled.value = false; + // newPasswordInput.value.focus(); + // newPasswordInput.value?.focus(); + } + + if (form.errors.old_password) { + form.reset('old_password'); + // oldPasswordInput.value?.focus(); + } + }, + }); +}; + +</script> + +<template> + <!-- <div class="p-7 text-gray-900 bg-white rounded-lg border border-gray-100 shadow dark:border-gray-600 dark:bg-secondary-dark dark:text-white"> --> + + + + + <!-- <div class="mb-4"> + <h3 class="text-dark text-md">{{ 'Update Password' }}</h3> + </div> --> + + <CardBox id="passwordForm" title="Change Password" :icon="mdiLock" form :show-header-icon="false"> + + + + <FormField label="Current password" help="Required. Your current password" + :class="{ 'text-red-400': form.errors.old_password }"> + <FormControl label="Current Password" id="current_password" ref="oldPasswordInput" + :placeholder="'Please Enter Current Password'" v-model="form.old_password" + :error="form.errors.old_password" type="password" class="block w-full" autocomplete="current-password"> + <div class="text-red-400 text-sm" v-if="form.errors.old_password"> + {{ form.errors.old_password }} + </div> + </FormControl> + </FormField> + + + <!-- <div class="col-span-6 sm:col-span-4"> --> + <!-- <FormControl label="New Password" id="password" :placeholder="'Please Enter New Password'" + ref="newPasswordInput" v-model="form.new_password" type="password" class="block w-full" + autocomplete="new-password" :error="form.errors.new_password"> + <div class="text-red-400 text-sm" v-if="form.errors.new_password"> + {{ form.errors.new_password }} + </div> + </FormControl> --> + <PasswordMeter ref="newPasswordInput" v-model="form.new_password" :errors="form.errors" + @score="handleScore" /> + + <!-- </div> --> + + <FormField label="Confirm password" help="Required. New password one more time" + :class="{ 'text-red-400': form.errors.confirm_password }"> + <FormControl label="Confirm Password" :placeholder="'Please Enter Confirm Password'" id="confirm_password" + v-model="form.confirm_password" type="password" class="block w-full" autocomplete="new-password" + :error="form.errors.confirm_password"> + <div class="text-red-400 text-sm" v-if="form.errors.confirm_password"> + {{ form.errors.confirm_password }} + </div> + </FormControl> + </FormField> + + <!-- <BaseDivider /> --> + + <template #footer> + + + + <div class="flex items-center justify-end gap-3 mt-5"> + <ActionMessage v-if="flash.message" :on="form.recentlySuccessful" color="success"> + {{ flash.message }} + </ActionMessage> + <ActionMessage v-if="flash.warning" :on="form.recentlySuccessful" color="warning"> + {{ flash.warning }} + </ActionMessage> + <BaseButtons> + <BaseButton type="submit" color="info" label="Change password" @click.prevent="updatePassword()" + :disabled="form.processing == true || enabled == false" /> + </BaseButtons> + </div> + </template> + </CardBox> +</template> \ No newline at end of file diff --git a/resources/js/Pages/profile/partials/update-profile-information-form.vue b/resources/js/Pages/profile/partials/update-profile-information-form.vue new file mode 100644 index 0000000..5a5fdff --- /dev/null +++ b/resources/js/Pages/profile/partials/update-profile-information-form.vue @@ -0,0 +1,179 @@ +<script lang="ts" setup> +import { computed, Ref, ref } from 'vue'; +import { useForm, usePage } from '@inertiajs/vue3'; +import ActionMessage from '@/Components/action-message.vue' +import FormControl from '@/Components/FormControl.vue'; +import FormField from '@/Components/FormField.vue'; +import CardBox from '@/Components/CardBox.vue'; +import { mdiAccount } from '@mdi/js'; +import BaseButton from '@/Components/BaseButton.vue'; +import BaseButtons from '@/Components/BaseButtons.vue' +import { stardust } from '@eidellev/adonis-stardust/client'; +import AvatarInput from '@/Components/avatar-input.vue'; + +const props = defineProps({ + user: { + type: Object, + required: true, + }, + defaultUrl: { + type: String, + required: false, + }, +}); + +const errors: Ref<any> = computed(() => { + return usePage().props.errors || {} +}); +const flash: Ref<any> = computed(() => { + return usePage().props.flash; +}); + +const fullName = computed(() => `${props.user.first_name} ${props.user.last_name}`); +const recentlyHasError = ref(false); + +const form = useForm({ + first_name: props.user.first_name, + last_name: props.user.last_name, + login: props.user.login, + email: props.user.email, + // mobile: `${props.user.mobile}`, + avatar: undefined as File | undefined, +}); + +// const verificationLinkSent = ref(null); +const avatarInput: Ref<HTMLInputElement | null> = ref(null); + +const updateProfileInformation = () => { + // if (avatarInput.value) { + // form.avatar = avatarInput.value?.files ? avatarInput.value.files[0] : undefined; + // } + + form.put(stardust.route('settings.profile.update', [props.user.id]), { + errorBag: 'updateProfileInformation', + preserveScroll: true, + onSuccess: () => { + // clearPhotoFileInput(); + }, + onError: () => { + if (form.errors.avatar) { + if (avatarInput.value) { + avatarInput.value.value = ''; + } + } + recentlyHasError.value = true + setTimeout(() => { + recentlyHasError.value = false + }, 5000) + }, + }); +}; +// const sendEmailVerification = () => { +// verificationLinkSent.value = true; +// }; + +</script> + +<template> + <CardBox id="passwordForm" title="Basic Info" :icon="mdiAccount" form :show-header-icon="false"> + <!-- <FormValidationErrors v-bind:errors="errors" /> --> + + <AvatarInput class="h-24 w-24 rounded-full" v-model="form.avatar" ref="avatarInput" + :default-src="defaultUrl ? defaultUrl : '/api/avatar?name=' + fullName + '&size=50'"> + </AvatarInput> + <div class="text-red-400 text-sm" v-if="errors.avatar && Array.isArray(errors.avatar)"> + {{ errors.avatar.join(', ') }} + </div> + + <FormField label="First Name" :class="{ 'text-red-400': form.errors.first_name }"> + <FormControl id="first_name" label="First Name" v-model="form.first_name" :error="form.errors.first_name" + type="text" :placeholder="'First Name'" class="w-full" autocomplete="first_name"> + <div class="text-red-400 text-sm" v-if="errors.first_name && Array.isArray(errors.first_name)"> + {{ errors.first_name.join(', ') }} + </div> + </FormControl> + </FormField> + + <FormField label="Last Name" :class="{ 'text-red-400': form.errors.last_name }"> + <FormControl id="last_name" label="Last Name" v-model="form.last_name" :error="form.errors.last_name" + type="text" :placeholder="'Last Name'" class="w-full" autocomplete="last_name"> + <div class="text-red-400 text-sm" v-if="errors.last_name && Array.isArray(errors.last_name)"> + {{ errors.last_name.join(', ') }} + </div> + </FormControl> + </FormField> + + <FormField label="Username" :class="{ 'text-red-400': form.errors.login }"> + <FormControl id="username" label="Username" v-model="form.login" class="w-full" + :is-read-only="true"> + <div class="text-red-400 text-sm" v-if="errors.login && Array.isArray(errors.login)"> + {{ errors.login.join(', ') }} + </div> + </FormControl> + + </FormField> + + <FormField label="Enter Email"> + <FormControl v-model="form.email" type="text" placeholder="Email" :errors="form.errors.email" + :is-read-only="true"> + <div class="text-red-400 text-sm" v-if="errors.email && Array.isArray(errors.email)"> + {{ errors.email.join(', ') }} + </div> + </FormControl> + </FormField> + + <!-- Email --> + <!-- <div> + <FormControl label="Email" id="email" v-model="form.email" :readonly="!user.is_super_admin" + :disabled="!user.is_super_admin" class="w-full" /> + + <div v-if="user.email_verified_at === null"> + <p class="text-sm mt-2"> + {{ 'Your email address is unverified.' }} + + <Link :href="route('verification.send')" method="post" as="button" + class="underline text-gray-600 hover:text-gray-900" @click.prevent="sendEmailVerification"> + {{ 'Click here to re-send the verification email.' }} + </Link> + </p> + + <div v-show="verificationLinkSent" class="mt-2 font-medium text-sm text-green-600"> + {{ 'A new verification link has been sent to your email address.' }} + </div> + </div> + </div> --> + + <!-- <div class="relative"> + <FormControl label="Mobile" id="mobile" class="w-full" input-class="w-full pl-5" v-model="form.mobile" + :readonly="!user.is_super_admin" :disabled="!user.is_super_admin" /> + <span class="absolute top-9 left-0 inline-flex items-center ml-2 font-bold text-secondary-light">+</span> + </div> --> + + <!-- <div class="col-span-2 flex justify-end items-center mt-5"> --> + <template #footer> + <div class="flex items-center justify-end gap-3 "> + <ActionMessage :on="recentlyHasError" color="warning"> + <ul class="list-disc list-inside space-y-2 text-sm"> + <li v-for="(messages, field) in errors" :key="field" class="flex flex-col"> + <span class="font-semibold capitalize text-gray-700 dark:text-gray-300">{{ field }}:</span> + <span class="text-red-600 dark:text-red-400 ml-4">{{ messages.join(', ') }}</span> + </li> + </ul> + </ActionMessage> + <ActionMessage v-if="flash.message" :on="form.recentlySuccessful" color="success"> + {{ flash.message }} + </ActionMessage> + <ActionMessage v-if="flash.warning" :on="form.recentlySuccessful" color="warning"> + {{ flash.warning }} + </ActionMessage> + <BaseButtons> + <BaseButton type="submit" color="info" label="Save Changes" + @click.prevent="updateProfileInformation" :disabled="form.processing" /> + </BaseButtons> + </div> + </template> + </CardBox> + + <!-- </div> + </div> --> +</template> \ No newline at end of file diff --git a/resources/js/Pages/profile/show.vue b/resources/js/Pages/profile/show.vue new file mode 100644 index 0000000..12714f9 --- /dev/null +++ b/resources/js/Pages/profile/show.vue @@ -0,0 +1,139 @@ +<script lang="ts" setup> + +/*========================================================================================= + File Name: show profile + + ---------------------------------------------------------------------------------------- + Author: Arno Kaimbacher + Author URL: https://jakint.at/ +==========================================================================================*/ + + + +// import { useUrlSearchParams } from '@vueuse/core'; +// import { usePage } from '@inertiajs/vue3'; +import { TabGroup, TabList, Tab, TabPanels, TabPanel } from '@headlessui/vue'; +import { ref, computed } from 'vue'; +// import IconRounded from '@/Components/IconRounded.vue'; + + +// import AdminLayout from '@/Layouts/AdminLayout.vue'; +// import AppLayout from '@/Layouts/AppLayout.vue'; +import LayoutAuthenticated from '@/Layouts/LayoutAuthenticated.vue'; + +import UpdatePasswordForm from '@/Pages/profile/partials/update-password-form.vue'; +import UpdateProfileInformationForm from '@/Pages/profile/partials/update-profile-information-form.vue'; +import { usePage } from '@inertiajs/vue3'; +import SectionMain from '@/Components/SectionMain.vue'; +import SectionTitleLineWithButton from '@/Components/SectionTitleLineWithButton.vue'; +import { mdiAccount, mdiArrowLeftBoldOutline, mdiFormTextboxPassword } from '@mdi/js'; +import { stardust } from '@eidellev/adonis-stardust/client'; +import BaseButton from '@/Components/BaseButton.vue'; +import BaseIcon from '@/Components/BaseIcon.vue'; + +defineProps({ + user: { + type: Object, + required: true, + }, + defaultUrl: { + type: String, + required: true, + }, +}); + +const tabs = [ + { id: 1, title: 'My Profile', icon: mdiAccount }, + { id: 2, title: 'Change Password', icon: mdiFormTextboxPassword }, +]; + +// const params = useUrlSearchParams('history'); +const selectedTab = ref(0); + +function changeTab(index: number) { + selectedTab.value = index; +} + +const user = computed(() => usePage().props.user); +// const sessions = computed(() => usePage().props.sessions); +// const Layout = computed(() => (user.value.is_super_admin ? AdminLayout : AppLayout)); +</script> + +<template> + <Component :is="LayoutAuthenticated"> + + <Head title="Profile"></Head> + <SectionMain> + <SectionTitleLineWithButton :icon="mdiAccount" title="Profile" main> + <BaseButton :route-name="stardust.route('dashboard')" :icon="mdiArrowLeftBoldOutline" label="Back" + color="white" rounded-full small /> + </SectionTitleLineWithButton> + + <!-- <NotificationBar v-if="flash.message" color="success" :icon="mdiAlertBoxOutline"> + {{ flash.message }} + </NotificationBar> --> + + <div class=""> + <TabGroup :selectedIndex="selectedTab" @change="changeTab"> + <TabList class="flex space-x-7 p-1"> + <Tab v-for="(tab, index) in tabs" as="template" :key="index" v-slot="{ selected }"> + <button :class="[ + 'inline-flex items-center justify-center font-semibold focus:outline-none disabled:opacity-25 pb-2', + selected + ? 'text-dark border-b-2 border-primary transition-all duration-500 inline-block' + : 'text-light', + ]"> + <!-- <IconRounded :name="tab.icon" class="h-5 mr-3" /> --> + <BaseIcon :path="tab.icon" class="flex-none" w="w-12" /> + {{ tab.title }} + </button> + </Tab> + </TabList> + <transition v-show="selectedTab >= 0" enter-active-class="transition duration-100 ease-out" + enter-from-class="transform scale-95 opacity-0" enter-to-class="transform scale-100 opacity-100" + leave-active-class="transition duration-75 ease-in" + leave-from-class="transform scale-100 opacity-100" + leave-to-class="transform scale-95 opacity-0"> + <TabPanels class="mt-2"> + <TabPanel :key="Date.now().toString() + 1" :class="[]"> + <div class="grid grid-cols-1 lg:grid-cols-2 gap-6"> + <UpdateProfileInformationForm class="p-5" :user="user" :default-url="defaultUrl" /> + </div> + </TabPanel> + <TabPanel :key="Date.now().toString() + 3" :class="[]"> + <!-- <div class="grid grid-cols-1 md:grid-cols-2 gap-x-5 gap-y-7 items-start"> --> + <div class="grid grid-cols-1 lg:grid-cols-2 gap-6"> + <UpdatePasswordForm class="p-5" /> + </div> + </TabPanel> + + </TabPanels> + </transition> + </TabGroup> + </div> + </SectionMain> + </Component> +</template> + +<style> +.cool-link::after { + content: ''; + display: block; + width: 0; + height: 2px; + background: #fda92d; + transition: width 0.3s; +} + +.cool-link:hover::after { + width: 100%; +} +</style> + +<!-- <script> +import { defineComponent } from 'vue'; + +export default defineComponent({ + layout: false, +}); +</script> --> \ No newline at end of file diff --git a/resources/js/Stores/main.ts b/resources/js/Stores/main.ts index 8d51088..3a31057 100644 --- a/resources/js/Stores/main.ts +++ b/resources/js/Stores/main.ts @@ -2,6 +2,56 @@ import { defineStore } from 'pinia'; import axios from 'axios'; import { Dataset } from '@/Dataset'; import menu from '@/menu'; +// import type Person from '#models/person'; + +export interface User { + id: number; + login: string; + firstName: string; + lastName: string; + email: string; + password: string; + created_at: DateTime; + updatedAt: DateTime; + lastLoginAt: DateTime; + isActive: boolean; + isVerified: boolean; + roles: string[]; + permissions: string[]; + settings: Record<string, any>; + profile: { + avatar: string; + bio: string; + location: string; + website: string; + social: { + twitter: string; + facebook: string; + linkedin: string; + github: string; + } + }; + metadata: Record<string, any>; + verifyPassword: (plainPassword: string) => Promise<boolean>; +} + +interface DateTime { + get: (unit: keyof DateTime) => number; + getPossibleOffsets: () => DateTime[]; + toRelativeCalendar: (options?: ToRelativeCalendarOptions) => string | null; + toFormat: (format: string) => string; + toISO: () => string; + toJSON: () => string; + toString: () => string; + toLocaleString: (options?: Intl.DateTimeFormatOptions) => string; + toUTC: () => DateTime; + toLocal: () => DateTime; + valueOf: () => number; + toMillis: () => number; + toSeconds: () => number; + toUnixInteger: () => number; +} + export interface Person { id: number; @@ -9,10 +59,12 @@ export interface Person { email: string; name_type: string; identifier_orcid: string; - datasetCount: string; + dataset_count: number; created_at: string; } + + interface TransactionItem { amount: number; account: string; @@ -61,7 +113,7 @@ export const MainService = defineStore('main', { isFieldFocusRegistered: false, /* Sample data for starting dashboard(commonly used) */ - clients: [], + clients: [] as Array<User>, history: [] as Array<TransactionItem>, // api based data @@ -146,7 +198,8 @@ export const MainService = defineStore('main', { } }) .catch((error) => { - alert(error.message); + // alert(error.message); + throw error; }); }, @@ -184,17 +237,18 @@ export const MainService = defineStore('main', { this.totpState = state; }, - async fetchChartData(year: string) { + fetchChartData() { // sampleDataKey= authors or datasets axios - .get(`/api/statistic/${year}`) + .get(`/api/statistic`) .then((r) => { if (r.data) { this.graphData = r.data; } }) .catch((error) => { - alert(error.message); + // alert(error.message); + throw error; }); }, diff --git a/resources/js/app.ts b/resources/js/app.ts index 50f019b..67186e6 100644 --- a/resources/js/app.ts +++ b/resources/js/app.ts @@ -2,15 +2,16 @@ import '../css/app.css'; import { createApp, h } from 'vue'; import { Inertia } from '@inertiajs/inertia'; -import { createInertiaApp } from '@inertiajs/vue3'; +import { Head, Link, createInertiaApp } from '@inertiajs/vue3'; // import DefaultLayout from '@/Layouts/Default.vue'; import { createPinia } from 'pinia'; import { StyleService } from '@/Stores/style.service'; import { LayoutService } from '@/Stores/layout'; import { LocaleStore } from '@/Stores/locale'; +import { MainService } from './Stores/main'; import { darkModeKey, styleKey } from '@/config'; -// import type { DefineComponent } from 'vue'; -// import { resolvePageComponent } from '@adonisjs/inertia/helpers'; +import type { DefineComponent } from 'vue'; +import { resolvePageComponent } from '@adonisjs/inertia/helpers'; const pinia = createPinia(); // import i18n from './i18n'; import { EmitterPlugin } from '@/EmitterDirective'; @@ -36,33 +37,21 @@ createInertiaApp({ progress: { // color: '#4B5563', color: '#22C55E', + showSpinner: true, }, - // Webpack - // resolve: (name) => require(`./Pages/${name}`), - // resolve: (name) => require(`./Pages/${name}.vue`), - // add default layout - // resolve: (name) => { - // const page = require(`./Pages/${name}.vue`).default; + // Webpack + // resolve: async (name: string) => { + // // Dynamically import the Vue component using import + // const { default: page } = await import(`./Pages/${name}.vue`); + // // const page = require(`./Pages/${name}.vue`).default; // // if (!page.layout) { // // page.layout = DefaultLayout; // // } // return page; // }, - resolve: async (name: string) => { - // Dynamically import the Vue component using import - const { default: page } = await import(`./Pages/${name}.vue`); - // const page = require(`./Pages/${name}.vue`).default; - // if (!page.layout) { - // page.layout = DefaultLayout; - // } - return page; + resolve: (name) => { + return resolvePageComponent(`./Pages/${name}.vue`, import.meta.glob<DefineComponent>('./Pages/**/*.vue')); }, - // resolve: (name) => { - // return resolvePageComponent( - // `./Pages/${name}.vue`, - // import.meta.glob<DefineComponent>('./pages/**/*.vue'), - // ) - // }, setup({ el, App, props, plugin }) { const app = createApp({ render: () => h(App, props) }) @@ -72,11 +61,19 @@ createInertiaApp({ .use(EmitterPlugin); // .component('inertia-link', Link) + app.component('Head', Head); + app.component('Link', Link); + + // Listen for navigation event to handle layout changes + // window.addEventListener('inertia:navigate', () => { + // layoutService.isAsideMobileExpanded = false; + // layoutService.isAsideLgActive = false; + // }); + asyncPlugin.install('settings').then(() => { app.mount(el); }); }, - }); const styleService = StyleService(pinia); @@ -84,7 +81,7 @@ const layoutService = LayoutService(pinia); const localeService = LocaleStore(pinia); localeService.initializeLocale(); -// const mainService = MainService(pinia); +const mainService = MainService(pinia); // mainService.setUser(user); /* App style */ @@ -94,6 +91,12 @@ styleService.setStyle(localStorage[styleKey] ?? 'basic'); if ((!localStorage[darkModeKey] && window.matchMedia('(prefers-color-scheme: dark)').matches) || localStorage[darkModeKey] === '1') { styleService.setDarkMode(true); } +// mainService.fetch('clients'); +// mainService.fetch('history'); +mainService.fetchApi('clients'); +mainService.fetchApi('authors'); +mainService.fetchApi('datasets'); +mainService.fetchChartData(); /* Collapse mobile aside menu on route change */ Inertia.on('navigate', () => { diff --git a/resources/js/logo.svg b/resources/js/logo.svg new file mode 100644 index 0000000..c0b62d6 --- /dev/null +++ b/resources/js/logo.svg @@ -0,0 +1,9 @@ +<svg width="3420" height="756" viewBox="0 0 3420 756" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> +<rect width="3420" height="756" fill="url(#pattern0_280_776)"/> +<defs> +<pattern id="pattern0_280_776" patternContentUnits="objectBoundingBox" width="1" height="1"> +<use xlink:href="#image0_280_776" transform="scale(0.000292398 0.00132275)"/> +</pattern> +<image id="image0_280_776" width="3420" height="756" preserveAspectRatio="none" xlink:href=""/> +</defs> +</svg> diff --git a/resources/js/menu.ts b/resources/js/menu.ts index 576f340..16d08d7 100644 --- a/resources/js/menu.ts +++ b/resources/js/menu.ts @@ -12,6 +12,7 @@ import { mdiShieldCrownOutline, mdiLicense, mdiFileDocument, + mdiLibraryShelves } from '@mdi/js'; export default [ @@ -27,6 +28,11 @@ export default [ icon: mdiLock, label: 'Security', }, + { + route: 'settings.profile.edit', + icon: mdiLock, + label: 'Profile', + }, // { // route: 'dataset.create', // icon: mdiPublish, @@ -106,6 +112,11 @@ export default [ icon: mdiPublish, label: 'Create Dataset', }, + // { + // route: 'dataset.categorize', + // icon: mdiLibraryShelves, + // label: 'Library Classification', + // }, ], }, { @@ -145,18 +156,13 @@ export default [ // label: 'Create Dataset', // }, ], - }, + }, // { - // route: 'dataset.create', - // icon: mdiDatabasePlus, - // label: 'Create Dataset', + // href: '', + // icon: mdiGithub, + // label: 'Forgejo', + // target: '_blank', // }, - { - href: 'https://gitea.geologie.ac.at/geolba/tethys', - icon: mdiGithub, - label: 'Gitea', - target: '_blank', - }, { href: '/oai', icon: mdiAccountEye, diff --git a/resources/js/tsconfig.json b/resources/js/tsconfig.json index 8dacb19..6b180e3 100644 --- a/resources/js/tsconfig.json +++ b/resources/js/tsconfig.json @@ -15,5 +15,5 @@ }, }, "include": ["./**/*.ts", "./**/*.vue"], - "exclude": ["./utils/*.js"], + "exclude": ["./utils/*.js", "./utils/Timer.js", "./utils/focusTrap.js"], } diff --git a/resources/js/types/models.ts b/resources/js/types/models.ts new file mode 100644 index 0000000..f94dbff --- /dev/null +++ b/resources/js/types/models.ts @@ -0,0 +1,16 @@ +/* -------------------------------------------------------------------------- + Types and Interfaces +-------------------------------------------------------------------------- */ +export interface Collection { + id: number; + name: string; + number: string; + parent_id?: number | null; + inUse?: boolean; +} + +export interface CollectionRole { + id: number; + name: string; + collections?: Collection[]; +} \ No newline at end of file diff --git a/resources/js/utils/tethyscloud-l10n/date.ts b/resources/js/utils/tethyscloud-l10n/date.ts index a17a3c9..2ef59e2 100644 --- a/resources/js/utils/tethyscloud-l10n/date.ts +++ b/resources/js/utils/tethyscloud-l10n/date.ts @@ -1,16 +1,15 @@ - /** * Get the first day of the week * * @return {number} */ export function getFirstDay(): number { - if (typeof window.firstDay === 'undefined') { - console.warn('No firstDay found') - return 1 - } + if (typeof window.firstDay === 'undefined') { + console.warn('No firstDay found'); + return 1; + } - return window.firstDay + return window.firstDay; } /** @@ -19,20 +18,12 @@ export function getFirstDay(): number { * @return {string[]} */ export function getDayNames(): string[] { - if (typeof window.dayNames === 'undefined') { - console.warn('No dayNames found') - return [ - 'Sunday', - 'Monday', - 'Tuesday', - 'Wednesday', - 'Thursday', - 'Friday', - 'Saturday', - ] - } + if (typeof window.dayNames === 'undefined') { + console.warn('No dayNames found'); + return ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday']; + } - return window.dayNames + return window.dayNames; } /** @@ -41,12 +32,12 @@ export function getDayNames(): string[] { * @return {string[]} */ export function getDayNamesShort(): string[] { - if (typeof window.dayNamesShort === 'undefined') { - console.warn('No dayNamesShort found') - return ['Sun.', 'Mon.', 'Tue.', 'Wed.', 'Thu.', 'Fri.', 'Sat.'] - } + if (typeof window.dayNamesShort === 'undefined') { + console.warn('No dayNamesShort found'); + return ['Sun.', 'Mon.', 'Tue.', 'Wed.', 'Thu.', 'Fri.', 'Sat.']; + } - return window.dayNamesShort + return window.dayNamesShort; } /** @@ -55,12 +46,12 @@ export function getDayNamesShort(): string[] { * @return {string[]} */ export function getDayNamesMin(): string[] { - if (typeof window.dayNamesMin === 'undefined') { - console.warn('No dayNamesMin found') - return ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'] - } + if (typeof window.dayNamesMin === 'undefined') { + console.warn('No dayNamesMin found'); + return ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa']; + } - return window.dayNamesMin + return window.dayNamesMin; } /** @@ -69,25 +60,12 @@ export function getDayNamesMin(): string[] { * @return {string[]} */ export function getMonthNames(): string[] { - if (typeof window.monthNames === 'undefined') { - console.warn('No monthNames found') - return [ - 'January', - 'February', - 'March', - 'April', - 'May', - 'June', - 'July', - 'August', - 'September', - 'October', - 'November', - 'December', - ] - } + if (typeof window.monthNames === 'undefined') { + console.warn('No monthNames found'); + return ['January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September', 'October', 'November', 'December']; + } - return window.monthNames + return window.monthNames; } /** @@ -96,23 +74,10 @@ export function getMonthNames(): string[] { * @return {string[]} */ export function getMonthNamesShort(): string[] { - if (typeof window.monthNamesShort === 'undefined') { - console.warn('No monthNamesShort found') - return [ - 'Jan.', - 'Feb.', - 'Mar.', - 'Apr.', - 'May.', - 'Jun.', - 'Jul.', - 'Aug.', - 'Sep.', - 'Oct.', - 'Nov.', - 'Dec.', - ] - } + if (typeof window.monthNamesShort === 'undefined') { + console.warn('No monthNamesShort found'); + return ['Jan.', 'Feb.', 'Mar.', 'Apr.', 'May.', 'Jun.', 'Jul.', 'Aug.', 'Sep.', 'Oct.', 'Nov.', 'Dec.']; + } - return window.monthNamesShort + return window.monthNamesShort; } diff --git a/resources/js/utils/tethyscloud-l10n/translation.ts b/resources/js/utils/tethyscloud-l10n/translation.ts index c2de254..a6f2253 100644 --- a/resources/js/utils/tethyscloud-l10n/translation.ts +++ b/resources/js/utils/tethyscloud-l10n/translation.ts @@ -1,8 +1,8 @@ import type { Translations } from './registry'; -import { getLanguage, getBrowserLocale } from './locale'; -import { getAppTranslations, hasAppTranslations, registerAppTranslations, unregisterAppTranslations } from './registry'; +import { getLanguage } from './locale'; +import { getAppTranslations, registerAppTranslations, unregisterAppTranslations } from './registry'; // import { generateFilePath } from '@nextcloud/router'; -import axios from 'axios'; +// import axios from 'axios'; // import DOMPurify from 'dompurify'; import escapeHTML from 'escape-html'; @@ -206,7 +206,7 @@ export async function loadTranslations(appName: string) { try { // Making a GET request using Axios // const response = await axios.get(url); - const response = await import(`@/apps/settings/l18n/${locale}`); + const response = await import(`@/apps/settings/l18n/${locale}.js`); // Check if the response data contains translations const bundle = response.default; if (typeof bundle.translations === 'object') { diff --git a/resources/views/app.edge b/resources/views/app.edge index eb1fc19..8b97aae 100644 --- a/resources/views/app.edge +++ b/resources/views/app.edge @@ -7,6 +7,12 @@ <meta name="msapplication-TileColor" content="#da532c"> <meta name="theme-color" content="#ffffff"> + <link rel="icon" type="image/svg+xml" href="favicon.svg"> + <link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png"> + <link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png"> + <link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png"> + <link rel="manifest" href="/site.webmanifest"> + <!-- <link rel="icon" href="/apps/theming/favicon/settings?v=ad28c447"> --> <input type="hidden" id="initial-state-firstrunwizard-desktop" value="Imh0dHBzOi8vZ2l0ZWEuZ2VvbG9naWUuYWMuYXQvZ2VvbGJhL3RldGh5cy5iYWNrZW5kIg=="> @@ -22,15 +28,17 @@ <input type="hidden" id="initial-state-settings-cronErrors" value="IiI="> <input type="hidden" id="initial-state-settings-cliBasedCronPossible" value="dHJ1ZQ=="> <input type="hidden" id="initial-state-settings-cliBasedCronUser" value="Ind3dy1kYXRhIg=="> + @vite(['resources/js/app.ts']) @routes('test') + @inertiaHead </head> <body> @inertia({ as: 'div', class: 'h-full' }) - @entryPointStyles('app') - @entryPointScripts('app') + {{-- @entryPointStyles('app') + @entryPointScripts('app') --}} </body> </html> \ No newline at end of file diff --git a/start/env.ts b/start/env.ts index 7b66602..371f1cf 100644 --- a/start/env.ts +++ b/start/env.ts @@ -11,39 +11,37 @@ | and hence do not rename or move this file to a different location. | */ -import { Env } from "@adonisjs/core/env" +import { Env } from '@adonisjs/core/env'; -export default await Env.create(new URL("../", import.meta.url), { - HOST: Env.schema.string({ format: 'host' }), - PORT: Env.schema.number(), - APP_KEY: Env.schema.string(), - APP_NAME: Env.schema.string(), - CACHE_VIEWS: Env.schema.boolean(), - SESSION_DRIVER: Env.schema.enum(["cookie", "memory"] as const), +export default await Env.create(new URL('../', import.meta.url), { + HOST: Env.schema.string({ format: 'host' }), + PORT: Env.schema.number(), + APP_KEY: Env.schema.string(), + APP_NAME: Env.schema.string(), + CACHE_VIEWS: Env.schema.boolean(), + SESSION_DRIVER: Env.schema.enum(['cookie', 'memory'] as const), - DRIVE_DISK: Env.schema.enum(['local'] as const), - NODE_ENV: Env.schema.enum(['development', 'production', 'test'] as const), - DB_CONNECTION: Env.schema.string(), - PG_HOST: Env.schema.string({ format: 'host' }), - PG_PORT: Env.schema.number(), - PG_USER: Env.schema.string(), - PG_PASSWORD: Env.schema.string.optional(), - PG_DB_NAME: Env.schema.string(), + DRIVE_DISK: Env.schema.enum(['local'] as const), + NODE_ENV: Env.schema.enum(['development', 'production', 'test'] as const), + DB_CONNECTION: Env.schema.string(), + PG_HOST: Env.schema.string({ format: 'host' }), + PG_PORT: Env.schema.number(), + PG_USER: Env.schema.string(), + PG_PASSWORD: Env.schema.string.optional(), + PG_DB_NAME: Env.schema.string(), - REDIS_HOST: Env.schema.string({ format: 'host' }), - REDIS_PORT: Env.schema.number(), + REDIS_HOST: Env.schema.string({ format: 'host' }), + REDIS_PORT: Env.schema.number(), - HASH_DRIVER: Env.schema.enum(["scrypt", "argon", "bcrypt", "laravel", undefined] as const), - OAI_LIST_SIZE: Env.schema.number(), + HASH_DRIVER: Env.schema.enum(['scrypt', 'argon', 'bcrypt', 'laravel', undefined] as const), + OAI_LIST_SIZE: Env.schema.number(), - /* + /* |---------------------------------------------------------- | Variables for configuring the mail package |---------------------------------------------------------- */ - SMTP_HOST: Env.schema.string.optional(), - SMTP_PORT: Env.schema.string.optional(), - RESEND_API_KEY: Env.schema.string.optional() - -}) - + SMTP_HOST: Env.schema.string.optional(), + SMTP_PORT: Env.schema.string.optional(), + RESEND_API_KEY: Env.schema.string.optional(), +}); diff --git a/start/kernel.ts b/start/kernel.ts index e0af507..b294bb5 100644 --- a/start/kernel.ts +++ b/start/kernel.ts @@ -24,8 +24,8 @@ server.use([ () => import('@adonisjs/static/static_middleware'), // () => import('@adonisjs/cors/cors_middleware'), () => import('@adonisjs/inertia/inertia_middleware'), + () => import('@adonisjs/vite/vite_middleware'), ]); - /** * The router middleware stack runs middleware on all the HTTP * requests with a registered route. diff --git a/start/routes.ts b/start/routes.ts index 5cb7e50..ad28a87 100644 --- a/start/routes.ts +++ b/start/routes.ts @@ -41,150 +41,425 @@ import db from '@adonisjs/lucid/services/db'; // Import the DB service router.get('/health', ({ response }: HttpContext) => response.noContent()); // OAI routes -router.group(() => { - router.get('/oai', [OaiController, 'index']).as('get'); - router.post('/oai', [OaiController, 'index']).as('post'); -}).as('oai'); +router + .group(() => { + router.get('/oai', [OaiController, 'index']).as('get'); + router.post('/oai', [OaiController, 'index']).as('post'); + }) + .as('oai'); // Welcome route -router.get('/welcome', async ({ view }: HttpContext) => { - return view.render('welcome'); -}).as('welcome'); +router + .get('/welcome', async ({ view }: HttpContext) => { + return view.render('welcome'); + }) + .as('welcome'); // Dashboard route -router.get('/', async ({ response }: HttpContext) => { - return response.redirect().toRoute('apps.dashboard'); -}).as('dashboard'); +router + .get('/', async ({ response }: HttpContext) => { + return response.redirect().toRoute('apps.dashboard'); + }) + .as('dashboard'); // Apps group -router.group(() => { - router.get('/dashboard', async ({ inertia }: HttpContext) => { - return inertia.render('Dashboard'); - }).as('dashboard'); +router + .group(() => { + router + .get('/dashboard', async ({ inertia }: HttpContext) => { + return inertia.render('Dashboard'); + }) + .as('dashboard'); - router.get('/map', async ({ inertia }: HttpContext) => { - return inertia.render('Map'); - }).as('map'); + router + .get('/map', async ({ inertia }: HttpContext) => { + return inertia.render('Map'); + }) + .as('map'); - router.get('/', async ({ inertia }: HttpContext) => { - const users = await User.query().orderBy('login'); - return inertia.render('App', { - testing: 'this is a test', - users: users, - }); - }).as('index'); + router + .get('/', async ({ inertia }: HttpContext) => { + const users = await User.query().orderBy('login'); + return inertia.render('App', { + testing: 'this is a test', + users: users, + }); + }) + .as('index'); - router.get('/register', async ({ inertia }: HttpContext) => { - return inertia.render('register-view/register-view-component'); - }).as('register.show'); + router + .get('/register', async ({ inertia }: HttpContext) => { + return inertia.render('register-view/register-view-component'); + }) + .as('register.show'); - router.post('/register', async ({ request, response }: HttpContext) => { - const data = await request.validateUsing(authValidator); - return response.redirect().toRoute('app.index'); - }).as('register.store'); -}).prefix('apps').as('apps').use(middleware.auth()); + router + .post('/register', async ({ request, response }: HttpContext) => { + await request.validateUsing(authValidator); + return response.redirect().toRoute('app.index'); + }) + .as('register.store'); + }) + .prefix('apps') + .as('apps') + .use(middleware.auth()); // Auth routes -router.get('/app/login', ({ inertia }: HttpContext) => { - return inertia.render('Auth/Login'); -}).as('app.login.show'); +router + .get('/app/login', async ({ inertia }: HttpContext) => { + try { + await db.connection().rawQuery('SELECT 1'); + } catch (error) { + if (error.code === 'ECONNREFUSED') { + throw error; + } + } + return inertia.render('Auth/Login'); + }) + .as('app.login.show'); router.post('/app/login', [AuthController, 'login']).as('login.store'); router.post('/app/twoFactorChallenge', [AuthController, 'twoFactorChallenge']).as('login.twoFactorChallenge'); router.post('/signout', [AuthController, 'logout']).as('logout'); // Administrator routes -router.group(() => { - router.get('/settings', async ({ inertia }: HttpContext) => { - const updatedConfigValue = await db.from('appconfigs') - .select('configvalue') - .where('appid', 'backgroundjob') - .where('configkey', 'lastjob') - .first(); - return inertia.render('Admin/Settings', { - lastCron: updatedConfigValue?.configvalue || '', - }); - }).as('overview'); +router + .group(() => { + router + .get('/settings', async ({ inertia }: HttpContext) => { + const updatedConfigValue = await db + .from('appconfigs') + .select('configvalue') + .where('appid', 'backgroundjob') + .where('configkey', 'lastjob') + .first(); + return inertia.render('Admin/Settings', { + lastCron: updatedConfigValue?.configvalue || '', + }); + }) + .as('overview'); - router.post('/mail/store', [MailSettingsController, 'setMailSettings']).as('mail.store').use(middleware.can(['user-create'])); - router.post('/mail/send', [MailSettingsController, 'sendTestMail']).as('mail.send').use(middleware.can(['user-create'])); + router + .post('/mail/store', [MailSettingsController, 'setMailSettings']) + .as('mail.store') + .use(middleware.can(['user-create'])); + router + .post('/mail/send', [MailSettingsController, 'sendTestMail']) + .as('mail.send') + .use(middleware.can(['user-create'])); - // User routes - router.get('/user', [AdminuserController, 'index']).as('user.index').use(middleware.can(['user-list'])); - router.get('/user/create', [AdminuserController, 'create']).as('user.create').use(middleware.can(['user-create'])); - router.post('/user/store', [AdminuserController, 'store']).as('user.store').use(middleware.can(['user-create'])); - router.get('/user/:id', [AdminuserController, 'show']).as('user.show').where('id', router.matchers.number()); - router.get('/user/:id/edit', [AdminuserController, 'edit']).as('user.edit').where('id', router.matchers.number()).use(middleware.can(['user-edit'])); - router.put('/user/:id/update', [AdminuserController, 'update']).as('user.update').where('id', router.matchers.number()).use(middleware.can(['user-edit'])); - router.delete('/user/:id', [AdminuserController, 'destroy']).as('user.destroy').where('id', router.matchers.number()).use(middleware.can(['user-delete'])); + // User routes + router + .get('/user', [AdminuserController, 'index']) + .as('user.index') + .use(middleware.can(['user-list'])); + router + .get('/user/create', [AdminuserController, 'create']) + .as('user.create') + .use(middleware.can(['user-create'])); + router + .post('/user/store', [AdminuserController, 'store']) + .as('user.store') + .use(middleware.can(['user-create'])); + router.get('/user/:id', [AdminuserController, 'show']).as('user.show').where('id', router.matchers.number()); + router + .get('/user/:id/edit', [AdminuserController, 'edit']) + .as('user.edit') + .where('id', router.matchers.number()) + .use(middleware.can(['user-edit'])); + router + .put('/user/:id/update', [AdminuserController, 'update']) + .as('user.update') + .where('id', router.matchers.number()) + .use(middleware.can(['user-edit'])); + router + .delete('/user/:id', [AdminuserController, 'destroy']) + .as('user.destroy') + .where('id', router.matchers.number()) + .use(middleware.can(['user-delete'])); - // Role routes - router.get('/role', [RoleController, 'index']).as('role.index').use(middleware.can(['user-list'])); - router.get('/role/create', [RoleController, 'create']).as('role.create').use(middleware.can(['user-create'])); - router.post('/role/store', [RoleController, 'store']).as('role.store').use(middleware.can(['user-create'])); - router.get('/role/:id', [RoleController, 'show']).as('role.show').where('id', router.matchers.number()); - router.get('/role/:id/edit', [RoleController, 'edit']).as('role.edit').where('id', router.matchers.number()).use(middleware.can(['user-edit'])); - router.put('/role/:id/update', [RoleController, 'update']).as('role.update').where('id', router.matchers.number()).use(middleware.can(['user-edit'])); + // Role routes + router + .get('/role', [RoleController, 'index']) + .as('role.index') + .use(middleware.can(['user-list'])); + router + .get('/role/create', [RoleController, 'create']) + .as('role.create') + .use(middleware.can(['user-create'])); + router + .post('/role/store', [RoleController, 'store']) + .as('role.store') + .use(middleware.can(['user-create'])); + router.get('/role/:id', [RoleController, 'show']).as('role.show').where('id', router.matchers.number()); + router + .get('/role/:id/edit', [RoleController, 'edit']) + .as('role.edit') + .where('id', router.matchers.number()) + .use(middleware.can(['user-edit'])); + router + .put('/role/:id/update', [RoleController, 'update']) + .as('role.update') + .where('id', router.matchers.number()) + .use(middleware.can(['user-edit'])); - // License routes - router.get('/license', [LicenseController, 'index']).as('license.index'); - router.get('/license/:id/down', [LicenseController, 'down']).as('license.down').where('id', router.matchers.number()).use(middleware.can(['settings'])); - router.get('/license/:id/up', [LicenseController, 'up']).as('license.up').where('id', router.matchers.number()).use(middleware.can(['settings'])); + // License routes + router.get('/license', [LicenseController, 'index']).as('license.index'); + router + .get('/license/:id/down', [LicenseController, 'down']) + .as('license.down') + .where('id', router.matchers.number()) + .use(middleware.can(['settings'])); + router + .get('/license/:id/up', [LicenseController, 'up']) + .as('license.up') + .where('id', router.matchers.number()) + .use(middleware.can(['settings'])); - // Mimetype routes - router.get('/mimetype', [MimetypeController, 'index']).as('mimetype.index'); - router.get('/mimetype/create', [MimetypeController, 'create']).as('mimetype.create').use(middleware.can(['settings'])); - router.post('/mimetype/store', [MimetypeController, 'store']).as('mimetype.store').use(middleware.can(['settings'])); - router.get('/mimetype/:id/down', [MimetypeController, 'down']).as('mimetype.down').where('id', router.matchers.number()).use(middleware.can(['settings'])); - router.get('/mimetype/:id/up', [MimetypeController, 'up']).as('mimetype.up').where('id', router.matchers.number()).use(middleware.can(['settings'])); - router.get('/mimetype/:id/delete', [MimetypeController, 'delete']).as('mimetype.delete').use([middleware.auth(), middleware.can(['dataset-delete'])]); - router.delete('/mimetype/:id/deleteStore', [MimetypeController, 'deleteStore']).as('mimetype.deleteStore').use([middleware.auth(), middleware.can(['settings'])]); -}).prefix('admin').as('settings').use([middleware.auth(), middleware.is(['administrator', 'moderator'])]); + // Mimetype routes + router.get('/mimetype', [MimetypeController, 'index']).as('mimetype.index'); + router + .get('/mimetype/create', [MimetypeController, 'create']) + .as('mimetype.create') + .use(middleware.can(['settings'])); + router + .post('/mimetype/store', [MimetypeController, 'store']) + .as('mimetype.store') + .use(middleware.can(['settings'])); + router + .get('/mimetype/:id/down', [MimetypeController, 'down']) + .as('mimetype.down') + .where('id', router.matchers.number()) + .use(middleware.can(['settings'])); + router + .get('/mimetype/:id/up', [MimetypeController, 'up']) + .as('mimetype.up') + .where('id', router.matchers.number()) + .use(middleware.can(['settings'])); + router + .get('/mimetype/:id/delete', [MimetypeController, 'delete']) + .as('mimetype.delete') + .use([middleware.auth(), middleware.can(['dataset-delete'])]); + router + .delete('/mimetype/:id/deleteStore', [MimetypeController, 'deleteStore']) + .as('mimetype.deleteStore') + .use([middleware.auth(), middleware.can(['settings'])]); + }) + .prefix('admin') + .as('settings') + .use([middleware.auth(), middleware.is(['administrator', 'moderator'])]); router.get('/settings/user/security', [UserController, 'accountInfo']).as('settings.user').use(middleware.auth()); router.post('/settings/user/store', [UserController, 'accountInfoStore']).as('account.password.store').use(middleware.auth()); +router.get('/settings/profile/edit', [UserController, 'profile']).as('settings.profile.edit').use(middleware.auth()); +router + .put('/settings/profile/:id/update', [UserController, 'profileUpdate']) + .as('settings.profile.update') + .where('id', router.matchers.number()) + .use(middleware.auth()); +router.put('/settings/password/update', [UserController, 'passwordUpdate']).as('settings.password.update').use(middleware.auth()); // Submitter routes -router.group(() => { - router.get('/dataset', [DatasetController, 'index']).as('dataset.list').use([middleware.auth(), middleware.can(['dataset-list'])]); - router.get('/dataset/create', [DatasetController, 'create']).as('dataset.create').use([middleware.auth(), middleware.can(['dataset-submit'])]); - router.post('/dataset/first/first-step', [DatasetController, 'firstStep']).as('dataset.first.step').use([middleware.auth(), middleware.can(['dataset-submit'])]); - router.post('/dataset/second/second-step', [DatasetController, 'secondStep']).as('dataset.second.step').use([middleware.auth(), middleware.can(['dataset-submit'])]); - router.post('/dataset/second/third-step', [DatasetController, 'thirdStep']).as('dataset.third.step').use([middleware.auth(), middleware.can(['dataset-submit'])]); - router.post('/dataset/submit', [DatasetController, 'store']).as('dataset.submit').use([middleware.auth(), middleware.can(['dataset-submit'])]); - router.get('/dataset/:id/release', [DatasetController, 'release']).as('dataset.release').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-edit'])]); - router.put('/dataset/:id/releaseupdate', [DatasetController, 'releaseUpdate']).as('dataset.releaseUpdate').use([middleware.auth(), middleware.can(['dataset-edit'])]); - router.get('/dataset/:id/edit', [DatasetController, 'edit']).as('dataset.edit').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-edit'])]); - router.put('/dataset/:id/update', [DatasetController, 'update']).as('dataset.update').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-edit'])]); - router.get('/dataset/:id/delete', [DatasetController, 'delete']).as('dataset.delete').use([middleware.auth(), middleware.can(['dataset-delete'])]); - router.put('/dataset/:id/deleteupdate', [DatasetController, 'deleteUpdate']).as('dataset.deleteUpdate').use([middleware.auth(), middleware.can(['dataset-delete'])]); - router.get('/person', [PersonController, 'index']).as('person.index').use([middleware.auth()]); - router.get('/dataset/categorize', ({ inertia }: HttpContext) => { - return inertia.render('Submitter/Dataset/Category'); - }); -}).prefix('submitter'); +router + .group(() => { + router + .get('/dataset', [DatasetController, 'index']) + .as('dataset.list') + .use([middleware.auth(), middleware.can(['dataset-list'])]); + router + .get('/dataset/create', [DatasetController, 'create']) + .as('dataset.create') + .use([middleware.auth(), middleware.can(['dataset-submit'])]); + router + .post('/dataset/first/first-step', [DatasetController, 'firstStep']) + .as('dataset.first.step') + .use([middleware.auth(), middleware.can(['dataset-submit'])]); + router + .post('/dataset/second/second-step', [DatasetController, 'secondStep']) + .as('dataset.second.step') + .use([middleware.auth(), middleware.can(['dataset-submit'])]); + router + .post('/dataset/second/third-step', [DatasetController, 'thirdStep']) + .as('dataset.third.step') + .use([middleware.auth(), middleware.can(['dataset-submit'])]); + router + .post('/dataset/submit', [DatasetController, 'store']) + .as('dataset.submit') + .use([middleware.auth(), middleware.can(['dataset-submit'])]); + router + .get('/dataset/:id/release', [DatasetController, 'release']) + .as('dataset.release') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-edit'])]); + router + .put('/dataset/:id/releaseupdate', [DatasetController, 'releaseUpdate']) + .as('dataset.releaseUpdate') + .use([middleware.auth(), middleware.can(['dataset-edit'])]); + router + .get('/dataset/:id/edit', [DatasetController, 'edit']) + .as('dataset.edit') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-edit'])]); + router + .put('/dataset/:id/update', [DatasetController, 'update']) + .as('dataset.update') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-edit'])]); + router + .get('/dataset/:id/delete', [DatasetController, 'delete']) + .as('dataset.delete') + .use([middleware.auth(), middleware.can(['dataset-delete'])]); + router + .put('/dataset/:id/deleteupdate', [DatasetController, 'deleteUpdate']) + .as('dataset.deleteUpdate') + .use([middleware.auth(), middleware.can(['dataset-delete'])]); + router.get('/person', [PersonController, 'index']).as('person.index').use([middleware.auth()]); + router + .get('/dataset/:id/categorize', [DatasetController, 'categorize']) + .as('dataset.categorize') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-edit'])]); + router + .put('/dataset/:id/categorizeUpdate', [DatasetController, 'categorizeUpdate']) + .as('dataset.categorizeUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-edit'])]); + }) + .prefix('submitter'); // Editor routes -router.group(() => { - router.get('/dataset', [EditorDatasetController, 'index']).as('editor.dataset.list').use([middleware.auth(), middleware.can(['dataset-editor-list'])]); - router.get('dataset/:id/receive', [EditorDatasetController, 'receive']).as('editor.dataset.receive').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-receive'])]); - router.put('dataset/:id/receive', [EditorDatasetController, 'receiveUpdate']).as('editor.dataset.receiveUpdate').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-receive'])]); - router.get('dataset/:id/approve', [EditorDatasetController, 'approve']).as('editor.dataset.approve').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-approve'])]); - router.put('dataset/:id/approve', [EditorDatasetController, 'approveUpdate']).as('editor.dataset.approveUpdate').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-approve'])]); - router.get('dataset/:id/reject', [EditorDatasetController, 'reject']).as('editor.dataset.reject').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-editor-reject'])]); - router.put('dataset/:id/reject', [EditorDatasetController, 'rejectUpdate']).as('editor.dataset.rejectUpdate').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-editor-reject'])]); - router.get('dataset/:id/publish', [EditorDatasetController, 'publish']).as('editor.dataset.publish').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-publish'])]); - router.put('dataset/:id/publish', [EditorDatasetController, 'publishUpdate']).as('editor.dataset.publishUpdate').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-publish'])]); - router.get('dataset/:id/doi', [EditorDatasetController, 'doiCreate']).as('editor.dataset.doi').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-publish'])]); - router.put('dataset/:publish_id/doi', [EditorDatasetController, 'doiStore']).as('editor.dataset.doiStore').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-publish'])]); - router.put('/dataset/:id/update', [EditorDatasetController, 'update']).as('editor.dataset.update').use([middleware.auth(), middleware.can(['dataset-editor-edit'])]); -}).prefix('editor'); +router + .group(() => { + router + .get('/dataset', [EditorDatasetController, 'index']) + .as('editor.dataset.list') + .use([middleware.auth(), middleware.can(['dataset-editor-list'])]); + router + .get('dataset/:id/receive', [EditorDatasetController, 'receive']) + .as('editor.dataset.receive') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-receive'])]); + router + .put('dataset/:id/receive', [EditorDatasetController, 'receiveUpdate']) + .as('editor.dataset.receiveUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-receive'])]); + router + .get('dataset/:id/approve', [EditorDatasetController, 'approve']) + .as('editor.dataset.approve') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-approve'])]); + router + .put('dataset/:id/approve', [EditorDatasetController, 'approveUpdate']) + .as('editor.dataset.approveUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-approve'])]); + router + .get('dataset/:id/reject', [EditorDatasetController, 'reject']) + .as('editor.dataset.reject') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-reject'])]); + router + .put('dataset/:id/reject', [EditorDatasetController, 'rejectUpdate']) + .as('editor.dataset.rejectUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-reject'])]); + + router + .get('/dataset/:id/edit', [EditorDatasetController, 'edit']) + .as('editor.dataset.edit') + // .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-update'])]); + router + .put('/dataset/:id/update', [EditorDatasetController, 'update']) + .as('editor.dataset.update') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-update'])]); + + router + .get('/dataset/:id/categorize', [EditorDatasetController, 'categorize']) + .as('editor.dataset.categorize') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-update'])]); + router + .put('/dataset/:id/categorizeUpdate', [EditorDatasetController, 'categorizeUpdate']) + .as('editor.dataset.categorizeUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-update'])]); + + router + .get('/file/download/:id', [EditorDatasetController, 'download']) + .as('editor.file.download') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-update'])]); + router + .get('dataset/:id/rejectToReviewer', [EditorDatasetController, 'rejectToReviewer']) + .as('editor.dataset.rejectToReviewer') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-reject'])]); + router + .put('dataset/:id/rejectToReviewer', [EditorDatasetController, 'rejectToReviewerUpdate']) + .as('editor.dataset.rejectToReviewerUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-editor-reject'])]); + router + .get('dataset/:id/publish', [EditorDatasetController, 'publish']) + .as('editor.dataset.publish') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-publish'])]); + router + .put('dataset/:id/publish', [EditorDatasetController, 'publishUpdate']) + .as('editor.dataset.publishUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-publish'])]); + router + .get('dataset/:id/doi', [EditorDatasetController, 'doiCreate']) + .as('editor.dataset.doi') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-publish'])]); + router + .put('dataset/:publish_id/doi', [EditorDatasetController, 'doiStore']) + .as('editor.dataset.doiStore') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-publish'])]); + // router + // .put('/dataset/:id/update', [EditorDatasetController, 'update']) + // .as('editor.dataset.update') + // .use([middleware.auth(), middleware.can(['dataset-editor-edit'])]); + }) + .prefix('editor'); // Reviewer routes -router.group(() => { - router.get('/dataset', [ReviewerDatasetController, 'index']).as('reviewer.dataset.list').use([middleware.auth(), middleware.can(['dataset-review-list'])]); - router.get('dataset/:id/review', [ReviewerDatasetController, 'review']).as('reviewer.dataset.review').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-review'])]); - router.put('dataset/:id/review', [ReviewerDatasetController, 'reviewUpdate']).as('reviewer.dataset.reviewUpdate').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-review'])]); - router.get('dataset/:id/reject', [ReviewerDatasetController, 'reject']).as('reviewer.dataset.reject').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-review-reject'])]); - router.put('dataset/:id/reject', [ReviewerDatasetController, 'rejectUpdate']).as('reviewer.dataset.rejectUpdate').where('id', router.matchers.number()).use([middleware.auth(), middleware.can(['dataset-review-reject'])]); -}).prefix('reviewer'); +router + .group(() => { + router + .get('/dataset', [ReviewerDatasetController, 'index']) + .as('reviewer.dataset.list') + .use([middleware.auth(), middleware.can(['dataset-review-list'])]); + router + .get('dataset/:id/review', [ReviewerDatasetController, 'review']) + .as('reviewer.dataset.review') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-review'])]); + router + .put('dataset/:id/review', [ReviewerDatasetController, 'reviewUpdate']) + .as('reviewer.dataset.reviewUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-review'])]); + router + .get('/file/download/:id', [ReviewerDatasetController, 'download']) + .as('reviewer.file.download') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-review'])]); + router + .get('dataset/:id/reject', [ReviewerDatasetController, 'reject']) + .as('reviewer.dataset.reject') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-review-reject'])]); + router + .put('dataset/:id/reject', [ReviewerDatasetController, 'rejectUpdate']) + .as('reviewer.dataset.rejectUpdate') + .where('id', router.matchers.number()) + .use([middleware.auth(), middleware.can(['dataset-review-reject'])]); + }) + .prefix('reviewer'); diff --git a/start/routes/api.ts b/start/routes/api.ts index 477f69a..adc7f0e 100644 --- a/start/routes/api.ts +++ b/start/routes/api.ts @@ -6,30 +6,36 @@ import HomeController from '#controllers/Http/Api/HomeController'; import FileController from '#controllers/Http/Api/FileController'; import AvatarController from '#controllers/Http/Api/AvatarController'; import UserController from '#controllers/Http/Api/UserController'; -import { middleware } from '../kernel.js' +import CollectionsController from '#controllers/Http/Api/collections_controller'; +import { middleware } from '../kernel.js'; // API -router.group(() => { - +router + .group(() => { + router.get('clients', [UserController, 'getSubmitters']).as('client.index'); + router.get('authors', [AuthorsController, 'index']).as('author.index'); + router.get('datasets', [DatasetController, 'index']).as('dataset.index'); + router.get('persons', [AuthorsController, 'persons']).as('author.persons'); - - router.get('authors', [AuthorsController, "index"]).as('author.index'); - router.get('datasets', [DatasetController, "index"]).as('dataset.index'); - router.get('persons', [AuthorsController, "persons"]).as('author.persons'); - - router.get('/dataset', [DatasetController, "findAll"]).as('dataset.findAll'); - router.get('/dataset/:publish_id', [DatasetController, "findOne"]).as('dataset.findOne'); - router.get('/sitelinks/:year', [HomeController, "findDocumentsPerYear"]); - router.get('/years', [HomeController, "findYears"]); - router.get('/statistic/:year', [HomeController, "findPublicationsPerMonth"]); + router.get('/dataset', [DatasetController, 'findAll']).as('dataset.findAll'); + router.get('/dataset/:publish_id', [DatasetController, 'findOne']).as('dataset.findOne'); + router.get('/sitelinks/:year', [HomeController, 'findDocumentsPerYear']); + router.get('/years', [HomeController, 'findYears']); + router.get('/statistic', [HomeController, 'findPublicationsPerMonth']); - router.get('/download/:id', [FileController, "findOne"]).as('file.findOne'); + router.get('/download/:id', [FileController, 'findOne']).as('file.findOne'); router.get('/avatar/:name/:background?/:textColor?/:size?', [AvatarController, 'generateAvatar']); - - router.post('/twofactor_totp/settings/enable/:state/:code?', [UserController, 'enable']).as('apps.twofactor_totp.enable') .use(middleware.auth()); - router.post('/twofactor_backupcodes/settings/create', [UserController, 'createCodes']).as('apps.twofactor_backupcodes.create') .use(middleware.auth()); - -}) + router + .post('/twofactor_totp/settings/enable/:state/:code?', [UserController, 'enable']) + .as('apps.twofactor_totp.enable') + .use(middleware.auth()); + router + .post('/twofactor_backupcodes/settings/create', [UserController, 'createCodes']) + .as('apps.twofactor_backupcodes.create') + .use(middleware.auth()); + + router.get('collections/:id', [CollectionsController, 'show']).as('collection.show') + }) // .namespace('App/Controllers/Http/Api') .prefix('api'); diff --git a/start/rules/allowed_extensions_mimetypes.ts b/start/rules/allowed_extensions_mimetypes.ts index 9cd6dd6..88ff43e 100644 --- a/start/rules/allowed_extensions_mimetypes.ts +++ b/start/rules/allowed_extensions_mimetypes.ts @@ -9,27 +9,21 @@ import vine from '@vinejs/vine'; // import { VineString } from '@vinejs/vine'; import { VineMultipartFile, isBodyParserFile } from '#providers/vinejs_provider'; import type { MultipartFile } from '@adonisjs/core/bodyparser'; -// import db from '@adonisjs/lucid/services/db'; import MimeType from '#models/mime_type'; /** * Options accepted by the unique rule */ -// type Options = { -// mainLanguageField: string; -// typeField: string; -// }; type Options = { // size: string | number; // extnames: string[]; - clientNameSizeLimit: number; + // clientNameSizeLimit?: number; + allowedExtensions: string[]; + allowedMimeTypes?: string[]; }; -async function allowedMimetypeExtensions(file: VineMultipartFile | unknown, options: Options | unknown, field: FieldContext) { - // if (typeof value !== 'string' && typeof value != 'number') { - // return; - // } - +// async function allowedMimetypeExtensions(file: VineMultipartFile | unknown, options: Options | unknown, field: FieldContext) { +async function allowedMimetypeExtensions(file: VineMultipartFile | unknown, options: Options, field: FieldContext) { if (!isBodyParserFile(file)) { return; } @@ -38,7 +32,15 @@ async function allowedMimetypeExtensions(file: VineMultipartFile | unknown, opti const fileExtension = validatedFile?.extname?.toLocaleLowerCase() as string; // Get file extension from the file // validate if file extension is allowed in combination with mimetype - const mimeRecord = await MimeType.query().select('file_extension').where('name', mimeType).andWhere('enabled', true).first(); + let mimeRecord = await MimeType.query().select('file_extension').where('name', mimeType).andWhere('enabled', true).first(); + + if (!mimeRecord) { + mimeRecord = await MimeType.query() + .select('file_extension') + .whereRaw("? = ANY (string_to_array(alternate_mimetype, '|'))", [mimeType]) + .andWhere('enabled', true) + .first(); + } if (!mimeRecord) { const allowedMimetypes = await MimeType.query().select('name').where('enabled', true); @@ -52,14 +54,19 @@ async function allowedMimetypeExtensions(file: VineMultipartFile | unknown, opti field, ); } else { - const allowedExtensions = mimeRecord.file_extension.split('|'); + let allowedExtensions: string[] = []; + if (options && options.allowedExtensions) { + allowedExtensions = options.allowedExtensions; + } else { + allowedExtensions = mimeRecord.file_extension.split('|'); + } // Validate if the file's extension is in the allowed extensions if (!allowedExtensions.includes(fileExtension)) { //throw new Error(`File extension ${fileExtension} is not allowed for MIME type ${mimeType}`); field.report( `File extension ${fileExtension} is not allowed for MIME type ${mimeType}. Allowed extensions are: ${mimeRecord.file_extension}`, - 'allowedMimetypeExtensions', - field + 'allowedMimetypeExtensions', + field, ); } // if (validatedFile.clientName.length > options.clientNameSizeLimit) { diff --git a/start/rules/array_contains_types.ts b/start/rules/array_contains_types.ts new file mode 100644 index 0000000..1fe5994 --- /dev/null +++ b/start/rules/array_contains_types.ts @@ -0,0 +1,80 @@ +import { FieldContext } from '@vinejs/vine/types'; +import vine, { VineArray } from '@vinejs/vine'; +import { SchemaTypes } from '@vinejs/vine/types'; + +type Options = { + typeA: string; + typeB: string; +}; + +/** + * Custom rule to validate an array of titles contains at least one title + * with type 'main' and one with type 'translated'. + * + * This rule expects the validated value to be an array of objects, + * where each object has a "type" property. + */ +async function arrayContainsTypes(value: unknown, options: Options, field: FieldContext) { + if (!Array.isArray(value)) { + field.report(`The {{field}} must be an array of titles.`, 'array.titlesContainsMainAndTranslated', field); + return false; + } + + const typeAExpected = options.typeA.toLowerCase(); + const typeBExpected = options.typeB.toLowerCase(); + + // const hasMain = value.some((title: any) => { + // return typeof title === 'object' && title !== null && String(title.type).toLowerCase() === 'main'; + // }); + + // const hasTranslated = value.some((title: any) => { + // return typeof title === 'object' && title !== null && String(title.type).toLowerCase() === 'translated'; + // }); + const hasTypeA = value.some((item: any) => { + return typeof item === 'object' && item !== null && String(item.type).toLowerCase() === typeAExpected; + }); + + const hasTypeB = value.some((item: any) => { + return typeof item === 'object' && item !== null && String(item.type).toLowerCase() === typeBExpected; + }); + if (!hasTypeA || !hasTypeB) { + let errorMessage = `The ${field.getFieldPath()} array must have at least one '${options.typeA}' item and one '${options.typeB}' item.`; + + // Check for specific field names to produce a more readable message. + if (field.getFieldPath() === 'titles') { + // For titles we expect one main and minimum one translated title. + if (!hasTypeA && !hasTypeB) { + errorMessage = 'For titles, define at least one main title and at least one Translated title as MAIN TITLE.'; + } else if (!hasTypeA) { + errorMessage = 'For titles, define at least one main title.'; + } else if (!hasTypeB) { + errorMessage = 'For Titles, define at least one Translated title as MAIN TITLE.'; + } + } else if (field.getFieldPath() === 'descriptions') { + // For descriptions we expect one abstracts description and minimum one translated description. + if (!hasTypeA && !hasTypeB) { + errorMessage = 'For descriptions, define at least one abstract and at least one Translated description as MAIN ABSTRACT.'; + } else if (!hasTypeA) { + errorMessage = 'For descriptions, define at least one abstract.'; + } else if (!hasTypeB) { + errorMessage = 'For Descriptions, define at least one Translated description as MAIN ABSTRACT.'; + } + } + + field.report(errorMessage, 'array.containsTypes', field, options); + return false; + } + return true; +} + +export const arrayContainsMainAndTranslatedRule = vine.createRule(arrayContainsTypes); + +declare module '@vinejs/vine' { + interface VineArray<Schema extends SchemaTypes> { + arrayContainsTypes(options: Options): this; + } +} + +VineArray.macro('arrayContainsTypes', function <Schema extends SchemaTypes>(this: VineArray<Schema>, options: Options) { + return this.use(arrayContainsMainAndTranslatedRule(options)); +}); diff --git a/start/rules/file_scan.ts b/start/rules/file_scan.ts index b8a3b30..53383ea 100644 --- a/start/rules/file_scan.ts +++ b/start/rules/file_scan.ts @@ -3,37 +3,16 @@ | Preloaded File - node ace make:preload rules/fileScan |-------------------------------------------------------------------------- |*/ - import { FieldContext } from '@vinejs/vine/types'; import vine, { errors } from '@vinejs/vine'; - -// import { VineString } from '@vinejs/vine'; import { VineMultipartFile, isBodyParserFile } from '#providers/vinejs_provider'; import type { MultipartFile } from '@adonisjs/core/bodyparser'; import ClamScan from 'clamscan'; -/** - * Options accepted by the unique rule - */ -// type Options = { -// mainLanguageField: string; -// typeField: string; -// }; type Options = { - // size: string | number; - // extnames: string[]; removeInfected: boolean; - // debugMode?: boolean; - // scanRecursively?: boolean; host?: string; port?: number; - // clamdscan: { - // active: boolean; - // host: string; - // port: number; - // multiscan: boolean; - // }; - // preference: string; }; async function fileScan(file: VineMultipartFile | unknown, options: Options, field: FieldContext) { @@ -44,42 +23,47 @@ async function fileScan(file: VineMultipartFile | unknown, options: Options, fie return; } const validatedFile = file as MultipartFile; - + try { - await scanFileForViruses(validatedFile.tmpPath, options.host, options.port); //, 'gitea.lan', 3310); - // await this.scanFileForViruses("/tmp/testfile.txt"); + await scanFileForViruses(validatedFile.tmpPath, options); } catch (error) { // If the file is infected or there's an error scanning the file, throw a validation exception // throw error; field.report(`Upload error. Code: ${error.code} message: ${error.messages.uploadError}`, 'fileScan', field); } } -async function scanFileForViruses(filePath: string | undefined, host?: string, port?: number): Promise<void> { - // const clamscan = await (new ClamScan().init()); + +async function scanFileForViruses(filePath: string | undefined, options: Options): Promise<void> { + if (!filePath) { + throw new errors.E_VALIDATION_ERROR({ uploadError: 'File path is undefined!' }); + } const opts: ClamScan.Options = { - removeInfected: true, // If true, removes infected files - debugMode: false, // Whether or not to log info/debug/error msgs to the console + removeInfected: options.removeInfected, // If true, removes infected files + debugMode: false, // If true, deep scan folders recursively scanRecursively: true, // If true, deep scan folders recursively clamdscan: { active: true, // If true, this module will consider using the clamdscan binary - host, - port, + host: options.host, + port: options.port, multiscan: true, // Scan using all available cores! Yay! }, - preference: 'clamdscan', // If clamdscan is found and active, it will be used by default }; + const clamscan = await new ClamScan().init(opts); return new Promise(async (resolve, reject) => { try { - const clamscan = await new ClamScan().init(opts); // You can re-use the `clamscan` object as many times as you want // const version = await clamscan.getVersion(); // console.log(`ClamAV Version: ${version}`); - const { file, isInfected, viruses } = await clamscan.isInfected(filePath); + const result = await clamscan.isInfected(filePath); + if (!result || typeof result.isInfected === 'undefined') { + reject(new errors.E_VALIDATION_ERROR({ uploadError: 'Unexpected response from virus scan!' })); + return; + } + const { file, isInfected, viruses } = result; if (isInfected) { - console.log(`${file} is infected with ${viruses}!`); - // reject(new ValidationException(true, { 'upload error': `File ${file} is infected!` })); - reject(new errors.E_VALIDATION_ERROR({ uploadError: `File ${file} is infected!` })); + console.log(`${file} is infected with ${viruses}!`); // reject(new ValidationException(true, { 'upload error': `File ${file} is infected!` })); + reject(new errors.E_VALIDATION_ERROR({ uploadError: `File ${file} is infected with ${viruses}!` })); } else { resolve(); } diff --git a/start/rules/referenceValidation.ts b/start/rules/referenceValidation.ts new file mode 100644 index 0000000..dd4030a --- /dev/null +++ b/start/rules/referenceValidation.ts @@ -0,0 +1,166 @@ +import { FieldContext } from '@vinejs/vine/types'; +import vine from '@vinejs/vine'; +import { VineString } from '@vinejs/vine'; +import { default as axios } from 'axios'; +import { ReferenceIdentifierTypes } from '#contracts/enums'; + +type Options = { + typeField: string; +}; + +// Function to check if DOI exists using the DOI API +async function checkDoiExists(doi: string): Promise<boolean> { + try { + const response = await axios.get(`${doi}`); + return response.status === 200; // If status is 200, DOI is valid + } catch (error) { + return false; // If request fails, DOI does not exist + } +} + +// Function to check if ISBN exists using the Open Library API +// async function checkIsbnExists(isbn: string): Promise<boolean> { +// try { +// const response = await axios.get(`https://isbnsearch.org/isbn/${isbn}`); +// return response.status === 200 && response.data.includes('ISBN'); // Check if response contains ISBN information +// } catch (error) { +// return false; // If request fails, ISBN does not exist +// } +// } + +async function checkIsbnExists(isbn: string): Promise<boolean> { + // Try Open Library first + try { + const response = await axios.get(`https://openlibrary.org/api/books?bibkeys=ISBN:${isbn}&format=json&jscmd=data`); + const data = response.data; + if (Object.keys(data).length > 0) { + return true; + } + } catch (error) { + // If an error occurs, continue to the next API + } + + // Fallback to Google Books API + try { + const response = await axios.get(`https://www.googleapis.com/books/v1/volumes?q=isbn:${isbn}`); + const data = response.data; + if (data.totalItems > 0) { + return true; + } + } catch (error) { + // If an error occurs, continue to the next API + } + + // Lastly use the Koha library by scraping HTML + try { + const response = await axios.get(`https://bibliothek.geosphere.at/cgi-bin/koha/opac-search.pl?idx=nb&q=${isbn}`); + const html = response.data; + // Check if zero results are explicitly indicated (German or English) + if (html.includes('Keine Treffer gefunden!') || html.includes('Your search returned 0 results')) { + return false; + } + // Try to extract the count from German message + let match = html.match(/Ihre Suche erzielte\s*(\d+)\s*Treffer/); + + // If not found, try the English equivalent + if (!match) { + match = html.match(/Your search returned\s*(\d+)\s*results/); + } + + if (match && match[1]) { + const count = parseInt(match[1], 10); + return count > 0; + } + + // Fallback: if no match is found, return false + return false; + } catch (error) { + return false; + } +} + +async function validateReference(value: unknown, options: Options, field: FieldContext) { + if (typeof value !== 'string') { + return; + } + + const type = field.parent[options.typeField]; + + if (type === ReferenceIdentifierTypes.URL) { + if (!/^https?:\/\/[^\s$.?#].[^\s]*$/.test(value)) { + field.report('The {{ field }} must be a valid URL', 'validateReference', field); + } else { + try { + const exists = await checkDoiExists(value); + if (!exists) { + field.report('The {{ field }} must be an existing URL', 'validateReference', field); + } + } catch (error) { + field.report('Error checking URL existence: ' + error.message, 'validateReference', field); + } + } + } + // Check if the value does not match the DOI pattern + // The regex pattern ^10.\d{4,9}\/[-._;()/:a-zA-Z0-9]+$ is designed to match valid DOI formats. + // - ^10. ensures that the string starts with "10." + // - \d{4,9} matches a sequence of 4 to 9 digits. + // - \/ matches the literal forward slash character. + // - [-._;()/:a-zA-Z0-9]+ matches one or more characters that can include uppercase and lowercase letters, digits, and a set of special characters. + // The i flag at the end of the regex makes the matching case-insensitive, meaning it will match both uppercase and lowercase letters. + // If the value does not match this pattern, the code inside the if block will execute. + else if (type === ReferenceIdentifierTypes.DOI) { + // Extract the DOI from the URL if it starts with 'https://doi.org/' + const doiPattern = /^https:\/\/doi\.org\/(10.\d{4,9}\/[-._;()/:a-zA-Z0-9]+)$/i; + const match = value.match(doiPattern); + const doi = match ? match[1] : value; + // Check if the extracted DOI or the value itself matches the DOI patter + if (!/^10.\d{4,9}\/[-._;()/:a-zA-Z0-9]+$/i.test(doi)) { + field.report('The {{ field }} must be a valid DOI', 'validateReference', field); + } else { + try { + const exists = await checkDoiExists(value); + if (!exists) { + field.report('The {{ field }} must be an existing DOI', 'validateReference', field); + } + } catch (error) { + field.report('Error checking DOI existence: ' + error.message, 'validateReference', field); + } + } + } else if (type === ReferenceIdentifierTypes.ISBN) { + const isbnRegex = /^(?:\d{1,5}-\d{1,7}-\d{1,7}-[\dX]$|97[89]-\d{1,5}-\d{1,7}-\d{1,7}-\d)$/; + if (!isbnRegex.test(value)) { + field.report('Invalid {{ field }}. Expected format: 978-3-16-148410-0 or similar.', 'validateReference', field); + } else { + try { + const exists = await checkIsbnExists(value); + if (!exists) { + field.report('The {{ field }} must be an existing ISBN', 'validateReference', field); + } + } catch (error) { + field.report('Error checking ISBN existence: ' + error.message, 'validateReference', field); + } + } + } else if (type === ReferenceIdentifierTypes.Handle && !/^\d{2,}.\d{4,9}\/[-._;()/:a-zA-Z0-9]+$/.test(value)) { + /// Extract the Handle from the URL if it contains '/handle/' + field.report('The {{ field }} must be a valid Handle', 'validateReference', field); + } else if ( + type === ReferenceIdentifierTypes.URN && + !/^urn:[a-zA-Z0-9][a-zA-Z0-9-]{0,31}:[a-zA-Z0-9()+,\-.:=@;$_!*'%/?#]+$/.test(value) + ) { + field.report('The {{ field }} must be a valid URN', 'validateReference', field); + } else if (type === ReferenceIdentifierTypes.ISSN && !/^\d{4}-\d{3}[\dxX]$/.test(value)) { + field.report('The {{ field }} must be a valid ISSN', 'validateReference', field); + } +} + +export const validateReferenceRule = vine.createRule(validateReference); + +declare module '@vinejs/vine' { + interface VineString { + validateReference(options: Options): this; + } +} + +VineString.macro('validateReference', function (this: VineString, options: Options) { + return this.use(validateReferenceRule(options)); +}); diff --git a/start/rules/translated_language.ts b/start/rules/translated_language.ts index 99312f3..08b6b23 100644 --- a/start/rules/translated_language.ts +++ b/start/rules/translated_language.ts @@ -17,7 +17,7 @@ type Options = { }; async function translatedLanguage(value: unknown, options: Options, field: FieldContext) { - if (typeof value !== 'string' && typeof value != 'number') { + if (typeof value !== 'string' && typeof value !== 'number') { return; } diff --git a/start/rules/unique.ts b/start/rules/unique.ts index 4661c19..f9a2b25 100644 --- a/start/rules/unique.ts +++ b/start/rules/unique.ts @@ -15,11 +15,11 @@ import { VineString, VineNumber } from '@vinejs/vine'; type Options = { table: string; column: string; - whereNot?: ((field: FieldContext) => string); + whereNot?: (field: FieldContext) => string; }; async function isUnique(value: unknown, options: Options, field: FieldContext) { - if (typeof value !== 'string' && typeof value != 'number') { + if (typeof value !== 'string' && typeof value !== 'number') { return; } @@ -37,13 +37,11 @@ async function isUnique(value: unknown, options: Options, field: FieldContext) { // report that value is NOT unique field.report('The {{ field }} field is not unique', 'isUnique', field); // field.report(messages.unique, "isUnique", field); - - } + } } export const isUniqueRule = vine.createRule(isUnique); - declare module '@vinejs/vine' { interface VineString { isUnique(options: Options): this; @@ -58,4 +56,4 @@ VineString.macro('isUnique', function (this: VineString, options: Options) { }); VineNumber.macro('isUnique', function (this: VineNumber, options: Options) { return this.use(isUniqueRule(options)); -}); \ No newline at end of file +}); diff --git a/start/rules/unique_person.ts b/start/rules/unique_person.ts index a939336..747bdbc 100644 --- a/start/rules/unique_person.ts +++ b/start/rules/unique_person.ts @@ -19,9 +19,8 @@ type Options = { idField: string; }; - async function isUniquePerson(value: unknown, options: Options, field: FieldContext) { - if (typeof value !== 'string' && typeof value != 'number') { + if (typeof value !== 'string' && typeof value !== 'number') { return; } @@ -40,13 +39,11 @@ async function isUniquePerson(value: unknown, options: Options, field: FieldCont if (result) { // report that value is NOT unique field.report('The {{ field }} field is not unique', 'isUnique', field); - - } + } } export const isUniquePersonRule = vine.createRule(isUniquePerson); - declare module '@vinejs/vine' { interface VineString { isUniquePerson(options: Options): this; @@ -61,4 +58,4 @@ VineString.macro('isUniquePerson', function (this: VineString, options: Options) }); VineNumber.macro('isUniquePerson', function (this: VineNumber, options: Options) { return this.use(isUniquePersonRule(options)); -}); \ No newline at end of file +}); diff --git a/start/rules/valid_mimetype.ts b/start/rules/valid_mimetype.ts new file mode 100644 index 0000000..4a9eeea --- /dev/null +++ b/start/rules/valid_mimetype.ts @@ -0,0 +1,29 @@ +import { FieldContext } from '@vinejs/vine/types'; +import vine from '@vinejs/vine'; +import { VineString } from '@vinejs/vine'; + +async function isValidMimetype(value: unknown, options: unknown, field: FieldContext) { + if (typeof value !== 'string') { + return; + } + + // Regex pattern to match valid mimetypes (e.g., "application/json", "text/html") + const mimetypePattern = /^[a-zA-Z0-9!#$&^_.+-]+\/[a-zA-Z0-9!#$&^_.+-]+$/; + if (!mimetypePattern.test(value)) { + field.report('The given value is not a valid mimetype', 'isValidMimetype', field); + } +} + +export const isValidMimetypeRule = vine.createRule(isValidMimetype); + +declare module '@vinejs/vine' { + interface VineString { + isValidMimetype(): this; + } +} + +VineString.macro('isValidMimetype', function (this: VineString) { + return this.use(isValidMimetypeRule()); +}); + + diff --git a/start/validator.ts b/start/validator.ts index 05fea83..6ac9133 100644 --- a/start/validator.ts +++ b/start/validator.ts @@ -13,20 +13,18 @@ import vine from '@vinejs/vine'; // import db from '@adonisjs/lucid/services/db'; import { VanillaErrorReporter } from '#validators/vanilla_error_reporter'; - // vine.messagesProvider = new SimpleMessagesProvider({ // // Applicable for all fields // 'required': 'The {{ field }} field is required', // 'string': 'The value of {{ field }} field must be a string', // 'email': 'The value is not a valid email address', - + // // 'contacts.0.email.required': 'The primary email of the contact is required', // // 'contacts.*.email.required': 'Contact email is required', // 'permissions.minLength': 'at least {{ options.minLength }} permission must be defined', // 'permissions.*.number': 'Define permissions as valid numbers', // }) vine.errorReporter = () => new VanillaErrorReporter(); - // /** // * Options accepted by the unique rule diff --git a/tailwind.config.js b/tailwind.config.js index 5a6d183..e41b62d 100644 --- a/tailwind.config.js +++ b/tailwind.config.js @@ -12,6 +12,10 @@ module.exports = { gray: 'gray', }, extend: { + backgroundImage: { + 'radio-checked': "url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath fill='%23fff' d='M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10,10 0 0,0 12,2Z' /%3E%3C/svg%3E\")", + 'checkbox-checked': "url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 1 1'%3E%3Cpath style='fill:%23fff' d='M 0.04038059,0.6267767 0.14644661,0.52071068 0.42928932,0.80355339 0.3232233,0.90961941 z M 0.21715729,0.80355339 0.85355339,0.16715729 0.95961941,0.2732233 0.3232233,0.90961941 z'%3E%3C/path%3E%3C/svg%3E\")", + }, colors: { 'primary': '#22C55E', 'inprogress': 'rgb(94 234 212)', @@ -102,7 +106,19 @@ module.exports = { { values: theme('asideScrollbars') }, ); }), + plugin(function({ addUtilities }) { + const newUtilities = { + '.drag-none': { + '-webkit-user-drag': 'none', + '-khtml-user-drag': 'none', + '-moz-user-drag': 'none', + '-o-user-drag': 'none', + 'user-drag': 'none', + }, + } + addUtilities(newUtilities) + }), // As of Tailwind CSS v3.3, the `@tailwindcss/line-clamp` plugin is now included by default // require('@tailwindcss/line-clamp'), ], -}; +}; \ No newline at end of file diff --git a/tests/bootstrap.ts b/tests/bootstrap.ts index eb6901a..0cd357c 100644 --- a/tests/bootstrap.ts +++ b/tests/bootstrap.ts @@ -12,10 +12,10 @@ import testUtils from '@adonisjs/core/services/test_utils'; // import { assert, runFailedTests, specReporter, apiClient } from '@japa/preset-adonis'; import { assert } from '@japa/assert'; -import { apiClient } from '@japa/api-client'; +// import { apiClient } from '@japa/api-client'; import { pluginAdonisJS } from '@japa/plugin-adonisjs'; import app from '@adonisjs/core/services/app'; - +// import env from '#start/env' /* |-------------------------------------------------------------------------- | Japa Plugins @@ -28,7 +28,13 @@ import app from '@adonisjs/core/services/app'; | */ // export const plugins: Required<Config>['plugins'] = [assert(), runFailedTests(), apiClient()]; -export const plugins: Config['plugins'] = [assert(), apiClient(), pluginAdonisJS(app)]; +export const plugins: Config['plugins'] = [ + assert(), + // apiClient({ + // baseURL: `http://${env.get('HOST')}:${env.get('PORT')}`, + // }), + pluginAdonisJS(app), +]; /* |-------------------------------------------------------------------------- @@ -59,11 +65,18 @@ export const reporters: Required<Config>['reporters'] = { */ export const runnerHooks: Pick<Required<Config>, 'setup' | 'teardown'> = { setup: [ + () => { + console.log('running before all the tests'); + }, // () => testUtils.ace().loadCommands(), () => testUtils.db().migrate(), // () => testUtils.httpServer().start(), ], - teardown: [], + teardown: [ + () => { + console.log('running after all the tests'); + }, + ], }; /* diff --git a/tests/functional/dataset_controller.spec.ts b/tests/functional/dataset_controller.spec.ts index 9363fec..68ea9aa 100644 --- a/tests/functional/dataset_controller.spec.ts +++ b/tests/functional/dataset_controller.spec.ts @@ -8,7 +8,7 @@ import User from '#models/user'; import Role from '#models/role'; import Permission from '#models/permission'; import { TestContext } from '@japa/runner/core'; -const BASE_URL = `http://${process.env.HOST}:${process.env.PORT}` +const BASE_URL = `http://${process.env.HOST}:${process.env.PORT}`; test.group('DatasetController', (group) => { // Write your test here @@ -24,8 +24,6 @@ test.group('DatasetController', (group) => { // server = await supertest(BASE_URL); // }); - - test('should render dataset release page', async ({ assert }: TestContext) => { var testAgent = supertest(BASE_URL); @@ -35,19 +33,18 @@ test.group('DatasetController', (group) => { email: 'alice@email.com', password: 'password', }); - + const role = await Role.create({ name: 'administrator', display_name: 'admin', - description: 'User has access to all system functionality' + description: 'User has access to all system functionality', }); await user.related('roles').attach([role.id]); - const permission = await Permission.create({ name: 'dataset-edit', display_name: 'edit dataset', - description: 'allow role to edit datasets' + description: 'allow role to edit datasets', }); await role.related('permissions').attach([permission.id]); diff --git a/tests/functional/hello_world.spec.ts b/tests/functional/hello_world.spec.ts index 2978df8..2f3721e 100644 --- a/tests/functional/hello_world.spec.ts +++ b/tests/functional/hello_world.spec.ts @@ -1,8 +1,8 @@ -import { test } from '@japa/runner'; +// import { test } from '@japa/runner'; -test('display welcome page', async ({ client }) => { - const response = await client.get('/welcome'); +// test('display welcome page', async ({ client }) => { +// const response = await client.get('/welcome'); - response.assertStatus(200); - response.assertTextIncludes('<h1 class="title"> It Works! </h1>'); -}); +// response.assertStatus(200); +// response.assertTextIncludes('<h1 class="title"> It Works! </h1>'); +// }); diff --git a/tests/functional/referenceValidation.spec.ts b/tests/functional/referenceValidation.spec.ts new file mode 100644 index 0000000..17648ab --- /dev/null +++ b/tests/functional/referenceValidation.spec.ts @@ -0,0 +1,215 @@ +import { test } from '@japa/runner'; +import { ReferenceIdentifierTypes } from '#contracts/enums'; +import vine from '@vinejs/vine'; + +// node ace test functional --groups "ReferenceValidation" +test.group('ReferenceValidation', () => { + test('validate valid DOI', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + reference: 'https://doi.org/10.24341/tethys.236', + type: ReferenceIdentifierTypes.DOI, + }; + + try { + const payload = await validator.validate(data); + assert.deepEqual(payload, data); + } catch { + assert.isTrue(false); + } + }); + + test('validate invalid DOI', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + reference: 'https://doi.org/invalid-doi', + type: ReferenceIdentifierTypes.DOI, + }; + + let payload = {}; + try { + payload = await validator.validate(data); + } catch { + assert.notDeepEqual(payload, data); + } + }); + + test('validate valid Handle', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + reference: '20.5000/abc123', + type: ReferenceIdentifierTypes.Handle, + }; + + try { + const payload = await validator.validate(data); + assert.deepEqual(payload, data); + } catch { + assert.isTrue(false); + } + }); + test('validate valid ISBN', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + // reference: '978-3-85316-090-9', + // reference: '9783853160909', + // reference: '978-3-900312-64-0', // Geologische Karte der Republik Österreich 1 : 50.000 + reference: '3-90031-264-8', // Geologische Karte der Republik Österreich 1 : 50.000 + type: ReferenceIdentifierTypes.ISBN, + }; + + try { + const payload = await validator.validate(data); + assert.deepEqual(payload, data); + } catch { + assert.isTrue(false); + } + }); + + test('validate invalid ISBN', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + reference: 'invalid-isbn', + type: ReferenceIdentifierTypes.ISBN, + }; + + let payload = {}; + try { + payload = await validator.validate(data); + } catch { + assert.notDeepEqual(payload, data); + } + }); + + test('validate valid URN', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + reference: 'urn:isbn:0451450523', + type: ReferenceIdentifierTypes.URN, + }; + + try { + const payload = await validator.validate(data); + assert.deepEqual(payload, data); + } catch { + assert.isTrue(false); + } + }); + + test('validate invalid URN', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + reference: 'invalid-urn', + type: ReferenceIdentifierTypes.URN, + }; + + let payload = {}; + try { + payload = await validator.validate(data); + } catch { + assert.notDeepEqual(payload, data); + } + }); + + test('validate valid ISSN', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + reference: '1234-567X', + type: ReferenceIdentifierTypes.ISSN, + }; + + try { + const payload = await validator.validate(data); + assert.deepEqual(payload, data); + } catch { + assert.isTrue(false); + } + }); + + test('validate invalid ISSN', async ({ assert }) => { + const validator = vine.compile( + vine.object({ + reference: vine.string().validateReference({ typeField: 'type' }), + type: vine.enum(Object.values(ReferenceIdentifierTypes)), + }), + ); + + const data = { + reference: 'invalid-issn', + type: ReferenceIdentifierTypes.ISSN, + }; + + let payload = {}; + try { + payload = await validator.validate(data); + } catch { + assert.notDeepEqual(payload, data); + } + }); + // test('validate invalid Handle', async ({ assert }) => { + // const validator = vine.compile( + // vine.object({ + // reference: vine.string().validateReference({ typeField: 'type' }), + // type: vine.enum(Object.values(ReferenceIdentifierTypes)), + // }) + // ); + + // const data = { + // reference: 'invalid-handle', + // type: ReferenceIdentifierTypes.Handle, + // }; + + // const result = await validator.validate(data); + // assert.isFalse(result.valid); + // }); + + // Add more tests for other reference types as needed +}); diff --git a/tsconfig.json b/tsconfig.json index 3a18d09..7def8a0 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -10,8 +10,8 @@ "node_modules", "build", "public", - "./resources/js/**/*.vue", - "./resources/js/**/*.ts" + "./resources/js/**/*", + // "./resources/js/**/*.ts" ], // "exclude": ["./inertia/**/*"] "compilerOptions": { @@ -19,9 +19,10 @@ "outDir": "build", "rootDir": "./", "sourceMap": true, + "sourceRoot": "./build", "experimentalDecorators": true, "strictPropertyInitialization": false, - "target": "esnext", + "target": "esnext", // Update to a version that supports top-level await "skipLibCheck": true, "esModuleInterop": true, "allowSyntheticDefaultImports": true, diff --git a/vite.config.ts b/vite.config.ts new file mode 100644 index 0000000..24889bb --- /dev/null +++ b/vite.config.ts @@ -0,0 +1,78 @@ +import { defineConfig } from 'vite'; +import adonisjs from '@adonisjs/vite/client'; +// import { getDirname } from '@adonisjs/core/helpers'; +import vue from '@vitejs/plugin-vue'; +import path from 'path'; +// import tailwind from '@tailwindcss/postcss'; +// import autoprefixer from 'autoprefixer'; +// import postcssNesting from 'postcss-nesting'; +import inertia from '@adonisjs/inertia/client'; + +export default defineConfig({ + plugins: [ + inertia(), + vue(), + adonisjs({ + /** + * Entrypoints of your application. Each entrypoint will + * result in a separate bundle. + */ + entrypoints: ['resources/js/app.ts', 'resources/css/app.css'], + + /** + * Paths to watch and reload the browser on file change + */ + reload: ['resources/views/**/*.edge'], + }), + ], + server: { + port: 5173, + // host: '127.0.0.1' + }, + + // css: { + // postcss: { + // plugins: [ + // postcssNesting(), + // tailwind(), + // autoprefixer(), + // ], + // }, + // }, + /** + * Define aliases for importing modules from + * your frontend code + */ + resolve: { + alias: { + '@': path.resolve('./resources/js/'), + '~': path.resolve(__dirname, 'node_modules/'), + }, + }, + + // optimizeDeps: { + // esbuildOptions: { + // target: 'esnext' + // }, + // include: ['resources/js/**/*.{vue,js,jsx,ts,tsx}'], + // exclude: ['node_modules', 'app'], + // }, + + build: { + sourcemap: true, + outDir: 'public/assets', + emptyOutDir: true, + manifest: true, + rollupOptions: { + input: 'resources/js/app.ts', + }, + }, + // build: { + // outDir: 'public/assets', + // emptyOutDir: true, + // manifest: true, + // rollupOptions: { + // input: path.resolve(__dirname, 'resources/js/app.ts'), + // }, + // }, +}); diff --git a/webpack.config.cjs b/webpack.config.cjs index b85fac3..83ed320 100644 --- a/webpack.config.cjs +++ b/webpack.config.cjs @@ -302,18 +302,18 @@ Encore.addLoader({ // vue$: 'vue/dist/vue.runtime.esm-bundler.js', // }); -Encore.addLoader(babelLoader) - // Encore.enableTypeScriptLoader(config => { - // // Loader-specific options - // config.configFile = 'resources/js/tsconfig.json'; - // config.appendTsSuffixTo = [/\.vue$/]; - // config.transpileOnly = true; - // config.happyPackMode = false; - // }, { - // // Directly change the exclude rule - // exclude: /node_modules/, +// Encore.addLoader(babelLoader) + Encore.enableTypeScriptLoader(config => { + // Loader-specific options + config.configFile = 'resources/js/tsconfig.json'; + config.appendTsSuffixTo = [/\.vue$/]; + config.transpileOnly = true; + config.happyPackMode = false; + }, { + // Directly change the exclude rule + exclude: /node_modules/, - // }) + }) .addAliases({ '@': join(__dirname, 'resources/js'), 'vue$': 'vue/dist/vue.runtime.esm-bundler.js',