From a966d71fa0973a0ecb8d37eed1b124e534c8a5a8 Mon Sep 17 00:00:00 2001 From: Developer Date: Sun, 30 Nov 2025 06:44:57 -0800 Subject: [PATCH] . --- .claude/settings.local.json | 12 +- .../identity-service/.env.development | 6 + .../services/identity-service/.env.example | 6 + .../identity-service/package-lock.json | 10493 ++++++++++++++++ .../services/identity-service/package.json | 2 + .../identity-service/prisma/schema.prisma | 23 + .../identity-service/src/api/api.module.ts | 4 +- .../src/api/controllers/deposit.controller.ts | 90 + .../api/controllers/referrals.controller.ts | 68 + .../identity-service/src/api/dto/index.ts | 133 + .../identity-service/src/app.module.ts | 47 +- .../src/application/application.module.ts | 3 + .../auto-create-account.handler.ts | 32 +- .../src/application/commands/index.ts | 68 + .../application/services/deposit.service.ts | 194 + .../user-application.service.referral.spec.ts | 634 + .../services/user-application.service.ts | 175 + .../src/domain/domain.module.ts | 6 +- .../user-account.repository.interface.ts | 23 + .../src/domain/services/index.ts | 65 +- .../services/wallet-generator.service.ts | 112 +- .../external/backup/backup-client.service.ts | 192 + .../infrastructure/external/backup/index.ts | 2 + .../backup/mpc-share-storage.service.ts | 89 + .../blockchain/blockchain-query.service.ts | 178 + .../wallet-generator.service.impl.ts | 74 +- .../infrastructure/infrastructure.module.ts | 15 + .../entities/user-account.entity.ts | 6 +- .../mappers/user-account.mapper.ts | 6 +- .../user-account.repository.impl.ts | 54 +- .../wallet-service/DEVELOPMENT_GUIDE.md | 757 ++ .../assets/images/Background@2x.png | Bin 0 -> 155472 bytes frontend/mobile-app/assets/images/Button.svg | 3 + frontend/mobile-app/assets/images/Button1.svg | 4 + .../mobile-app/assets/images/Container.svg | 3 + .../mobile-app/assets/images/Container1.svg | 3 + .../mobile-app/assets/images/Frame1@3x.png | Bin 0 -> 363 bytes .../mobile-app/assets/images/Frame@3x.png | Bin 0 -> 363 bytes frontend/mobile-app/assets/images/Margin.svg | 3 + frontend/mobile-app/assets/images/Vector.svg | 3 + frontend/mobile-app/assets/images/Vector1.svg | 3 + frontend/mobile-app/docs/mpc_share_backup.md | 448 + .../lib/core/di/injection_container.dart | 14 + .../lib/core/network/api_client.dart | 2 +- .../lib/core/services/account_service.dart | 114 +- .../lib/core/services/deposit_service.dart | 123 + .../lib/core/services/mpc_share_service.dart | 323 + .../lib/core/services/referral_service.dart | 188 + .../lib/core/storage/storage_keys.dart | 5 +- .../pages/backup_mnemonic_page.dart | 521 +- .../presentation/pages/onboarding_page.dart | 184 +- .../pages/verify_mnemonic_page.dart | 222 +- .../pages/wallet_created_page.dart | 419 +- .../presentation/pages/deposit_usdt_page.dart | 618 + .../pages/planting_location_page.dart | 496 + .../pages/planting_quantity_page.dart | 577 + .../widgets/planting_confirm_dialog.dart | 280 + .../presentation/pages/profile_page.dart | 2 +- .../share/presentation/pages/share_page.dart | 351 + .../mobile-app/lib/routes/app_router.dart | 58 + .../mobile-app/lib/routes/route_names.dart | 4 + .../mobile-app/lib/routes/route_paths.dart | 4 + frontend/mobile-app/pubspec.lock | 24 + frontend/mobile-app/pubspec.yaml | 1 + .../core/services/account_service_test.dart | 380 + .../core/services/mpc_share_service_test.dart | 343 + 66 files changed, 18589 insertions(+), 703 deletions(-) create mode 100644 backend/services/identity-service/package-lock.json create mode 100644 backend/services/identity-service/src/api/controllers/deposit.controller.ts create mode 100644 backend/services/identity-service/src/api/controllers/referrals.controller.ts create mode 100644 backend/services/identity-service/src/application/services/deposit.service.ts create mode 100644 backend/services/identity-service/src/application/services/user-application.service.referral.spec.ts create mode 100644 backend/services/identity-service/src/infrastructure/external/backup/backup-client.service.ts create mode 100644 backend/services/identity-service/src/infrastructure/external/backup/index.ts create mode 100644 backend/services/identity-service/src/infrastructure/external/backup/mpc-share-storage.service.ts create mode 100644 backend/services/identity-service/src/infrastructure/external/blockchain/blockchain-query.service.ts create mode 100644 backend/services/wallet-service/DEVELOPMENT_GUIDE.md create mode 100644 frontend/mobile-app/assets/images/Background@2x.png create mode 100644 frontend/mobile-app/assets/images/Button.svg create mode 100644 frontend/mobile-app/assets/images/Button1.svg create mode 100644 frontend/mobile-app/assets/images/Container.svg create mode 100644 frontend/mobile-app/assets/images/Container1.svg create mode 100644 frontend/mobile-app/assets/images/Frame1@3x.png create mode 100644 frontend/mobile-app/assets/images/Frame@3x.png create mode 100644 frontend/mobile-app/assets/images/Margin.svg create mode 100644 frontend/mobile-app/assets/images/Vector.svg create mode 100644 frontend/mobile-app/assets/images/Vector1.svg create mode 100644 frontend/mobile-app/docs/mpc_share_backup.md create mode 100644 frontend/mobile-app/lib/core/services/deposit_service.dart create mode 100644 frontend/mobile-app/lib/core/services/mpc_share_service.dart create mode 100644 frontend/mobile-app/lib/core/services/referral_service.dart create mode 100644 frontend/mobile-app/lib/features/deposit/presentation/pages/deposit_usdt_page.dart create mode 100644 frontend/mobile-app/lib/features/planting/presentation/pages/planting_location_page.dart create mode 100644 frontend/mobile-app/lib/features/planting/presentation/pages/planting_quantity_page.dart create mode 100644 frontend/mobile-app/lib/features/planting/presentation/widgets/planting_confirm_dialog.dart create mode 100644 frontend/mobile-app/lib/features/share/presentation/pages/share_page.dart create mode 100644 frontend/mobile-app/test/core/services/account_service_test.dart create mode 100644 frontend/mobile-app/test/core/services/mpc_share_service_test.dart diff --git a/.claude/settings.local.json b/.claude/settings.local.json index 2361f390..092a6bce 100644 --- a/.claude/settings.local.json +++ b/.claude/settings.local.json @@ -4,7 +4,17 @@ "Bash(dir:*)", "Bash(tree:*)", "Bash(find:*)", - "Bash(ls -la \"c:\\Users\\dong\\Desktop\\rwadurian\\backend\\services\"\" 2>/dev/null || dir \"c:UsersdongDesktoprwadurianbackendservices\"\")" + "Bash(ls -la \"c:\\Users\\dong\\Desktop\\rwadurian\\backend\\services\"\" 2>/dev/null || dir \"c:UsersdongDesktoprwadurianbackendservices\"\")", + "Bash(mkdir:*)", + "Bash(npm run build:*)", + "Bash(npx nest build)", + "Bash(npm install)", + "Bash(npx prisma migrate dev:*)", + "Bash(npx jest:*)", + "Bash(flutter test:*)", + "Bash(flutter analyze:*)", + "Bash(findstr:*)", + "Bash(flutter pub get:*)" ], "deny": [], "ask": [] diff --git a/backend/services/identity-service/.env.development b/backend/services/identity-service/.env.development index 4429d445..7e06e058 100644 --- a/backend/services/identity-service/.env.development +++ b/backend/services/identity-service/.env.development @@ -32,3 +32,9 @@ WALLET_ENCRYPTION_SALT="dev-wallet-salt" # 调用路径: identity-service → mpc-service (NestJS) → mpc-system (Go) MPC_SERVICE_URL="http://localhost:3001" MPC_MODE="local" # local 使用本地模拟,remote 调用 mpc-service + +# Backup Service (MPC 备份分片存储) +# 安全要求: 必须部署在与 identity-service 不同的物理服务器上! +BACKUP_SERVICE_URL="http://localhost:3002" +BACKUP_SERVICE_ENABLED="true" +SERVICE_JWT_SECRET="dev-service-jwt-secret-key" diff --git a/backend/services/identity-service/.env.example b/backend/services/identity-service/.env.example index 1ecc9faf..e6795ca5 100644 --- a/backend/services/identity-service/.env.example +++ b/backend/services/identity-service/.env.example @@ -32,3 +32,9 @@ WALLET_ENCRYPTION_SALT="rwa-wallet-salt-change-in-production" # 调用路径: identity-service → mpc-service (NestJS) → mpc-system (Go) MPC_SERVICE_URL="http://localhost:3001" MPC_MODE="local" # local 使用本地模拟,remote 调用 mpc-service + +# Backup Service (MPC 备份分片存储) +# 安全要求: 必须部署在与 identity-service 不同的物理服务器上! +BACKUP_SERVICE_URL="http://backup-server:3002" +BACKUP_SERVICE_ENABLED="true" +SERVICE_JWT_SECRET="your-service-jwt-secret-change-in-production" diff --git a/backend/services/identity-service/package-lock.json b/backend/services/identity-service/package-lock.json new file mode 100644 index 00000000..e6356a3f --- /dev/null +++ b/backend/services/identity-service/package-lock.json @@ -0,0 +1,10493 @@ +{ + "name": "identity-service", + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "identity-service", + "version": "1.0.0", + "license": "UNLICENSED", + "dependencies": { + "@nestjs/axios": "^3.0.0", + "@nestjs/common": "^10.0.0", + "@nestjs/config": "^3.1.1", + "@nestjs/core": "^10.0.0", + "@nestjs/jwt": "^10.2.0", + "@nestjs/microservices": "^10.0.0", + "@nestjs/passport": "^10.0.0", + "@nestjs/platform-express": "^10.0.0", + "@nestjs/schedule": "^4.0.0", + "@nestjs/swagger": "^7.1.17", + "@prisma/client": "^5.7.0", + "@scure/bip32": "^1.3.2", + "@scure/bip39": "^1.2.1", + "bech32": "^2.0.0", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.0", + "ethers": "^6.9.0", + "ioredis": "^5.3.2", + "jsonwebtoken": "^9.0.0", + "kafkajs": "^2.2.4", + "passport-jwt": "^4.0.1", + "reflect-metadata": "^0.1.13", + "rxjs": "^7.8.1", + "uuid": "^9.0.0" + }, + "devDependencies": { + "@nestjs/cli": "^10.0.0", + "@nestjs/schematics": "^10.0.0", + "@nestjs/testing": "^10.0.0", + "@types/express": "^4.17.17", + "@types/jest": "^29.5.2", + "@types/jsonwebtoken": "^9.0.0", + "@types/node": "^20.3.1", + "@types/passport-jwt": "^4.0.0", + "@types/supertest": "^6.0.0", + "@types/uuid": "^9.0.0", + "@typescript-eslint/eslint-plugin": "^6.0.0", + "@typescript-eslint/parser": "^6.0.0", + "eslint": "^8.42.0", + "eslint-config-prettier": "^9.0.0", + "eslint-plugin-prettier": "^5.0.0", + "jest": "^29.5.0", + "prettier": "^3.0.0", + "prisma": "^5.7.0", + "source-map-support": "^0.5.21", + "supertest": "^6.3.3", + "ts-jest": "^29.1.0", + "ts-loader": "^9.4.3", + "ts-node": "^10.9.1", + "tsconfig-paths": "^4.2.0", + "typescript": "^5.1.3" + } + }, + "node_modules/@adraffy/ens-normalize": { + "version": "1.10.1", + "resolved": "https://registry.npmjs.org/@adraffy/ens-normalize/-/ens-normalize-1.10.1.tgz", + "integrity": "sha512-96Z2IP3mYmF1Xg2cDm8f1gWGf/HUVedQ3FMifV4kG/PQ4yEP51xDtRAEfhVNt5f/uzpNkZHwWQuUcu6D6K+Ekw==", + "license": "MIT" + }, + "node_modules/@angular-devkit/core": { + "version": "17.3.11", + "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-17.3.11.tgz", + "integrity": "sha512-vTNDYNsLIWpYk2I969LMQFH29GTsLzxNk/0cLw5q56ARF0v5sIWfHYwGTS88jdDqIpuuettcSczbxeA7EuAmqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "8.12.0", + "ajv-formats": "2.1.1", + "jsonc-parser": "3.2.1", + "picomatch": "4.0.1", + "rxjs": "7.8.1", + "source-map": "0.7.4" + }, + "engines": { + "node": "^18.13.0 || >=20.9.0", + "npm": "^6.11.0 || ^7.5.6 || >=8.0.0", + "yarn": ">= 1.13.0" + }, + "peerDependencies": { + "chokidar": "^3.5.2" + }, + "peerDependenciesMeta": { + "chokidar": { + "optional": true + } + } + }, + "node_modules/@angular-devkit/core/node_modules/rxjs": { + "version": "7.8.1", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.1.tgz", + "integrity": "sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.1.0" + } + }, + "node_modules/@angular-devkit/schematics": { + "version": "17.3.11", + "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-17.3.11.tgz", + "integrity": "sha512-I5wviiIqiFwar9Pdk30Lujk8FczEEc18i22A5c6Z9lbmhPQdTroDnEQdsfXjy404wPe8H62s0I15o4pmMGfTYQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@angular-devkit/core": "17.3.11", + "jsonc-parser": "3.2.1", + "magic-string": "0.30.8", + "ora": "5.4.1", + "rxjs": "7.8.1" + }, + "engines": { + "node": "^18.13.0 || >=20.9.0", + "npm": "^6.11.0 || ^7.5.6 || >=8.0.0", + "yarn": ">= 1.13.0" + } + }, + "node_modules/@angular-devkit/schematics-cli": { + "version": "17.3.11", + "resolved": "https://registry.npmjs.org/@angular-devkit/schematics-cli/-/schematics-cli-17.3.11.tgz", + "integrity": "sha512-kcOMqp+PHAKkqRad7Zd7PbpqJ0LqLaNZdY1+k66lLWmkEBozgq8v4ASn/puPWf9Bo0HpCiK+EzLf0VHE8Z/y6Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@angular-devkit/core": "17.3.11", + "@angular-devkit/schematics": "17.3.11", + "ansi-colors": "4.1.3", + "inquirer": "9.2.15", + "symbol-observable": "4.0.0", + "yargs-parser": "21.1.1" + }, + "bin": { + "schematics": "bin/schematics.js" + }, + "engines": { + "node": "^18.13.0 || >=20.9.0", + "npm": "^6.11.0 || ^7.5.6 || >=8.0.0", + "yarn": ">= 1.13.0" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/chalk": { + "version": "5.6.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.6.2.tgz", + "integrity": "sha512-7NzBL0rN6fMUW+f7A6Io4h40qQlG+xGmtMxfbnH/K7TAtt8JQWVQK+6g0UXKMeVJoyV5EkkNsErQ8pVD3bLHbA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.17.0 || ^14.13 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/cli-width": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-4.1.0.tgz", + "integrity": "sha512-ouuZd4/dm2Sw5Gmqy6bGyNNNe1qt9RpmxveLSO7KcgsTnU7RXfsw+/bukWGo1abgBiMAic068rclZsO4IWmmxQ==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">= 12" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/inquirer": { + "version": "9.2.15", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-9.2.15.tgz", + "integrity": "sha512-vI2w4zl/mDluHt9YEQ/543VTCwPKWiHzKtm9dM2V0NdFcqEexDAjUHzO1oA60HRNaVifGXXM1tRRNluLVHa0Kg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@ljharb/through": "^2.3.12", + "ansi-escapes": "^4.3.2", + "chalk": "^5.3.0", + "cli-cursor": "^3.1.0", + "cli-width": "^4.1.0", + "external-editor": "^3.1.0", + "figures": "^3.2.0", + "lodash": "^4.17.21", + "mute-stream": "1.0.0", + "ora": "^5.4.1", + "run-async": "^3.0.0", + "rxjs": "^7.8.1", + "string-width": "^4.2.3", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^6.2.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/mute-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-1.0.0.tgz", + "integrity": "sha512-avsJQhyd+680gKXyG/sQc0nXaC6rBkPOfyHYcFb9+hdkqQkR9bdnkJ0AMZhke0oesPqIO+mFFJ+IdBc7mst4IA==", + "dev": true, + "license": "ISC", + "engines": { + "node": "^14.17.0 || ^16.13.0 || >=18.0.0" + } + }, + "node_modules/@angular-devkit/schematics-cli/node_modules/run-async": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-3.0.0.tgz", + "integrity": "sha512-540WwVDOMxA6dN6We19EcT9sc3hkXPw5mzRNGM3FkdN/vtE9NFvj5lFAPNwUDmJjXidm3v7TC1cTE7t17Ulm1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/@angular-devkit/schematics/node_modules/rxjs": { + "version": "7.8.1", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.1.tgz", + "integrity": "sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.1.0" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.27.1.tgz", + "integrity": "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.27.1", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.5.tgz", + "integrity": "sha512-6uFXyCayocRbqhZOB+6XcuZbkMNimwfVGFji8CTZnCzOHVGvDqzvitu1re2AU5LROliz7eQPhB8CpAMvnx9EjA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.5.tgz", + "integrity": "sha512-e7jT4DxYvIDLk1ZHmU/m/mB19rex9sv0c2ftBtjSBv+kVM/902eh0fINUzD7UwLLNR+jU585GxUJ8/EBfAM5fw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.5", + "@babel/helper-compilation-targets": "^7.27.2", + "@babel/helper-module-transforms": "^7.28.3", + "@babel/helpers": "^7.28.4", + "@babel/parser": "^7.28.5", + "@babel/template": "^7.27.2", + "@babel/traverse": "^7.28.5", + "@babel/types": "^7.28.5", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/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.28.5", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.5.tgz", + "integrity": "sha512-3EwLFhZ38J4VyIP6WNtt2kUdW9dokXA9Cr4IVIFHuCpZ3H8/YFOl5JjZHisrn1fATPBmKKqXzDFvh9fUwHz6CQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.28.5", + "@babel/types": "^7.28.5", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.27.2", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.2.tgz", + "integrity": "sha512-2+1thGUUWWjLTYTHZWK1n8Yga0ijBz1XAhUXcKy81rd5g6yh7hGqMp45v7cadSbEHc9G3OTv45SyneRN3ps4DQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.27.2", + "@babel/helper-validator-option": "^7.27.1", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "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-globals": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", + "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.27.1.tgz", + "integrity": "sha512-0gSFWUPNXNopqtIPQvlD5WgXYI5GY2kP2cCvoT8kczjbfcfuIljTbcWrulD1CIPIX2gt1wghbDy08yE1p+/r3w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.27.1", + "@babel/types": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.28.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.3.tgz", + "integrity": "sha512-gytXUbs8k2sXS9PnQptz5o0QnpLL51SwASIORY6XaBKF88nsOT0Zw9szLqlSGQDP/4TljBAD5y98p2U1fqkdsw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.27.1", + "@babel/helper-validator-identifier": "^7.27.1", + "@babel/traverse": "^7.28.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.27.1.tgz", + "integrity": "sha512-1gn1Up5YXka3YYAHGKpbideQ5Yjf1tDa9qYcgysz+cNCXukyLl6DjPXhD3VRwSb8c0J9tA4b2+rHEZtc6R0tlw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", + "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.28.4", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.4.tgz", + "integrity": "sha512-HFN59MmQXGHVyYadKLVumYsA9dBFun/ldYxipEjzA4196jpLZd8UjEEBLkbEkvfYreDqJhZxYAWFPtrfhNpj4w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.4" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.5.tgz", + "integrity": "sha512-KKBU1VGYR7ORr3At5HAtUQ+TV3SzRCXmA/8OdDZiLDBIZxVyzXuztPjfLd3BV1PRAQGCMWWSHYhL0F8d5uHBDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.5" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", + "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-bigint": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", + "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-properties": { + "version": "7.12.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", + "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.12.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-static-block": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-static-block/-/plugin-syntax-class-static-block-7.14.5.tgz", + "integrity": "sha512-b+YyPmr6ldyNnM6sqYeMWE+bgJcJpO6yS4QD7ymxgH34GBPNDM/THBh8iunyvKIZztiwLH4CJZ0RxTk9emgpjw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-attributes": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.27.1.tgz", + "integrity": "sha512-oFT0FrKHgF53f4vOsZGi2Hh3I35PfSmVs4IBFLFj4dnafP+hIWDLg3VyKmUHfLoLHlyxY4C7DGtmHuJgn+IGww==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-meta": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", + "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", + "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.27.1.tgz", + "integrity": "sha512-y8YTNIeKoyhGd9O0Jiyzyyqk8gdjnumGTQPsz0xOZOQ2RmkVJeZ1vmmfIvFEKqucBG6axJGBZDE/7iI5suUI/w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", + "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-nullish-coalescing-operator": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", + "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", + "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", + "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", + "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-chaining": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", + "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-private-property-in-object": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-private-property-in-object/-/plugin-syntax-private-property-in-object-7.14.5.tgz", + "integrity": "sha512-0wVnp9dxJ72ZUJDV27ZfbSj6iHLoytYZmh3rFcxNnvsJF3ktkzLDZPy/mA17HGsaQT3/DQsWYX1f1QGWkCoVUg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-top-level-await": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", + "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-typescript": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.27.1.tgz", + "integrity": "sha512-xfYCBMxveHrRMnAWl1ZlPXOZjzkN82THFvLhQhFXFt81Z5HnN+EtUkZhv/zcKpmT3fzmWZB0ywiBrbC3vogbwQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/template": { + "version": "7.27.2", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.27.2.tgz", + "integrity": "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/parser": "^7.27.2", + "@babel/types": "^7.27.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.5.tgz", + "integrity": "sha512-TCCj4t55U90khlYkVV/0TfkJkAkUg3jZFA3Neb7unZT8CPok7iiRfaX0F+WnqWqt7OxhOn0uBKXCw4lbL8W0aQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.27.1", + "@babel/generator": "^7.28.5", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.28.5", + "@babel/template": "^7.27.2", + "@babel/types": "^7.28.5", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.5.tgz", + "integrity": "sha512-qQ5m48eI/MFLQ5PxQj4PFaprjyCTLI37ElWMmNs0K8Lk3dVeOdNpB3ks8jc7yM5CDmVC73eMVk/trk3fgmrUpA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@bcoe/v8-coverage": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", + "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@borewit/text-codec": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/@borewit/text-codec/-/text-codec-0.1.1.tgz", + "integrity": "sha512-5L/uBxmjaCIX5h8Z+uu+kA9BQLkc/Wl06UGR5ajNRxu+/XjonB5i8JpgFMrPj3LXTCPA0pv8yxUvbUi+QthGGA==", + "license": "MIT", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Borewit" + } + }, + "node_modules/@colors/colors": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.5.0.tgz", + "integrity": "sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==", + "dev": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.1.90" + } + }, + "node_modules/@cspotcode/source-map-support": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz", + "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/trace-mapping": "0.3.9" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@cspotcode/source-map-support/node_modules/@jridgewell/trace-mapping": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", + "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.0.3", + "@jridgewell/sourcemap-codec": "^1.4.10" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.9.0", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.0.tgz", + "integrity": "sha512-ayVFHdtZ+hsq1t2Dy24wCmGXGe4q9Gu3smhLYALJrr473ZH27MsnSL+LKUlimp4BWJqMDMLmPpx/Q9R3OAlL4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "eslint-visitor-keys": "^3.4.3" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-2.1.4.tgz", + "integrity": "sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^9.6.0", + "globals": "^13.19.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/eslintrc/node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/@eslint/eslintrc/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/@eslint/eslintrc/node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true, + "license": "MIT" + }, + "node_modules/@eslint/eslintrc/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/@eslint/js": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.1.tgz", + "integrity": "sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + } + }, + "node_modules/@humanwhocodes/config-array": { + "version": "0.13.0", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.13.0.tgz", + "integrity": "sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw==", + "deprecated": "Use @eslint/config-array instead", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanwhocodes/object-schema": "^2.0.3", + "debug": "^4.3.1", + "minimatch": "^3.0.5" + }, + "engines": { + "node": ">=10.10.0" + } + }, + "node_modules/@humanwhocodes/config-array/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/@humanwhocodes/config-array/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/object-schema": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", + "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", + "deprecated": "Use @eslint/object-schema instead", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/@ioredis/commands": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/@ioredis/commands/-/commands-1.4.0.tgz", + "integrity": "sha512-aFT2yemJJo+TZCmieA7qnYGQooOS7QfNmYrzGtsYd3g9j5iDP8AimYYAesf79ohjbLG12XxC4nG5DyEnC88AsQ==", + "license": "MIT" + }, + "node_modules/@isaacs/cliui": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", + "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^5.1.2", + "string-width-cjs": "npm:string-width@^4.2.0", + "strip-ansi": "^7.0.1", + "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", + "wrap-ansi": "^8.1.0", + "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@isaacs/cliui/node_modules/ansi-regex": { + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.2.2.tgz", + "integrity": "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-regex?sponsor=1" + } + }, + "node_modules/@isaacs/cliui/node_modules/ansi-styles": { + "version": "6.2.3", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.3.tgz", + "integrity": "sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@isaacs/cliui/node_modules/emoji-regex": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", + "dev": true, + "license": "MIT" + }, + "node_modules/@isaacs/cliui/node_modules/string-width": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", + "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", + "dev": true, + "license": "MIT", + "dependencies": { + "eastasianwidth": "^0.2.0", + "emoji-regex": "^9.2.2", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@isaacs/cliui/node_modules/strip-ansi": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.2.tgz", + "integrity": "sha512-gmBGslpoQJtgnMAvOVqGZpEz9dyoKTCzy2nfz/n8aIFhN/jCE/rCmcxabB6jOOHV+0WNnylOxaxBQPSvcWklhA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^6.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/strip-ansi?sponsor=1" + } + }, + "node_modules/@isaacs/cliui/node_modules/wrap-ansi": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", + "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^6.1.0", + "string-width": "^5.0.1", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/@istanbuljs/load-nyc-config": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", + "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "camelcase": "^5.3.1", + "find-up": "^4.1.0", + "get-package-type": "^0.1.0", + "js-yaml": "^3.13.1", + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "license": "MIT", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/@istanbuljs/load-nyc-config/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", + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/js-yaml": { + "version": "3.14.2", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.2.tgz", + "integrity": "sha512-PMSmkqxr106Xa156c2M265Z+FTrPl+oxd/rgOQy2tijQeK5TxQ43psO1ZCwhVOSdnn+RzkzlRz/eY4BgJBYVpg==", + "dev": true, + "license": "MIT", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/@istanbuljs/load-nyc-config/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", + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/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", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@istanbuljs/load-nyc-config/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", + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-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==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/schema": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", + "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/console": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-29.7.0.tgz", + "integrity": "sha512-5Ni4CU7XHQi32IJ398EEP4RrB8eV09sXP2ROqD4bksHrnTree52PsxvX8tpL8LvTZ3pFzXyPbNQReSN41CAhOg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/core": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/core/-/core-29.7.0.tgz", + "integrity": "sha512-n7aeXWKMnGtDA48y8TLWJPJmLmmZ642Ceo78cYWEpiD7FzDgmNDV/GCVRorPABdXLJZ/9wzzgZAlHjXjxDHGsg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/reporters": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-changed-files": "^29.7.0", + "jest-config": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-resolve-dependencies": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "jest-watcher": "^29.7.0", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/environment": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-29.7.0.tgz", + "integrity": "sha512-aQIfHDq33ExsN4jP1NWGXhxgQ/wixs60gDiKO+XVMd8Mn0NWPWgc34ZQDTb2jKaUWQ7MuwoitXAsN2XVXNMpAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-8uMeAMycttpva3P1lBHB8VciS9V0XAr3GymPpipdyQXbBcuhkLQOSe8E/p92RyAdToS6ZD1tFkX+CkhoECE0dQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "expect": "^29.7.0", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/expect-utils/-/expect-utils-29.7.0.tgz", + "integrity": "sha512-GlsNBWiFQFCVi9QVSx7f5AgMeLxe9YCCs5PuP2O2LdjDAA8Jh9eX7lA1Jq/xdXw3Wb3hyvlFNfZIfcRetSzYcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-get-type": "^29.6.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/fake-timers": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-29.7.0.tgz", + "integrity": "sha512-q4DH1Ha4TTFPdxLsqDXK1d3+ioSL7yL5oCMJZgDYm6i+6CygW5E5xVr/D1HdsGxjt1ZWSfUAs9OxSB/BNelWrQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@sinonjs/fake-timers": "^10.0.2", + "@types/node": "*", + "jest-message-util": "^29.7.0", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/globals": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-29.7.0.tgz", + "integrity": "sha512-mpiz3dutLbkW2MNFubUGUEVLkTGiqW6yLVTA+JbP6fI6J5iL9Y0Nlg8k95pcF8ctKwCS7WVxteBs29hhfAotzQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/types": "^29.6.3", + "jest-mock": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/reporters": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-29.7.0.tgz", + "integrity": "sha512-DApq0KJbJOEzAFYjHADNNxAE3KbhxQB1y5Kplb5Waqw6zVbuWatSnMjE5gs8FUgEPmNsnZA3NCWl9NG0ia04Pg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@bcoe/v8-coverage": "^0.2.3", + "@jest/console": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@jridgewell/trace-mapping": "^0.3.18", + "@types/node": "*", + "chalk": "^4.0.0", + "collect-v8-coverage": "^1.0.0", + "exit": "^0.1.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "istanbul-lib-coverage": "^3.0.0", + "istanbul-lib-instrument": "^6.0.0", + "istanbul-lib-report": "^3.0.0", + "istanbul-lib-source-maps": "^4.0.0", + "istanbul-reports": "^3.1.3", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "slash": "^3.0.0", + "string-length": "^4.0.1", + "strip-ansi": "^6.0.0", + "v8-to-istanbul": "^9.0.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/reporters/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/@jest/reporters/node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/@jest/reporters/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/@jest/schemas": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.6.3.tgz", + "integrity": "sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@sinclair/typebox": "^0.27.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/source-map": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-29.6.3.tgz", + "integrity": "sha512-MHjT95QuipcPrpLM+8JMSzFx6eHp5Bm+4XeFDJlwsvVBjmKNiIAvasGK2fxz2WbGRlnvqehFbh07MMa7n3YJnw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.18", + "callsites": "^3.0.0", + "graceful-fs": "^4.2.9" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/test-result": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-29.7.0.tgz", + "integrity": "sha512-Fdx+tv6x1zlkJPcWXmMDAG2HBnaR9XPSd5aDWQVsfrZmLVT3lU1cwyxLgRmXR9yrq4NBoEm9BMsfgFzTQAbJYA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "collect-v8-coverage": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/test-sequencer": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-29.7.0.tgz", + "integrity": "sha512-GQwJ5WZVrKnOJuiYiAF52UNUJXgTZx1NHjFSEB0qEMmSZKAkdMoIzw/Cj6x6NF4AvV23AUqDpFzQkN/eYCYTxw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/test-result": "^29.7.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/transform": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-29.7.0.tgz", + "integrity": "sha512-ok/BTPFzFKVMwO5eOHRrvnBVHdRy9IrsrW1GpMaQ9MCnilNLXQKmAX8s1YXDFaai9xJpac2ySzV0YeRRECr2Vw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/types": "^29.6.3", + "@jridgewell/trace-mapping": "^0.3.18", + "babel-plugin-istanbul": "^6.1.1", + "chalk": "^4.0.0", + "convert-source-map": "^2.0.0", + "fast-json-stable-stringify": "^2.1.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "micromatch": "^4.0.4", + "pirates": "^4.0.4", + "slash": "^3.0.0", + "write-file-atomic": "^4.0.2" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "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/@jridgewell/gen-mapping": { + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", + "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", + "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/source-map": { + "version": "0.3.11", + "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.11.tgz", + "integrity": "sha512-ZMp1V8ZFcPG5dIWnQLr3NSI1MiCU7UETdS/A0G8V/XWHvJv3ZsFqutJn1Y5RPmAPX6F3BiE397OqveU/9NCuIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.25" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", + "dev": true, + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@ljharb/through": { + "version": "2.3.14", + "resolved": "https://registry.npmjs.org/@ljharb/through/-/through-2.3.14.tgz", + "integrity": "sha512-ajBvlKpWucBB17FuQYUShqpqy8GRgYEpJW0vWJbUu1CV9lWyrDCapy0lScU8T8Z6qn49sSwJB3+M+evYIdGg+A==", + "dev": true, + "license": "MIT", + "dependencies": { + "call-bind": "^1.0.8" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/@lukeed/csprng": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@lukeed/csprng/-/csprng-1.1.0.tgz", + "integrity": "sha512-Z7C/xXCiGWsg0KuKsHTKJxbWhpI3Vs5GwLfOean7MGyVFGqdRgBbAjOCh6u4bbjPc/8MJ2pZmK/0DLdCbivLDA==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/@microsoft/tsdoc": { + "version": "0.15.1", + "resolved": "https://registry.npmjs.org/@microsoft/tsdoc/-/tsdoc-0.15.1.tgz", + "integrity": "sha512-4aErSrCR/On/e5G2hDP0wjooqDdauzEbIq8hIkIe5pXV0rtWJZvdCEKL0ykZxex+IxIwBp0eGeV48hQN07dXtw==", + "license": "MIT" + }, + "node_modules/@nestjs/axios": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/@nestjs/axios/-/axios-3.1.3.tgz", + "integrity": "sha512-RZ/63c1tMxGLqyG3iOCVt7A72oy4x1eM6QEhd4KzCYpaVWW0igq0WSREeRoEZhIxRcZfDfIIkvsOMiM7yfVGZQ==", + "license": "MIT", + "peerDependencies": { + "@nestjs/common": "^7.0.0 || ^8.0.0 || ^9.0.0 || ^10.0.0", + "axios": "^1.3.1", + "rxjs": "^6.0.0 || ^7.0.0" + } + }, + "node_modules/@nestjs/cli": { + "version": "10.4.9", + "resolved": "https://registry.npmjs.org/@nestjs/cli/-/cli-10.4.9.tgz", + "integrity": "sha512-s8qYd97bggqeK7Op3iD49X2MpFtW4LVNLAwXFkfbRxKME6IYT7X0muNTJ2+QfI8hpbNx9isWkrLWIp+g5FOhiA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@angular-devkit/core": "17.3.11", + "@angular-devkit/schematics": "17.3.11", + "@angular-devkit/schematics-cli": "17.3.11", + "@nestjs/schematics": "^10.0.1", + "chalk": "4.1.2", + "chokidar": "3.6.0", + "cli-table3": "0.6.5", + "commander": "4.1.1", + "fork-ts-checker-webpack-plugin": "9.0.2", + "glob": "10.4.5", + "inquirer": "8.2.6", + "node-emoji": "1.11.0", + "ora": "5.4.1", + "tree-kill": "1.2.2", + "tsconfig-paths": "4.2.0", + "tsconfig-paths-webpack-plugin": "4.2.0", + "typescript": "5.7.2", + "webpack": "5.97.1", + "webpack-node-externals": "3.0.0" + }, + "bin": { + "nest": "bin/nest.js" + }, + "engines": { + "node": ">= 16.14" + }, + "peerDependencies": { + "@swc/cli": "^0.1.62 || ^0.3.0 || ^0.4.0 || ^0.5.0", + "@swc/core": "^1.3.62" + }, + "peerDependenciesMeta": { + "@swc/cli": { + "optional": true + }, + "@swc/core": { + "optional": true + } + } + }, + "node_modules/@nestjs/cli/node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/@nestjs/cli/node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/@nestjs/cli/node_modules/typescript": { + "version": "5.7.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.7.2.tgz", + "integrity": "sha512-i5t66RHxDvVN40HfDd1PsEThGNnlMCMT3jMUuoh9/0TaqWevNontacunWyN02LA9/fIbEWlcHZcgTKb9QoaLfg==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/@nestjs/cli/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==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/eslint-scope": "^3.7.7", + "@types/estree": "^1.0.6", + "@webassemblyjs/ast": "^1.14.1", + "@webassemblyjs/wasm-edit": "^1.14.1", + "@webassemblyjs/wasm-parser": "^1.14.1", + "acorn": "^8.14.0", + "browserslist": "^4.24.0", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^5.17.1", + "es-module-lexer": "^1.2.1", + "eslint-scope": "5.1.1", + "events": "^3.2.0", + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.2.11", + "json-parse-even-better-errors": "^2.3.1", + "loader-runner": "^4.2.0", + "mime-types": "^2.1.27", + "neo-async": "^2.6.2", + "schema-utils": "^3.2.0", + "tapable": "^2.1.1", + "terser-webpack-plugin": "^5.3.10", + "watchpack": "^2.4.1", + "webpack-sources": "^3.2.3" + }, + "bin": { + "webpack": "bin/webpack.js" + }, + "engines": { + "node": ">=10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependenciesMeta": { + "webpack-cli": { + "optional": true + } + } + }, + "node_modules/@nestjs/common": { + "version": "10.4.20", + "resolved": "https://registry.npmjs.org/@nestjs/common/-/common-10.4.20.tgz", + "integrity": "sha512-hxJxZF7jcKGuUzM9EYbuES80Z/36piJbiqmPy86mk8qOn5gglFebBTvcx7PWVbRNSb4gngASYnefBj/Y2HAzpQ==", + "license": "MIT", + "peer": true, + "dependencies": { + "file-type": "20.4.1", + "iterare": "1.2.1", + "tslib": "2.8.1", + "uid": "2.0.2" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/nest" + }, + "peerDependencies": { + "class-transformer": "*", + "class-validator": "*", + "reflect-metadata": "^0.1.12 || ^0.2.0", + "rxjs": "^7.1.0" + }, + "peerDependenciesMeta": { + "class-transformer": { + "optional": true + }, + "class-validator": { + "optional": true + } + } + }, + "node_modules/@nestjs/config": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/@nestjs/config/-/config-3.3.0.tgz", + "integrity": "sha512-pdGTp8m9d0ZCrjTpjkUbZx6gyf2IKf+7zlkrPNMsJzYZ4bFRRTpXrnj+556/5uiI6AfL5mMrJc2u7dB6bvM+VA==", + "license": "MIT", + "dependencies": { + "dotenv": "16.4.5", + "dotenv-expand": "10.0.0", + "lodash": "4.17.21" + }, + "peerDependencies": { + "@nestjs/common": "^8.0.0 || ^9.0.0 || ^10.0.0", + "rxjs": "^7.1.0" + } + }, + "node_modules/@nestjs/core": { + "version": "10.4.20", + "resolved": "https://registry.npmjs.org/@nestjs/core/-/core-10.4.20.tgz", + "integrity": "sha512-kRdtyKA3+Tu70N3RQ4JgmO1E3LzAMs/eppj7SfjabC7TgqNWoS4RLhWl4BqmsNVmjj6D5jgfPVtHtgYkU3AfpQ==", + "hasInstallScript": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@nuxtjs/opencollective": "0.3.2", + "fast-safe-stringify": "2.1.1", + "iterare": "1.2.1", + "path-to-regexp": "3.3.0", + "tslib": "2.8.1", + "uid": "2.0.2" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/nest" + }, + "peerDependencies": { + "@nestjs/common": "^10.0.0", + "@nestjs/microservices": "^10.0.0", + "@nestjs/platform-express": "^10.0.0", + "@nestjs/websockets": "^10.0.0", + "reflect-metadata": "^0.1.12 || ^0.2.0", + "rxjs": "^7.1.0" + }, + "peerDependenciesMeta": { + "@nestjs/microservices": { + "optional": true + }, + "@nestjs/platform-express": { + "optional": true + }, + "@nestjs/websockets": { + "optional": true + } + } + }, + "node_modules/@nestjs/jwt": { + "version": "10.2.0", + "resolved": "https://registry.npmjs.org/@nestjs/jwt/-/jwt-10.2.0.tgz", + "integrity": "sha512-x8cG90SURkEiLOehNaN2aRlotxT0KZESUliOPKKnjWiyJOcWurkF3w345WOX0P4MgFzUjGoZ1Sy0aZnxeihT0g==", + "license": "MIT", + "dependencies": { + "@types/jsonwebtoken": "9.0.5", + "jsonwebtoken": "9.0.2" + }, + "peerDependencies": { + "@nestjs/common": "^8.0.0 || ^9.0.0 || ^10.0.0" + } + }, + "node_modules/@nestjs/jwt/node_modules/@types/jsonwebtoken": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/@types/jsonwebtoken/-/jsonwebtoken-9.0.5.tgz", + "integrity": "sha512-VRLSGzik+Unrup6BsouBeHsf4d1hOEgYWTm/7Nmw1sXoN1+tRly/Gy/po3yeahnP4jfnQWWAhQAqcNfH7ngOkA==", + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@nestjs/mapped-types": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nestjs/mapped-types/-/mapped-types-2.0.5.tgz", + "integrity": "sha512-bSJv4pd6EY99NX9CjBIyn4TVDoSit82DUZlL4I3bqNfy5Gt+gXTa86i3I/i0iIV9P4hntcGM5GyO+FhZAhxtyg==", + "license": "MIT", + "peerDependencies": { + "@nestjs/common": "^8.0.0 || ^9.0.0 || ^10.0.0", + "class-transformer": "^0.4.0 || ^0.5.0", + "class-validator": "^0.13.0 || ^0.14.0", + "reflect-metadata": "^0.1.12 || ^0.2.0" + }, + "peerDependenciesMeta": { + "class-transformer": { + "optional": true + }, + "class-validator": { + "optional": true + } + } + }, + "node_modules/@nestjs/microservices": { + "version": "10.4.20", + "resolved": "https://registry.npmjs.org/@nestjs/microservices/-/microservices-10.4.20.tgz", + "integrity": "sha512-zu/o84Z0uTUClNnGIGfIjcrO3z6T60h/pZPSJK50o4mehbEvJ76fijj6R/WTW0VP+1N16qOv/NsiYLKJA5Cc3w==", + "license": "MIT", + "peer": true, + "dependencies": { + "iterare": "1.2.1", + "tslib": "2.8.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/nest" + }, + "peerDependencies": { + "@grpc/grpc-js": "*", + "@nestjs/common": "^10.0.0", + "@nestjs/core": "^10.0.0", + "@nestjs/websockets": "^10.0.0", + "amqp-connection-manager": "*", + "amqplib": "*", + "cache-manager": "*", + "ioredis": "*", + "kafkajs": "*", + "mqtt": "*", + "nats": "*", + "reflect-metadata": "^0.1.12 || ^0.2.0", + "rxjs": "^7.1.0" + }, + "peerDependenciesMeta": { + "@grpc/grpc-js": { + "optional": true + }, + "@nestjs/websockets": { + "optional": true + }, + "amqp-connection-manager": { + "optional": true + }, + "amqplib": { + "optional": true + }, + "cache-manager": { + "optional": true + }, + "ioredis": { + "optional": true + }, + "kafkajs": { + "optional": true + }, + "mqtt": { + "optional": true + }, + "nats": { + "optional": true + } + } + }, + "node_modules/@nestjs/passport": { + "version": "10.0.3", + "resolved": "https://registry.npmjs.org/@nestjs/passport/-/passport-10.0.3.tgz", + "integrity": "sha512-znJ9Y4S8ZDVY+j4doWAJ8EuuVO7SkQN3yOBmzxbGaXbvcSwFDAdGJ+OMCg52NdzIO4tQoN4pYKx8W6M0ArfFRQ==", + "license": "MIT", + "peerDependencies": { + "@nestjs/common": "^8.0.0 || ^9.0.0 || ^10.0.0", + "passport": "^0.4.0 || ^0.5.0 || ^0.6.0 || ^0.7.0" + } + }, + "node_modules/@nestjs/platform-express": { + "version": "10.4.20", + "resolved": "https://registry.npmjs.org/@nestjs/platform-express/-/platform-express-10.4.20.tgz", + "integrity": "sha512-rh97mX3rimyf4xLMLHuTOBKe6UD8LOJ14VlJ1F/PTd6C6ZK9Ak6EHuJvdaGcSFQhd3ZMBh3I6CuujKGW9pNdIg==", + "license": "MIT", + "peer": true, + "dependencies": { + "body-parser": "1.20.3", + "cors": "2.8.5", + "express": "4.21.2", + "multer": "2.0.2", + "tslib": "2.8.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/nest" + }, + "peerDependencies": { + "@nestjs/common": "^10.0.0", + "@nestjs/core": "^10.0.0" + } + }, + "node_modules/@nestjs/schedule": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/@nestjs/schedule/-/schedule-4.1.2.tgz", + "integrity": "sha512-hCTQ1lNjIA5EHxeu8VvQu2Ed2DBLS1GSC6uKPYlBiQe6LL9a7zfE9iVSK+zuK8E2odsApteEBmfAQchc8Hx0Gg==", + "license": "MIT", + "dependencies": { + "cron": "3.2.1", + "uuid": "11.0.3" + }, + "peerDependencies": { + "@nestjs/common": "^8.0.0 || ^9.0.0 || ^10.0.0", + "@nestjs/core": "^8.0.0 || ^9.0.0 || ^10.0.0" + } + }, + "node_modules/@nestjs/schedule/node_modules/uuid": { + "version": "11.0.3", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-11.0.3.tgz", + "integrity": "sha512-d0z310fCWv5dJwnX1Y/MncBAqGMKEzlBb1AOf7z9K8ALnd0utBX/msg/fA0+sbyN1ihbMsLhrBlnl1ak7Wa0rg==", + "funding": [ + "https://github.com/sponsors/broofa", + "https://github.com/sponsors/ctavan" + ], + "license": "MIT", + "bin": { + "uuid": "dist/esm/bin/uuid" + } + }, + "node_modules/@nestjs/schematics": { + "version": "10.2.3", + "resolved": "https://registry.npmjs.org/@nestjs/schematics/-/schematics-10.2.3.tgz", + "integrity": "sha512-4e8gxaCk7DhBxVUly2PjYL4xC2ifDFexCqq1/u4TtivLGXotVk0wHdYuPYe1tHTHuR1lsOkRbfOCpkdTnigLVg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@angular-devkit/core": "17.3.11", + "@angular-devkit/schematics": "17.3.11", + "comment-json": "4.2.5", + "jsonc-parser": "3.3.1", + "pluralize": "8.0.0" + }, + "peerDependencies": { + "typescript": ">=4.8.2" + } + }, + "node_modules/@nestjs/schematics/node_modules/jsonc-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-3.3.1.tgz", + "integrity": "sha512-HUgH65KyejrUFPvHFPbqOY0rsFip3Bo5wb4ngvdi1EpCYWUQDC5V+Y7mZws+DLkr4M//zQJoanu1SP+87Dv1oQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@nestjs/swagger": { + "version": "7.4.2", + "resolved": "https://registry.npmjs.org/@nestjs/swagger/-/swagger-7.4.2.tgz", + "integrity": "sha512-Mu6TEn1M/owIvAx2B4DUQObQXqo2028R2s9rSZ/hJEgBK95+doTwS0DjmVA2wTeZTyVtXOoN7CsoM5pONBzvKQ==", + "license": "MIT", + "dependencies": { + "@microsoft/tsdoc": "^0.15.0", + "@nestjs/mapped-types": "2.0.5", + "js-yaml": "4.1.0", + "lodash": "4.17.21", + "path-to-regexp": "3.3.0", + "swagger-ui-dist": "5.17.14" + }, + "peerDependencies": { + "@fastify/static": "^6.0.0 || ^7.0.0", + "@nestjs/common": "^9.0.0 || ^10.0.0", + "@nestjs/core": "^9.0.0 || ^10.0.0", + "class-transformer": "*", + "class-validator": "*", + "reflect-metadata": "^0.1.12 || ^0.2.0" + }, + "peerDependenciesMeta": { + "@fastify/static": { + "optional": true + }, + "class-transformer": { + "optional": true + }, + "class-validator": { + "optional": true + } + } + }, + "node_modules/@nestjs/testing": { + "version": "10.4.20", + "resolved": "https://registry.npmjs.org/@nestjs/testing/-/testing-10.4.20.tgz", + "integrity": "sha512-nMkRDukDKskdPruM6EsgMq7yJua+CPZM6I6FrLP8yXw8BiVSPv9Nm0CtcGGwt3kgZF9hfxKjGqLjsvVBsv6Vfw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tslib": "2.8.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/nest" + }, + "peerDependencies": { + "@nestjs/common": "^10.0.0", + "@nestjs/core": "^10.0.0", + "@nestjs/microservices": "^10.0.0", + "@nestjs/platform-express": "^10.0.0" + }, + "peerDependenciesMeta": { + "@nestjs/microservices": { + "optional": true + }, + "@nestjs/platform-express": { + "optional": true + } + } + }, + "node_modules/@noble/curves": { + "version": "1.9.7", + "resolved": "https://registry.npmjs.org/@noble/curves/-/curves-1.9.7.tgz", + "integrity": "sha512-gbKGcRUYIjA3/zCCNaWDciTMFI0dCkvou3TL8Zmy5Nc7sJ47a0jtOeZoTaMxkuqRo9cRhjOdZJXegxYE5FN/xw==", + "license": "MIT", + "dependencies": { + "@noble/hashes": "1.8.0" + }, + "engines": { + "node": "^14.21.3 || >=16" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/@noble/hashes": { + "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" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nuxtjs/opencollective": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/@nuxtjs/opencollective/-/opencollective-0.3.2.tgz", + "integrity": "sha512-um0xL3fO7Mf4fDxcqx9KryrB7zgRM5JSlvGN5AGkP6JLM5XEKyjeAiPbNxdXVXQ16isuAhYpvP88NgL2BGd6aA==", + "license": "MIT", + "dependencies": { + "chalk": "^4.1.0", + "consola": "^2.15.0", + "node-fetch": "^2.6.1" + }, + "bin": { + "opencollective": "bin/opencollective.js" + }, + "engines": { + "node": ">=8.0.0", + "npm": ">=5.0.0" + } + }, + "node_modules/@paralleldrive/cuid2": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/@paralleldrive/cuid2/-/cuid2-2.3.1.tgz", + "integrity": "sha512-XO7cAxhnTZl0Yggq6jOgjiOHhbgcO4NqFqwSmQpjK3b6TEE6Uj/jfSk6wzYyemh3+I0sHirKSetjQwn5cZktFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@noble/hashes": "^1.1.5" + } + }, + "node_modules/@pkgjs/parseargs": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", + "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", + "dev": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=14" + } + }, + "node_modules/@pkgr/core": { + "version": "0.2.9", + "resolved": "https://registry.npmjs.org/@pkgr/core/-/core-0.2.9.tgz", + "integrity": "sha512-QNqXyfVS2wm9hweSYD2O7F0G06uurj9kZ96TRQE5Y9hU7+tgdZwIkbAKc5Ocy1HxEY2kuDQa6cQ1WRs/O5LFKA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.20.0 || ^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/pkgr" + } + }, + "node_modules/@prisma/client": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/client/-/client-5.22.0.tgz", + "integrity": "sha512-M0SVXfyHnQREBKxCgyo7sffrKttwE6R8PMq330MIUF0pTwjUhLbW84pFDlf06B27XyCR++VtjugEnIHdr07SVA==", + "hasInstallScript": true, + "license": "Apache-2.0", + "engines": { + "node": ">=16.13" + }, + "peerDependencies": { + "prisma": "*" + }, + "peerDependenciesMeta": { + "prisma": { + "optional": true + } + } + }, + "node_modules/@prisma/debug": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/debug/-/debug-5.22.0.tgz", + "integrity": "sha512-AUt44v3YJeggO2ZU5BkXI7M4hu9BF2zzH2iF2V5pyXT/lRTyWiElZ7It+bRH1EshoMRxHgpYg4VB6rCM+mG5jQ==", + "devOptional": true, + "license": "Apache-2.0" + }, + "node_modules/@prisma/engines": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/engines/-/engines-5.22.0.tgz", + "integrity": "sha512-UNjfslWhAt06kVL3CjkuYpHAWSO6L4kDCVPegV6itt7nD1kSJavd3vhgAEhjglLJJKEdJ7oIqDJ+yHk6qO8gPA==", + "devOptional": true, + "hasInstallScript": true, + "license": "Apache-2.0", + "dependencies": { + "@prisma/debug": "5.22.0", + "@prisma/engines-version": "5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2", + "@prisma/fetch-engine": "5.22.0", + "@prisma/get-platform": "5.22.0" + } + }, + "node_modules/@prisma/engines-version": { + "version": "5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2", + "resolved": "https://registry.npmjs.org/@prisma/engines-version/-/engines-version-5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2.tgz", + "integrity": "sha512-2PTmxFR2yHW/eB3uqWtcgRcgAbG1rwG9ZriSvQw+nnb7c4uCr3RAcGMb6/zfE88SKlC1Nj2ziUvc96Z379mHgQ==", + "devOptional": true, + "license": "Apache-2.0" + }, + "node_modules/@prisma/fetch-engine": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/fetch-engine/-/fetch-engine-5.22.0.tgz", + "integrity": "sha512-bkrD/Mc2fSvkQBV5EpoFcZ87AvOgDxbG99488a5cexp5Ccny+UM6MAe/UFkUC0wLYD9+9befNOqGiIJhhq+HbA==", + "devOptional": true, + "license": "Apache-2.0", + "dependencies": { + "@prisma/debug": "5.22.0", + "@prisma/engines-version": "5.22.0-44.605197351a3c8bdd595af2d2a9bc3025bca48ea2", + "@prisma/get-platform": "5.22.0" + } + }, + "node_modules/@prisma/get-platform": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/@prisma/get-platform/-/get-platform-5.22.0.tgz", + "integrity": "sha512-pHhpQdr1UPFpt+zFfnPazhulaZYCUqeIcPpJViYoq9R+D/yw4fjE+CtnsnKzPYm0ddUbeXUzjGVGIRVgPDCk4Q==", + "devOptional": true, + "license": "Apache-2.0", + "dependencies": { + "@prisma/debug": "5.22.0" + } + }, + "node_modules/@scure/base": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@scure/base/-/base-1.2.6.tgz", + "integrity": "sha512-g/nm5FgUa//MCj1gV09zTJTaM6KBAHqLN907YVQqf7zC49+DcO4B1so4ZX07Ef10Twr6nuqYEH9GEggFXA4Fmg==", + "license": "MIT", + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/@scure/bip32": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/@scure/bip32/-/bip32-1.7.0.tgz", + "integrity": "sha512-E4FFX/N3f4B80AKWp5dP6ow+flD1LQZo/w8UnLGYZO674jS6YnYeepycOOksv+vLPSpgN35wgKgy+ybfTb2SMw==", + "license": "MIT", + "dependencies": { + "@noble/curves": "~1.9.0", + "@noble/hashes": "~1.8.0", + "@scure/base": "~1.2.5" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/@scure/bip39": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/@scure/bip39/-/bip39-1.6.0.tgz", + "integrity": "sha512-+lF0BbLiJNwVlev4eKelw1WWLaiKXw7sSl8T6FvBlWkdX+94aGJ4o8XjUdlyhTCjd8c+B3KT3JfS8P0bLRNU6A==", + "license": "MIT", + "dependencies": { + "@noble/hashes": "~1.8.0", + "@scure/base": "~1.2.5" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/@sinclair/typebox": { + "version": "0.27.8", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.27.8.tgz", + "integrity": "sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@sinonjs/commons": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-3.0.1.tgz", + "integrity": "sha512-K3mCHKQ9sVh8o1C9cxkwxaOmXoAMlDxC1mYyHrjqOWEcBjYr76t96zL2zlj5dUGZ3HSw240X1qgH3Mjf1yJWpQ==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "type-detect": "4.0.8" + } + }, + "node_modules/@sinonjs/fake-timers": { + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-10.3.0.tgz", + "integrity": "sha512-V4BG07kuYSUkTCSBHG8G8TNhM+F19jXFWnQtzj+we8DrkpSBCee9Z3Ms8yiGer/dlmhe35/Xdgyo3/0rQKg7YA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@sinonjs/commons": "^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/@tsconfig/node10": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.12.tgz", + "integrity": "sha512-UCYBaeFvM11aU2y3YPZ//O5Rhj+xKyzy7mvcIoAjASbigy8mHMryP5cK7dgjlz2hWxh1g5pLw084E0a/wlUSFQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node12": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz", + "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node14": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz", + "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node16": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.4.tgz", + "integrity": "sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz", + "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz", + "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.2" + } + }, + "node_modules/@types/body-parser": { + "version": "1.19.6", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.6.tgz", + "integrity": "sha512-HLFeCYgz89uk22N5Qg3dvGvsv46B8GLvKKo1zKG4NybA8U2DiEO3w9lqGg29t/tfLRJpJ6iQxnVw4OnB7MoM9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "node_modules/@types/connect": { + "version": "3.4.38", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", + "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/cookiejar": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@types/cookiejar/-/cookiejar-2.1.5.tgz", + "integrity": "sha512-he+DHOWReW0nghN24E1WUqM0efK4kI9oTqDm6XmK8ZPe2djZ90BSNdGnIyCLzCPw7/pogPlGbzI2wHGGmi4O/Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/eslint": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-9.6.1.tgz", + "integrity": "sha512-FXx2pKgId/WyYo2jXw63kk7/+TY7u7AziEJxJAnSFzHlqTAS3Ync6SvgYAN/k4/PQpnnVuzoMuVnByKK2qp0ag==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@types/estree": "*", + "@types/json-schema": "*" + } + }, + "node_modules/@types/eslint-scope": { + "version": "3.7.7", + "resolved": "https://registry.npmjs.org/@types/eslint-scope/-/eslint-scope-3.7.7.tgz", + "integrity": "sha512-MzMFlSLBqNF2gcHWO0G1vP/YQyfvrxZ0bF+u7mzUdZ1/xK4A4sru+nraZz5i3iEIk1l1uyicaDVTB4QbbEkAYg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/eslint": "*", + "@types/estree": "*" + } + }, + "node_modules/@types/estree": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", + "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/express": { + "version": "4.17.25", + "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.25.tgz", + "integrity": "sha512-dVd04UKsfpINUnK0yBoYHDF3xu7xVH4BuDotC/xGuycx4CgbP48X/KF/586bcObxT0HENHXEU8Nqtu6NR+eKhw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "^4.17.33", + "@types/qs": "*", + "@types/serve-static": "^1" + } + }, + "node_modules/@types/express-serve-static-core": { + "version": "4.19.7", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.19.7.tgz", + "integrity": "sha512-FvPtiIf1LfhzsaIXhv/PHan/2FeQBbtBDtfX2QfvPxdUelMDEckK08SM6nqo1MIZY3RUlfA+HV8+hFUSio78qg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*", + "@types/send": "*" + } + }, + "node_modules/@types/graceful-fs": { + "version": "4.1.9", + "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.9.tgz", + "integrity": "sha512-olP3sd1qOEe5dXTSaFvQG+02VdRXcdytWLAZsAq1PecU8uqQAhkrnbli7DagjtXKW/Bl7YJbUsa8MPcuc8LHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/http-errors": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.5.tgz", + "integrity": "sha512-r8Tayk8HJnX0FztbZN7oVqGccWgw98T/0neJphO91KkmOzug1KkofZURD4UaD5uH8AqcFLfdPErnBod0u71/qg==", + "dev": true, + "license": "MIT" + }, + "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": "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/@types/jest": { + "version": "29.5.14", + "resolved": "https://registry.npmjs.org/@types/jest/-/jest-29.5.14.tgz", + "integrity": "sha512-ZN+4sdnLUbo8EVvVc2ao0GFW6oVrQRPn4K2lglySj7APvSrgzxHiNNK99us4WDMi57xxA2yggblIAMNhXOotLQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "expect": "^29.0.0", + "pretty-format": "^29.0.0" + } + }, + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/jsonwebtoken": { + "version": "9.0.10", + "resolved": "https://registry.npmjs.org/@types/jsonwebtoken/-/jsonwebtoken-9.0.10.tgz", + "integrity": "sha512-asx5hIG9Qmf/1oStypjanR7iKTv0gXQ1Ov/jfrX6kS/EO0OFni8orbmGCn0672NHR3kXHwpAwR+B368ZGN/2rA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/ms": "*", + "@types/node": "*" + } + }, + "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==", + "license": "MIT" + }, + "node_modules/@types/methods": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@types/methods/-/methods-1.1.4.tgz", + "integrity": "sha512-ymXWVrDiCxTBE3+RIrrP533E70eA+9qu7zdWoHuOmGujkYtzf4HQF96b8nwHLqhuf4ykX61IGRIB38CC6/sImQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/mime": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.5.tgz", + "integrity": "sha512-/pyBZWSLD2n0dcHE3hq8s8ZvcETHtEuF+3E7XVt0Ig2nvsVQXdghHVcEkIWjy9A0wKfTn97a/PSDYohKIlnP/w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/ms": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@types/ms/-/ms-2.1.0.tgz", + "integrity": "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "20.19.25", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.25.tgz", + "integrity": "sha512-ZsJzA5thDQMSQO788d7IocwwQbI8B5OPzmqNvpf3NY/+MHDAS759Wo0gd2WQeXYt5AAAQjzcrTVC6SKCuYgoCQ==", + "license": "MIT", + "peer": true, + "dependencies": { + "undici-types": "~6.21.0" + } + }, + "node_modules/@types/passport": { + "version": "1.0.17", + "resolved": "https://registry.npmjs.org/@types/passport/-/passport-1.0.17.tgz", + "integrity": "sha512-aciLyx+wDwT2t2/kJGJR2AEeBz0nJU4WuRX04Wu9Dqc5lSUtwu0WERPHYsLhF9PtseiAMPBGNUOtFjxZ56prsg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/express": "*" + } + }, + "node_modules/@types/passport-jwt": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/@types/passport-jwt/-/passport-jwt-4.0.1.tgz", + "integrity": "sha512-Y0Ykz6nWP4jpxgEUYq8NoVZeCQPo1ZndJLfapI249g1jHChvRfZRO/LS3tqu26YgAS/laI1qx98sYGz0IalRXQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/jsonwebtoken": "*", + "@types/passport-strategy": "*" + } + }, + "node_modules/@types/passport-strategy": { + "version": "0.2.38", + "resolved": "https://registry.npmjs.org/@types/passport-strategy/-/passport-strategy-0.2.38.tgz", + "integrity": "sha512-GC6eMqqojOooq993Tmnmp7AUTbbQSgilyvpCYQjT+H6JfG/g6RGc7nXEniZlp0zyKJ0WUdOiZWLBZft9Yug1uA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/express": "*", + "@types/passport": "*" + } + }, + "node_modules/@types/qs": { + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.14.0.tgz", + "integrity": "sha512-eOunJqu0K1923aExK6y8p6fsihYEn/BYuQ4g0CxAAgFc4b/ZLN4CrsRZ55srTdqoiLzU2B2evC+apEIxprEzkQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/range-parser": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.7.tgz", + "integrity": "sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-FmgJfu+MOcQ370SD0ev7EI8TlCAfKYU+B4m5T3yXc1CiRN94g/SZPtsCkk506aUDtlMnFZvasDwHHUcZUEaYuA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/send": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@types/send/-/send-1.2.1.tgz", + "integrity": "sha512-arsCikDvlU99zl1g69TcAB3mzZPpxgw0UQnaHeC1Nwb015xp8bknZv5rIfri9xTOcMuaVgvabfIRA7PSZVuZIQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/serve-static": { + "version": "1.15.10", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.10.tgz", + "integrity": "sha512-tRs1dB+g8Itk72rlSI2ZrW6vZg0YrLI81iQSTkMmOqnqCaNr/8Ek4VwWcN5vZgCYWbg/JJSGBlUaYGAOP73qBw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/http-errors": "*", + "@types/node": "*", + "@types/send": "<1" + } + }, + "node_modules/@types/serve-static/node_modules/@types/send": { + "version": "0.17.6", + "resolved": "https://registry.npmjs.org/@types/send/-/send-0.17.6.tgz", + "integrity": "sha512-Uqt8rPBE8SY0RK8JB1EzVOIZ32uqy8HwdxCnoCOsYrvnswqmFZ/k+9Ikidlk/ImhsdvBsloHbAlewb2IEBV/Og==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/mime": "^1", + "@types/node": "*" + } + }, + "node_modules/@types/stack-utils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.3.tgz", + "integrity": "sha512-9aEbYZ3TbYMznPdcdr3SmIrLXwC/AKZXQeCf9Pgao5CKb8CyHuEX5jzWPTkvregvhRJHcpRO6BFoGW9ycaOkYw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/superagent": { + "version": "8.1.9", + "resolved": "https://registry.npmjs.org/@types/superagent/-/superagent-8.1.9.tgz", + "integrity": "sha512-pTVjI73witn+9ILmoJdajHGW2jkSaOzhiFYF1Rd3EQ94kymLqB9PjD9ISg7WaALC7+dCHT0FGe9T2LktLq/3GQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/cookiejar": "^2.1.5", + "@types/methods": "^1.1.4", + "@types/node": "*", + "form-data": "^4.0.0" + } + }, + "node_modules/@types/supertest": { + "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": { + "@types/methods": "^1.1.4", + "@types/superagent": "^8.1.0" + } + }, + "node_modules/@types/uuid": { + "version": "9.0.8", + "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-9.0.8.tgz", + "integrity": "sha512-jg+97EGIcY9AGHJJRaaPVgetKDsrTgbRjQ5Msgjh/DQKEFl0DtyRr/VCOyD1T2R1MNeWPK/u7JoGhlDZnKBAfA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/validator": { + "version": "13.15.10", + "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.15.10.tgz", + "integrity": "sha512-T8L6i7wCuyoK8A/ZeLYt1+q0ty3Zb9+qbSSvrIVitzT3YjZqkTZ40IbRsPanlB4h1QB3JVL1SYCdR6ngtFYcuA==", + "license": "MIT" + }, + "node_modules/@types/yargs": { + "version": "17.0.35", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.35.tgz", + "integrity": "sha512-qUHkeCyQFxMXg79wQfTtfndEC+N9ZZg76HJftDJp+qH2tV7Gj4OJi7l+PiWwJ+pWtW8GwSmqsDj/oymhrTWXjg==", + "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": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-6.21.0.tgz", + "integrity": "sha512-oy9+hTPCUFpngkEZUSzbf9MxI65wbKFoQYsgPdILTfbUldp5ovUuphZVe4i30emU9M/kP+T64Di0mxl7dSw3MA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/regexpp": "^4.5.1", + "@typescript-eslint/scope-manager": "6.21.0", + "@typescript-eslint/type-utils": "6.21.0", + "@typescript-eslint/utils": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", + "debug": "^4.3.4", + "graphemer": "^1.4.0", + "ignore": "^5.2.4", + "natural-compare": "^1.4.0", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^6.0.0 || ^6.0.0-alpha", + "eslint": "^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-6.21.0.tgz", + "integrity": "sha512-tbsV1jPne5CkFQCgPBcDOt30ItF7aJoZL997JSF7MhGQqOeT3svWRYxiqlfA5RUdlHN6Fi+EI9bxqbdyAUZjYQ==", + "dev": true, + "license": "BSD-2-Clause", + "peer": true, + "dependencies": { + "@typescript-eslint/scope-manager": "6.21.0", + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/typescript-estree": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/scope-manager": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-6.21.0.tgz", + "integrity": "sha512-OwLUIWZJry80O99zvqXVEioyniJMa+d2GrqpUTqi5/v5D5rOrppJVBPa0yKCblcigC0/aYAzxxqQ1B+DS2RYsg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-6.21.0.tgz", + "integrity": "sha512-rZQI7wHfao8qMX3Rd3xqeYSMCL3SoiSQLBATSiVKARdFGCYSRvmViieZjqc58jKgs8Y8i9YvVVhRbHSTA4VBag==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/typescript-estree": "6.21.0", + "@typescript-eslint/utils": "6.21.0", + "debug": "^4.3.4", + "ts-api-utils": "^1.0.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/types": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-6.21.0.tgz", + "integrity": "sha512-1kFmZ1rOm5epu9NZEZm1kckCDGj5UJEf7P1kliH4LKu/RkwpsfqqGmY2OOcUs18lSlQBKLDYBOGxRVtrMN5lpg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-6.21.0.tgz", + "integrity": "sha512-6npJTkZcO+y2/kr+z0hc4HwNfrrP4kNYh57ek7yCNlrBjWQ1Y0OS7jiZTkgumrvkX5HkEKXFZkkdFNkaW2wmUQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/visitor-keys": "6.21.0", + "debug": "^4.3.4", + "globby": "^11.1.0", + "is-glob": "^4.0.3", + "minimatch": "9.0.3", + "semver": "^7.5.4", + "ts-api-utils": "^1.0.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-6.21.0.tgz", + "integrity": "sha512-NfWVaC8HP9T8cbKQxHcsJBY5YE1O33+jpMwN45qzWWaPDZgLIbo12toGMWnmhvCpd3sIxkpDw3Wv1B3dYrbDQQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@types/json-schema": "^7.0.12", + "@types/semver": "^7.5.0", + "@typescript-eslint/scope-manager": "6.21.0", + "@typescript-eslint/types": "6.21.0", + "@typescript-eslint/typescript-estree": "6.21.0", + "semver": "^7.5.4" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^7.0.0 || ^8.0.0" + } + }, + "node_modules/@typescript-eslint/visitor-keys": { + "version": "6.21.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-6.21.0.tgz", + "integrity": "sha512-JJtkDduxLi9bivAB+cYOVMtbkqdPOhZ+ZI5LC47MIRrDV4Yn2o+ZnW10Nkmr28xRpSpdJ6Sm42Hjf2+REYXm0A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "6.21.0", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^16.0.0 || >=18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@ungap/structured-clone": { + "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/@webassemblyjs/ast": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.14.1.tgz", + "integrity": "sha512-nuBEDgQfm1ccRp/8bCQrx1frohyufl4JlbMMZ4P1wpeOfDhF6FQkxZJ1b/e+PLwr6X1Nhw6OLme5usuBWYBvuQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@webassemblyjs/helper-numbers": "1.13.2", + "@webassemblyjs/helper-wasm-bytecode": "1.13.2" + } + }, + "node_modules/@webassemblyjs/floating-point-hex-parser": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.13.2.tgz", + "integrity": "sha512-6oXyTOzbKxGH4steLbLNOu71Oj+C8Lg34n6CqRvqfS2O71BxY6ByfMDRhBytzknj9yGUPVJ1qIKhRlAwO1AovA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@webassemblyjs/helper-api-error": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.13.2.tgz", + "integrity": "sha512-U56GMYxy4ZQCbDZd6JuvvNV/WFildOjsaWD3Tzzvmw/mas3cXzRJPMjP83JqEsgSbyrmaGjBfDtV7KDXV9UzFQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@webassemblyjs/helper-buffer": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.14.1.tgz", + "integrity": "sha512-jyH7wtcHiKssDtFPRB+iQdxlDf96m0E39yb0k5uJVhFGleZFoNw1c4aeIcVUPPbXUVJ94wwnMOAqUHyzoEPVMA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@webassemblyjs/helper-numbers": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-numbers/-/helper-numbers-1.13.2.tgz", + "integrity": "sha512-FE8aCmS5Q6eQYcV3gI35O4J789wlQA+7JrqTTpJqn5emA4U2hvwJmvFRC0HODS+3Ye6WioDklgd6scJ3+PLnEA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@webassemblyjs/floating-point-hex-parser": "1.13.2", + "@webassemblyjs/helper-api-error": "1.13.2", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/helper-wasm-bytecode": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.13.2.tgz", + "integrity": "sha512-3QbLKy93F0EAIXLh0ogEVR6rOubA9AoZ+WRYhNbFyuB70j3dRdwH9g+qXhLAO0kiYGlg3TxDV+I4rQTr/YNXkA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@webassemblyjs/helper-wasm-section": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.14.1.tgz", + "integrity": "sha512-ds5mXEqTJ6oxRoqjhWDU83OgzAYjwsCV8Lo/N+oRsNDmx/ZDpqalmrtgOMkHwxsG0iI//3BwWAErYRHtgn0dZw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@webassemblyjs/ast": "1.14.1", + "@webassemblyjs/helper-buffer": "1.14.1", + "@webassemblyjs/helper-wasm-bytecode": "1.13.2", + "@webassemblyjs/wasm-gen": "1.14.1" + } + }, + "node_modules/@webassemblyjs/ieee754": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.13.2.tgz", + "integrity": "sha512-4LtOzh58S/5lX4ITKxnAK2USuNEvpdVV9AlgGQb8rJDHaLeHciwG4zlGr0j/SNWlr7x3vO1lDEsuePvtcDNCkw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@xtuc/ieee754": "^1.2.0" + } + }, + "node_modules/@webassemblyjs/leb128": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.13.2.tgz", + "integrity": "sha512-Lde1oNoIdzVzdkNEAWZ1dZ5orIbff80YPdHx20mrHwHrVNNTjNr8E3xz9BdpcGqRQbAEa+fkrCb+fRFTl/6sQw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/utf8": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.13.2.tgz", + "integrity": "sha512-3NQWGjKTASY1xV5m7Hr0iPeXD9+RDobLll3T9d2AO+g3my8xy5peVyjSag4I50mR1bBSN/Ct12lo+R9tJk0NZQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@webassemblyjs/wasm-edit": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.14.1.tgz", + "integrity": "sha512-RNJUIQH/J8iA/1NzlE4N7KtyZNHi3w7at7hDjvRNm5rcUXa00z1vRz3glZoULfJ5mpvYhLybmVcwcjGrC1pRrQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@webassemblyjs/ast": "1.14.1", + "@webassemblyjs/helper-buffer": "1.14.1", + "@webassemblyjs/helper-wasm-bytecode": "1.13.2", + "@webassemblyjs/helper-wasm-section": "1.14.1", + "@webassemblyjs/wasm-gen": "1.14.1", + "@webassemblyjs/wasm-opt": "1.14.1", + "@webassemblyjs/wasm-parser": "1.14.1", + "@webassemblyjs/wast-printer": "1.14.1" + } + }, + "node_modules/@webassemblyjs/wasm-gen": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.14.1.tgz", + "integrity": "sha512-AmomSIjP8ZbfGQhumkNvgC33AY7qtMCXnN6bL2u2Js4gVCg8fp735aEiMSBbDR7UQIj90n4wKAFUSEd0QN2Ukg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@webassemblyjs/ast": "1.14.1", + "@webassemblyjs/helper-wasm-bytecode": "1.13.2", + "@webassemblyjs/ieee754": "1.13.2", + "@webassemblyjs/leb128": "1.13.2", + "@webassemblyjs/utf8": "1.13.2" + } + }, + "node_modules/@webassemblyjs/wasm-opt": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.14.1.tgz", + "integrity": "sha512-PTcKLUNvBqnY2U6E5bdOQcSM+oVP/PmrDY9NzowJjislEjwP/C4an2303MCVS2Mg9d3AJpIGdUFIQQWbPds0Sw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@webassemblyjs/ast": "1.14.1", + "@webassemblyjs/helper-buffer": "1.14.1", + "@webassemblyjs/wasm-gen": "1.14.1", + "@webassemblyjs/wasm-parser": "1.14.1" + } + }, + "node_modules/@webassemblyjs/wasm-parser": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.14.1.tgz", + "integrity": "sha512-JLBl+KZ0R5qB7mCnud/yyX08jWFw5MsoalJ1pQ4EdFlgj9VdXKGuENGsiCIjegI1W7p91rUlcB/LB5yRJKNTcQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@webassemblyjs/ast": "1.14.1", + "@webassemblyjs/helper-api-error": "1.13.2", + "@webassemblyjs/helper-wasm-bytecode": "1.13.2", + "@webassemblyjs/ieee754": "1.13.2", + "@webassemblyjs/leb128": "1.13.2", + "@webassemblyjs/utf8": "1.13.2" + } + }, + "node_modules/@webassemblyjs/wast-printer": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.14.1.tgz", + "integrity": "sha512-kPSSXE6De1XOR820C90RIo2ogvZG+c3KiHzqUoO/F34Y2shGzesfqv7o57xrxovZJH/MetF5UjroJ/R/3isoiw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@webassemblyjs/ast": "1.14.1", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@xtuc/ieee754": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz", + "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/@xtuc/long": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@xtuc/long/-/long-4.2.2.tgz", + "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==", + "dev": true, + "license": "Apache-2.0" + }, + "node_modules/accepts": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", + "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", + "license": "MIT", + "dependencies": { + "mime-types": "~2.1.34", + "negotiator": "0.6.3" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/acorn": { + "version": "8.15.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", + "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-import-phases": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/acorn-import-phases/-/acorn-import-phases-1.0.4.tgz", + "integrity": "sha512-wKmbr/DDiIXzEOiWrTTUcDm24kQ2vGfZQvM2fwg2vXqR5uW6aapr7ObPtj1th32b9u90/Pf4AItvdTh42fBmVQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10.13.0" + }, + "peerDependencies": { + "acorn": "^8.14.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/acorn-walk": { + "version": "8.3.4", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.4.tgz", + "integrity": "sha512-ueEepnujpqee2o5aIYnvHU6C0A42MNdsIDeqy5BydrkuC5R1ZuUFnm27EeFJGoEHJQgn3uleRvmTXaJgfXbt4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "acorn": "^8.11.0" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/aes-js": { + "version": "4.0.0-beta.5", + "resolved": "https://registry.npmjs.org/aes-js/-/aes-js-4.0.0-beta.5.tgz", + "integrity": "sha512-G965FqalsNyrPqgEGON7nIx1e/OVENSgiEIzyC63haUMuvNnwIgIjMs52hlTCKhkBny7A2ORNlfY9Zu+jmGk1Q==", + "license": "MIT" + }, + "node_modules/ajv": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-8.12.0.tgz", + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "fast-deep-equal": "^3.1.1", + "json-schema-traverse": "^1.0.0", + "require-from-string": "^2.0.2", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ajv-formats": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ajv-formats/-/ajv-formats-2.1.1.tgz", + "integrity": "sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "^8.0.0" + }, + "peerDependencies": { + "ajv": "^8.0.0" + }, + "peerDependenciesMeta": { + "ajv": { + "optional": true + } + } + }, + "node_modules/ajv-keywords": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-5.1.0.tgz", + "integrity": "sha512-YCS/JNFAUyr5vAuhk1DWm1CBxRHW9LbJ2ozWeemrIqpbsqKjHVxYPyi5GC0rjZIT5JxJ3virVTS8wk4i/Z+krw==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-deep-equal": "^3.1.3" + }, + "peerDependencies": { + "ajv": "^8.8.2" + } + }, + "node_modules/ansi-colors": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.3.tgz", + "integrity": "sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-escapes/node_modules/type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/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/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dev": true, + "license": "ISC", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/anymatch/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/append-field": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/append-field/-/append-field-1.0.0.tgz", + "integrity": "sha512-klpgFSWLW1ZEs8svjfb7g4qWY0YS5imI82dTg+QahUvJ8YqAY0P10Uk8tTyh9ZGuYEZEMaeJYCF5BFuX552hsw==", + "license": "MIT" + }, + "node_modules/arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true, + "license": "MIT" + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "license": "Python-2.0" + }, + "node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==", + "license": "MIT" + }, + "node_modules/array-timsort": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/array-timsort/-/array-timsort-1.0.3.tgz", + "integrity": "sha512-/+3GRL7dDAGEfM6TseQk/U+mi18TU2Ms9I3UlLdUMhz2hbvGNTKdj9xniwXfUqgYhHxRx0+8UnKkvlNwVU+cWQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/asap": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", + "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==", + "dev": true, + "license": "MIT" + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", + "license": "MIT" + }, + "node_modules/axios": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/axios/-/axios-1.13.2.tgz", + "integrity": "sha512-VPk9ebNqPcy5lRGuSlKx752IlDatOjT9paPlm8A7yOuW2Fbvp4X3JznJtT4f0GzGLLiWE9W8onz51SqLYwzGaA==", + "license": "MIT", + "peer": true, + "dependencies": { + "follow-redirects": "^1.15.6", + "form-data": "^4.0.4", + "proxy-from-env": "^1.1.0" + } + }, + "node_modules/babel-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-29.7.0.tgz", + "integrity": "sha512-BrvGY3xZSwEcCzKvKsCi2GgHqDqsYkOP4/by5xCgIwGXQxIEh+8ew3gmrE1y7XRR6LHZIj6yLYnUi/mm2KXKBg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/transform": "^29.7.0", + "@types/babel__core": "^7.1.14", + "babel-plugin-istanbul": "^6.1.1", + "babel-preset-jest": "^29.6.3", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.8.0" + } + }, + "node_modules/babel-plugin-istanbul": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", + "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-instrument": "^5.0.4", + "test-exclude": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-istanbul/node_modules/istanbul-lib-instrument": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.2.1.tgz", + "integrity": "sha512-pzqtp31nLv/XFOzXGuvhCb8qhjmTVo5vjVk19XE4CRlSWz0KoeJ3bw9XsA7nOp9YBf4qHjwBxkDzKcME/J29Yg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.12.3", + "@babel/parser": "^7.14.7", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-istanbul/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-jest-hoist": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-29.6.3.tgz", + "integrity": "sha512-ESAc/RJvGTFEzRwOTT4+lNDk/GNHMkKbNzsvT0qKRfDyyYTskxB5rnU2njIDYVxXCBHHEI1c0YwHob3WaYujOg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.3.3", + "@babel/types": "^7.3.3", + "@types/babel__core": "^7.1.14", + "@types/babel__traverse": "^7.0.6" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/babel-preset-current-node-syntax": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.2.0.tgz", + "integrity": "sha512-E/VlAEzRrsLEb2+dv8yp3bo4scof3l9nR4lrld+Iy5NyVqgVYUJnDAmunkhPMisRI32Qc4iRiz425d8vM++2fg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.12.13", + "@babel/plugin-syntax-class-static-block": "^7.14.5", + "@babel/plugin-syntax-import-attributes": "^7.24.7", + "@babel/plugin-syntax-import-meta": "^7.10.4", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-private-property-in-object": "^7.14.5", + "@babel/plugin-syntax-top-level-await": "^7.14.5" + }, + "peerDependencies": { + "@babel/core": "^7.0.0 || ^8.0.0-0" + } + }, + "node_modules/babel-preset-jest": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-29.6.3.tgz", + "integrity": "sha512-0B3bhxR6snWXJZtR/RliHTDPRgn1sNHOR0yVtq/IiQFyuOVjFS+wuio/R4gSNkyYmKmJB4wGZv2NZanmKmTnNA==", + "dev": true, + "license": "MIT", + "dependencies": { + "babel-plugin-jest-hoist": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/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/baseline-browser-mapping": { + "version": "2.8.32", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.8.32.tgz", + "integrity": "sha512-OPz5aBThlyLFgxyhdwf/s2+8ab3OvT7AdTNvKHBwpXomIYeXqpUUuT8LrdtxZSsWJ4R4CU1un4XGh5Ez3nlTpw==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.js" + } + }, + "node_modules/bech32": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/bech32/-/bech32-2.0.0.tgz", + "integrity": "sha512-LcknSilhIGatDAsY1ak2I8VtGaHNhgMSYVxFrGLXv+xLHytaKZKcaUJJUE7qmBr7h33o5YQwP55pMI0xmkpJwg==", + "license": "MIT" + }, + "node_modules/binary-extensions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", + "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/bl": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, + "node_modules/body-parser": { + "version": "1.20.3", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz", + "integrity": "sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==", + "license": "MIT", + "dependencies": { + "bytes": "3.1.2", + "content-type": "~1.0.5", + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "on-finished": "2.4.1", + "qs": "6.13.0", + "raw-body": "2.5.2", + "type-is": "~1.6.18", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/body-parser/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "license": "MIT", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/body-parser/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", + "license": "MIT" + }, + "node_modules/brace-expansion": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", + "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/browserslist": { + "version": "4.28.0", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.0.tgz", + "integrity": "sha512-tbydkR/CxfMwelN0vwdP/pLkDwyAASZ+VfWm4EOwlB6SWhx1sYnWLqo8N5j0rAzPfzfRaxt0mM/4wPU/Su84RQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "peer": true, + "dependencies": { + "baseline-browser-mapping": "^2.8.25", + "caniuse-lite": "^1.0.30001754", + "electron-to-chromium": "^1.5.249", + "node-releases": "^2.0.27", + "update-browserslist-db": "^1.1.4" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/bs-logger": { + "version": "0.2.6", + "resolved": "https://registry.npmjs.org/bs-logger/-/bs-logger-0.2.6.tgz", + "integrity": "sha512-pd8DCoxmbgc7hyPKOvxtqNcjYoOsABPQdcCUjGp3d42VR2CX1ORhk2A87oqqu5R1kk+76nsxZupkmyd+MVtCog==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-json-stable-stringify": "2.x" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/bser": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", + "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "node-int64": "^0.4.0" + } + }, + "node_modules/buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "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.1.13" + } + }, + "node_modules/buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==", + "license": "BSD-3-Clause" + }, + "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==", + "license": "MIT" + }, + "node_modules/busboy": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz", + "integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==", + "dependencies": { + "streamsearch": "^1.1.0" + }, + "engines": { + "node": ">=10.16.0" + } + }, + "node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "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.2", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", + "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/call-bound": { + "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.2", + "get-intrinsic": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001757", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001757.tgz", + "integrity": "sha512-r0nnL/I28Zi/yjk1el6ilj27tKcdjLsNqAOZr0yVjWPrSQyHgKI2INaEWw21bAQSv2LXRt1XuCS/GomNpWOxsQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/char-regex": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", + "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true, + "license": "MIT" + }, + "node_modules/chokidar": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", + "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/chrome-trace-event": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.4.tgz", + "integrity": "sha512-rNjApaLzuwaOTjCiT8lSDdGN1APCiqkChLMJxJPWLunPAt5fy8xgU9/jNOchV84wfIxrA0lRQB7oCT8jrn/wrQ==", + "dev": true, + "license": "MIT", + "engines": { + "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/cjs-module-lexer": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.4.3.tgz", + "integrity": "sha512-9z8TZaGM1pfswYeXrUpzPrkx8UnWYdhJclsiYMm6x/w5+nN+8Tf/LnAgfLGQCm59qAOxU8WwHEq2vNwF6i4j+Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/class-transformer": { + "version": "0.5.1", + "resolved": "https://registry.npmjs.org/class-transformer/-/class-transformer-0.5.1.tgz", + "integrity": "sha512-SQa1Ws6hUbfC98vKGxZH3KFY0Y1lm5Zm0SY8XX9zbK7FJCyVEac3ATW0RIpwzW+oOfmHE5PMPufDG9hCfoEOMw==", + "license": "MIT", + "peer": true + }, + "node_modules/class-validator": { + "version": "0.14.3", + "resolved": "https://registry.npmjs.org/class-validator/-/class-validator-0.14.3.tgz", + "integrity": "sha512-rXXekcjofVN1LTOSw+u4u9WXVEUvNBVjORW154q/IdmYWy1nMbOU9aNtZB0t8m+FJQ9q91jlr2f9CwwUFdFMRA==", + "license": "MIT", + "peer": true, + "dependencies": { + "@types/validator": "^13.15.3", + "libphonenumber-js": "^1.11.1", + "validator": "^13.15.20" + } + }, + "node_modules/cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "license": "MIT", + "dependencies": { + "restore-cursor": "^3.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cli-spinners": { + "version": "2.9.2", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.9.2.tgz", + "integrity": "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/cli-table3": { + "version": "0.6.5", + "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.6.5.tgz", + "integrity": "sha512-+W/5efTR7y5HRD7gACw9yQjqMVvEMLBHmboM/kPWam+H+Hmyrgjh6YncVKK122YZkXrLudzTuAukUw9FnMf7IQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "string-width": "^4.2.0" + }, + "engines": { + "node": "10.* || >= 12.*" + }, + "optionalDependencies": { + "@colors/colors": "1.5.0" + } + }, + "node_modules/cli-width": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-3.0.0.tgz", + "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">= 10" + } + }, + "node_modules/cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/cliui/node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/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/cluster-key-slot": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/cluster-key-slot/-/cluster-key-slot-1.1.2.tgz", + "integrity": "sha512-RMr0FhtfXemyinomL4hrWcYJxmX6deFdCxpJzhDttxgO1+bcCnkk+9drydLVDmAMG7NE6aN/fl4F7ucU/90gAA==", + "license": "Apache-2.0", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/collect-v8-coverage": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.3.tgz", + "integrity": "sha512-1L5aqIkwPfiodaMgQunkF1zRhNqifHBmtbbbxcr6yVxxBnliw4TDOW6NxpO8DJLgJ16OT+Y4ztZqP6p/FtXnAw==", + "dev": true, + "license": "MIT" + }, + "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/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/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "license": "MIT", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/commander": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz", + "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 6" + } + }, + "node_modules/comment-json": { + "version": "4.2.5", + "resolved": "https://registry.npmjs.org/comment-json/-/comment-json-4.2.5.tgz", + "integrity": "sha512-bKw/r35jR3HGt5PEPm1ljsQQGyCrR8sFGNiN5L+ykDHdpO8Smxkrkla9Yi6NkQyUrb8V54PGhfMs6NrIwtxtdw==", + "dev": true, + "license": "MIT", + "dependencies": { + "array-timsort": "^1.0.3", + "core-util-is": "^1.0.3", + "esprima": "^4.0.1", + "has-own-prop": "^2.0.0", + "repeat-string": "^1.6.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/component-emitter": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.1.tgz", + "integrity": "sha512-T0+barUSQRTUQASh8bx02dl+DhF54GtIDY13Y3m9oWTklKbb3Wv974meRpeZ3lp1JpLVECWWNHC4vaG2XHXouQ==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true, + "license": "MIT" + }, + "node_modules/concat-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-2.0.0.tgz", + "integrity": "sha512-MWufYdFw53ccGjCA+Ol7XJYpAlW6/prSMzuPOTRnJGcGzuhLn4Scrz7qf6o8bROZ514ltazcIFJZevcfbo0x7A==", + "engines": [ + "node >= 6.0" + ], + "license": "MIT", + "dependencies": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.0.2", + "typedarray": "^0.0.6" + } + }, + "node_modules/consola": { + "version": "2.15.3", + "resolved": "https://registry.npmjs.org/consola/-/consola-2.15.3.tgz", + "integrity": "sha512-9vAdYbHj6x2fLKC4+oPH0kFzY/orMZyG2Aj+kNylHxKGJ/Ed4dpNyAQYwJOdqO4zdM7XpVHmyejQDcQHrnuXbw==", + "license": "MIT" + }, + "node_modules/content-disposition": { + "version": "0.5.4", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", + "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", + "license": "MIT", + "dependencies": { + "safe-buffer": "5.2.1" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-type": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", + "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true, + "license": "MIT" + }, + "node_modules/cookie": { + "version": "0.7.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz", + "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==", + "license": "MIT" + }, + "node_modules/cookiejar": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.4.tgz", + "integrity": "sha512-LDx6oHrK+PhzLKJU9j5S7/Y3jM/mUHvD/DeI1WQmJn652iPC5Y4TBzC9l+5OMOXlyTTA+SmVUPm0HQUwpD5Jqw==", + "dev": true, + "license": "MIT" + }, + "node_modules/core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/cors": { + "version": "2.8.5", + "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", + "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", + "license": "MIT", + "dependencies": { + "object-assign": "^4", + "vary": "^1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/cosmiconfig": { + "version": "8.3.6", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-8.3.6.tgz", + "integrity": "sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==", + "dev": true, + "license": "MIT", + "dependencies": { + "import-fresh": "^3.3.0", + "js-yaml": "^4.1.0", + "parse-json": "^5.2.0", + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/d-fischer" + }, + "peerDependencies": { + "typescript": ">=4.9.5" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/create-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", + "integrity": "sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "prompts": "^2.0.1" + }, + "bin": { + "create-jest": "bin/create-jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/create-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/cron": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/cron/-/cron-3.2.1.tgz", + "integrity": "sha512-w2n5l49GMmmkBFEsH9FIDhjZ1n1QgTMOCMGuQtOXs5veNiosZmso6bQGuqOJSYAXXrG84WQFVneNk+Yt0Ua9iw==", + "license": "MIT", + "dependencies": { + "@types/luxon": "~3.4.0", + "luxon": "~3.5.0" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/dedent": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.7.0.tgz", + "integrity": "sha512-HGFtf8yhuhGhqO07SV79tRp+br4MnbdjeVxotpn1QBl30pcLLCQjX5b2295ll0fv8RKDKsmWYrl05usHM9CewQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "babel-plugin-macros": "^3.1.0" + }, + "peerDependenciesMeta": { + "babel-plugin-macros": { + "optional": true + } + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", + "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "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/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/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "license": "MIT", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/denque": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/denque/-/denque-2.1.0.tgz", + "integrity": "sha512-HVQE3AAb/pxF8fQAoiqpvg9i3evqug3hoiwakOyZAwJm+6vZehbkYXZ0l4JxS+I3QxM97v5aaRNhj8v5oBhekw==", + "license": "Apache-2.0", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/destroy": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", + "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", + "license": "MIT", + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/detect-newline": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", + "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/dezalgo": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/dezalgo/-/dezalgo-1.0.4.tgz", + "integrity": "sha512-rXSP0bf+5n0Qonsb+SVVfNfIsimO4HEtmnIpPHY8Q1UCzKlQrDMfdobr8nJOOsRgWCyMRqeSBQzmWUMq7zvVig==", + "dev": true, + "license": "ISC", + "dependencies": { + "asap": "^2.0.0", + "wrappy": "1" + } + }, + "node_modules/diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/diff-sequences": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", + "integrity": "sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/dotenv": { + "version": "16.4.5", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.5.tgz", + "integrity": "sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://dotenvx.com" + } + }, + "node_modules/dotenv-expand": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/dotenv-expand/-/dotenv-expand-10.0.0.tgz", + "integrity": "sha512-GopVGCpVS1UKH75VKHGuQFqS1Gusej0z4FyQkPdwjil2gNIv+LNsqBlboOzpJFZKVT95GkCyWJbBSdFEFUWI2A==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + } + }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-errors": "^1.3.0", + "gopd": "^1.2.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/eastasianwidth": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", + "dev": true, + "license": "MIT" + }, + "node_modules/ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "license": "Apache-2.0", + "dependencies": { + "safe-buffer": "^5.0.1" + } + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", + "license": "MIT" + }, + "node_modules/electron-to-chromium": { + "version": "1.5.262", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.262.tgz", + "integrity": "sha512-NlAsMteRHek05jRUxUR0a5jpjYq9ykk6+kO0yRaMi5moe7u0fVIOeQ3Y30A8dIiWFBNUoQGi1ljb1i5VtS9WQQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/emittery": { + "version": "0.13.1", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.13.1.tgz", + "integrity": "sha512-DeWwawk6r5yR9jFgnDKYt4sLS0LmHJJi3ZOnb5/JdbYwj3nW+FxQnHIjhBKz8YLC7oRNPVM9NQ47I3CVx34eqQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sindresorhus/emittery?sponsor=1" + } + }, + "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/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/enhanced-resolve": { + "version": "5.18.3", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.18.3.tgz", + "integrity": "sha512-d4lC8xfavMeBjzGr2vECC3fsGXziXZQyJxD868h2M/mBI3PwAuODxAkLkq5HYuvrPYcUtiLzsTo8U3PgX3Ocww==", + "dev": true, + "license": "MIT", + "dependencies": { + "graceful-fs": "^4.2.4", + "tapable": "^2.2.0" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/error-ex": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.4.tgz", + "integrity": "sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/es-define-property": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", + "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-module-lexer": { + "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==", + "dev": true, + "license": "MIT" + }, + "node_modules/es-object-atoms": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", + "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-set-tostringtag": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", + "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.6", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", + "license": "MIT" + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "8.57.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.1.tgz", + "integrity": "sha512-ypowyDxpVSYpkXr9WPv2PAZCtNip1Mv5KTW0SCurXv/9iOpcrH9PaqUElksqEB6pChqHGDRCFTyrZlGhnLNGiA==", + "deprecated": "This version is no longer supported. Please see https://eslint.org/version-support for other options.", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.6.1", + "@eslint/eslintrc": "^2.1.4", + "@eslint/js": "8.57.1", + "@humanwhocodes/config-array": "^0.13.0", + "@humanwhocodes/module-importer": "^1.0.1", + "@nodelib/fs.walk": "^1.2.8", + "@ungap/structured-clone": "^1.2.0", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "doctrine": "^3.0.0", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^7.2.2", + "eslint-visitor-keys": "^3.4.3", + "espree": "^9.6.1", + "esquery": "^1.4.2", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "globals": "^13.19.0", + "graphemer": "^1.4.0", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "is-path-inside": "^3.0.3", + "js-yaml": "^4.1.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.4.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3", + "strip-ansi": "^6.0.1", + "text-table": "^0.2.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-config-prettier": { + "version": "9.1.2", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-9.1.2.tgz", + "integrity": "sha512-iI1f+D2ViGn+uvv5HuHVUamg8ll4tN+JRHGc6IJi4TP9Kl976C57fzPXgseXNs8v0iA8aSJpHsTWjDb9QJamGQ==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "eslint-config-prettier": "bin/cli.js" + }, + "peerDependencies": { + "eslint": ">=7.0.0" + } + }, + "node_modules/eslint-plugin-prettier": { + "version": "5.5.4", + "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-5.5.4.tgz", + "integrity": "sha512-swNtI95SToIz05YINMA6Ox5R057IMAmWZ26GqPxusAp1TZzj+IdY9tXNWWD3vkF/wEqydCONcwjTFpxybBqZsg==", + "dev": true, + "license": "MIT", + "dependencies": { + "prettier-linter-helpers": "^1.0.0", + "synckit": "^0.11.7" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint-plugin-prettier" + }, + "peerDependencies": { + "@types/eslint": ">=8.0.0", + "eslint": ">=8.0.0", + "eslint-config-prettier": ">= 7.0.0 <10.0.0 || >=10.1.0", + "prettier": ">=3.0.0" + }, + "peerDependenciesMeta": { + "@types/eslint": { + "optional": true + }, + "eslint-config-prettier": { + "optional": true + } + } + }, + "node_modules/eslint-scope": { + "version": "7.2.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.2.2.tgz", + "integrity": "sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint/node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/eslint/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/eslint/node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/eslint/node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true, + "license": "MIT" + }, + "node_modules/eslint/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/espree": { + "version": "9.6.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.6.1.tgz", + "integrity": "sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "acorn": "^8.9.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^3.4.1" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "license": "BSD-2-Clause", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/ethers": { + "version": "6.15.0", + "resolved": "https://registry.npmjs.org/ethers/-/ethers-6.15.0.tgz", + "integrity": "sha512-Kf/3ZW54L4UT0pZtsY/rf+EkBU7Qi5nnhonjUb8yTXcxH3cdcWrV2cRyk0Xk/4jK6OoHhxxZHriyhje20If2hQ==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/ethers-io/" + }, + { + "type": "individual", + "url": "https://www.buymeacoffee.com/ricmoo" + } + ], + "license": "MIT", + "dependencies": { + "@adraffy/ens-normalize": "1.10.1", + "@noble/curves": "1.2.0", + "@noble/hashes": "1.3.2", + "@types/node": "22.7.5", + "aes-js": "4.0.0-beta.5", + "tslib": "2.7.0", + "ws": "8.17.1" + }, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/ethers/node_modules/@noble/curves": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@noble/curves/-/curves-1.2.0.tgz", + "integrity": "sha512-oYclrNgRaM9SsBUBVbb8M6DTV7ZHRTKugureoYEncY5c65HOmRzvSiTE3y5CYaPYJA/GVkrhXEoF0M3Ya9PMnw==", + "license": "MIT", + "dependencies": { + "@noble/hashes": "1.3.2" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/ethers/node_modules/@noble/hashes": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.3.2.tgz", + "integrity": "sha512-MVC8EAQp7MvEcm30KWENFjgR+Mkmf+D189XJTkFIlwohU5hcBbn1ZkKq7KVTi2Hme3PMGF390DaL52beVrIihQ==", + "license": "MIT", + "engines": { + "node": ">= 16" + }, + "funding": { + "url": "https://paulmillr.com/funding/" + } + }, + "node_modules/ethers/node_modules/@types/node": { + "version": "22.7.5", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.7.5.tgz", + "integrity": "sha512-jML7s2NAzMWc//QSJ1a3prpk78cOPchGvXJsC3C6R6PSMoooztvRVQEz89gmBTBY1SPMaqo5teB4uNHPdetShQ==", + "license": "MIT", + "dependencies": { + "undici-types": "~6.19.2" + } + }, + "node_modules/ethers/node_modules/tslib": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.7.0.tgz", + "integrity": "sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA==", + "license": "0BSD" + }, + "node_modules/ethers/node_modules/undici-types": { + "version": "6.19.8", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.19.8.tgz", + "integrity": "sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw==", + "license": "MIT" + }, + "node_modules/events": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.x" + } + }, + "node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/execa?sponsor=1" + } + }, + "node_modules/execa/node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/expect-utils": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/express": { + "version": "4.21.2", + "resolved": "https://registry.npmjs.org/express/-/express-4.21.2.tgz", + "integrity": "sha512-28HqgMZAmih1Czt9ny7qr6ek2qddF4FclbMzwhCREB6OFfH+rXAnuNCwo1/wFvrtbgsQDb4kSbX9de9lFbrXnA==", + "license": "MIT", + "dependencies": { + "accepts": "~1.3.8", + "array-flatten": "1.1.1", + "body-parser": "1.20.3", + "content-disposition": "0.5.4", + "content-type": "~1.0.4", + "cookie": "0.7.1", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "2.0.0", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "1.3.1", + "fresh": "0.5.2", + "http-errors": "2.0.0", + "merge-descriptors": "1.0.3", + "methods": "~1.1.2", + "on-finished": "2.4.1", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.12", + "proxy-addr": "~2.0.7", + "qs": "6.13.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.2.1", + "send": "0.19.0", + "serve-static": "1.16.2", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/express/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "license": "MIT", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/express/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", + "license": "MIT" + }, + "node_modules/express/node_modules/path-to-regexp": { + "version": "0.1.12", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz", + "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==", + "license": "MIT" + }, + "node_modules/external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "license": "MIT", + "dependencies": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-diff": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.3.0.tgz", + "integrity": "sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==", + "dev": true, + "license": "Apache-2.0" + }, + "node_modules/fast-glob": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", + "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.8" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-safe-stringify": { + "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==", + "license": "MIT" + }, + "node_modules/fastq": { + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", + "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/fb-watchman": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.2.tgz", + "integrity": "sha512-p5161BqbuCaSnB8jIbzQHOlpgsPmK5rJVDfDKO91Axs5NC1uu3HRQm6wt9cd9/+GtQQIO53JdGXXoyDpTAsgYA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "bser": "2.1.1" + } + }, + "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": "3.2.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "license": "MIT", + "dependencies": { + "escape-string-regexp": "^1.0.5" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/figures/node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "license": "MIT", + "dependencies": { + "flat-cache": "^3.0.4" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/file-type": { + "version": "20.4.1", + "resolved": "https://registry.npmjs.org/file-type/-/file-type-20.4.1.tgz", + "integrity": "sha512-hw9gNZXUfZ02Jo0uafWLaFVPter5/k2rfcrjFJJHX/77xtSDOfJuEFb6oKlFV86FLP1SuyHMW1PSk0U9M5tKkQ==", + "license": "MIT", + "dependencies": { + "@tokenizer/inflate": "^0.2.6", + "strtok3": "^10.2.0", + "token-types": "^6.0.0", + "uint8array-extras": "^1.4.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sindresorhus/file-type?sponsor=1" + } + }, + "node_modules/fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dev": true, + "license": "MIT", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/finalhandler": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.1.tgz", + "integrity": "sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==", + "license": "MIT", + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "on-finished": "2.4.1", + "parseurl": "~1.3.3", + "statuses": "2.0.1", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/finalhandler/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "license": "MIT", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/finalhandler/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==", + "license": "MIT" + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "dev": true, + "license": "MIT", + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" + } + }, + "node_modules/flatted": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", + "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", + "dev": true, + "license": "ISC" + }, + "node_modules/follow-redirects": { + "version": "1.15.11", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.11.tgz", + "integrity": "sha512-deG2P0JfjrTxl50XGCDyfI97ZGVCxIpfKYmfyrQ54n5FO/0gfIES8C/Psl6kWVDolizcaaxZJnTS0QSMxvnsBQ==", + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "license": "MIT", + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/foreground-child": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.1.tgz", + "integrity": "sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==", + "dev": true, + "license": "ISC", + "dependencies": { + "cross-spawn": "^7.0.6", + "signal-exit": "^4.0.1" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/fork-ts-checker-webpack-plugin": { + "version": "9.0.2", + "resolved": "https://registry.npmjs.org/fork-ts-checker-webpack-plugin/-/fork-ts-checker-webpack-plugin-9.0.2.tgz", + "integrity": "sha512-Uochze2R8peoN1XqlSi/rGUkDQpRogtLFocP9+PGu68zk1BDAKXfdeCdyVZpgTk8V8WFVQXdEz426VKjXLO1Gg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.16.7", + "chalk": "^4.1.2", + "chokidar": "^3.5.3", + "cosmiconfig": "^8.2.0", + "deepmerge": "^4.2.2", + "fs-extra": "^10.0.0", + "memfs": "^3.4.1", + "minimatch": "^3.0.4", + "node-abort-controller": "^3.0.1", + "schema-utils": "^3.1.1", + "semver": "^7.3.5", + "tapable": "^2.2.1" + }, + "engines": { + "node": ">=12.13.0", + "yarn": ">=1.0.0" + }, + "peerDependencies": { + "typescript": ">3.6.0", + "webpack": "^5.11.0" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/fork-ts-checker-webpack-plugin/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/form-data": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.5.tgz", + "integrity": "sha512-8RipRLol37bNs2bhoV67fiTEvdTrbMUYcFTiy3+wuuOnUog2QBHCZWXDRijWQfAkhBj2Uf5UnVaiWwA5vdd82w==", + "license": "MIT", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "es-set-tostringtag": "^2.1.0", + "hasown": "^2.0.2", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/formidable": { + "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", + "once": "^1.4.0", + "qs": "^6.11.0" + }, + "funding": { + "url": "https://ko-fi.com/tunnckoCore/commissions" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "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==", + "dev": true, + "license": "MIT", + "dependencies": { + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/fs-monkey": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fs-monkey/-/fs-monkey-1.1.0.tgz", + "integrity": "sha512-QMUezzXWII9EV5aTFXW1UBVUO77wYPpjqIF8/AviUCThNeSYZykpoTixUeaNNBwmCev0AMDWMAni+f8Hxb1IFw==", + "dev": true, + "license": "Unlicense" + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true, + "license": "ISC" + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true, + "license": "ISC", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-intrinsic": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.1.1", + "function-bind": "^1.1.2", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-package-type": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", + "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/glob": { + "version": "10.4.5", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz", + "integrity": "sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==", + "dev": true, + "license": "ISC", + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^3.1.2", + "minimatch": "^9.0.4", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^1.11.1" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/glob-to-regexp": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz", + "integrity": "sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==", + "dev": true, + "license": "BSD-2-Clause" + }, + "node_modules/glob/node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/globals": { + "version": "13.24.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", + "integrity": "sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "type-fest": "^0.20.2" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/globby": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dev": true, + "license": "MIT", + "dependencies": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.9", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^3.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/gopd": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true, + "license": "MIT" + }, + "node_modules/handlebars": { + "version": "4.7.8", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.8.tgz", + "integrity": "sha512-vafaFqs8MZkRrSX7sFVUdo3ap/eNiLnb4IakshzvP56X5Nr1iGKAIqdX6tMlm6HcNRIkr6AxO5jFEoJzzpT8aQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "minimist": "^1.2.5", + "neo-async": "^2.6.2", + "source-map": "^0.6.1", + "wordwrap": "^1.0.0" + }, + "bin": { + "handlebars": "bin/handlebars" + }, + "engines": { + "node": ">=0.4.7" + }, + "optionalDependencies": { + "uglify-js": "^3.1.4" + } + }, + "node_modules/handlebars/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "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==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/has-own-prop": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-own-prop/-/has-own-prop-2.0.0.tgz", + "integrity": "sha512-Pq0h+hvsVm6dDEa8x82GnLSYHOzNDt7f0ddFa3FqcQlgzEiptPqL+XrOJNavjOzSYiYWIrgeVYYgGlLmnxwilQ==", + "dev": 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-symbols": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", + "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-tostringtag": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", + "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", + "license": "MIT", + "dependencies": { + "has-symbols": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "license": "MIT", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true, + "license": "MIT" + }, + "node_modules/http-errors": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", + "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", + "license": "MIT", + "dependencies": { + "depd": "2.0.0", + "inherits": "2.0.4", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "toidentifier": "1.0.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=10.17.0" + } + }, + "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==", + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "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": "BSD-3-Clause" + }, + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", + "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/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", + "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/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", + "dev": true, + "license": "ISC", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "license": "ISC" + }, + "node_modules/inquirer": { + "version": "8.2.6", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-8.2.6.tgz", + "integrity": "sha512-M1WuAmb7pn9zdFRtQYk26ZBoY043Sse0wVDdk4Bppr+JOXyQYybdtvK+l9wUibhtjdjvtoiNy8tk+EgsYIUqKg==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-escapes": "^4.2.1", + "chalk": "^4.1.1", + "cli-cursor": "^3.1.0", + "cli-width": "^3.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.21", + "mute-stream": "0.0.8", + "ora": "^5.4.1", + "run-async": "^2.4.0", + "rxjs": "^7.5.5", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6", + "wrap-ansi": "^6.0.1" + }, + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/ioredis": { + "version": "5.8.2", + "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-5.8.2.tgz", + "integrity": "sha512-C6uC+kleiIMmjViJINWk80sOQw5lEzse1ZmvD+S/s8p8CWapftSaC+kocGTx6xrbrJ4WmYQGC08ffHLr6ToR6Q==", + "license": "MIT", + "peer": true, + "dependencies": { + "@ioredis/commands": "1.4.0", + "cluster-key-slot": "^1.1.0", + "debug": "^4.3.4", + "denque": "^2.1.0", + "lodash.defaults": "^4.2.0", + "lodash.isarguments": "^3.1.0", + "redis-errors": "^1.2.0", + "redis-parser": "^3.0.0", + "standard-as-callback": "^2.1.0" + }, + "engines": { + "node": ">=12.22.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/ioredis" + } + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "license": "MIT", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", + "dev": true, + "license": "MIT" + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "license": "MIT", + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-core-module": { + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", + "dev": true, + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-generator-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", + "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-interactive": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz", + "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-path-inside": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.3.tgz", + "integrity": "sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-unicode-supported": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", + "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" + }, + "node_modules/istanbul-lib-coverage": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.2.tgz", + "integrity": "sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-6.0.3.tgz", + "integrity": "sha512-Vtgk7L/R2JHyyGW07spoFlB8/lpjiOLTjMdms6AFMraYt3BaJauod/NGrfnVG/y4Ix1JEuMRPDPEj2ua+zz1/Q==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.23.9", + "@babel/parser": "^7.23.9", + "@istanbuljs/schema": "^0.1.3", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", + "integrity": "sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^4.0.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", + "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/istanbul-reports": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.2.0.tgz", + "integrity": "sha512-HGYWWS/ehqTV3xN10i23tkPkpH46MLCIMFNCaaKNavAXTF1RkqxawEPtnjnGZ6XKSInBKkiOA5BKS+aZiY3AvA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/iterare": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/iterare/-/iterare-1.2.1.tgz", + "integrity": "sha512-RKYVTCjAnRthyJes037NX/IiqeidgN1xc3j1RjFfECFp28A1GVwK9nA+i0rJPaHqSZwygLzRnFlzUuHFoWWy+Q==", + "license": "ISC", + "engines": { + "node": ">=6" + } + }, + "node_modules/jackspeak": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz", + "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "@isaacs/cliui": "^8.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + }, + "optionalDependencies": { + "@pkgjs/parseargs": "^0.11.0" + } + }, + "node_modules/jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest/-/jest-29.7.0.tgz", + "integrity": "sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/types": "^29.6.3", + "import-local": "^3.0.2", + "jest-cli": "^29.7.0" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-changed-files": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-29.7.0.tgz", + "integrity": "sha512-fEArFiwf1BpQ+4bXSprcDc3/x4HSzL4al2tozwVpDFpsxALjLYdyiIK4e5Vz66GQJIbXJ82+35PtysofptNX2w==", + "dev": true, + "license": "MIT", + "dependencies": { + "execa": "^5.0.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-circus": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-29.7.0.tgz", + "integrity": "sha512-3E1nCMgipcTkCocFwM90XXQab9bS+GMsjdpmPrlelaxwD93Ad8iVEjX/vvHPdLPnFf+L40u+5+iutRdA1N9myw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "dedent": "^1.0.0", + "is-generator-fn": "^2.0.0", + "jest-each": "^29.7.0", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0", + "pretty-format": "^29.7.0", + "pure-rand": "^6.0.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-cli": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-29.7.0.tgz", + "integrity": "sha512-OVVobw2IubN/GSYsxETi+gOe7Ka59EFMR/twOU3Jb2GnKKeMGJB5SGUUrEz3SFVmJASUdZUzy83sLNNQ2gZslg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "create-jest": "^29.7.0", + "exit": "^0.1.2", + "import-local": "^3.0.2", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "yargs": "^17.3.1" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-config": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-29.7.0.tgz", + "integrity": "sha512-uXbpfeQ7R6TZBqI3/TxCU4q4ttk3u0PJeC+E0zbfSoSjq6bJ7buBPxzQPL0ifrkY4DNu4JUdk0ImlBUYi840eQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/test-sequencer": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-jest": "^29.7.0", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "deepmerge": "^4.2.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-circus": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "micromatch": "^4.0.4", + "parse-json": "^5.2.0", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@types/node": "*", + "ts-node": ">=9.0.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "ts-node": { + "optional": true + } + } + }, + "node_modules/jest-config/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/jest-config/node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/jest-config/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/jest-diff": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.7.0.tgz", + "integrity": "sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "diff-sequences": "^29.6.3", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-docblock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-29.7.0.tgz", + "integrity": "sha512-q617Auw3A612guyaFgsbFeYpNP5t2aoUNLwBUbc/0kD1R4t9ixDbyFTHd1nok4epoVFpr7PmeWHrhvuV3XaJ4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "detect-newline": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-29.7.0.tgz", + "integrity": "sha512-gns+Er14+ZrEoC5fhOfYCY1LOHHr0TI+rQUHZS8Ttw2l7gl+80eHc/gFf2Ktkw0+SIACDTeWvpFcv3B04VembQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "jest-util": "^29.7.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-environment-node": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-29.7.0.tgz", + "integrity": "sha512-DOSwCRqXirTOyheM+4d5YZOrWcdu0LNZ87ewUoywbcb2XR4wKgqiG8vNeYwhjFMbEkfju7wx2GYH0P2gevGvFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-haste-map": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-29.7.0.tgz", + "integrity": "sha512-fP8u2pyfqx0K1rGn1R9pyE0/KTn+G7PxktWidOBTqFPLYX0b9ksaMFkhK5vrS3DVun09pckLdlx90QthlW7AmA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/graceful-fs": "^4.1.3", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "micromatch": "^4.0.4", + "walker": "^1.0.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "optionalDependencies": { + "fsevents": "^2.3.2" + } + }, + "node_modules/jest-leak-detector": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-29.7.0.tgz", + "integrity": "sha512-kYA8IJcSYtST2BY9I+SMC32nDpBT3J2NvWJx8+JCuCdl/CR1I4EKUJROiP8XtCcxqgTTBGJNdbB1A8XRKbTetw==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz", + "integrity": "sha512-sBkD+Xi9DtcChsI3L3u0+N0opgPYnCRPtGcQYrgXmR+hmt/fYfWAL0xRXYU8eWOdfuLgBe0YCW3AFtnRLagq/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-message-util": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.7.0.tgz", + "integrity": "sha512-GBEV4GRADeP+qtB2+6u61stea8mGcOT4mCtrYISZwfu9/ISHFJ/5zOMXYbpBE9RsS5+Gb63DW4FgmnKJ79Kf6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^29.6.3", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-mock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-29.7.0.tgz", + "integrity": "sha512-ITOMZn+UkYS4ZFh83xYAOzWStloNzJFO2s8DWrE4lhtGD+AorgnbkiKERe4wQVBydIGPx059g6riW5Btp6Llnw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-pnp-resolver": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.3.tgz", + "integrity": "sha512-+3NpwQEnRoIBtx4fyhblQDPgJI0H1IEIkX7ShLUjPGA7TtUTvI1oiKi3SR4oBR0hQhQR80l4WAe5RrXBwWMA8w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "jest-resolve": "*" + }, + "peerDependenciesMeta": { + "jest-resolve": { + "optional": true + } + } + }, + "node_modules/jest-regex-util": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-29.6.3.tgz", + "integrity": "sha512-KJJBsRCyyLNWCNBOvZyRDnAIfUiRJ8v+hOBQYGn8gDyF3UegwiP4gwRR3/SDa42g1YbVycTidUF3rKjyLFDWbg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-29.7.0.tgz", + "integrity": "sha512-IOVhZSrg+UvVAshDSDtHyFCCBUl/Q3AAJv8iZ6ZjnZ74xzvwuzLXid9IIIPgTnY62SJjfuupMKZsZQRsCvxEgA==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-pnp-resolver": "^1.2.2", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "resolve": "^1.20.0", + "resolve.exports": "^2.0.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve-dependencies": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-29.7.0.tgz", + "integrity": "sha512-un0zD/6qxJ+S0et7WxeI3H5XSe9lTBBR7bOHCHXkKR6luG5mwDDlIzVQ0V5cZCuoTgEdcdwzTghYkTWfubi+nA==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-regex-util": "^29.6.3", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runner": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-29.7.0.tgz", + "integrity": "sha512-fsc4N6cPCAahybGBfTRcq5wFR6fpLznMg47sY5aDpsoejOcVYFb07AHuSnR0liMcPTgBsA3ZJL6kFOjPdoNipQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/environment": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "graceful-fs": "^4.2.9", + "jest-docblock": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-leak-detector": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-resolve": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-util": "^29.7.0", + "jest-watcher": "^29.7.0", + "jest-worker": "^29.7.0", + "p-limit": "^3.1.0", + "source-map-support": "0.5.13" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runner/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/jest-runner/node_modules/source-map-support": { + "version": "0.5.13", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.13.tgz", + "integrity": "sha512-SHSKFHadjVA5oR4PPqhtAVdcBWwRYVd6g6cAXnIbRiIwc2EhPrTuKUBdSLvlEKyIP3GCf89fltvcZiP9MMFA1w==", + "dev": true, + "license": "MIT", + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/jest-runtime": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-29.7.0.tgz", + "integrity": "sha512-gUnLjgwdGqW7B4LvOIkbKs9WGbn+QLqRQQ9juC6HndeDiezIwhDP+mhMwHWCEcfQ5RUXa6OPnFF8BJh5xegwwQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/globals": "^29.7.0", + "@jest/source-map": "^29.6.3", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "cjs-module-lexer": "^1.0.0", + "collect-v8-coverage": "^1.0.0", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-mock": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "slash": "^3.0.0", + "strip-bom": "^4.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runtime/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/jest-runtime/node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/jest-runtime/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/jest-snapshot": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-29.7.0.tgz", + "integrity": "sha512-Rm0BMWtxBcioHr1/OX5YCP8Uov4riHvKPknOGs804Zg9JGZgmIBkbtlxJC/7Z4msKYVbIJtfU+tKb8xlYNfdkw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@babel/generator": "^7.7.2", + "@babel/plugin-syntax-jsx": "^7.7.2", + "@babel/plugin-syntax-typescript": "^7.7.2", + "@babel/types": "^7.3.3", + "@jest/expect-utils": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0", + "chalk": "^4.0.0", + "expect": "^29.7.0", + "graceful-fs": "^4.2.9", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "natural-compare": "^1.4.0", + "pretty-format": "^29.7.0", + "semver": "^7.5.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "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/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-validate": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-29.7.0.tgz", + "integrity": "sha512-ZB7wHqaRGVw/9hST/OuFUReG7M8vKeq0/J2egIGLdvjHCmYqGARhzXmtgi+gVeZ5uXFF219aOc3Ls2yLg27tkw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "camelcase": "^6.2.0", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "leven": "^3.1.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate/node_modules/camelcase": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", + "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/jest-watcher": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-29.7.0.tgz", + "integrity": "sha512-49Fg7WXkU3Vl2h6LbLtMQ/HyB6rXSIX7SqvBLQmssRBGN9I0PNvPmAmCWSOY6SOvrjhI/F7/bGAv9RtnsPA03g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "jest-util": "^29.7.0", + "string-length": "^4.0.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "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/jest-worker/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/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "license": "MIT", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsesc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", + "dev": true, + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-schema-traverse": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz", + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "license": "MIT", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/jsonc-parser": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-3.2.1.tgz", + "integrity": "sha512-AilxAyFOAcK5wA1+LeaySVBrHsGQvUFCDWXKpZjzaL0PqW+xfBOttn8GNtWKFWqneyMZj41MWF9Kl6iPWLwgOA==", + "dev": true, + "license": "MIT" + }, + "node_modules/jsonfile": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.2.0.tgz", + "integrity": "sha512-FGuPw30AdOIUTRMC2OMRtQV+jkVj2cfPqSeWXv1NEAJ1qZ5zb1X6z1mFhbfOB/iy3ssJCD+3KuZ8r8C3uVFlAg==", + "dev": true, + "license": "MIT", + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/jsonwebtoken": { + "version": "9.0.2", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz", + "integrity": "sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==", + "license": "MIT", + "dependencies": { + "jws": "^3.2.2", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=12", + "npm": ">=6" + } + }, + "node_modules/jwa": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.2.tgz", + "integrity": "sha512-eeH5JO+21J78qMvTIDdBXidBd6nG2kZjg5Ohz/1fpa28Z4CcsWUzJ1ZZyFq/3z3N17aZy+ZuBoHljASbL1WfOw==", + "license": "MIT", + "dependencies": { + "buffer-equal-constant-time": "^1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/jws": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", + "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", + "license": "MIT", + "dependencies": { + "jwa": "^1.4.1", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/kafkajs": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/kafkajs/-/kafkajs-2.2.4.tgz", + "integrity": "sha512-j/YeapB1vfPT2iOIUn/vxdyKEuhuY2PxMBvf5JWux6iSaukAccrMtXEY/Lb7OvavDhOWME589bpLrEdnVHjfjA==", + "license": "MIT", + "peer": true, + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/leven": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", + "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/libphonenumber-js": { + "version": "1.12.29", + "resolved": "https://registry.npmjs.org/libphonenumber-js/-/libphonenumber-js-1.12.29.tgz", + "integrity": "sha512-P2aLrbeqHbmh8+9P35LXQfXOKc7XJ0ymUKl7tyeyQjdRNfzunXWxQXGc4yl3fUf28fqLRfPY+vIVvFXK7KEBTw==", + "license": "MIT" + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "dev": true, + "license": "MIT" + }, + "node_modules/loader-runner": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-4.3.1.tgz", + "integrity": "sha512-IWqP2SCPhyVFTBtRcgMHdzlf9ul25NwaFx4wCEH/KjAXuuHY4yNjvPXsBokp8jCB936PyWRaPKUNh8NvylLp2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.11.5" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "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.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==", + "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.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==", + "license": "MIT" + }, + "node_modules/lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==", + "license": "MIT" + }, + "node_modules/lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==", + "license": "MIT" + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==", + "license": "MIT" + }, + "node_modules/lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==", + "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", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==", + "license": "MIT" + }, + "node_modules/log-symbols": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", + "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.1.0", + "is-unicode-supported": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/luxon": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/luxon/-/luxon-3.5.0.tgz", + "integrity": "sha512-rh+Zjr6DNfUYR3bPwJEnuwDdqMbxZW7LOQfUN4B54+Cl+0o5zaU9RJ6bcidfDtC1cWCZXQ+nvX8bf6bAji37QQ==", + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/magic-string": { + "version": "0.30.8", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.8.tgz", + "integrity": "sha512-ISQTe55T2ao7XtlAStud6qwYPZjE4GK1S/BeVPus4jrq6JuOnQ00YKQC581RWhR122W7msZV263KzVeLoqidyQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.4.15" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/make-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", + "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", + "dev": true, + "license": "MIT", + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true, + "license": "ISC" + }, + "node_modules/makeerror": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", + "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tmpl": "1.0.5" + } + }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/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", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/memfs": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/memfs/-/memfs-3.5.3.tgz", + "integrity": "sha512-UERzLsxzllchadvbPs5aolHh65ISpKpM+ccLbOJ8/vvpBKmAWf+la7dXFy7Mr0ySHbdHrFv5kGFCUHHe6GFEmw==", + "dev": true, + "license": "Unlicense", + "dependencies": { + "fs-monkey": "^1.0.4" + }, + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", + "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true, + "license": "MIT" + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "license": "MIT", + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/micromatch/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/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "license": "MIT", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/minimatch": { + "version": "9.0.3", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", + "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/minipass": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", + "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=16 || 14 >=14.17" + } + }, + "node_modules/mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "license": "MIT", + "dependencies": { + "minimist": "^1.2.6" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "license": "MIT" + }, + "node_modules/multer": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/multer/-/multer-2.0.2.tgz", + "integrity": "sha512-u7f2xaZ/UG8oLXHvtF/oWTRvT44p9ecwBBqTwgJVq0+4BW1g8OW01TyMEGWBHbyMOYVHXslaut7qEQ1meATXgw==", + "license": "MIT", + "dependencies": { + "append-field": "^1.0.0", + "busboy": "^1.6.0", + "concat-stream": "^2.0.0", + "mkdirp": "^0.5.6", + "object-assign": "^4.1.1", + "type-is": "^1.6.18", + "xtend": "^4.0.2" + }, + "engines": { + "node": ">= 10.16.0" + } + }, + "node_modules/mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true, + "license": "ISC" + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true, + "license": "MIT" + }, + "node_modules/negotiator": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-abort-controller": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/node-abort-controller/-/node-abort-controller-3.1.1.tgz", + "integrity": "sha512-AGK2yQKIjRuqnc6VkX2Xj5d+QW8xZ87pa1UK6yA6ouUyuxfHuMP6umE5QK7UmTeOAymo+Zx1Fxiuw9rVx8taHQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-emoji": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/node-emoji/-/node-emoji-1.11.0.tgz", + "integrity": "sha512-wo2DpQkQp7Sjm2A0cq+sN7EHKO6Sl0ctXeBdFZrL9T9+UywORbufTcTZxom8YqpLQt/FqNMUkOpkZrJVYSKD3A==", + "dev": true, + "license": "MIT", + "dependencies": { + "lodash": "^4.17.21" + } + }, + "node_modules/node-fetch": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", + "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", + "license": "MIT", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + }, + "peerDependencies": { + "encoding": "^0.1.0" + }, + "peerDependenciesMeta": { + "encoding": { + "optional": true + } + } + }, + "node_modules/node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-releases": { + "version": "2.0.27", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz", + "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-inspect": { + "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" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "license": "MIT", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "license": "ISC", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/ora": { + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/ora/-/ora-5.4.1.tgz", + "integrity": "sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "bl": "^4.1.0", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-spinners": "^2.5.0", + "is-interactive": "^1.0.0", + "is-unicode-supported": "^0.1.0", + "log-symbols": "^4.1.0", + "strip-ansi": "^6.0.0", + "wcwidth": "^1.0.1" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/package-json-from-dist": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", + "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==", + "dev": true, + "license": "BlueOak-1.0.0" + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "license": "MIT", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/passport": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/passport/-/passport-0.7.0.tgz", + "integrity": "sha512-cPLl+qZpSc+ireUvt+IzqbED1cHHkDoVYMo30jbJIdOOjQ1MQYZBPiNvmi8UM6lJuOpTPXJGZQk0DtC4y61MYQ==", + "license": "MIT", + "peer": true, + "dependencies": { + "passport-strategy": "1.x.x", + "pause": "0.0.1", + "utils-merge": "^1.0.1" + }, + "engines": { + "node": ">= 0.4.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/jaredhanson" + } + }, + "node_modules/passport-jwt": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/passport-jwt/-/passport-jwt-4.0.1.tgz", + "integrity": "sha512-UCKMDYhNuGOBE9/9Ycuoyh7vP6jpeTp/+sfMJl7nLff/t6dps+iaeE0hhNkKN8/HZHcJ7lCdOyDxHdDoxoSvdQ==", + "license": "MIT", + "dependencies": { + "jsonwebtoken": "^9.0.0", + "passport-strategy": "^1.0.0" + } + }, + "node_modules/passport-strategy": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/passport-strategy/-/passport-strategy-1.0.0.tgz", + "integrity": "sha512-CB97UUvDKJde2V0KDWWB3lyf6PC3FaZP7YxZ2G8OAtn9p4HI9j9JLP9qjOGZFvyl8uwNT8qM+hGnz/n16NI7oA==", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true, + "license": "MIT" + }, + "node_modules/path-scurry": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz", + "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "lru-cache": "^10.2.0", + "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0" + }, + "engines": { + "node": ">=16 || 14 >=14.18" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/path-scurry/node_modules/lru-cache": { + "version": "10.4.3", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz", + "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/path-to-regexp": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-3.3.0.tgz", + "integrity": "sha512-qyCH421YQPS2WFDxDjftfc1ZR5WKQzVzqsp4n9M2kQhVOo/ByahFoUNJfl58kOcEGfQ//7weFTDhm+ss8Ecxgw==", + "license": "MIT" + }, + "node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/pause": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/pause/-/pause-0.0.1.tgz", + "integrity": "sha512-KG8UEiEVkR3wGEb4m5yZkVCzigAD+cVEJck2CzYZO37ZGJfctvVptVO192MwrtPhzONn6go8ylnOdMhKqi4nfg==" + }, + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "dev": true, + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.1.tgz", + "integrity": "sha512-xUXwsxNjwTQ8K3GnT4pCJm+xq3RUPQbmkYJTP5aFIfNIvbcc/4MUxgBaaRSZJ6yGJZiGSyYlM6MzwTsRk8SYCg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pirates": { + "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": { + "node": ">= 6" + } + }, + "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", + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/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", + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/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", + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/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", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/pkg-dir/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", + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pluralize": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-8.0.0.tgz", + "integrity": "sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/prettier": { + "version": "3.7.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.7.3.tgz", + "integrity": "sha512-QgODejq9K3OzoBbuyobZlUhznP5SKwPqp+6Q6xw6o8gnhr4O85L2U915iM2IDcfF2NPXVaM9zlo9tdwipnYwzg==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, + "node_modules/prettier-linter-helpers": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz", + "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-diff": "^1.1.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/prisma": { + "version": "5.22.0", + "resolved": "https://registry.npmjs.org/prisma/-/prisma-5.22.0.tgz", + "integrity": "sha512-vtpjW3XuYCSnMsNVBjLMNkTj6OZbudcPPTPYHqX0CJfpcdWciI1dM8uHETwmDxxiqEwCIE6WvXucWUetJgfu/A==", + "devOptional": true, + "hasInstallScript": true, + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "@prisma/engines": "5.22.0" + }, + "bin": { + "prisma": "build/index.js" + }, + "engines": { + "node": ">=16.13" + }, + "optionalDependencies": { + "fsevents": "2.3.3" + } + }, + "node_modules/prompts": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", + "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "kleur": "^3.0.3", + "sisteransi": "^1.0.5" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "license": "MIT", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", + "license": "MIT" + }, + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/pure-rand": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-6.1.0.tgz", + "integrity": "sha512-bVWawvoZoBYpp6yIoQtQXHZjmz35RSVHnUOTefl8Vcjr8snTPY1wnpSPMWekcFwbxI6gtmT7rSYPFvz71ldiOA==", + "dev": true, + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/dubzzz" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/fast-check" + } + ], + "license": "MIT" + }, + "node_modules/qs": { + "version": "6.13.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", + "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", + "license": "BSD-3-Clause", + "dependencies": { + "side-channel": "^1.0.6" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/randombytes": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "safe-buffer": "^5.1.0" + } + }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "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==", + "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/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, + "node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "license": "MIT", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/readdirp/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/redis-errors": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/redis-errors/-/redis-errors-1.2.0.tgz", + "integrity": "sha512-1qny3OExCf0UvUV/5wpYKf2YwPcOqXzkwKKSmKHiE6ZMQs5heeE/c8eXK+PNllPvmjgAbfnsbpkGZWy8cBpn9w==", + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/redis-parser": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/redis-parser/-/redis-parser-3.0.0.tgz", + "integrity": "sha512-DJnGAeenTdpMEH6uAJRK/uiyEIH9WVsUmoLwzudwGJUwZPp80PDBWPHXSAGNPwNvIXAbe7MSUB1zQFugFml66A==", + "license": "MIT", + "dependencies": { + "redis-errors": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/reflect-metadata": { + "version": "0.1.14", + "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.14.tgz", + "integrity": "sha512-ZhYeb6nRaXCfhnndflDK8qI6ZQ/YcWZCISRAWICW9XYqMUwjZM9Z0DveWX/ABN01oxSHwVxKQmxeYZSsm0jh5A==", + "license": "Apache-2.0", + "peer": true + }, + "node_modules/repeat-string": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", + "integrity": "sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/require-from-string": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve": { + "version": "1.22.11", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz", + "integrity": "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-core-module": "^2.16.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-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", + "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", + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/resolve.exports": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-2.0.3.tgz", + "integrity": "sha512-OcXjMsGdhL4XnbShKpAcSqPMzQoYkYyhbEaeSko47MjRP9NfEQMhZkXL1DoFlt9LWQn4YttrdnV6X2OiyzBi+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/restore-cursor/node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/reusify": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", + "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "deprecated": "Rimraf versions prior to v4 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rimraf/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/rimraf/node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/rimraf/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/run-async": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.1.tgz", + "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/rxjs": { + "version": "7.8.2", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.2.tgz", + "integrity": "sha512-dhKf903U/PQZY6boNNtAGdWbG85WAbjT/1xYoZIC7FAY0yWapOBQVsVrDl58W86//e1VpMNBtRV4MaXfdMySFA==", + "license": "Apache-2.0", + "peer": true, + "dependencies": { + "tslib": "^2.1.0" + } + }, + "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/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "license": "MIT" + }, + "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", + "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/schema-utils/node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/schema-utils/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", + "peerDependencies": { + "ajv": "^6.9.1" + } + }, + "node_modules/schema-utils/node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true, + "license": "MIT" + }, + "node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/send": { + "version": "0.19.0", + "resolved": "https://registry.npmjs.org/send/-/send-0.19.0.tgz", + "integrity": "sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==", + "license": "MIT", + "dependencies": { + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "2.0.0", + "mime": "1.6.0", + "ms": "2.1.3", + "on-finished": "2.4.1", + "range-parser": "~1.2.1", + "statuses": "2.0.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "license": "MIT", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/send/node_modules/debug/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "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/serialize-javascript": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.2.tgz", + "integrity": "sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "randombytes": "^2.1.0" + } + }, + "node_modules/serve-static": { + "version": "1.16.2", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.2.tgz", + "integrity": "sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==", + "license": "MIT", + "dependencies": { + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.19.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "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/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", + "license": "ISC" + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/side-channel": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", + "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3", + "side-channel-list": "^1.0.0", + "side-channel-map": "^1.0.1", + "side-channel-weakmap": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-list": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", + "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", + "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-weakmap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", + "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3", + "side-channel-map": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/signal-exit": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", + "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", + "dev": true, + "license": "MIT" + }, + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map": { + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.4.tgz", + "integrity": "sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">= 8" + } + }, + "node_modules/source-map-support": { + "version": "0.5.21", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", + "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", + "dev": true, + "license": "MIT", + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/source-map-support/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/stack-utils": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.6.tgz", + "integrity": "sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "escape-string-regexp": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/stack-utils/node_modules/escape-string-regexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", + "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/standard-as-callback": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/standard-as-callback/-/standard-as-callback-2.1.0.tgz", + "integrity": "sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A==", + "license": "MIT" + }, + "node_modules/statuses": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", + "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/streamsearch": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz", + "integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==", + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.2.0" + } + }, + "node_modules/string-length": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", + "integrity": "sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "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/string-width-cjs": { + "name": "string-width", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi-cjs": { + "name": "strip-ansi", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-bom": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", + "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-final-newline": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", + "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/strtok3": { + "version": "10.3.4", + "resolved": "https://registry.npmjs.org/strtok3/-/strtok3-10.3.4.tgz", + "integrity": "sha512-KIy5nylvC5le1OdaaoCJ07L+8iQzJHGH6pWDuzS+d07Cu7n1MZ2x26P8ZKIWfbK02+XIL8Mp4RkWeqdUCrDMfg==", + "license": "MIT", + "dependencies": { + "@tokenizer/token": "^0.3.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Borewit" + } + }, + "node_modules/superagent": { + "version": "8.1.2", + "resolved": "https://registry.npmjs.org/superagent/-/superagent-8.1.2.tgz", + "integrity": "sha512-6WTxW1EB6yCxV5VFOIPQruWGHqc3yI7hEmZK6h+pyk69Lk/Ut7rLUY6W/ONF2MjBuGjvmMiIpsrVJ2vjrHlslA==", + "deprecated": "Please upgrade to superagent v10.2.2+, see release notes at https://github.com/forwardemail/superagent/releases/tag/v10.2.2 - maintenance is supported by Forward Email @ https://forwardemail.net", + "dev": true, + "license": "MIT", + "dependencies": { + "component-emitter": "^1.3.0", + "cookiejar": "^2.1.4", + "debug": "^4.3.4", + "fast-safe-stringify": "^2.1.1", + "form-data": "^4.0.0", + "formidable": "^2.1.2", + "methods": "^1.1.2", + "mime": "2.6.0", + "qs": "^6.11.0", + "semver": "^7.3.8" + }, + "engines": { + "node": ">=6.4.0 <13 || >=14" + } + }, + "node_modules/superagent/node_modules/mime": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.6.0.tgz", + "integrity": "sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg==", + "dev": true, + "license": "MIT", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/supertest": { + "version": "6.3.4", + "resolved": "https://registry.npmjs.org/supertest/-/supertest-6.3.4.tgz", + "integrity": "sha512-erY3HFDG0dPnhw4U+udPfrzXa4xhSG+n4rxfRuZWCUvjFWwKl+OxWf/7zk50s84/fAAs7vf5QAb9uRa0cCykxw==", + "deprecated": "Please upgrade to supertest v7.1.3+, see release notes at https://github.com/forwardemail/supertest/releases/tag/v7.1.3 - maintenance is supported by Forward Email @ https://forwardemail.net", + "dev": true, + "license": "MIT", + "dependencies": { + "methods": "^1.1.2", + "superagent": "^8.1.2" + }, + "engines": { + "node": ">=6.4.0" + } + }, + "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==", + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/swagger-ui-dist": { + "version": "5.17.14", + "resolved": "https://registry.npmjs.org/swagger-ui-dist/-/swagger-ui-dist-5.17.14.tgz", + "integrity": "sha512-CVbSfaLpstV65OnSjbXfVd6Sta3q3F7Cj/yYuvHMp1P90LztOLs6PfUnKEVAeiIVQt9u2SaPwv0LiH/OyMjHRw==", + "license": "Apache-2.0" + }, + "node_modules/symbol-observable": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-4.0.0.tgz", + "integrity": "sha512-b19dMThMV4HVFynSAM1++gBHAbk2Tc/osgLIBZMKsyqh34jb2e8Os7T6ZW/Bt3pJFdBTd2JwAnAAEQV7rSNvcQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/synckit": { + "version": "0.11.11", + "resolved": "https://registry.npmjs.org/synckit/-/synckit-0.11.11.tgz", + "integrity": "sha512-MeQTA1r0litLUf0Rp/iisCaL8761lKAZHaimlbGK4j0HysC4PLfqygQj9srcs0m2RdtDYnF8UuYyKpbjHYp7Jw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@pkgr/core": "^0.2.9" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/synckit" + } + }, + "node_modules/tapable": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.0.tgz", + "integrity": "sha512-g9ljZiwki/LfxmQADO3dEY1CbpmXT5Hm2fJ+QaGKwSXUylMybePR7/67YW7jOrrvjEgL1Fmz5kzyAjWVWLlucg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/terser": { + "version": "5.44.1", + "resolved": "https://registry.npmjs.org/terser/-/terser-5.44.1.tgz", + "integrity": "sha512-t/R3R/n0MSwnnazuPpPNVO60LX0SKL45pyl9YlvxIdkH0Of7D5qM2EVe+yASRIlY5pZ73nclYJfNANGWPwFDZw==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "@jridgewell/source-map": "^0.3.3", + "acorn": "^8.15.0", + "commander": "^2.20.0", + "source-map-support": "~0.5.20" + }, + "bin": { + "terser": "bin/terser" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/terser-webpack-plugin": { + "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", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.25", + "jest-worker": "^27.4.5", + "schema-utils": "^4.3.0", + "serialize-javascript": "^6.0.2", + "terser": "^5.31.1" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^5.1.0" + }, + "peerDependenciesMeta": { + "@swc/core": { + "optional": true + }, + "esbuild": { + "optional": true + }, + "uglify-js": { + "optional": true + } + } + }, + "node_modules/terser-webpack-plugin/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", + "dependencies": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/terser-webpack-plugin/node_modules/schema-utils": { + "version": "4.3.3", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.3.3.tgz", + "integrity": "sha512-eflK8wEtyOE6+hsaRVPxvUKYCpRgzLqDTb8krvAsRIwOGlHoSgYLgBXoubGgLd2fT41/OUYdb48v4k4WWHQurA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/terser-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/terser/node_modules/commander": { + "version": "2.20.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/test-exclude": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", + "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", + "dev": true, + "license": "ISC", + "dependencies": { + "@istanbuljs/schema": "^0.1.2", + "glob": "^7.1.4", + "minimatch": "^3.0.4" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/test-exclude/node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/test-exclude/node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/test-exclude/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true, + "license": "MIT" + }, + "node_modules/through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==", + "dev": true, + "license": "MIT" + }, + "node_modules/tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "license": "MIT", + "dependencies": { + "os-tmpdir": "~1.0.2" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/tmpl": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", + "integrity": "sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "license": "MIT", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/token-types": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/token-types/-/token-types-6.1.1.tgz", + "integrity": "sha512-kh9LVIWH5CnL63Ipf0jhlBIy0UsrMj/NJDfpsy1SqOXlLKEVyXXYrnFxFT1yOOYVGBSApeVnjPw/sBz5BfEjAQ==", + "license": "MIT", + "dependencies": { + "@borewit/text-codec": "^0.1.0", + "@tokenizer/token": "^0.3.0", + "ieee754": "^1.2.1" + }, + "engines": { + "node": ">=14.16" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Borewit" + } + }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + "license": "MIT" + }, + "node_modules/tree-kill": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", + "dev": true, + "license": "MIT", + "bin": { + "tree-kill": "cli.js" + } + }, + "node_modules/ts-api-utils": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.4.3.tgz", + "integrity": "sha512-i3eMG77UTMD0hZhgRS562pv83RC6ukSAC2GMNWc+9dieh/+jDM5u5YG+NHX6VNDRHQcHwmsTHctP9LhbC3WxVw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=16" + }, + "peerDependencies": { + "typescript": ">=4.2.0" + } + }, + "node_modules/ts-jest": { + "version": "29.4.5", + "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-29.4.5.tgz", + "integrity": "sha512-HO3GyiWn2qvTQA4kTgjDcXiMwYQt68a1Y8+JuLRVpdIzm+UOLSHgl/XqR4c6nzJkq5rOkjc02O2I7P7l/Yof0Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "bs-logger": "^0.2.6", + "fast-json-stable-stringify": "^2.1.0", + "handlebars": "^4.7.8", + "json5": "^2.2.3", + "lodash.memoize": "^4.1.2", + "make-error": "^1.3.6", + "semver": "^7.7.3", + "type-fest": "^4.41.0", + "yargs-parser": "^21.1.1" + }, + "bin": { + "ts-jest": "cli.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || ^18.0.0 || >=20.0.0" + }, + "peerDependencies": { + "@babel/core": ">=7.0.0-beta.0 <8", + "@jest/transform": "^29.0.0 || ^30.0.0", + "@jest/types": "^29.0.0 || ^30.0.0", + "babel-jest": "^29.0.0 || ^30.0.0", + "jest": "^29.0.0 || ^30.0.0", + "jest-util": "^29.0.0 || ^30.0.0", + "typescript": ">=4.3 <6" + }, + "peerDependenciesMeta": { + "@babel/core": { + "optional": true + }, + "@jest/transform": { + "optional": true + }, + "@jest/types": { + "optional": true + }, + "babel-jest": { + "optional": true + }, + "esbuild": { + "optional": true + }, + "jest-util": { + "optional": true + } + } + }, + "node_modules/ts-jest/node_modules/type-fest": { + "version": "4.41.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-4.41.0.tgz", + "integrity": "sha512-TeTSQ6H5YHvpqVwBRcnLDCBnDOHWYu7IvGbHT6N8AOymcr9PJGjc1GTtiWZTYg0NCgYwvnYWEkVChQAr9bjfwA==", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=16" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ts-loader": { + "version": "9.5.4", + "resolved": "https://registry.npmjs.org/ts-loader/-/ts-loader-9.5.4.tgz", + "integrity": "sha512-nCz0rEwunlTZiy6rXFByQU1kVVpCIgUpc/psFiKVrUwrizdnIbRFu8w7bxhUF0X613DYwT4XzrZHpVyMe758hQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.1.0", + "enhanced-resolve": "^5.0.0", + "micromatch": "^4.0.0", + "semver": "^7.3.4", + "source-map": "^0.7.4" + }, + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "typescript": "*", + "webpack": "^5.0.0" + } + }, + "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==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@cspotcode/source-map-support": "^0.8.0", + "@tsconfig/node10": "^1.0.7", + "@tsconfig/node12": "^1.0.7", + "@tsconfig/node14": "^1.0.0", + "@tsconfig/node16": "^1.0.2", + "acorn": "^8.4.1", + "acorn-walk": "^8.1.1", + "arg": "^4.1.0", + "create-require": "^1.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "v8-compile-cache-lib": "^3.0.1", + "yn": "3.1.1" + }, + "bin": { + "ts-node": "dist/bin.js", + "ts-node-cwd": "dist/bin-cwd.js", + "ts-node-esm": "dist/bin-esm.js", + "ts-node-script": "dist/bin-script.js", + "ts-node-transpile-only": "dist/bin-transpile.js", + "ts-script": "dist/bin-script-deprecated.js" + }, + "peerDependencies": { + "@swc/core": ">=1.2.50", + "@swc/wasm": ">=1.2.50", + "@types/node": "*", + "typescript": ">=2.7" + }, + "peerDependenciesMeta": { + "@swc/core": { + "optional": true + }, + "@swc/wasm": { + "optional": true + } + } + }, + "node_modules/tsconfig-paths": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-4.2.0.tgz", + "integrity": "sha512-NoZ4roiN7LnbKn9QqE1amc9DJfzvZXxF4xDavcOWt1BPkdx+m+0gJuPM+S0vCe7zTJMYUP0R8pO2XMr+Y8oLIg==", + "dev": true, + "license": "MIT", + "dependencies": { + "json5": "^2.2.2", + "minimist": "^1.2.6", + "strip-bom": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/tsconfig-paths-webpack-plugin": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/tsconfig-paths-webpack-plugin/-/tsconfig-paths-webpack-plugin-4.2.0.tgz", + "integrity": "sha512-zbem3rfRS8BgeNK50Zz5SIQgXzLafiHjOwUAvk/38/o1jHn/V5QAgVUcz884or7WYcPaH3N2CIfUc2u0ul7UcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.1.0", + "enhanced-resolve": "^5.7.0", + "tapable": "^2.2.1", + "tsconfig-paths": "^4.1.2" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/tsconfig-paths/node_modules/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "license": "0BSD" + }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/type-detect": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", + "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "license": "MIT", + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==", + "license": "MIT" + }, + "node_modules/typescript": { + "version": "5.9.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", + "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", + "dev": true, + "license": "Apache-2.0", + "peer": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/uglify-js": { + "version": "3.19.3", + "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.19.3.tgz", + "integrity": "sha512-v3Xu+yuwBXisp6QYTcH4UbH+xYJXqnq2m/LtQVWKWzYc1iehYnLixoQDN9FH6/j9/oybfd6W9Ghwkl8+UMKTKQ==", + "dev": true, + "license": "BSD-2-Clause", + "optional": true, + "bin": { + "uglifyjs": "bin/uglifyjs" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/uid": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/uid/-/uid-2.0.2.tgz", + "integrity": "sha512-u3xV3X7uzvi5b1MncmZo3i2Aw222Zk1keqLA1YkHldREkAhAqi65wuPfe7lHx8H/Wzy+8CE7S7uS3jekIM5s8g==", + "license": "MIT", + "dependencies": { + "@lukeed/csprng": "^1.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/uint8array-extras": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/uint8array-extras/-/uint8array-extras-1.5.0.tgz", + "integrity": "sha512-rvKSBiC5zqCCiDZ9kAOszZcDvdAHwwIKJG33Ykj43OKcWsnmcBRL09YTU4nOeHZ8Y2a7l1MgTd08SBe9A8Qj6A==", + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/undici-types": { + "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/universalify": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", + "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/update-browserslist-db": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.1.4.tgz", + "integrity": "sha512-q0SPT4xyU84saUX+tomz1WLkxUbuaJnR1xWt17M7fJtEJigJeWUNGUqrauFXsHnqev9y9JTRGwk13tFBuKby4A==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.1" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", + "license": "MIT" + }, + "node_modules/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", + "license": "MIT", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/uuid": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.1.tgz", + "integrity": "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==", + "funding": [ + "https://github.com/sponsors/broofa", + "https://github.com/sponsors/ctavan" + ], + "license": "MIT", + "bin": { + "uuid": "dist/bin/uuid" + } + }, + "node_modules/v8-compile-cache-lib": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", + "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==", + "dev": true, + "license": "MIT" + }, + "node_modules/v8-to-istanbul": { + "version": "9.3.0", + "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-9.3.0.tgz", + "integrity": "sha512-kiGUalWN+rgBJ/1OHZsBtU4rXZOfj/7rKQxULKlIzwzQSvMJUUNgPwJEEh7gU6xEVxC0ahoOBvN2YI8GH6FNgA==", + "dev": true, + "license": "ISC", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.12", + "@types/istanbul-lib-coverage": "^2.0.1", + "convert-source-map": "^2.0.0" + }, + "engines": { + "node": ">=10.12.0" + } + }, + "node_modules/validator": { + "version": "13.15.23", + "resolved": "https://registry.npmjs.org/validator/-/validator-13.15.23.tgz", + "integrity": "sha512-4yoz1kEWqUjzi5zsPbAS/903QXSYp0UOtHsPpp7p9rHAw/W+dkInskAE386Fat3oKRROwO98d9ZB0G4cObgUyw==", + "license": "MIT", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/walker": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.8.tgz", + "integrity": "sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "makeerror": "1.0.12" + } + }, + "node_modules/watchpack": { + "version": "2.4.4", + "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.4.4.tgz", + "integrity": "sha512-c5EGNOiyxxV5qmTtAB7rbiXxi1ooX1pQKMLX/MIabJjRA0SJBQOjKF+KSVfHkr9U1cADPon0mRiVe/riyaiDUA==", + "dev": true, + "license": "MIT", + "dependencies": { + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.1.2" + }, + "engines": { + "node": ">=10.13.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", + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", + "license": "BSD-2-Clause" + }, + "node_modules/webpack": { + "version": "5.103.0", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.103.0.tgz", + "integrity": "sha512-HU1JOuV1OavsZ+mfigY0j8d1TgQgbZ6M+J75zDkpEAwYeXjWSqrGJtgnPblJjd/mAyTNQ7ygw0MiKOn6etz8yw==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@types/eslint-scope": "^3.7.7", + "@types/estree": "^1.0.8", + "@types/json-schema": "^7.0.15", + "@webassemblyjs/ast": "^1.14.1", + "@webassemblyjs/wasm-edit": "^1.14.1", + "@webassemblyjs/wasm-parser": "^1.14.1", + "acorn": "^8.15.0", + "acorn-import-phases": "^1.0.3", + "browserslist": "^4.26.3", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^5.17.3", + "es-module-lexer": "^1.2.1", + "eslint-scope": "5.1.1", + "events": "^3.2.0", + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.2.11", + "json-parse-even-better-errors": "^2.3.1", + "loader-runner": "^4.3.1", + "mime-types": "^2.1.27", + "neo-async": "^2.6.2", + "schema-utils": "^4.3.3", + "tapable": "^2.3.0", + "terser-webpack-plugin": "^5.3.11", + "watchpack": "^2.4.4", + "webpack-sources": "^3.3.3" + }, + "bin": { + "webpack": "bin/webpack.js" + }, + "engines": { + "node": ">=10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependenciesMeta": { + "webpack-cli": { + "optional": true + } + } + }, + "node_modules/webpack-node-externals": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/webpack-node-externals/-/webpack-node-externals-3.0.0.tgz", + "integrity": "sha512-LnL6Z3GGDPht/AigwRh2dvL9PQPFQ8skEpVrWZXLWBYmqcaojHNN0onvHzie6rq7EWKrrBfPYqNEzTJgiwEQDQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack-sources": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-3.3.3.tgz", + "integrity": "sha512-yd1RBzSGanHkitROoPFd6qsrxt+oFhg/129YzheDGqeustzX0vTZJZsSsQjVQC4yzBQ56K55XU8gaNCtIzOnTg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/webpack/node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/webpack/node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/webpack/node_modules/schema-utils": { + "version": "4.3.3", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-4.3.3.tgz", + "integrity": "sha512-eflK8wEtyOE6+hsaRVPxvUKYCpRgzLqDTb8krvAsRIwOGlHoSgYLgBXoubGgLd2fT41/OUYdb48v4k4WWHQurA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/json-schema": "^7.0.9", + "ajv": "^8.9.0", + "ajv-formats": "^2.1.1", + "ajv-keywords": "^5.1.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "license": "MIT", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs": { + "name": "wrap-ansi", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/write-file-atomic": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-4.0.2.tgz", + "integrity": "sha512-7KxauUdBmSdWnmpaGFg+ppNjKF8uNLry8LyzjauQDOVONfFLNKrKvQOxZ/VuTIcS/gge/YNahf5RIIQWTSarlg==", + "dev": true, + "license": "ISC", + "dependencies": { + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.7" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || >=16.0.0" + } + }, + "node_modules/write-file-atomic/node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/ws": { + "version": "8.17.1", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.17.1.tgz", + "integrity": "sha512-6XQFvXTkbfUOZOKKILFG1PDK2NDQs4azKQl26T0YS5CxqWLgXajbPZ+h4gZekJyRqFU8pvnbAbbs/3TgRPy+GQ==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "license": "MIT", + "engines": { + "node": ">=0.4" + } + }, + "node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=10" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "license": "ISC" + }, + "node_modules/yargs": { + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "cliui": "^8.0.1", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.3", + "y18n": "^5.0.5", + "yargs-parser": "^21.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs-parser": { + "version": "21.1.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=12" + } + }, + "node_modules/yn": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + } +} diff --git a/backend/services/identity-service/package.json b/backend/services/identity-service/package.json index 1a4bccc0..9efb49b3 100644 --- a/backend/services/identity-service/package.json +++ b/backend/services/identity-service/package.json @@ -47,6 +47,7 @@ "ethers": "^6.9.0", "ioredis": "^5.3.2", "kafkajs": "^2.2.4", + "jsonwebtoken": "^9.0.0", "passport-jwt": "^4.0.1", "reflect-metadata": "^0.1.13", "rxjs": "^7.8.1", @@ -59,6 +60,7 @@ "@types/express": "^4.17.17", "@types/jest": "^29.5.2", "@types/node": "^20.3.1", + "@types/jsonwebtoken": "^9.0.0", "@types/passport-jwt": "^4.0.0", "@types/supertest": "^6.0.0", "@types/uuid": "^9.0.0", diff --git a/backend/services/identity-service/prisma/schema.prisma b/backend/services/identity-service/prisma/schema.prisma index f0c12b31..e20e766d 100644 --- a/backend/services/identity-service/prisma/schema.prisma +++ b/backend/services/identity-service/prisma/schema.prisma @@ -226,3 +226,26 @@ model MpcSession { @@index([createdAt], name: "idx_session_created") @@map("mpc_sessions") } + +// 推荐链接 - 用于追踪不同渠道的邀请 +model ReferralLink { + linkId BigInt @id @default(autoincrement()) @map("link_id") + userId BigInt @map("user_id") + referralCode String @map("referral_code") @db.VarChar(10) + + shortCode String @unique @map("short_code") @db.VarChar(10) // 短链码 + channel String? @db.VarChar(50) // 渠道: wechat, telegram, twitter, etc. + campaignId String? @map("campaign_id") @db.VarChar(50) // 活动ID + + clickCount Int @default(0) @map("click_count") // 点击次数 + registerCount Int @default(0) @map("register_count") // 注册转化数 + + createdAt DateTime @default(now()) @map("created_at") + expiresAt DateTime? @map("expires_at") // 过期时间 (可选) + + @@index([userId], name: "idx_referral_link_user") + @@index([referralCode], name: "idx_referral_link_code") + @@index([channel], name: "idx_referral_link_channel") + @@index([createdAt], name: "idx_referral_link_created") + @@map("referral_links") +} diff --git a/backend/services/identity-service/src/api/api.module.ts b/backend/services/identity-service/src/api/api.module.ts index ab8c73ee..10b8b005 100644 --- a/backend/services/identity-service/src/api/api.module.ts +++ b/backend/services/identity-service/src/api/api.module.ts @@ -1,10 +1,12 @@ import { Module } from '@nestjs/common'; import { UserAccountController } from './controllers/user-account.controller'; import { AuthController } from './controllers/auth.controller'; +import { ReferralsController } from './controllers/referrals.controller'; +import { DepositController } from './controllers/deposit.controller'; import { ApplicationModule } from '@/application/application.module'; @Module({ imports: [ApplicationModule], - controllers: [UserAccountController, AuthController], + controllers: [UserAccountController, AuthController, ReferralsController, DepositController], }) export class ApiModule {} diff --git a/backend/services/identity-service/src/api/controllers/deposit.controller.ts b/backend/services/identity-service/src/api/controllers/deposit.controller.ts new file mode 100644 index 00000000..8d842835 --- /dev/null +++ b/backend/services/identity-service/src/api/controllers/deposit.controller.ts @@ -0,0 +1,90 @@ +import { Controller, Get, UseGuards, Request, Logger } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse, ApiBearerAuth } from '@nestjs/swagger'; +import { JwtAuthGuard } from '@/shared/guards/jwt-auth.guard'; +import { DepositService } from '@/application/services/deposit.service'; + +/** + * 充值地址响应 DTO + */ +class DepositAddressResponseDto { + kavaAddress: string | null; + bscAddress: string | null; + isValid: boolean; + message?: string; +} + +/** + * USDT 余额响应 DTO + */ +class UsdtBalanceDto { + chainType: string; + address: string; + balance: string; + rawBalance: string; + decimals: number; +} + +class BalanceResponseDto { + kava: UsdtBalanceDto | null; + bsc: UsdtBalanceDto | null; +} + +/** + * 充值控制器 + * + * 提供充值地址获取和余额查询 API + */ +@ApiTags('Deposit') +@Controller('api/deposit') +export class DepositController { + private readonly logger = new Logger(DepositController.name); + + constructor(private readonly depositService: DepositService) {} + + /** + * 获取充值地址 + * + * 返回用户的 KAVA 和 BSC 充值地址 + * 会验证地址签名,确保地址未被篡改 + */ + @Get('addresses') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: '获取充值地址' }) + @ApiResponse({ + status: 200, + description: '成功获取充值地址', + type: DepositAddressResponseDto, + }) + @ApiResponse({ status: 400, description: '获取失败' }) + @ApiResponse({ status: 401, description: '未授权' }) + async getDepositAddresses(@Request() req: any): Promise { + const userId = req.user.userId; + this.logger.log(`获取充值地址: userId=${userId}`); + + return this.depositService.getDepositAddresses(userId); + } + + /** + * 查询 USDT 余额 + * + * 实时查询 KAVA 和 BSC 链上的 USDT 余额 + */ + @Get('balances') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: '查询 USDT 余额' }) + @ApiResponse({ + status: 200, + description: '成功获取余额', + type: BalanceResponseDto, + }) + @ApiResponse({ status: 400, description: '查询失败' }) + @ApiResponse({ status: 401, description: '未授权' }) + async getUsdtBalances(@Request() req: any): Promise { + const userId = req.user.userId; + this.logger.log(`查询 USDT 余额: userId=${userId}`); + + return this.depositService.getUsdtBalances(userId); + } +} diff --git a/backend/services/identity-service/src/api/controllers/referrals.controller.ts b/backend/services/identity-service/src/api/controllers/referrals.controller.ts new file mode 100644 index 00000000..a2920103 --- /dev/null +++ b/backend/services/identity-service/src/api/controllers/referrals.controller.ts @@ -0,0 +1,68 @@ +import { Controller, Get, Post, Body, Query, UseGuards } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiBearerAuth, ApiResponse, ApiQuery } from '@nestjs/swagger'; +import { UserApplicationService } from '@/application/services/user-application.service'; +import { JwtAuthGuard, Public, CurrentUser, CurrentUserData } from '@/shared/guards/jwt-auth.guard'; +import { + ValidateReferralCodeQuery, GetReferralStatsQuery, GenerateReferralLinkCommand, +} from '@/application/commands'; +import { + GenerateReferralLinkDto, MeResponseDto, ReferralValidationResponseDto, + ReferralLinkResponseDto, ReferralStatsResponseDto, +} from '@/api/dto'; + +@ApiTags('Referrals') +@Controller() +@UseGuards(JwtAuthGuard) +export class ReferralsController { + constructor(private readonly userService: UserApplicationService) {} + + /** + * GET /api/me - 获取当前登录用户信息 + 推荐码 + */ + @Get('me') + @ApiBearerAuth() + @ApiOperation({ summary: '获取当前登录用户信息', description: '返回用户基本信息、推荐码和推荐链接' }) + @ApiResponse({ status: 200, type: MeResponseDto }) + async getMe(@CurrentUser() user: CurrentUserData): Promise { + return this.userService.getMe(user.userId); + } + + /** + * GET /api/referrals/validate - 校验推荐码是否合法 + */ + @Public() + @Get('referrals/validate') + @ApiOperation({ summary: '校验推荐码', description: '创建账号时校验推荐码是否合法' }) + @ApiQuery({ name: 'code', description: '推荐码', required: true }) + @ApiResponse({ status: 200, type: ReferralValidationResponseDto }) + async validateReferralCode(@Query('code') code: string): Promise { + return this.userService.validateReferralCode(new ValidateReferralCodeQuery(code)); + } + + /** + * POST /api/referrals/links - 为当前登录用户生成短链/渠道链接 + */ + @Post('referrals/links') + @ApiBearerAuth() + @ApiOperation({ summary: '生成推荐链接', description: '为当前登录用户生成短链/渠道链接' }) + @ApiResponse({ status: 201, type: ReferralLinkResponseDto }) + async generateReferralLink( + @CurrentUser() user: CurrentUserData, + @Body() dto: GenerateReferralLinkDto, + ): Promise { + return this.userService.generateReferralLink( + new GenerateReferralLinkCommand(user.userId, dto.channel, dto.campaignId), + ); + } + + /** + * GET /api/referrals/stats - 查询登录用户的邀请记录 + */ + @Get('referrals/stats') + @ApiBearerAuth() + @ApiOperation({ summary: '查询邀请统计', description: '查询登录用户的邀请记录和统计数据' }) + @ApiResponse({ status: 200, type: ReferralStatsResponseDto }) + async getReferralStats(@CurrentUser() user: CurrentUserData): Promise { + return this.userService.getReferralStats(new GetReferralStatsQuery(user.userId)); + } +} diff --git a/backend/services/identity-service/src/api/dto/index.ts b/backend/services/identity-service/src/api/dto/index.ts index 1344063a..5a6ccfd1 100644 --- a/backend/services/identity-service/src/api/dto/index.ts +++ b/backend/services/identity-service/src/api/dto/index.ts @@ -186,3 +186,136 @@ export class LoginResponseDto { @ApiProperty() refreshToken: string; } + +// ============ Referral DTOs ============ + +export class GenerateReferralLinkDto { + @ApiPropertyOptional({ description: '渠道标识: wechat, telegram, twitter 等' }) + @IsOptional() + @IsString() + channel?: string; + + @ApiPropertyOptional({ description: '活动ID' }) + @IsOptional() + @IsString() + campaignId?: string; +} + +export class MeResponseDto { + @ApiProperty() + userId: string; + + @ApiProperty({ description: '账户序列号' }) + accountSequence: number; + + @ApiProperty({ nullable: true }) + phoneNumber: string | null; + + @ApiProperty() + nickname: string; + + @ApiProperty({ nullable: true }) + avatarUrl: string | null; + + @ApiProperty({ description: '推荐码' }) + referralCode: string; + + @ApiProperty({ description: '完整推荐链接' }) + referralLink: string; + + @ApiProperty({ description: '钱包地址列表' }) + walletAddresses: Array<{ chainType: string; address: string }>; + + @ApiProperty() + kycStatus: string; + + @ApiProperty() + status: string; + + @ApiProperty() + registeredAt: Date; +} + +export class ReferralValidationResponseDto { + @ApiProperty({ description: '推荐码是否有效' }) + valid: boolean; + + @ApiPropertyOptional() + referralCode?: string; + + @ApiPropertyOptional({ description: '邀请人信息' }) + inviterInfo?: { + accountSequence: number; + nickname: string; + avatarUrl: string | null; + }; + + @ApiPropertyOptional({ description: '错误信息' }) + message?: string; +} + +export class ReferralLinkResponseDto { + @ApiProperty() + linkId: string; + + @ApiProperty() + referralCode: string; + + @ApiProperty({ description: '短链' }) + shortUrl: string; + + @ApiProperty({ description: '完整链接' }) + fullUrl: string; + + @ApiProperty({ nullable: true }) + channel: string | null; + + @ApiProperty({ nullable: true }) + campaignId: string | null; + + @ApiProperty() + createdAt: Date; +} + +export class InviteRecordDto { + @ApiProperty() + accountSequence: number; + + @ApiProperty() + nickname: string; + + @ApiProperty({ nullable: true }) + avatarUrl: string | null; + + @ApiProperty() + registeredAt: Date; + + @ApiProperty({ description: '1=直接邀请, 2=间接邀请' }) + level: number; +} + +export class ReferralStatsResponseDto { + @ApiProperty() + referralCode: string; + + @ApiProperty({ description: '总邀请人数' }) + totalInvites: number; + + @ApiProperty({ description: '直接邀请人数' }) + directInvites: number; + + @ApiProperty({ description: '间接邀请人数 (二级)' }) + indirectInvites: number; + + @ApiProperty({ description: '今日邀请' }) + todayInvites: number; + + @ApiProperty({ description: '本周邀请' }) + thisWeekInvites: number; + + @ApiProperty({ description: '本月邀请' }) + thisMonthInvites: number; + + @ApiProperty({ description: '最近邀请记录', type: [InviteRecordDto] }) + recentInvites: InviteRecordDto[]; +} diff --git a/backend/services/identity-service/src/app.module.ts b/backend/services/identity-service/src/app.module.ts index 91b72ddc..868fede1 100644 --- a/backend/services/identity-service/src/app.module.ts +++ b/backend/services/identity-service/src/app.module.ts @@ -1,6 +1,7 @@ import { Module, Global } from '@nestjs/common'; import { ConfigModule, ConfigService } from '@nestjs/config'; import { JwtModule } from '@nestjs/jwt'; +import { HttpModule } from '@nestjs/axios'; import { APP_FILTER, APP_INTERCEPTOR, APP_GUARD } from '@nestjs/core'; // Config @@ -9,6 +10,7 @@ import { appConfig, databaseConfig, jwtConfig, redisConfig, kafkaConfig, smsConf // Controllers import { UserAccountController } from '@/api/controllers/user-account.controller'; import { HealthController } from '@/api/controllers/health.controller'; +import { ReferralsController } from '@/api/controllers/referrals.controller'; // Application Services import { UserApplicationService } from '@/application/services/user-application.service'; @@ -19,13 +21,18 @@ import { AccountSequenceGeneratorService, UserValidatorService, WalletGeneratorService, } from '@/domain/services'; import { USER_ACCOUNT_REPOSITORY } from '@/domain/repositories/user-account.repository.interface'; +import { MPC_KEY_SHARE_REPOSITORY } from '@/domain/repositories/mpc-key-share.repository.interface'; // Infrastructure import { PrismaService } from '@/infrastructure/persistence/prisma/prisma.service'; import { UserAccountRepositoryImpl } from '@/infrastructure/persistence/repositories/user-account.repository.impl'; +import { MpcKeyShareRepositoryImpl } from '@/infrastructure/persistence/repositories/mpc-key-share.repository.impl'; import { RedisService } from '@/infrastructure/redis/redis.service'; import { EventPublisherService } from '@/infrastructure/kafka/event-publisher.service'; import { SmsService } from '@/infrastructure/external/sms/sms.service'; +import { MpcClientService, MpcWalletService } from '@/infrastructure/external/mpc'; +import { BackupClientService, MpcShareStorageService } from '@/infrastructure/external/backup'; +import { WalletGeneratorServiceImpl } from '@/infrastructure/external/blockchain/wallet-generator.service.impl'; // Shared import { GlobalExceptionFilter, TransformInterceptor } from '@/shared/filters/global-exception.filter'; @@ -34,8 +41,38 @@ import { JwtAuthGuard } from '@/shared/guards/jwt-auth.guard'; // ============ Infrastructure Module ============ @Global() @Module({ - providers: [PrismaService, RedisService, EventPublisherService, SmsService], - exports: [PrismaService, RedisService, EventPublisherService, SmsService], + imports: [ + HttpModule.register({ + timeout: 300000, + maxRedirects: 5, + }), + ], + providers: [ + PrismaService, + RedisService, + EventPublisherService, + SmsService, + MpcClientService, + MpcWalletService, + BackupClientService, + MpcShareStorageService, + // WalletGeneratorService 抽象类由 WalletGeneratorServiceImpl 实现 + WalletGeneratorServiceImpl, + { provide: WalletGeneratorService, useExisting: WalletGeneratorServiceImpl }, + { provide: MPC_KEY_SHARE_REPOSITORY, useClass: MpcKeyShareRepositoryImpl }, + ], + exports: [ + PrismaService, + RedisService, + EventPublisherService, + SmsService, + MpcClientService, + MpcWalletService, + BackupClientService, + MpcShareStorageService, + WalletGeneratorService, + MPC_KEY_SHARE_REPOSITORY, + ], }) export class InfrastructureModule {} @@ -46,13 +83,13 @@ export class InfrastructureModule {} { provide: USER_ACCOUNT_REPOSITORY, useClass: UserAccountRepositoryImpl }, AccountSequenceGeneratorService, UserValidatorService, - WalletGeneratorService, + // WalletGeneratorService 由 InfrastructureModule 提供 ], exports: [ USER_ACCOUNT_REPOSITORY, AccountSequenceGeneratorService, UserValidatorService, - WalletGeneratorService, + // WalletGeneratorService 由 InfrastructureModule 导出 ], }) export class DomainModule {} @@ -68,7 +105,7 @@ export class ApplicationModule {} // ============ API Module ============ @Module({ imports: [ApplicationModule], - controllers: [HealthController, UserAccountController], + controllers: [HealthController, UserAccountController, ReferralsController], }) export class ApiModule {} diff --git a/backend/services/identity-service/src/application/application.module.ts b/backend/services/identity-service/src/application/application.module.ts index f7a3f643..9ca565b2 100644 --- a/backend/services/identity-service/src/application/application.module.ts +++ b/backend/services/identity-service/src/application/application.module.ts @@ -1,6 +1,7 @@ import { Module } from '@nestjs/common'; import { UserApplicationService } from './services/user-application.service'; import { TokenService } from './services/token.service'; +import { DepositService } from './services/deposit.service'; import { AutoCreateAccountHandler } from './commands/auto-create-account/auto-create-account.handler'; import { RecoverByMnemonicHandler } from './commands/recover-by-mnemonic/recover-by-mnemonic.handler'; import { RecoverByPhoneHandler } from './commands/recover-by-phone/recover-by-phone.handler'; @@ -15,6 +16,7 @@ import { InfrastructureModule } from '@/infrastructure/infrastructure.module'; providers: [ UserApplicationService, TokenService, + DepositService, AutoCreateAccountHandler, RecoverByMnemonicHandler, RecoverByPhoneHandler, @@ -25,6 +27,7 @@ import { InfrastructureModule } from '@/infrastructure/infrastructure.module'; exports: [ UserApplicationService, TokenService, + DepositService, AutoCreateAccountHandler, RecoverByMnemonicHandler, RecoverByPhoneHandler, diff --git a/backend/services/identity-service/src/application/commands/auto-create-account/auto-create-account.handler.ts b/backend/services/identity-service/src/application/commands/auto-create-account/auto-create-account.handler.ts index 38588a1e..24b608f4 100644 --- a/backend/services/identity-service/src/application/commands/auto-create-account/auto-create-account.handler.ts +++ b/backend/services/identity-service/src/application/commands/auto-create-account/auto-create-account.handler.ts @@ -1,4 +1,4 @@ -import { Injectable, Inject } from '@nestjs/common'; +import { Injectable, Inject, Logger } from '@nestjs/common'; import { AutoCreateAccountCommand } from './auto-create-account.command'; import { UserAccountRepository, USER_ACCOUNT_REPOSITORY } from '@/domain/repositories/user-account.repository.interface'; import { UserAccount } from '@/domain/aggregates/user-account/user-account.aggregate'; @@ -8,9 +8,12 @@ import { TokenService } from '@/application/services/token.service'; import { EventPublisherService } from '@/infrastructure/kafka/event-publisher.service'; import { ApplicationError } from '@/shared/exceptions/domain.exception'; import { AutoCreateAccountResult } from '../index'; +import { MpcShareStorageService } from '@/infrastructure/external/backup/mpc-share-storage.service'; @Injectable() export class AutoCreateAccountHandler { + private readonly logger = new Logger(AutoCreateAccountHandler.name); + constructor( @Inject(USER_ACCOUNT_REPOSITORY) private readonly userRepository: UserAccountRepository, @@ -19,6 +22,7 @@ export class AutoCreateAccountHandler { private readonly walletGenerator: WalletGeneratorService, private readonly tokenService: TokenService, private readonly eventPublisher: EventPublisherService, + private readonly mpcShareStorage: MpcShareStorageService, ) {} async execute(command: AutoCreateAccountCommand): Promise { @@ -45,11 +49,27 @@ export class AutoCreateAccountHandler { city: CityCode.create(command.cityCode || 'DEFAULT'), }); - const { mnemonic, wallets } = this.walletGenerator.generateWalletSystem({ - userId: account.userId, + // 使用 MPC 2-of-3 生成三链钱包 + this.logger.log(`Generating MPC wallet for user=${account.userId.toString()}`); + const mpcResult = await this.walletGenerator.generateMpcWalletSystem({ + userId: account.userId.toString(), deviceId: command.deviceId, }); + // 将 MPC 钱包信息转换为领域实体 + const wallets = this.walletGenerator.convertToWalletEntities( + account.userId, + mpcResult.wallets, + ); + + // 保存备份分片到备份服务 + this.logger.log(`Storing backup share for user=${account.userId.toString()}`); + await this.mpcShareStorage.storeBackupShare({ + userId: account.userId.toString(), + shareData: mpcResult.backupShareData, + publicKey: mpcResult.publicKey, + }); + account.bindMultipleWalletAddresses(wallets); await this.userRepository.save(account); await this.userRepository.saveWallets(account.userId, Array.from(wallets.values())); @@ -63,11 +83,15 @@ export class AutoCreateAccountHandler { await this.eventPublisher.publishAll(account.domainEvents); account.clearDomainEvents(); + this.logger.log(`Account created successfully: userId=${account.userId.toString()}, seq=${account.accountSequence.value}`); + return { userId: account.userId.toString(), accountSequence: account.accountSequence.value, referralCode: account.referralCode.value, - mnemonic: mnemonic.value, + mnemonic: '', // MPC 模式下不再使用助记词 + clientShareData: mpcResult.clientShareData, // 客户端需安全存储此分片 + publicKey: mpcResult.publicKey, walletAddresses: { kava: wallets.get(ChainType.KAVA)!.address, dst: wallets.get(ChainType.DST)!.address, diff --git a/backend/services/identity-service/src/application/commands/index.ts b/backend/services/identity-service/src/application/commands/index.ts index b100ffdf..cd36068a 100644 --- a/backend/services/identity-service/src/application/commands/index.ts +++ b/backend/services/identity-service/src/application/commands/index.ts @@ -129,6 +129,22 @@ export class GetUserByReferralCodeQuery { constructor(public readonly referralCode: string) {} } +export class ValidateReferralCodeQuery { + constructor(public readonly referralCode: string) {} +} + +export class GetReferralStatsQuery { + constructor(public readonly userId: string) {} +} + +export class GenerateReferralLinkCommand { + constructor( + public readonly userId: string, + public readonly channel?: string, // 渠道标识: wechat, telegram, twitter 等 + public readonly campaignId?: string, // 活动ID + ) {} +} + // ============ Results ============ export interface AutoCreateAccountResult { userId: string; @@ -206,3 +222,55 @@ export interface UserBriefDTO { nickname: string; avatarUrl: string | null; } + +export interface ReferralCodeValidationResult { + valid: boolean; + referralCode?: string; + inviterInfo?: { + accountSequence: number; + nickname: string; + avatarUrl: string | null; + }; + message?: string; +} + +export interface ReferralLinkResult { + linkId: string; + referralCode: string; + shortUrl: string; + fullUrl: string; + channel: string | null; + campaignId: string | null; + createdAt: Date; +} + +export interface ReferralStatsResult { + referralCode: string; + totalInvites: number; // 总邀请人数 + directInvites: number; // 直接邀请人数 + indirectInvites: number; // 间接邀请人数 (二级) + todayInvites: number; // 今日邀请 + thisWeekInvites: number; // 本周邀请 + thisMonthInvites: number; // 本月邀请 + recentInvites: Array<{ // 最近邀请记录 + accountSequence: number; + nickname: string; + avatarUrl: string | null; + registeredAt: Date; + level: number; // 1=直接, 2=间接 + }>; +} + +export interface MeResult { + userId: string; + accountSequence: number; + phoneNumber: string | null; + nickname: string; + avatarUrl: string | null; + referralCode: string; + referralLink: string; // 完整推荐链接 + walletAddresses: Array<{ chainType: string; address: string }>; + kycStatus: string; + status: string; + registeredAt: Date; +} diff --git a/backend/services/identity-service/src/application/services/deposit.service.ts b/backend/services/identity-service/src/application/services/deposit.service.ts new file mode 100644 index 00000000..804f9891 --- /dev/null +++ b/backend/services/identity-service/src/application/services/deposit.service.ts @@ -0,0 +1,194 @@ +import { Injectable, Logger, BadRequestException } from '@nestjs/common'; +import { UserAccountRepositoryImpl } from '@/infrastructure/persistence/repositories/user-account.repository.impl'; +import { BlockchainQueryService, UsdtBalance } from '@/infrastructure/external/blockchain/blockchain-query.service'; +import { UserId, ChainType } from '@/domain/value-objects'; +import { WalletAddress } from '@/domain/entities/wallet-address.entity'; + +/** + * 充值地址响应 + */ +export interface DepositAddressResponse { + kavaAddress: string | null; + bscAddress: string | null; + isValid: boolean; + message?: string; +} + +/** + * 余额响应 + */ +export interface BalanceResponse { + kava: UsdtBalance | null; + bsc: UsdtBalance | null; +} + +/** + * 充值服务 + * + * 提供充值地址获取和余额查询功能 + */ +@Injectable() +export class DepositService { + private readonly logger = new Logger(DepositService.name); + + constructor( + private readonly userAccountRepository: UserAccountRepositoryImpl, + private readonly blockchainQueryService: BlockchainQueryService, + ) {} + + /** + * 获取用户的充值地址 + * + * 验证地址签名是否有效,防止使用被篡改的地址 + */ + async getDepositAddresses(userId: string): Promise { + this.logger.log(`获取充值地址: userId=${userId}`); + + const userAccount = await this.userAccountRepository.findById(UserId.create(userId)); + if (!userAccount) { + throw new BadRequestException('用户不存在'); + } + + const walletAddresses = userAccount.getAllWalletAddresses(); + if (!walletAddresses || walletAddresses.length === 0) { + return { + kavaAddress: null, + bscAddress: null, + isValid: false, + message: '充值账户异常:未找到钱包地址,请联系客服', + }; + } + + // 查找 KAVA 和 BSC 地址 + const kavaWallet = walletAddresses.find((w: WalletAddress) => w.chainType === ChainType.KAVA); + const bscWallet = walletAddresses.find((w: WalletAddress) => w.chainType === ChainType.BSC); + + // 验证地址签名 + const validationResults = await Promise.all([ + kavaWallet ? this.validateWalletAddress(kavaWallet) : Promise.resolve(true), + bscWallet ? this.validateWalletAddress(bscWallet) : Promise.resolve(true), + ]); + + const [kavaValid, bscValid] = validationResults; + + // 如果有任何一个地址验证失败,返回错误 + if (!kavaValid || !bscValid) { + this.logger.warn(`地址验证失败: userId=${userId}, kavaValid=${kavaValid}, bscValid=${bscValid}`); + return { + kavaAddress: null, + bscAddress: null, + isValid: false, + message: '充值账户异常:地址验证失败,请重试或联系客服', + }; + } + + // 检查地址状态 + if (kavaWallet && kavaWallet.status !== 'ACTIVE') { + this.logger.warn(`KAVA 地址状态异常: userId=${userId}, status=${kavaWallet.status}`); + return { + kavaAddress: null, + bscAddress: null, + isValid: false, + message: '充值账户异常:KAVA 地址已禁用,请联系客服', + }; + } + + if (bscWallet && bscWallet.status !== 'ACTIVE') { + this.logger.warn(`BSC 地址状态异常: userId=${userId}, status=${bscWallet.status}`); + return { + kavaAddress: null, + bscAddress: null, + isValid: false, + message: '充值账户异常:BSC 地址已禁用,请联系客服', + }; + } + + return { + kavaAddress: kavaWallet?.address || null, + bscAddress: bscWallet?.address || null, + isValid: true, + }; + } + + /** + * 查询用户的 USDT 余额 + */ + async getUsdtBalances(userId: string): Promise { + this.logger.log(`查询 USDT 余额: userId=${userId}`); + + // 先获取充值地址 + const depositAddresses = await this.getDepositAddresses(userId); + + if (!depositAddresses.isValid) { + throw new BadRequestException(depositAddresses.message || '获取充值地址失败'); + } + + const results: BalanceResponse = { + kava: null, + bsc: null, + }; + + // 查询 KAVA 余额 + if (depositAddresses.kavaAddress) { + try { + results.kava = await this.blockchainQueryService.getUsdtBalance( + 'KAVA', + depositAddresses.kavaAddress, + ); + } catch (error) { + this.logger.error(`查询 KAVA USDT 余额失败: ${error.message}`); + results.kava = { + chainType: 'KAVA', + address: depositAddresses.kavaAddress, + balance: '0', + rawBalance: '0', + decimals: 6, + }; + } + } + + // 查询 BSC 余额 + if (depositAddresses.bscAddress) { + try { + results.bsc = await this.blockchainQueryService.getUsdtBalance( + 'BSC', + depositAddresses.bscAddress, + ); + } catch (error) { + this.logger.error(`查询 BSC USDT 余额失败: ${error.message}`); + results.bsc = { + chainType: 'BSC', + address: depositAddresses.bscAddress, + balance: '0', + rawBalance: '0', + decimals: 18, + }; + } + } + + return results; + } + + /** + * 验证钱包地址签名 + */ + private async validateWalletAddress(wallet: any): Promise { + try { + // 如果没有签名数据 (旧版本创建的地址),暂时允许通过 + if (!wallet.publicKey || !wallet.mpcSignature?.r) { + this.logger.warn(`钱包地址无签名数据: chainType=${wallet.chainType}`); + return true; + } + + // 验证签名 + const isValid = await wallet.verifySignature(); + if (!isValid) { + this.logger.error(`钱包地址签名验证失败: chainType=${wallet.chainType}, address=${wallet.address}`); + } + return isValid; + } catch (error) { + this.logger.error(`验证钱包地址签名异常: ${error.message}`); + return false; + } + } +} diff --git a/backend/services/identity-service/src/application/services/user-application.service.referral.spec.ts b/backend/services/identity-service/src/application/services/user-application.service.referral.spec.ts new file mode 100644 index 00000000..0bb7c4e7 --- /dev/null +++ b/backend/services/identity-service/src/application/services/user-application.service.referral.spec.ts @@ -0,0 +1,634 @@ +import { Test, TestingModule } from '@nestjs/testing'; +import { UserApplicationService } from './user-application.service'; +import { USER_ACCOUNT_REPOSITORY, UserAccountRepository, ReferralLinkData, CreateReferralLinkParams } from '@/domain/repositories/user-account.repository.interface'; +import { MPC_KEY_SHARE_REPOSITORY, MpcKeyShareRepository } from '@/domain/repositories/mpc-key-share.repository.interface'; +import { UserAccount } from '@/domain/aggregates/user-account/user-account.aggregate'; +import { AccountSequence, ReferralCode, UserId, ProvinceCode, CityCode, AccountStatus, KYCStatus, DeviceInfo } from '@/domain/value-objects'; +import { ConfigService } from '@nestjs/config'; +import { ValidateReferralCodeQuery, GetReferralStatsQuery, GenerateReferralLinkCommand } from '@/application/commands'; +import { ApplicationError } from '@/shared/exceptions/domain.exception'; +import { AccountSequenceGeneratorService, UserValidatorService, WalletGeneratorService } from '@/domain/services'; +import { TokenService } from './token.service'; +import { RedisService } from '@/infrastructure/redis/redis.service'; +import { SmsService } from '@/infrastructure/external/sms/sms.service'; +import { EventPublisherService } from '@/infrastructure/kafka/event-publisher.service'; +import { MpcWalletService } from '@/infrastructure/external/mpc'; +import { BackupClientService } from '@/infrastructure/external/backup'; + +describe('UserApplicationService - Referral APIs', () => { + let service: UserApplicationService; + let mockUserRepository: jest.Mocked; + + // Helper function to create a test account using UserAccount.reconstruct + const createMockAccount = (params: { + userId?: string; + accountSequence?: number; + referralCode?: string; + nickname?: string; + avatarUrl?: string | null; + isActive?: boolean; + inviterSequence?: number | null; + registeredAt?: Date; + } = {}): UserAccount => { + const devices = [ + new DeviceInfo('device-001', 'Test Device', new Date(), new Date()), + ]; + + return UserAccount.reconstruct({ + userId: params.userId || '123456789', + accountSequence: params.accountSequence || 1, + devices, + phoneNumber: '13800138000', + nickname: params.nickname || '用户1', + avatarUrl: params.avatarUrl ?? null, + inviterSequence: params.inviterSequence ?? null, + referralCode: params.referralCode || 'ABC123', + province: '110000', + city: '110100', + address: null, + walletAddresses: [], + kycInfo: null, + kycStatus: KYCStatus.NOT_VERIFIED, + status: params.isActive !== false ? AccountStatus.ACTIVE : AccountStatus.FROZEN, + registeredAt: params.registeredAt || new Date(), + lastLoginAt: null, + updatedAt: new Date(), + }); + }; + + beforeEach(async () => { + mockUserRepository = { + save: jest.fn(), + saveWallets: jest.fn(), + findById: jest.fn(), + findByAccountSequence: jest.fn(), + findByDeviceId: jest.fn(), + findByPhoneNumber: jest.fn(), + findByReferralCode: jest.fn(), + findByWalletAddress: jest.fn(), + getMaxAccountSequence: jest.fn(), + getNextAccountSequence: jest.fn(), + findUsers: jest.fn(), + countUsers: jest.fn(), + findByInviterSequence: jest.fn(), + createReferralLink: jest.fn(), + findReferralLinksByUserId: jest.fn(), + }; + + const mockMpcKeyShareRepository: jest.Mocked = { + saveServerShare: jest.fn(), + findByUserId: jest.fn(), + findByPublicKey: jest.fn(), + updateStatus: jest.fn(), + rotateShare: jest.fn(), + }; + + const mockConfigService = { + get: jest.fn((key: string) => { + const config: Record = { + 'APP_BASE_URL': 'https://app.rwadurian.com', + 'MPC_MODE': 'local', + }; + return config[key]; + }), + }; + + const mockAccountSequenceGeneratorService = { + getNext: jest.fn().mockResolvedValue(AccountSequence.create(1)), + }; + + const mockUserValidatorService = { + validateUniquePhone: jest.fn(), + }; + + const mockWalletGeneratorService = { + generateWallets: jest.fn(), + }; + + const mockTokenService = { + generateAccessToken: jest.fn().mockReturnValue('mock-access-token'), + generateRefreshToken: jest.fn().mockReturnValue('mock-refresh-token'), + generateDeviceRefreshToken: jest.fn().mockReturnValue('mock-device-refresh-token'), + verifyRefreshToken: jest.fn(), + }; + + const mockRedisService = { + get: jest.fn(), + set: jest.fn(), + del: jest.fn(), + setWithExpiry: jest.fn(), + }; + + const mockSmsService = { + sendSmsCode: jest.fn(), + }; + + const mockEventPublisherService = { + publish: jest.fn(), + }; + + const mockMpcWalletService = { + generateMpcWallet: jest.fn(), + }; + + const mockBackupClientService = { + storeBackupShare: jest.fn(), + }; + + const module: TestingModule = await Test.createTestingModule({ + providers: [ + UserApplicationService, + { + provide: USER_ACCOUNT_REPOSITORY, + useValue: mockUserRepository, + }, + { + provide: MPC_KEY_SHARE_REPOSITORY, + useValue: mockMpcKeyShareRepository, + }, + { + provide: ConfigService, + useValue: mockConfigService, + }, + { + provide: AccountSequenceGeneratorService, + useValue: mockAccountSequenceGeneratorService, + }, + { + provide: UserValidatorService, + useValue: mockUserValidatorService, + }, + { + provide: WalletGeneratorService, + useValue: mockWalletGeneratorService, + }, + { + provide: TokenService, + useValue: mockTokenService, + }, + { + provide: RedisService, + useValue: mockRedisService, + }, + { + provide: SmsService, + useValue: mockSmsService, + }, + { + provide: EventPublisherService, + useValue: mockEventPublisherService, + }, + { + provide: MpcWalletService, + useValue: mockMpcWalletService, + }, + { + provide: BackupClientService, + useValue: mockBackupClientService, + }, + ], + }).compile(); + + service = module.get(UserApplicationService); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + // ============ GET /api/me Tests ============ + describe('getMe', () => { + it('should return current user info with referral code and link', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + accountSequence: 1, + nickname: '测试用户', + referralCode: 'ABC123', + }); + + mockUserRepository.findById.mockResolvedValue(mockAccount); + + const result = await service.getMe('123456789'); + + expect(result).toEqual({ + userId: '123456789', + accountSequence: 1, + phoneNumber: '138****8000', // masked + nickname: '测试用户', + avatarUrl: null, + referralCode: 'ABC123', + referralLink: 'https://app.rwadurian.com/invite/ABC123', + walletAddresses: [], + kycStatus: KYCStatus.NOT_VERIFIED, + status: AccountStatus.ACTIVE, + registeredAt: expect.any(Date), + }); + + expect(mockUserRepository.findById).toHaveBeenCalledWith(expect.any(UserId)); + }); + + it('should throw error when user not found', async () => { + mockUserRepository.findById.mockResolvedValue(null); + + // Use valid numeric string for userId + await expect(service.getMe('999999999')).rejects.toThrow(ApplicationError); + await expect(service.getMe('999999999')).rejects.toThrow('用户不存在'); + }); + }); + + // ============ GET /api/referrals/validate Tests ============ + describe('validateReferralCode', () => { + it('should return valid=true for existing active referral code', async () => { + const mockInviter = createMockAccount({ + accountSequence: 100, + nickname: '邀请人', + avatarUrl: 'https://example.com/avatar.jpg', + referralCode: 'INVTE1', + isActive: true, + }); + + mockUserRepository.findByReferralCode.mockResolvedValue(mockInviter); + + const result = await service.validateReferralCode( + new ValidateReferralCodeQuery('INVTE1') + ); + + expect(result).toEqual({ + valid: true, + referralCode: 'INVTE1', + inviterInfo: { + accountSequence: 100, + nickname: '邀请人', + avatarUrl: 'https://example.com/avatar.jpg', + }, + }); + }); + + it('should return valid=false for non-existent referral code', async () => { + mockUserRepository.findByReferralCode.mockResolvedValue(null); + + const result = await service.validateReferralCode( + new ValidateReferralCodeQuery('INVLD1') + ); + + expect(result).toEqual({ + valid: false, + message: '推荐码不存在', + }); + }); + + it('should return valid=false for frozen inviter account', async () => { + const frozenInviter = createMockAccount({ + referralCode: 'FROZN1', + isActive: false, + }); + + mockUserRepository.findByReferralCode.mockResolvedValue(frozenInviter); + + const result = await service.validateReferralCode( + new ValidateReferralCodeQuery('FROZN1') + ); + + expect(result).toEqual({ + valid: false, + message: '推荐人账户已冻结', + }); + }); + + it('should return valid=false for invalid referral code format', async () => { + const result = await service.validateReferralCode( + new ValidateReferralCodeQuery('invalid-format-too-long') + ); + + expect(result.valid).toBe(false); + expect(result.message).toBe('推荐码格式无效'); + }); + }); + + // ============ POST /api/referrals/links Tests ============ + describe('generateReferralLink', () => { + it('should generate a new referral link with channel', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + referralCode: 'ABC123', + }); + + const mockLinkData: ReferralLinkData = { + linkId: BigInt(1), + userId: BigInt(123456789), + referralCode: 'ABC123', + shortCode: 'XyZ789', + channel: 'wechat', + campaignId: null, + createdAt: new Date('2024-01-15T10:00:00Z'), + }; + + mockUserRepository.findById.mockResolvedValue(mockAccount); + mockUserRepository.createReferralLink.mockResolvedValue(mockLinkData); + + const result = await service.generateReferralLink( + new GenerateReferralLinkCommand('123456789', 'wechat') + ); + + expect(result).toEqual({ + linkId: '1', + referralCode: 'ABC123', + shortUrl: expect.stringMatching(/^https:\/\/app\.rwadurian\.com\/r\/[A-Za-z0-9]{6}$/), + fullUrl: 'https://app.rwadurian.com/invite/ABC123?ch=wechat', + channel: 'wechat', + campaignId: null, + createdAt: expect.any(Date), + }); + + expect(mockUserRepository.createReferralLink).toHaveBeenCalledWith({ + userId: expect.any(BigInt), + referralCode: 'ABC123', + shortCode: expect.any(String), + channel: 'wechat', + campaignId: null, + }); + }); + + it('should generate a referral link with campaign ID', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + referralCode: 'ABC123', + }); + + const mockLinkData: ReferralLinkData = { + linkId: BigInt(2), + userId: BigInt(123456789), + referralCode: 'ABC123', + shortCode: 'AbC456', + channel: 'telegram', + campaignId: 'spring2024', + createdAt: new Date('2024-01-15T10:00:00Z'), + }; + + mockUserRepository.findById.mockResolvedValue(mockAccount); + mockUserRepository.createReferralLink.mockResolvedValue(mockLinkData); + + const result = await service.generateReferralLink( + new GenerateReferralLinkCommand('123456789', 'telegram', 'spring2024') + ); + + expect(result.channel).toBe('telegram'); + expect(result.campaignId).toBe('spring2024'); + }); + + it('should generate link with default channel when not specified', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + referralCode: 'ABC123', + }); + + const mockLinkData: ReferralLinkData = { + linkId: BigInt(3), + userId: BigInt(123456789), + referralCode: 'ABC123', + shortCode: 'DeF789', + channel: null, + campaignId: null, + createdAt: new Date(), + }; + + mockUserRepository.findById.mockResolvedValue(mockAccount); + mockUserRepository.createReferralLink.mockResolvedValue(mockLinkData); + + const result = await service.generateReferralLink( + new GenerateReferralLinkCommand('123456789') + ); + + expect(result.fullUrl).toContain('ch=default'); + expect(result.channel).toBeNull(); + }); + + it('should throw error when user not found', async () => { + mockUserRepository.findById.mockResolvedValue(null); + + // Use valid numeric string for userId + await expect( + service.generateReferralLink(new GenerateReferralLinkCommand('999999999')) + ).rejects.toThrow(ApplicationError); + }); + }); + + // ============ GET /api/referrals/stats Tests ============ + describe('getReferralStats', () => { + it('should return referral stats with direct and indirect invites', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + accountSequence: 1, + referralCode: 'ABC123', + }); + + // Direct invites (invited by user 1) + const directInvite1 = createMockAccount({ + userId: '200000001', + accountSequence: 2, + nickname: '直接邀请1', + inviterSequence: 1, + registeredAt: new Date(), + }); + + const directInvite2 = createMockAccount({ + userId: '200000002', + accountSequence: 3, + nickname: '直接邀请2', + inviterSequence: 1, + registeredAt: new Date(), + }); + + // Indirect invite (invited by user 2, who was invited by user 1) + const indirectInvite1 = createMockAccount({ + userId: '300000001', + accountSequence: 4, + nickname: '间接邀请1', + inviterSequence: 2, + registeredAt: new Date(), + }); + + mockUserRepository.findById.mockResolvedValue(mockAccount); + mockUserRepository.findByInviterSequence + .mockResolvedValueOnce([directInvite1, directInvite2]) // Direct invites of user 1 + .mockResolvedValueOnce([indirectInvite1]) // Indirect invites via user 2 + .mockResolvedValueOnce([]); // Indirect invites via user 3 (none) + + const result = await service.getReferralStats( + new GetReferralStatsQuery('123456789') + ); + + expect(result).toEqual({ + referralCode: 'ABC123', + totalInvites: 3, // 2 direct + 1 indirect + directInvites: 2, + indirectInvites: 1, + todayInvites: expect.any(Number), + thisWeekInvites: expect.any(Number), + thisMonthInvites: expect.any(Number), + recentInvites: expect.arrayContaining([ + expect.objectContaining({ + accountSequence: expect.any(Number), + nickname: expect.any(String), + level: expect.any(Number), // 1 for direct, 2 for indirect + }), + ]), + }); + + expect(result.recentInvites.length).toBeLessThanOrEqual(20); + }); + + it('should return empty stats when no invites', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + accountSequence: 1, + referralCode: 'ABC123', + }); + + mockUserRepository.findById.mockResolvedValue(mockAccount); + mockUserRepository.findByInviterSequence.mockResolvedValue([]); + + const result = await service.getReferralStats( + new GetReferralStatsQuery('123456789') + ); + + expect(result).toEqual({ + referralCode: 'ABC123', + totalInvites: 0, + directInvites: 0, + indirectInvites: 0, + todayInvites: 0, + thisWeekInvites: 0, + thisMonthInvites: 0, + recentInvites: [], + }); + }); + + it('should correctly calculate time-based stats', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + accountSequence: 1, + referralCode: 'ABC123', + }); + + const now = new Date(); + const todayInvite = createMockAccount({ + accountSequence: 2, + nickname: '今日邀请', + inviterSequence: 1, + registeredAt: now, + }); + + const yesterdayInvite = createMockAccount({ + accountSequence: 3, + nickname: '昨日邀请', + inviterSequence: 1, + registeredAt: new Date(now.getTime() - 24 * 60 * 60 * 1000), // yesterday + }); + + const lastMonthInvite = createMockAccount({ + accountSequence: 4, + nickname: '上月邀请', + inviterSequence: 1, + registeredAt: new Date(now.getFullYear(), now.getMonth() - 1, 15), + }); + + mockUserRepository.findById.mockResolvedValue(mockAccount); + mockUserRepository.findByInviterSequence + .mockResolvedValueOnce([todayInvite, yesterdayInvite, lastMonthInvite]) + .mockResolvedValue([]); // No second-level invites + + const result = await service.getReferralStats( + new GetReferralStatsQuery('123456789') + ); + + expect(result.directInvites).toBe(3); + expect(result.todayInvites).toBe(1); // Only today's invite + }); + + it('should throw error when user not found', async () => { + mockUserRepository.findById.mockResolvedValue(null); + + // Use valid numeric string for userId + await expect( + service.getReferralStats(new GetReferralStatsQuery('999999999')) + ).rejects.toThrow(ApplicationError); + }); + + it('should sort recent invites by registration date (newest first)', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + accountSequence: 1, + referralCode: 'ABC123', + }); + + const now = new Date(); + const oldInvite = createMockAccount({ + accountSequence: 2, + nickname: '旧邀请', + inviterSequence: 1, + registeredAt: new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000), // 7 days ago + }); + + const newInvite = createMockAccount({ + accountSequence: 3, + nickname: '新邀请', + inviterSequence: 1, + registeredAt: now, + }); + + mockUserRepository.findById.mockResolvedValue(mockAccount); + mockUserRepository.findByInviterSequence + .mockResolvedValueOnce([oldInvite, newInvite]) + .mockResolvedValue([]); + + const result = await service.getReferralStats( + new GetReferralStatsQuery('123456789') + ); + + // Newest should be first + expect(result.recentInvites[0].nickname).toBe('新邀请'); + expect(result.recentInvites[1].nickname).toBe('旧邀请'); + }); + }); + + // ============ Short Code Generation Tests ============ + describe('generateShortCode (private method behavior)', () => { + it('should generate short codes with 6 characters', async () => { + const mockAccount = createMockAccount({ + userId: '123456789', + referralCode: 'ABC123', + }); + + mockUserRepository.findById.mockResolvedValue(mockAccount); + + // Generate multiple links and verify short codes have correct length + for (let i = 0; i < 5; i++) { + const mockLinkData: ReferralLinkData = { + linkId: BigInt(i + 1), + userId: BigInt(123456789), + referralCode: 'ABC123', + shortCode: `code${i}`, + channel: null, + campaignId: null, + createdAt: new Date(), + }; + + mockUserRepository.createReferralLink.mockResolvedValueOnce(mockLinkData); + + await service.generateReferralLink( + new GenerateReferralLinkCommand('123456789', `channel${i}`) + ); + } + + // All generated short codes should have 6 characters + const createReferralLinkCalls = mockUserRepository.createReferralLink.mock.calls; + createReferralLinkCalls.forEach((call) => { + const params = call[0] as CreateReferralLinkParams; + expect(params.shortCode).toHaveLength(6); + // Should not contain confusing characters (I, l, O, 0, 1) + expect(params.shortCode).not.toMatch(/[IlO01]/); + }); + }); + }); +}); diff --git a/backend/services/identity-service/src/application/services/user-application.service.ts b/backend/services/identity-service/src/application/services/user-application.service.ts index 29b8a20b..c51c67c0 100644 --- a/backend/services/identity-service/src/application/services/user-application.service.ts +++ b/backend/services/identity-service/src/application/services/user-application.service.ts @@ -15,14 +15,17 @@ import { RedisService } from '@/infrastructure/redis/redis.service'; import { SmsService } from '@/infrastructure/external/sms/sms.service'; import { EventPublisherService } from '@/infrastructure/kafka/event-publisher.service'; import { MpcWalletService } from '@/infrastructure/external/mpc'; +import { BackupClientService } from '@/infrastructure/external/backup'; import { ApplicationError } from '@/shared/exceptions/domain.exception'; import { AutoCreateAccountCommand, RecoverByMnemonicCommand, RecoverByPhoneCommand, AutoLoginCommand, RegisterCommand, LoginCommand, BindPhoneNumberCommand, UpdateProfileCommand, SubmitKYCCommand, ReviewKYCCommand, RemoveDeviceCommand, SendSmsCodeCommand, GetMyProfileQuery, GetMyDevicesQuery, GetUserByReferralCodeQuery, + ValidateReferralCodeQuery, GetReferralStatsQuery, GenerateReferralLinkCommand, AutoCreateAccountResult, RecoverAccountResult, AutoLoginResult, RegisterResult, LoginResult, UserProfileDTO, DeviceDTO, UserBriefDTO, + ReferralCodeValidationResult, ReferralLinkResult, ReferralStatsResult, MeResult, } from '../commands'; @Injectable() @@ -38,6 +41,7 @@ export class UserApplicationService { private readonly validatorService: UserValidatorService, private readonly walletGenerator: WalletGeneratorService, private readonly mpcWalletService: MpcWalletService, + private readonly backupClient: BackupClientService, private readonly tokenService: TokenService, private readonly redisService: RedisService, private readonly smsService: SmsService, @@ -122,6 +126,18 @@ export class UserApplicationService { }); this.logger.log(`Server MPC share saved for user: ${account.userId.toString()}`); + // 10. 保存备份 MPC 分片到 backup-service (异地服务器) + // 注意: backup-service 必须部署在不同物理服务器,否则 MPC 安全性失效 + if (mpcResult.backupShareData) { + await this.backupClient.storeBackupShare({ + userId: account.userId.toString(), + accountSequence: account.accountSequence.value, + publicKey: mpcResult.publicKey, + encryptedShareData: mpcResult.backupShareData, + }); + this.logger.log(`Backup MPC share sent to backup-service for user: ${account.userId.toString()}`); + } + // 11. 生成 Token const tokens = await this.tokenService.generateTokenPair({ userId: account.userId.toString(), @@ -481,4 +497,163 @@ export class UserApplicationService { private generateSmsCode(): string { return String(Math.floor(100000 + Math.random() * 900000)); } + + // ============ 推荐/分享相关 API ============ + + /** + * 获取当前登录用户信息 (GET /api/me) + */ + async getMe(userId: string): Promise { + const account = await this.userRepository.findById(UserId.create(userId)); + if (!account) throw new ApplicationError('用户不存在'); + + const baseUrl = 'https://app.rwadurian.com'; // TODO: 从配置读取 + const referralLink = `${baseUrl}/invite/${account.referralCode.value}`; + + return { + userId: account.userId.toString(), + accountSequence: account.accountSequence.value, + phoneNumber: account.phoneNumber?.masked() || null, + nickname: account.nickname, + avatarUrl: account.avatarUrl, + referralCode: account.referralCode.value, + referralLink, + walletAddresses: account.getAllWalletAddresses().map((wa) => ({ + chainType: wa.chainType, + address: wa.address, + })), + kycStatus: account.kycStatus, + status: account.status, + registeredAt: account.registeredAt, + }; + } + + /** + * 验证推荐码是否有效 (GET /api/referrals/validate) + */ + async validateReferralCode(query: ValidateReferralCodeQuery): Promise { + try { + const referralCode = ReferralCode.create(query.referralCode); + const account = await this.userRepository.findByReferralCode(referralCode); + + if (!account) { + return { valid: false, message: '推荐码不存在' }; + } + + if (!account.isActive) { + return { valid: false, message: '推荐人账户已冻结' }; + } + + return { + valid: true, + referralCode: account.referralCode.value, + inviterInfo: { + accountSequence: account.accountSequence.value, + nickname: account.nickname, + avatarUrl: account.avatarUrl, + }, + }; + } catch (error) { + return { valid: false, message: '推荐码格式无效' }; + } + } + + /** + * 生成推荐链接 (POST /api/referrals/links) + */ + async generateReferralLink(command: GenerateReferralLinkCommand): Promise { + const account = await this.userRepository.findById(UserId.create(command.userId)); + if (!account) throw new ApplicationError('用户不存在'); + + // 生成短链码 (6位随机字符) + const shortCode = this.generateShortCode(); + const baseUrl = 'https://app.rwadurian.com'; // TODO: 从配置读取 + + // 保存到数据库 + const link = await this.userRepository.createReferralLink({ + userId: account.userId.value, + referralCode: account.referralCode.value, + shortCode, + channel: command.channel || null, + campaignId: command.campaignId || null, + }); + + return { + linkId: link.linkId.toString(), + referralCode: account.referralCode.value, + shortUrl: `${baseUrl}/r/${shortCode}`, + fullUrl: `${baseUrl}/invite/${account.referralCode.value}?ch=${command.channel || 'default'}`, + channel: command.channel || null, + campaignId: command.campaignId || null, + createdAt: link.createdAt, + }; + } + + /** + * 查询邀请统计 (GET /api/referrals/stats) + */ + async getReferralStats(query: GetReferralStatsQuery): Promise { + const account = await this.userRepository.findById(UserId.create(query.userId)); + if (!account) throw new ApplicationError('用户不存在'); + + // 查询直接邀请的用户 + const directInvites = await this.userRepository.findByInviterSequence(account.accountSequence); + + // 查询间接邀请 (二级) + let indirectInvites: typeof directInvites = []; + for (const invite of directInvites) { + const secondLevel = await this.userRepository.findByInviterSequence(invite.accountSequence); + indirectInvites = indirectInvites.concat(secondLevel); + } + + // 时间统计 + const now = new Date(); + const todayStart = new Date(now.getFullYear(), now.getMonth(), now.getDate()); + const weekStart = new Date(todayStart); + weekStart.setDate(weekStart.getDate() - weekStart.getDay()); + const monthStart = new Date(now.getFullYear(), now.getMonth(), 1); + + const todayInvites = directInvites.filter(u => u.registeredAt >= todayStart).length; + const thisWeekInvites = directInvites.filter(u => u.registeredAt >= weekStart).length; + const thisMonthInvites = directInvites.filter(u => u.registeredAt >= monthStart).length; + + // 合并并排序最近邀请 + interface InviteRecord { + account: UserAccount; + level: number; + } + const allInvites: InviteRecord[] = [ + ...directInvites.map(u => ({ account: u, level: 1 })), + ...indirectInvites.map(u => ({ account: u, level: 2 })), + ].sort((a, b) => b.account.registeredAt.getTime() - a.account.registeredAt.getTime()).slice(0, 20); + + return { + referralCode: account.referralCode.value, + totalInvites: directInvites.length + indirectInvites.length, + directInvites: directInvites.length, + indirectInvites: indirectInvites.length, + todayInvites, + thisWeekInvites, + thisMonthInvites, + recentInvites: allInvites.map(({ account: u, level }) => ({ + accountSequence: u.accountSequence.value, + nickname: u.nickname, + avatarUrl: u.avatarUrl, + registeredAt: u.registeredAt, + level, + })), + }; + } + + /** + * 生成短链码 + */ + private generateShortCode(): string { + const chars = 'ABCDEFGHJKLMNPQRSTUVWXYZabcdefghjkmnpqrstuvwxyz23456789'; + let result = ''; + for (let i = 0; i < 6; i++) { + result += chars.charAt(Math.floor(Math.random() * chars.length)); + } + return result; + } } diff --git a/backend/services/identity-service/src/domain/domain.module.ts b/backend/services/identity-service/src/domain/domain.module.ts index fcf40c9b..e8c4912d 100644 --- a/backend/services/identity-service/src/domain/domain.module.ts +++ b/backend/services/identity-service/src/domain/domain.module.ts @@ -1,5 +1,5 @@ import { Module } from '@nestjs/common'; -import { AccountSequenceGeneratorService, UserValidatorService, WalletGeneratorService } from './services'; +import { AccountSequenceGeneratorService, UserValidatorService } from './services'; import { UserAccountFactory } from './aggregates/user-account/user-account.factory'; import { USER_ACCOUNT_REPOSITORY } from './repositories/user-account.repository.interface'; import { UserAccountRepositoryImpl } from '@/infrastructure/persistence/repositories/user-account.repository.impl'; @@ -11,14 +11,14 @@ import { InfrastructureModule } from '@/infrastructure/infrastructure.module'; { provide: USER_ACCOUNT_REPOSITORY, useClass: UserAccountRepositoryImpl }, AccountSequenceGeneratorService, UserValidatorService, - WalletGeneratorService, + // WalletGeneratorService 由 InfrastructureModule 提供 UserAccountFactory, ], exports: [ USER_ACCOUNT_REPOSITORY, AccountSequenceGeneratorService, UserValidatorService, - WalletGeneratorService, + // WalletGeneratorService 由 InfrastructureModule 导出 UserAccountFactory, ], }) diff --git a/backend/services/identity-service/src/domain/repositories/user-account.repository.interface.ts b/backend/services/identity-service/src/domain/repositories/user-account.repository.interface.ts index 3584bb49..c41171c4 100644 --- a/backend/services/identity-service/src/domain/repositories/user-account.repository.interface.ts +++ b/backend/services/identity-service/src/domain/repositories/user-account.repository.interface.ts @@ -9,6 +9,24 @@ export interface Pagination { limit: number; } +export interface ReferralLinkData { + linkId: bigint; + userId: bigint; + referralCode: string; + shortCode: string; + channel: string | null; + campaignId: string | null; + createdAt: Date; +} + +export interface CreateReferralLinkParams { + userId: bigint; + referralCode: string; + shortCode: string; + channel: string | null; + campaignId: string | null; +} + export interface UserAccountRepository { save(account: UserAccount): Promise; saveWallets(userId: UserId, wallets: WalletAddress[]): Promise; @@ -25,6 +43,11 @@ export interface UserAccountRepository { pagination?: Pagination, ): Promise; countUsers(filters?: { status?: AccountStatus; kycStatus?: KYCStatus }): Promise; + + // 推荐相关 + findByInviterSequence(inviterSequence: AccountSequence): Promise; + createReferralLink(params: CreateReferralLinkParams): Promise; + findReferralLinksByUserId(userId: UserId): Promise; } export const USER_ACCOUNT_REPOSITORY = Symbol('USER_ACCOUNT_REPOSITORY'); diff --git a/backend/services/identity-service/src/domain/services/index.ts b/backend/services/identity-service/src/domain/services/index.ts index ba520cd1..a0f27e17 100644 --- a/backend/services/identity-service/src/domain/services/index.ts +++ b/backend/services/identity-service/src/domain/services/index.ts @@ -1,11 +1,14 @@ import { Injectable, Inject } from '@nestjs/common'; -import { createHash } from 'crypto'; -import { UserAccount } from '@/domain/aggregates/user-account/user-account.aggregate'; -import { WalletAddress } from '@/domain/entities/wallet-address.entity'; import { UserAccountRepository, USER_ACCOUNT_REPOSITORY } from '@/domain/repositories/user-account.repository.interface'; -import { - AccountSequence, PhoneNumber, ReferralCode, ChainType, Mnemonic, UserId, -} from '@/domain/value-objects'; +import { AccountSequence, PhoneNumber, ReferralCode, ChainType } from '@/domain/value-objects'; + +// 导出 WalletGeneratorService 和相关类型 +export { + WalletGeneratorService, + MpcWalletGenerationParams, + MpcWalletGenerationResult, + ChainWalletInfo, +} from './wallet-generator.service'; // ============ ValidationResult ============ export class ValidationResult { @@ -69,53 +72,3 @@ export class UserValidatorService { return ValidationResult.success(); } } - -// ============ WalletGeneratorService ============ -@Injectable() -export class WalletGeneratorService { - generateWalletSystem(params: { userId: UserId; deviceId: string }): { - mnemonic: Mnemonic; - wallets: Map; - } { - const mnemonic = Mnemonic.generate(); - const encryptionKey = this.deriveEncryptionKey(params.deviceId, params.userId.toString()); - - const wallets = new Map(); - const chains = [ChainType.KAVA, ChainType.DST, ChainType.BSC]; - - for (const chainType of chains) { - const wallet = WalletAddress.createFromMnemonic({ - userId: params.userId, - chainType, - mnemonic, - encryptionKey, - }); - wallets.set(chainType, wallet); - } - - return { mnemonic, wallets }; - } - - recoverWalletSystem(params: { userId: UserId; mnemonic: Mnemonic; deviceId: string }): Map { - const encryptionKey = this.deriveEncryptionKey(params.deviceId, params.userId.toString()); - const wallets = new Map(); - const chains = [ChainType.KAVA, ChainType.DST, ChainType.BSC]; - - for (const chainType of chains) { - const wallet = WalletAddress.createFromMnemonic({ - userId: params.userId, - chainType, - mnemonic: params.mnemonic, - encryptionKey, - }); - wallets.set(chainType, wallet); - } - - return wallets; - } - - private deriveEncryptionKey(deviceId: string, userId: string): string { - const input = `${deviceId}:${userId}`; - return createHash('sha256').update(input).digest('hex'); - } -} diff --git a/backend/services/identity-service/src/domain/services/wallet-generator.service.ts b/backend/services/identity-service/src/domain/services/wallet-generator.service.ts index a9e1eaaa..afd7df39 100644 --- a/backend/services/identity-service/src/domain/services/wallet-generator.service.ts +++ b/backend/services/identity-service/src/domain/services/wallet-generator.service.ts @@ -1,44 +1,73 @@ -import { Injectable } from '@nestjs/common'; -import { createHash } from 'crypto'; -import { WalletAddress } from '@/domain/entities/wallet-address.entity'; +import { WalletAddress, MpcSignature } from '@/domain/entities/wallet-address.entity'; import { ChainType, Mnemonic, UserId } from '@/domain/value-objects'; -@Injectable() -export class WalletGeneratorService { - generateWalletSystem(params: { userId: UserId; deviceId: string }): { - mnemonic: Mnemonic; - wallets: Map; - } { - const mnemonic = Mnemonic.generate(); - const encryptionKey = this.deriveEncryptionKey(params.deviceId, params.userId.toString()); +/** + * MPC 钱包生成参数 + */ +export interface MpcWalletGenerationParams { + userId: string; + deviceId: string; +} +/** + * 链钱包信息 + */ +export interface ChainWalletInfo { + chainType: 'KAVA' | 'DST' | 'BSC'; + address: string; + publicKey: string; + addressDigest: string; + signature: MpcSignature; +} + +/** + * MPC 钱包生成结果 + */ +export interface MpcWalletGenerationResult { + publicKey: string; + serverShareData: string; + clientShareData: string; + backupShareData: string; + wallets: ChainWalletInfo[]; + sessionId: string; +} + +/** + * 钱包生成服务接口 (端口) + * + * 定义钱包生成的业务接口,由基础设施层实现 + */ +export abstract class WalletGeneratorService { + /** + * 使用 MPC 2-of-3 生成三链钱包 + * + * @param params 用户ID和设备ID + * @returns MPC 钱包生成结果,包含分片和签名信息 + */ + abstract generateMpcWalletSystem(params: MpcWalletGenerationParams): Promise; + + /** + * 将 MPC 钱包信息转换为领域实体 + * + * @param userId 用户ID + * @param walletInfos MPC 钱包信息数组 + * @returns 钱包地址实体 Map + */ + convertToWalletEntities( + userId: UserId, + walletInfos: ChainWalletInfo[], + ): Map { const wallets = new Map(); - const chains = [ChainType.KAVA, ChainType.DST, ChainType.BSC]; - for (const chainType of chains) { - const wallet = WalletAddress.createFromMnemonic({ - userId: params.userId, + for (const info of walletInfos) { + const chainType = ChainType[info.chainType as keyof typeof ChainType]; + const wallet = WalletAddress.createMpc({ + userId, chainType, - mnemonic, - encryptionKey, - }); - wallets.set(chainType, wallet); - } - - return { mnemonic, wallets }; - } - - recoverWalletSystem(params: { userId: UserId; mnemonic: Mnemonic; deviceId: string }): Map { - const encryptionKey = this.deriveEncryptionKey(params.deviceId, params.userId.toString()); - const wallets = new Map(); - const chains = [ChainType.KAVA, ChainType.DST, ChainType.BSC]; - - for (const chainType of chains) { - const wallet = WalletAddress.createFromMnemonic({ - userId: params.userId, - chainType, - mnemonic: params.mnemonic, - encryptionKey, + address: info.address, + publicKey: info.publicKey, + addressDigest: info.addressDigest, + signature: info.signature, }); wallets.set(chainType, wallet); } @@ -46,8 +75,13 @@ export class WalletGeneratorService { return wallets; } - private deriveEncryptionKey(deviceId: string, userId: string): string { - const input = `${deviceId}:${userId}`; - return createHash('sha256').update(input).digest('hex'); - } + /** + * @deprecated MPC 模式下不再使用助记词恢复 + * 此方法保留用于向后兼容旧版账户恢复流程 + */ + abstract recoverWalletSystem(params: { + userId: UserId; + mnemonic: Mnemonic; + deviceId: string; + }): Map; } diff --git a/backend/services/identity-service/src/infrastructure/external/backup/backup-client.service.ts b/backend/services/identity-service/src/infrastructure/external/backup/backup-client.service.ts new file mode 100644 index 00000000..d0eccd7d --- /dev/null +++ b/backend/services/identity-service/src/infrastructure/external/backup/backup-client.service.ts @@ -0,0 +1,192 @@ +/** + * Backup Client Service + * + * 与 backup-service 通信的客户端服务 + * 负责存储和获取 MPC Backup Share (Party 2) + * + * 安全要求: backup-service 必须部署在与 identity-service 不同的物理服务器上 + */ + +import { Injectable, Logger } from '@nestjs/common'; +import { HttpService } from '@nestjs/axios'; +import { ConfigService } from '@nestjs/config'; +import { firstValueFrom } from 'rxjs'; +import * as jwt from 'jsonwebtoken'; + +export interface StoreBackupShareParams { + userId: string; + accountSequence: number; + publicKey: string; + encryptedShareData: string; +} + +export interface RetrieveBackupShareParams { + userId: string; + publicKey: string; + recoveryToken: string; + deviceId?: string; +} + +export interface BackupShareResult { + encryptedShareData: string; + partyIndex: number; + publicKey: string; +} + +@Injectable() +export class BackupClientService { + private readonly logger = new Logger(BackupClientService.name); + private readonly backupServiceUrl: string; + private readonly serviceJwtSecret: string; + private readonly enabled: boolean; + + constructor( + private readonly httpService: HttpService, + private readonly configService: ConfigService, + ) { + this.backupServiceUrl = this.configService.get('BACKUP_SERVICE_URL', 'http://localhost:3002'); + this.serviceJwtSecret = this.configService.get('SERVICE_JWT_SECRET', ''); + this.enabled = this.configService.get('BACKUP_SERVICE_ENABLED', 'false') === 'true'; + } + + /** + * 检查 backup-service 是否启用 + */ + isEnabled(): boolean { + return this.enabled && !!this.serviceJwtSecret; + } + + /** + * 存储备份分片到 backup-service + */ + async storeBackupShare(params: StoreBackupShareParams): Promise { + if (!this.isEnabled()) { + this.logger.warn('Backup service is disabled, skipping backup share storage'); + return; + } + + this.logger.log(`Storing backup share for user: ${params.userId}`); + + try { + const serviceToken = this.generateServiceToken(); + + await firstValueFrom( + this.httpService.post( + `${this.backupServiceUrl}/backup-share/store`, + { + userId: params.userId, + accountSequence: params.accountSequence, + publicKey: params.publicKey, + encryptedShareData: params.encryptedShareData, + }, + { + headers: { + 'Content-Type': 'application/json', + 'X-Service-Token': serviceToken, + }, + timeout: 30000, // 30秒超时 + }, + ), + ); + + this.logger.log(`Backup share stored successfully for user: ${params.userId}`); + } catch (error) { + this.logger.error(`Failed to store backup share for user: ${params.userId}`, error); + // 不抛出异常,允许账户创建继续 + // 可以通过补偿任务稍后重试 + } + } + + /** + * 从 backup-service 获取备份分片 (用于账户恢复) + */ + async retrieveBackupShare(params: RetrieveBackupShareParams): Promise { + if (!this.isEnabled()) { + this.logger.warn('Backup service is disabled'); + return null; + } + + this.logger.log(`Retrieving backup share for user: ${params.userId}`); + + try { + const serviceToken = this.generateServiceToken(); + + const response = await firstValueFrom( + this.httpService.post( + `${this.backupServiceUrl}/backup-share/retrieve`, + { + userId: params.userId, + publicKey: params.publicKey, + recoveryToken: params.recoveryToken, + deviceId: params.deviceId, + }, + { + headers: { + 'Content-Type': 'application/json', + 'X-Service-Token': serviceToken, + }, + timeout: 30000, + }, + ), + ); + + this.logger.log(`Backup share retrieved successfully for user: ${params.userId}`); + return response.data; + } catch (error) { + this.logger.error(`Failed to retrieve backup share for user: ${params.userId}`, error); + throw new Error(`Failed to retrieve backup share: ${error.message}`); + } + } + + /** + * 撤销备份分片 (用于密钥轮换或账户注销) + */ + async revokeBackupShare(userId: string, publicKey: string, reason: string): Promise { + if (!this.isEnabled()) { + this.logger.warn('Backup service is disabled'); + return; + } + + this.logger.log(`Revoking backup share for user: ${userId}, reason: ${reason}`); + + try { + const serviceToken = this.generateServiceToken(); + + await firstValueFrom( + this.httpService.post( + `${this.backupServiceUrl}/backup-share/revoke`, + { + userId, + publicKey, + reason, + }, + { + headers: { + 'Content-Type': 'application/json', + 'X-Service-Token': serviceToken, + }, + timeout: 30000, + }, + ), + ); + + this.logger.log(`Backup share revoked successfully for user: ${userId}`); + } catch (error) { + this.logger.error(`Failed to revoke backup share for user: ${userId}`, error); + } + } + + /** + * 生成服务间认证 JWT + */ + private generateServiceToken(): string { + return jwt.sign( + { + service: 'identity-service', + iat: Math.floor(Date.now() / 1000), + }, + this.serviceJwtSecret, + { expiresIn: '5m' }, + ); + } +} diff --git a/backend/services/identity-service/src/infrastructure/external/backup/index.ts b/backend/services/identity-service/src/infrastructure/external/backup/index.ts new file mode 100644 index 00000000..6fa59ab6 --- /dev/null +++ b/backend/services/identity-service/src/infrastructure/external/backup/index.ts @@ -0,0 +1,2 @@ +export * from './backup-client.service'; +export * from './mpc-share-storage.service'; diff --git a/backend/services/identity-service/src/infrastructure/external/backup/mpc-share-storage.service.ts b/backend/services/identity-service/src/infrastructure/external/backup/mpc-share-storage.service.ts new file mode 100644 index 00000000..16332a50 --- /dev/null +++ b/backend/services/identity-service/src/infrastructure/external/backup/mpc-share-storage.service.ts @@ -0,0 +1,89 @@ +/** + * MPC Share Storage Service + * + * 封装 MPC 分片存储逻辑,提供简化接口给应用层 + */ + +import { Injectable, Logger } from '@nestjs/common'; +import { BackupClientService } from './backup-client.service'; + +export interface MpcStoreBackupShareParams { + userId: string; + shareData: string; + publicKey: string; + accountSequence?: number; +} + +export interface MpcRetrieveBackupShareParams { + userId: string; + publicKey: string; + recoveryToken: string; + deviceId?: string; +} + +export interface MpcBackupShareData { + encryptedShareData: string; + partyIndex: number; + publicKey: string; +} + +@Injectable() +export class MpcShareStorageService { + private readonly logger = new Logger(MpcShareStorageService.name); + + constructor(private readonly backupClient: BackupClientService) {} + + /** + * 存储备份分片 + * + * @param params 分片存储参数 + */ + async storeBackupShare(params: MpcStoreBackupShareParams): Promise { + this.logger.log(`Storing backup share for user=${params.userId}`); + + await this.backupClient.storeBackupShare({ + userId: params.userId, + accountSequence: params.accountSequence || 0, + publicKey: params.publicKey, + encryptedShareData: params.shareData, + }); + } + + /** + * 获取备份分片 (用于账户恢复) + * + * @param params 分片获取参数 + * @returns 备份分片数据或 null + */ + async retrieveBackupShare( + params: MpcRetrieveBackupShareParams, + ): Promise { + this.logger.log(`Retrieving backup share for user=${params.userId}`); + + return this.backupClient.retrieveBackupShare(params); + } + + /** + * 撤销备份分片 + * + * @param userId 用户ID + * @param publicKey MPC 公钥 + * @param reason 撤销原因 + */ + async revokeBackupShare( + userId: string, + publicKey: string, + reason: string, + ): Promise { + this.logger.log(`Revoking backup share for user=${userId}`); + + await this.backupClient.revokeBackupShare(userId, publicKey, reason); + } + + /** + * 检查备份服务是否可用 + */ + isEnabled(): boolean { + return this.backupClient.isEnabled(); + } +} diff --git a/backend/services/identity-service/src/infrastructure/external/blockchain/blockchain-query.service.ts b/backend/services/identity-service/src/infrastructure/external/blockchain/blockchain-query.service.ts new file mode 100644 index 00000000..a625e924 --- /dev/null +++ b/backend/services/identity-service/src/infrastructure/external/blockchain/blockchain-query.service.ts @@ -0,0 +1,178 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { ethers, JsonRpcProvider, Contract } from 'ethers'; + +/** + * USDT 余额查询结果 + */ +export interface UsdtBalance { + chainType: string; + address: string; + balance: string; // 格式化后的余额 (带小数) + rawBalance: string; // 原始余额 (wei) + decimals: number; +} + +/** + * 链配置 + */ +interface ChainConfig { + rpcUrl: string; + usdtContract: string; + decimals: number; + name: string; +} + +/** + * ERC20 合约 ABI (仅需 balanceOf) + */ +const ERC20_ABI = [ + 'function balanceOf(address owner) view returns (uint256)', + 'function decimals() view returns (uint8)', +]; + +/** + * 区块链查询服务 + * + * 用于查询 KAVA EVM 和 BSC 链上的 USDT 余额 + */ +@Injectable() +export class BlockchainQueryService { + private readonly logger = new Logger(BlockchainQueryService.name); + + /** + * 链配置 + */ + private readonly chainConfigs: Record = { + KAVA: { + rpcUrl: process.env.KAVA_RPC_URL || 'https://evm.kava.io', + // KAVA EVM 原生 USDT 合约地址 (Tether 官方发行) + usdtContract: '0x919C1c267BC06a7039e03fcc2eF738525769109c', + decimals: 6, + name: 'KAVA EVM', + }, + BSC: { + rpcUrl: process.env.BSC_RPC_URL || 'https://bsc-dataseed.binance.org', + // BSC USDT 合约地址 (Binance-Peg) + usdtContract: '0x55d398326f99059fF775485246999027B3197955', + decimals: 18, + name: 'BSC', + }, + }; + + /** + * Provider 缓存 + */ + private providers: Map = new Map(); + + /** + * 获取或创建 Provider + */ + private getProvider(chainType: string): JsonRpcProvider { + if (this.providers.has(chainType)) { + return this.providers.get(chainType)!; + } + + const config = this.chainConfigs[chainType]; + if (!config) { + throw new Error(`不支持的链类型: ${chainType}`); + } + + const provider = new JsonRpcProvider(config.rpcUrl, undefined, { + staticNetwork: true, + }); + this.providers.set(chainType, provider); + return provider; + } + + /** + * 查询单个地址的 USDT 余额 + * + * @param chainType 链类型 (KAVA 或 BSC) + * @param address EVM 地址 + */ + async getUsdtBalance(chainType: string, address: string): Promise { + const config = this.chainConfigs[chainType]; + if (!config) { + throw new Error(`不支持的链类型: ${chainType}`); + } + + // 验证地址格式 + if (!ethers.isAddress(address)) { + throw new Error(`无效的 EVM 地址: ${address}`); + } + + try { + this.logger.debug(`查询 ${config.name} USDT 余额: ${address}`); + + const provider = this.getProvider(chainType); + const contract = new Contract(config.usdtContract, ERC20_ABI, provider); + + const rawBalance = await contract.balanceOf(address); + const balance = ethers.formatUnits(rawBalance, config.decimals); + + this.logger.debug(`${config.name} USDT 余额: ${balance}`); + + return { + chainType, + address, + balance, + rawBalance: rawBalance.toString(), + decimals: config.decimals, + }; + } catch (error) { + this.logger.error(`查询 ${config.name} USDT 余额失败: ${error.message}`); + throw new Error(`查询余额失败: ${error.message}`); + } + } + + /** + * 批量查询多个地址的 USDT 余额 + * + * @param addresses 地址列表 { chainType, address }[] + */ + async getMultipleUsdtBalances( + addresses: Array<{ chainType: string; address: string }>, + ): Promise { + const results = await Promise.allSettled( + addresses.map(({ chainType, address }) => this.getUsdtBalance(chainType, address)), + ); + + return results.map((result, index) => { + if (result.status === 'fulfilled') { + return result.value; + } + // 失败时返回零余额 + this.logger.warn(`查询余额失败 [${addresses[index].chainType}:${addresses[index].address}]: ${result.reason}`); + return { + chainType: addresses[index].chainType, + address: addresses[index].address, + balance: '0', + rawBalance: '0', + decimals: this.chainConfigs[addresses[index].chainType]?.decimals || 6, + }; + }); + } + + /** + * 查询原生代币余额 (KAVA / BNB) + */ + async getNativeBalance(chainType: string, address: string): Promise { + const config = this.chainConfigs[chainType]; + if (!config) { + throw new Error(`不支持的链类型: ${chainType}`); + } + + if (!ethers.isAddress(address)) { + throw new Error(`无效的 EVM 地址: ${address}`); + } + + try { + const provider = this.getProvider(chainType); + const rawBalance = await provider.getBalance(address); + return ethers.formatEther(rawBalance); + } catch (error) { + this.logger.error(`查询 ${config.name} 原生代币余额失败: ${error.message}`); + throw new Error(`查询余额失败: ${error.message}`); + } + } +} diff --git a/backend/services/identity-service/src/infrastructure/external/blockchain/wallet-generator.service.impl.ts b/backend/services/identity-service/src/infrastructure/external/blockchain/wallet-generator.service.impl.ts index 44262993..be576bb0 100644 --- a/backend/services/identity-service/src/infrastructure/external/blockchain/wallet-generator.service.impl.ts +++ b/backend/services/identity-service/src/infrastructure/external/blockchain/wallet-generator.service.impl.ts @@ -1,35 +1,61 @@ -import { Injectable } from '@nestjs/common'; +import { Injectable, Logger } from '@nestjs/common'; import { createHash } from 'crypto'; +import { + WalletGeneratorService, + MpcWalletGenerationParams, + MpcWalletGenerationResult, +} from '@/domain/services/wallet-generator.service'; import { WalletAddress } from '@/domain/entities/wallet-address.entity'; import { ChainType, Mnemonic, UserId } from '@/domain/value-objects'; +import { MpcWalletService } from '@/infrastructure/external/mpc/mpc-wallet.service'; +/** + * 钱包生成服务实现 + * + * 使用 MPC 2-of-3 协议生成三链钱包地址并签名 + */ @Injectable() -export class WalletGeneratorServiceImpl { - generateWalletSystem(params: { userId: UserId; deviceId: string }): { - mnemonic: Mnemonic; - wallets: Map; - } { - const mnemonic = Mnemonic.generate(); - const encryptionKey = this.deriveEncryptionKey(params.deviceId, params.userId.toString()); +export class WalletGeneratorServiceImpl extends WalletGeneratorService { + private readonly logger = new Logger(WalletGeneratorServiceImpl.name); - const wallets = new Map(); - const chains = [ChainType.KAVA, ChainType.DST, ChainType.BSC]; - - for (const chainType of chains) { - const wallet = WalletAddress.createFromMnemonic({ - userId: params.userId, - chainType, - mnemonic, - encryptionKey, - }); - wallets.set(chainType, wallet); - } - - return { mnemonic, wallets }; + constructor(private readonly mpcWalletService: MpcWalletService) { + super(); } - recoverWalletSystem(params: { userId: UserId; mnemonic: Mnemonic; deviceId: string }): Map { - const encryptionKey = this.deriveEncryptionKey(params.deviceId, params.userId.toString()); + /** + * 使用 MPC 2-of-3 生成三链钱包 + */ + async generateMpcWalletSystem( + params: MpcWalletGenerationParams, + ): Promise { + this.logger.log(`Generating MPC wallet system for user=${params.userId}`); + + const result = await this.mpcWalletService.generateMpcWallet(params); + + this.logger.log( + `MPC wallet system generated: ${result.wallets.length} wallets, sessionId=${result.sessionId}`, + ); + + return result; + } + + /** + * @deprecated MPC 模式下不再使用助记词恢复 + * 此方法保留用于向后兼容旧版账户恢复流程 + */ + recoverWalletSystem(params: { + userId: UserId; + mnemonic: Mnemonic; + deviceId: string; + }): Map { + this.logger.warn( + 'recoverWalletSystem is deprecated - MPC mode does not use mnemonic recovery', + ); + + const encryptionKey = this.deriveEncryptionKey( + params.deviceId, + params.userId.toString(), + ); const wallets = new Map(); const chains = [ChainType.KAVA, ChainType.DST, ChainType.BSC]; diff --git a/backend/services/identity-service/src/infrastructure/infrastructure.module.ts b/backend/services/identity-service/src/infrastructure/infrastructure.module.ts index 1a39019b..29cb3b98 100644 --- a/backend/services/identity-service/src/infrastructure/infrastructure.module.ts +++ b/backend/services/identity-service/src/infrastructure/infrastructure.module.ts @@ -8,8 +8,11 @@ import { RedisService } from './redis/redis.service'; import { EventPublisherService } from './kafka/event-publisher.service'; import { SmsService } from './external/sms/sms.service'; import { WalletGeneratorServiceImpl } from './external/blockchain/wallet-generator.service.impl'; +import { BlockchainQueryService } from './external/blockchain/blockchain-query.service'; import { MpcClientService, MpcWalletService } from './external/mpc'; +import { BackupClientService, MpcShareStorageService } from './external/backup'; import { MPC_KEY_SHARE_REPOSITORY } from '@/domain/repositories/mpc-key-share.repository.interface'; +import { WalletGeneratorService } from '@/domain/services/wallet-generator.service'; @Global() @Module({ @@ -30,9 +33,17 @@ import { MPC_KEY_SHARE_REPOSITORY } from '@/domain/repositories/mpc-key-share.re RedisService, EventPublisherService, SmsService, + // WalletGeneratorService 抽象类由 WalletGeneratorServiceImpl 实现 WalletGeneratorServiceImpl, + { + provide: WalletGeneratorService, + useExisting: WalletGeneratorServiceImpl, + }, + BlockchainQueryService, MpcClientService, MpcWalletService, + BackupClientService, + MpcShareStorageService, ], exports: [ PrismaService, @@ -46,8 +57,12 @@ import { MPC_KEY_SHARE_REPOSITORY } from '@/domain/repositories/mpc-key-share.re EventPublisherService, SmsService, WalletGeneratorServiceImpl, + WalletGeneratorService, + BlockchainQueryService, MpcClientService, MpcWalletService, + BackupClientService, + MpcShareStorageService, ], }) export class InfrastructureModule {} diff --git a/backend/services/identity-service/src/infrastructure/persistence/entities/user-account.entity.ts b/backend/services/identity-service/src/infrastructure/persistence/entities/user-account.entity.ts index b1fb63ea..25284c27 100644 --- a/backend/services/identity-service/src/infrastructure/persistence/entities/user-account.entity.ts +++ b/backend/services/identity-service/src/infrastructure/persistence/entities/user-account.entity.ts @@ -38,7 +38,11 @@ export interface WalletAddressEntity { userId: bigint; chainType: string; address: string; - encryptedMnemonic: string | null; + publicKey: string; + addressDigest: string; + mpcSignatureR: string; + mpcSignatureS: string; + mpcSignatureV: number; status: string; boundAt: Date; } diff --git a/backend/services/identity-service/src/infrastructure/persistence/mappers/user-account.mapper.ts b/backend/services/identity-service/src/infrastructure/persistence/mappers/user-account.mapper.ts index 52882eb8..f30ac1bb 100644 --- a/backend/services/identity-service/src/infrastructure/persistence/mappers/user-account.mapper.ts +++ b/backend/services/identity-service/src/infrastructure/persistence/mappers/user-account.mapper.ts @@ -17,7 +17,11 @@ export class UserAccountMapper { userId: w.userId.toString(), chainType: w.chainType as ChainType, address: w.address, - encryptedMnemonic: w.encryptedMnemonic || '', + publicKey: w.publicKey, + addressDigest: w.addressDigest, + mpcSignatureR: w.mpcSignatureR, + mpcSignatureS: w.mpcSignatureS, + mpcSignatureV: w.mpcSignatureV, status: w.status as AddressStatus, boundAt: w.boundAt, }), diff --git a/backend/services/identity-service/src/infrastructure/persistence/repositories/user-account.repository.impl.ts b/backend/services/identity-service/src/infrastructure/persistence/repositories/user-account.repository.impl.ts index d1df08bb..e11b86b1 100644 --- a/backend/services/identity-service/src/infrastructure/persistence/repositories/user-account.repository.impl.ts +++ b/backend/services/identity-service/src/infrastructure/persistence/repositories/user-account.repository.impl.ts @@ -1,6 +1,8 @@ import { Injectable } from '@nestjs/common'; import { PrismaService } from '@/infrastructure/persistence/prisma/prisma.service'; -import { UserAccountRepository, Pagination } from '@/domain/repositories/user-account.repository.interface'; +import { + UserAccountRepository, Pagination, ReferralLinkData, CreateReferralLinkParams, +} from '@/domain/repositories/user-account.repository.interface'; import { UserAccount } from '@/domain/aggregates/user-account/user-account.aggregate'; import { WalletAddress } from '@/domain/entities/wallet-address.entity'; import { @@ -250,4 +252,54 @@ export class UserAccountRepositoryImpl implements UserAccountRepository { updatedAt: data.updatedAt, }); } + + // ============ 推荐相关 ============ + + async findByInviterSequence(inviterSequence: AccountSequence): Promise { + const data = await this.prisma.userAccount.findMany({ + where: { inviterSequence: BigInt(inviterSequence.value) }, + include: { devices: true, walletAddresses: true }, + orderBy: { registeredAt: 'desc' }, + }); + return data.map((d) => this.toDomain(d)); + } + + async createReferralLink(params: CreateReferralLinkParams): Promise { + const result = await this.prisma.referralLink.create({ + data: { + userId: params.userId, + referralCode: params.referralCode, + shortCode: params.shortCode, + channel: params.channel, + campaignId: params.campaignId, + }, + }); + + return { + linkId: result.linkId, + userId: result.userId, + referralCode: result.referralCode, + shortCode: result.shortCode, + channel: result.channel, + campaignId: result.campaignId, + createdAt: result.createdAt, + }; + } + + async findReferralLinksByUserId(userId: UserId): Promise { + const results = await this.prisma.referralLink.findMany({ + where: { userId: BigInt(userId.value) }, + orderBy: { createdAt: 'desc' }, + }); + + return results.map((r) => ({ + linkId: r.linkId, + userId: r.userId, + referralCode: r.referralCode, + shortCode: r.shortCode, + channel: r.channel, + campaignId: r.campaignId, + createdAt: r.createdAt, + })); + } } diff --git a/backend/services/wallet-service/DEVELOPMENT_GUIDE.md b/backend/services/wallet-service/DEVELOPMENT_GUIDE.md new file mode 100644 index 00000000..a6f5bb44 --- /dev/null +++ b/backend/services/wallet-service/DEVELOPMENT_GUIDE.md @@ -0,0 +1,757 @@ +# Wallet Service 开发指导 + +## 项目概述 + +Wallet Service 是 RWA 榴莲女皇平台的钱包账本微服务,负责管理用户的平台内部余额、充值入账、提现、资金流水记账等功能。 + +## 技术栈 + +- **框架**: NestJS +- **数据库**: PostgreSQL + Prisma ORM +- **架构**: DDD + Hexagonal Architecture (六边形架构) +- **语言**: TypeScript + +## 架构参考 + +请参考 `identity-service` 的架构模式,保持一致性: + +``` +wallet-service/ +├── prisma/ +│ └── schema.prisma # 数据库模型 +├── src/ +│ ├── api/ # Presentation Layer (API层) +│ │ ├── controllers/ # HTTP 控制器 +│ │ │ ├── wallet.controller.ts +│ │ │ ├── ledger.controller.ts +│ │ │ ├── deposit.controller.ts +│ │ │ └── settlement.controller.ts +│ │ ├── dto/ # 数据传输对象 +│ │ │ ├── wallet.dto.ts +│ │ │ ├── ledger.dto.ts +│ │ │ ├── deposit.dto.ts +│ │ │ └── settlement.dto.ts +│ │ └── api.module.ts +│ │ +│ ├── application/ # Application Layer (应用层) +│ │ ├── commands/ # 命令对象 +│ │ │ ├── handle-deposit.command.ts +│ │ │ ├── deduct-for-planting.command.ts +│ │ │ ├── allocate-funds.command.ts +│ │ │ ├── add-rewards.command.ts +│ │ │ ├── settle-rewards.command.ts +│ │ │ └── withdraw.command.ts +│ │ ├── queries/ # 查询对象 +│ │ │ ├── get-my-wallet.query.ts +│ │ │ └── get-my-ledger.query.ts +│ │ └── services/ +│ │ └── wallet-application.service.ts +│ │ +│ ├── domain/ # Domain Layer (领域层) +│ │ ├── aggregates/ # 聚合根 +│ │ │ ├── wallet-account.aggregate.ts +│ │ │ ├── ledger-entry.aggregate.ts +│ │ │ ├── deposit-order.aggregate.ts +│ │ │ └── settlement-order.aggregate.ts +│ │ ├── value-objects/ # 值对象 +│ │ │ ├── wallet-id.vo.ts +│ │ │ ├── money.vo.ts +│ │ │ ├── balance.vo.ts +│ │ │ ├── wallet-balances.vo.ts +│ │ │ ├── wallet-rewards.vo.ts +│ │ │ ├── hashpower.vo.ts +│ │ │ ├── asset-type.enum.ts +│ │ │ ├── chain-type.enum.ts +│ │ │ ├── wallet-status.enum.ts +│ │ │ ├── ledger-entry-type.enum.ts +│ │ │ ├── deposit-status.enum.ts +│ │ │ └── settlement-status.enum.ts +│ │ ├── events/ # 领域事件 +│ │ │ ├── deposit-completed.event.ts +│ │ │ ├── withdrawal-requested.event.ts +│ │ │ ├── reward-moved-to-settleable.event.ts +│ │ │ ├── reward-expired.event.ts +│ │ │ └── settlement-completed.event.ts +│ │ ├── repositories/ # 仓储接口 (Port) +│ │ │ ├── wallet-account.repository.interface.ts +│ │ │ ├── ledger-entry.repository.interface.ts +│ │ │ ├── deposit-order.repository.interface.ts +│ │ │ └── settlement-order.repository.interface.ts +│ │ └── services/ # 领域服务 +│ │ └── wallet-ledger.service.ts +│ │ +│ ├── infrastructure/ # Infrastructure Layer (基础设施层) +│ │ ├── persistence/ # 持久化实现 (Adapter) +│ │ │ ├── entities/ # Prisma 实体映射 +│ │ │ ├── mappers/ # 领域模型与实体的映射 +│ │ │ └── repositories/ # 仓储实现 +│ │ └── infrastructure.module.ts +│ │ +│ ├── app.module.ts +│ └── main.ts +├── .env.development +├── .env.example +├── package.json +└── tsconfig.json +``` + +--- + +## 第一阶段:项目初始化 + +### 1.1 创建 NestJS 项目 + +```bash +cd backend/services +npx @nestjs/cli new wallet-service --skip-git --package-manager npm +cd wallet-service +``` + +### 1.2 安装依赖 + +```bash +npm install @nestjs/config @prisma/client class-validator class-transformer uuid +npm install -D prisma @types/uuid +``` + +### 1.3 配置环境变量 + +创建 `.env.development`: +```env +DATABASE_URL="postgresql://postgres:postgres@localhost:5432/rwadurian_wallet?schema=public" +NODE_ENV=development +PORT=3002 +``` + +创建 `.env.example`: +```env +DATABASE_URL="postgresql://user:password@host:5432/database?schema=public" +NODE_ENV=development +PORT=3002 +``` + +--- + +## 第二阶段:数据库设计 (Prisma Schema) + +### 2.1 创建 prisma/schema.prisma + +```prisma +generator client { + provider = "prisma-client-js" +} + +datasource db { + provider = "postgresql" + url = env("DATABASE_URL") +} + +// ============================================ +// 钱包账户表 (状态表) +// ============================================ +model WalletAccount { + id BigInt @id @default(autoincrement()) @map("wallet_id") + userId BigInt @unique @map("user_id") + + // USDT 余额 + usdtAvailable Decimal @default(0) @map("usdt_available") @db.Decimal(20, 8) + usdtFrozen Decimal @default(0) @map("usdt_frozen") @db.Decimal(20, 8) + + // DST 余额 + dstAvailable Decimal @default(0) @map("dst_available") @db.Decimal(20, 8) + dstFrozen Decimal @default(0) @map("dst_frozen") @db.Decimal(20, 8) + + // BNB 余额 + bnbAvailable Decimal @default(0) @map("bnb_available") @db.Decimal(20, 8) + bnbFrozen Decimal @default(0) @map("bnb_frozen") @db.Decimal(20, 8) + + // OG 余额 + ogAvailable Decimal @default(0) @map("og_available") @db.Decimal(20, 8) + ogFrozen Decimal @default(0) @map("og_frozen") @db.Decimal(20, 8) + + // RWAD 余额 + rwadAvailable Decimal @default(0) @map("rwad_available") @db.Decimal(20, 8) + rwadFrozen Decimal @default(0) @map("rwad_frozen") @db.Decimal(20, 8) + + // 算力 + hashpower Decimal @default(0) @map("hashpower") @db.Decimal(20, 8) + + // 待领取收益 + pendingUsdt Decimal @default(0) @map("pending_usdt") @db.Decimal(20, 8) + pendingHashpower Decimal @default(0) @map("pending_hashpower") @db.Decimal(20, 8) + pendingExpireAt DateTime? @map("pending_expire_at") + + // 可结算收益 + settleableUsdt Decimal @default(0) @map("settleable_usdt") @db.Decimal(20, 8) + settleableHashpower Decimal @default(0) @map("settleable_hashpower") @db.Decimal(20, 8) + + // 已结算总额 + settledTotalUsdt Decimal @default(0) @map("settled_total_usdt") @db.Decimal(20, 8) + settledTotalHashpower Decimal @default(0) @map("settled_total_hashpower") @db.Decimal(20, 8) + + // 已过期总额 + expiredTotalUsdt Decimal @default(0) @map("expired_total_usdt") @db.Decimal(20, 8) + expiredTotalHashpower Decimal @default(0) @map("expired_total_hashpower") @db.Decimal(20, 8) + + // 状态 + status String @default("ACTIVE") @map("status") @db.VarChar(20) + + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + @@map("wallet_accounts") + @@index([userId]) + @@index([usdtAvailable(sort: Desc)]) + @@index([hashpower(sort: Desc)]) + @@index([status]) +} + +// ============================================ +// 账本流水表 (行为表, append-only) +// ============================================ +model LedgerEntry { + id BigInt @id @default(autoincrement()) @map("entry_id") + userId BigInt @map("user_id") + + // 流水类型 + entryType String @map("entry_type") @db.VarChar(50) + + // 金额变动 (正数入账, 负数支出) + amount Decimal @map("amount") @db.Decimal(20, 8) + assetType String @map("asset_type") @db.VarChar(20) + + // 余额快照 (操作后余额) + balanceAfter Decimal? @map("balance_after") @db.Decimal(20, 8) + + // 关联引用 + refOrderId String? @map("ref_order_id") @db.VarChar(100) + refTxHash String? @map("ref_tx_hash") @db.VarChar(100) + + // 备注 + memo String? @map("memo") @db.VarChar(500) + + // 扩展数据 + payloadJson Json? @map("payload_json") + + createdAt DateTime @default(now()) @map("created_at") + + @@map("wallet_ledger_entries") + @@index([userId, createdAt(sort: Desc)]) + @@index([entryType]) + @@index([assetType]) + @@index([refOrderId]) + @@index([refTxHash]) + @@index([createdAt]) +} + +// ============================================ +// 充值订单表 +// ============================================ +model DepositOrder { + id BigInt @id @default(autoincrement()) @map("order_id") + userId BigInt @map("user_id") + + // 充值信息 + chainType String @map("chain_type") @db.VarChar(20) + amount Decimal @map("amount") @db.Decimal(20, 8) + txHash String @unique @map("tx_hash") @db.VarChar(100) + + // 状态 + status String @default("PENDING") @map("status") @db.VarChar(20) + confirmedAt DateTime? @map("confirmed_at") + + createdAt DateTime @default(now()) @map("created_at") + + @@map("deposit_orders") + @@index([userId]) + @@index([txHash]) + @@index([status]) + @@index([chainType]) +} + +// ============================================ +// 结算订单表 +// ============================================ +model SettlementOrder { + id BigInt @id @default(autoincrement()) @map("order_id") + userId BigInt @map("user_id") + + // 结算信息 + usdtAmount Decimal @map("usdt_amount") @db.Decimal(20, 8) + settleCurrency String @map("settle_currency") @db.VarChar(10) + + // SWAP 信息 + swapTxHash String? @map("swap_tx_hash") @db.VarChar(100) + receivedAmount Decimal? @map("received_amount") @db.Decimal(20, 8) + + // 状态 + status String @default("PENDING") @map("status") @db.VarChar(20) + settledAt DateTime? @map("settled_at") + + createdAt DateTime @default(now()) @map("created_at") + + @@map("settlement_orders") + @@index([userId]) + @@index([status]) + @@index([settleCurrency]) + @@index([createdAt]) +} +``` + +### 2.2 初始化数据库 + +```bash +npx prisma migrate dev --name init +npx prisma generate +``` + +--- + +## 第三阶段:领域层实现 + +### 3.1 值对象 (Value Objects) + +#### 3.1.1 asset-type.enum.ts +```typescript +export enum AssetType { + USDT = 'USDT', + DST = 'DST', + BNB = 'BNB', + OG = 'OG', + RWAD = 'RWAD', + HASHPOWER = 'HASHPOWER', +} +``` + +#### 3.1.2 chain-type.enum.ts +```typescript +export enum ChainType { + KAVA = 'KAVA', + DST = 'DST', + BSC = 'BSC', +} +``` + +#### 3.1.3 wallet-status.enum.ts +```typescript +export enum WalletStatus { + ACTIVE = 'ACTIVE', + FROZEN = 'FROZEN', + CLOSED = 'CLOSED', +} +``` + +#### 3.1.4 ledger-entry-type.enum.ts +```typescript +export enum LedgerEntryType { + DEPOSIT_KAVA = 'DEPOSIT_KAVA', + DEPOSIT_BSC = 'DEPOSIT_BSC', + PLANT_PAYMENT = 'PLANT_PAYMENT', + REWARD_PENDING = 'REWARD_PENDING', + REWARD_TO_SETTLEABLE = 'REWARD_TO_SETTLEABLE', + REWARD_EXPIRED = 'REWARD_EXPIRED', + REWARD_SETTLED = 'REWARD_SETTLED', + TRANSFER_TO_POOL = 'TRANSFER_TO_POOL', + SWAP_EXECUTED = 'SWAP_EXECUTED', + WITHDRAWAL = 'WITHDRAWAL', + TRANSFER_IN = 'TRANSFER_IN', + TRANSFER_OUT = 'TRANSFER_OUT', + FREEZE = 'FREEZE', + UNFREEZE = 'UNFREEZE', +} +``` + +#### 3.1.5 deposit-status.enum.ts +```typescript +export enum DepositStatus { + PENDING = 'PENDING', + CONFIRMED = 'CONFIRMED', + FAILED = 'FAILED', +} +``` + +#### 3.1.6 settlement-status.enum.ts +```typescript +export enum SettlementStatus { + PENDING = 'PENDING', + SWAPPING = 'SWAPPING', + COMPLETED = 'COMPLETED', + FAILED = 'FAILED', +} + +export enum SettleCurrency { + BNB = 'BNB', + OG = 'OG', + USDT = 'USDT', + DST = 'DST', +} +``` + +#### 3.1.7 money.vo.ts +```typescript +export class Money { + private constructor( + public readonly amount: number, + public readonly currency: string, + ) { + if (amount < 0) { + throw new Error('Money amount cannot be negative'); + } + } + + static create(amount: number, currency: string): Money { + return new Money(amount, currency); + } + + static USDT(amount: number): Money { + return new Money(amount, 'USDT'); + } + + static zero(currency: string = 'USDT'): Money { + return new Money(0, currency); + } + + add(other: Money): Money { + this.ensureSameCurrency(other); + return new Money(this.amount + other.amount, this.currency); + } + + subtract(other: Money): Money { + this.ensureSameCurrency(other); + if (this.amount < other.amount) { + throw new Error('Insufficient balance'); + } + return new Money(this.amount - other.amount, this.currency); + } + + lessThan(other: Money): boolean { + this.ensureSameCurrency(other); + return this.amount < other.amount; + } + + greaterThan(other: Money): boolean { + this.ensureSameCurrency(other); + return this.amount > other.amount; + } + + isZero(): boolean { + return this.amount === 0; + } + + private ensureSameCurrency(other: Money): void { + if (this.currency !== other.currency) { + throw new Error('Currency mismatch'); + } + } +} +``` + +#### 3.1.8 balance.vo.ts +```typescript +import { Money } from './money.vo'; + +export class Balance { + constructor( + public readonly available: Money, + public readonly frozen: Money, + ) {} + + static zero(): Balance { + return new Balance(Money.zero(), Money.zero()); + } + + add(amount: Money): Balance { + return new Balance(this.available.add(amount), this.frozen); + } + + deduct(amount: Money): Balance { + if (this.available.lessThan(amount)) { + throw new Error('Insufficient available balance'); + } + return new Balance(this.available.subtract(amount), this.frozen); + } + + freeze(amount: Money): Balance { + if (this.available.lessThan(amount)) { + throw new Error('Insufficient available balance to freeze'); + } + return new Balance( + this.available.subtract(amount), + this.frozen.add(amount), + ); + } + + unfreeze(amount: Money): Balance { + if (this.frozen.lessThan(amount)) { + throw new Error('Insufficient frozen balance to unfreeze'); + } + return new Balance( + this.available.add(amount), + this.frozen.subtract(amount), + ); + } + + get total(): Money { + return this.available.add(this.frozen); + } +} +``` + +#### 3.1.9 hashpower.vo.ts +```typescript +export class Hashpower { + private constructor(public readonly value: number) { + if (value < 0) { + throw new Error('Hashpower cannot be negative'); + } + } + + static create(value: number): Hashpower { + return new Hashpower(value); + } + + static zero(): Hashpower { + return new Hashpower(0); + } + + add(other: Hashpower): Hashpower { + return new Hashpower(this.value + other.value); + } + + subtract(other: Hashpower): Hashpower { + if (this.value < other.value) { + throw new Error('Insufficient hashpower'); + } + return new Hashpower(this.value - other.value); + } + + isZero(): boolean { + return this.value === 0; + } +} +``` + +### 3.2 聚合根 (Aggregates) + +#### 3.2.1 wallet-account.aggregate.ts + +实现 `WalletAccount` 聚合根,包含: +- 余额管理 (USDT/DST/BNB/OG/RWAD/算力) +- 收益汇总 (待领取/可结算/已结算/过期) +- 核心领域行为:deposit, deduct, freeze, unfreeze, addPendingReward, movePendingToSettleable, settleReward, withdraw + +#### 3.2.2 ledger-entry.aggregate.ts + +实现 `LedgerEntry` 聚合根 (append-only),包含: +- 流水类型 +- 金额变动 +- 余额快照 +- 关联引用 + +### 3.3 仓储接口 (Repository Interfaces) + +#### 3.3.1 wallet-account.repository.interface.ts +```typescript +export interface IWalletAccountRepository { + save(wallet: WalletAccount): Promise; + findById(walletId: bigint): Promise; + findByUserId(userId: bigint): Promise; + getOrCreate(userId: bigint): Promise; + findByUserIds(userIds: bigint[]): Promise>; +} + +export const WALLET_ACCOUNT_REPOSITORY = Symbol('IWalletAccountRepository'); +``` + +#### 3.3.2 ledger-entry.repository.interface.ts +```typescript +export interface ILedgerEntryRepository { + save(entry: LedgerEntry): Promise; + saveAll(entries: LedgerEntry[]): Promise; + findByUserId(userId: bigint, filters?: LedgerFilters, pagination?: Pagination): Promise; + findByRefOrderId(refOrderId: string): Promise; +} + +export const LEDGER_ENTRY_REPOSITORY = Symbol('ILedgerEntryRepository'); +``` + +--- + +## 第四阶段:基础设施层实现 + +### 4.1 实体映射 (Entities) + +创建 Prisma 实体到领域模型的映射类。 + +### 4.2 仓储实现 (Repository Implementations) + +实现所有仓储接口,使用 Prisma Client 进行数据库操作。 + +--- + +## 第五阶段:应用层实现 + +### 5.1 命令对象 (Commands) + +```typescript +// handle-deposit.command.ts +export class HandleDepositCommand { + constructor( + public readonly userId: string, + public readonly amount: number, + public readonly chainType: ChainType, + public readonly txHash: string, + ) {} +} + +// deduct-for-planting.command.ts +export class DeductForPlantingCommand { + constructor( + public readonly userId: string, + public readonly amount: number, + public readonly orderId: string, + ) {} +} +``` + +### 5.2 应用服务 (Application Service) + +实现 `WalletApplicationService`,包含所有用例: +- handleDeposit - 处理充值 +- deductForPlanting - 认种扣款 +- allocateFunds - 资金分配 +- addRewards - 增加奖励 +- movePendingToSettleable - 待领取→可结算 +- settleRewards - 结算收益 +- getMyWallet - 查询我的钱包 +- getMyLedger - 查询我的流水 + +--- + +## 第六阶段:API层实现 + +### 6.1 DTO 定义 + +```typescript +// wallet.dto.ts +export class WalletDTO { + walletId: string; + userId: string; + balances: BalancesDTO; + rewards: RewardsDTO; + status: string; +} + +export class BalancesDTO { + usdt: BalanceDTO; + dst: BalanceDTO; + bnb: BalanceDTO; + og: BalanceDTO; + rwad: BalanceDTO; + hashpower: number; +} + +export class BalanceDTO { + available: number; + frozen: number; +} +``` + +### 6.2 控制器实现 + +```typescript +// wallet.controller.ts +@Controller('wallet') +export class WalletController { + constructor(private readonly walletService: WalletApplicationService) {} + + @Get('my-wallet') + @UseGuards(JwtAuthGuard) + async getMyWallet(@CurrentUser() user: User): Promise { + return this.walletService.getMyWallet({ userId: user.id }); + } +} + +// ledger.controller.ts +@Controller('wallet/ledger') +export class LedgerController { + constructor(private readonly walletService: WalletApplicationService) {} + + @Get('my-ledger') + @UseGuards(JwtAuthGuard) + async getMyLedger( + @CurrentUser() user: User, + @Query() query: GetMyLedgerQueryDTO, + ): Promise { + return this.walletService.getMyLedger({ + userId: user.id, + ...query, + }); + } +} +``` + +--- + +## 第七阶段:模块配置 + +### 7.1 app.module.ts + +```typescript +@Module({ + imports: [ + ConfigModule.forRoot({ + isGlobal: true, + envFilePath: `.env.${process.env.NODE_ENV || 'development'}`, + }), + ApiModule, + InfrastructureModule, + ], +}) +export class AppModule {} +``` + +--- + +## 关键业务规则 (不变式) + +1. **余额不能为负**: 任何币种的可用余额不能为负数 +2. **流水表只能追加**: 账本流水表只能 INSERT,不能 UPDATE/DELETE +3. **每笔流水必须有余额快照**: 每笔流水必须记录操作后的余额快照 +4. **结算金额不能超过可结算余额**: 结算金额必须 ≤ 可结算余额 + +--- + +## API 端点汇总 + +| 方法 | 路径 | 描述 | 认证 | +|------|------|------|------| +| GET | /wallet/my-wallet | 查询我的钱包 | 需要 | +| GET | /wallet/ledger/my-ledger | 查询我的流水 | 需要 | +| POST | /wallet/deposit | 充值入账 (内部) | 服务间 | +| POST | /wallet/withdraw | 提现申请 | 需要 | +| POST | /wallet/settle | 结算收益 | 需要 | + +--- + +## 开发顺序建议 + +1. 项目初始化和 Prisma Schema +2. 值对象实现 +3. 聚合根实现 +4. 仓储接口定义 +5. 仓储实现 +6. 领域服务实现 +7. 应用服务实现 +8. DTO 和控制器实现 +9. 模块配置和测试 + +--- + +## 注意事项 + +1. 所有金额使用 `Decimal(20, 8)` 存储,避免浮点数精度问题 +2. 流水表是 append-only,不允许更新或删除 +3. 每次余额变动都要创建对应的流水记录 +4. 使用事务确保余额和流水的一致性 +5. 参考 identity-service 的代码风格和命名规范 diff --git a/frontend/mobile-app/assets/images/Background@2x.png b/frontend/mobile-app/assets/images/Background@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..327ca2ca3f3465130b3af94849a4dc6051ef896c GIT binary patch literal 155472 zcmYhgb8sb0@HHIUwsT`+W7|${tQ*}p*=)SAZQI<~+}QTUHa5;C-}8G>-}^_`)ST+m zb-HG%X1Y64O+^+Ji3kY-0s>WDPD%p;0y5x#3<2&xq-3E^)_()z#T+0*k@*n(`)B1NjIMItO zK|gM;toaHcXlYA}dAyO9I0E#GvR&Oi2yv)i77-D~@DyVL9uWh|m%zX7@BKO-fxQG* z9LT_NEREJ+m{I44{dZgLo-H4pnROpwu<*r)N#g1D`#a{XFxQRW*Uf8w)AL5PQ1zwX z6UT?B(AK}!tJV5-f8*YN-OleF;P(zx-T3O!<%4sV$Cr4ot)73~;P= z`oB_UkJ|s!5!nCa(TuWdclh)FkiOQ*ZU!l_`2+%f>@&WY2>(x*>d~Kdqg}1}_%5kx zX#u>qJ9NG`ga5w*H{t6+!!543L`bQ#lidFi^}k7MVFz_RI72AAs{fPw|Hk|D;u^Yl z%zM6M|1Y8&Z^Aq(0)HDL4E@6M|B%Zp?5~;x1JeH=?pj>>4ccz#|Krqp5c*EiZ~WEQ zIllRS1byraTyK*G{6DnQxQF(s1FzYikUG$K-}pRCus$vS_!~Iz4Bb5ajGfJQ=(QG# zFl`l$#z3bXy5jvZZu{x5)#SK3HM{E8qUS1uYyKh)x#Fa1hX}UYadpgF5W?IqLC54k6w8{URFhw+;f%J}BXnVT2F;Yop zQyeo#&6+_cBWN6OiT7gPTut!F**WUTe_+V|ZWI#N3mcI;91}_9ipO%O(vTS2qQ)UJ$e`iz`^}TmsY<682dk{ih zvHx);{+2DnNk^sy<|h_OcwRk*0)T*Ngg3M`gq8Z|?=hTomfU-leA*wAfAQxE?C-ka z$C(I`6NECtoLtq_hdEYL_hmGGi3ILgNL^r=_}t3p9JYT)?aARrxux38USKLnjoQOC zOUYMR0K9TxWC+M)M9r(-2yVG~92fqW52!~X#B`H#c0R;ZZ>nJq!jO#T%Ik!v4-=eN?3~*M_?{T##@)bW?m~^Qp3n6-MKx<&$_7Fk9unUC&>n_8Nom?BJ$&fvgygh2?cZisU=@8q# zPVYG>1}f!^ZrbBxWNhIM(Ykl1l8zFmIwSZoVz4Jg+rO_Y@4}9h2t_+u?n2ySF#!N& zzjVu%EwBf&plV#MOBKnS;h4x=*nflV8u=F%b(=h~9yAXz7!lk@6r-h5Wh&bUKo_D? zh9zz?pO3KoQGT44snv*HMZL|PL2Z;~NyjPCs!hULCPL9~g)%4E>j4pdN*U5uwyk*( z+%}m4nLFF)2}QW6y0caT8_7<%Ta)I^8!J=1PL`o&a(0+}tOXWayDs5h=C<79l%Z-s zZ2hGeVK}|NT}V)92uHx^15Am%K5MT;Ki9H4-YSh*SvXDscKq@XI4T!Xg-7jioOQQ0 z0kn@R$cih=F^CUDKc{;!w{KB{5=6-VghsZs?@z;IX@53mvG#gY&Hp12OO7U*5r#hSvZQi3tY({MgDvP{Q1 zGA-ghfe%KNEKAaz9RG+wD{s)Xo8_ILTydg=(>6A21PSh;KE^_LvETSV75q56J+$_w z7kz@+K!4|kvM|EPPwirMR8r~iSr1{8P`hOw?#Rz!wK z@d=1S>o~45+;tr1J3n1WbKR25D~ng;Cd(eC7uL`(kz37vCIE`$V@MN}3x~5{1-5Nv zS~Q-Zrxx10itv7v@!p|~S~;Cl%!Y!VrG+jlr?4ph%Gqcy#ec|Zf==B(XaeQKL%8Up`;@xm*sFZo3`5%hHInN% z9{0R01Cx-;rL5kL=7W2V%|Z*@wWre^cuW&nZuz-P%M0n{Q+$IYrFTS{i7jRt_}=W; zSNk?Q%ip@LaPD*%fRZ`jnBVf%xmYD}=1@LoU2;&RI^){vQ}Dm%zaM!WIpiS^JUbzPi=d$nBGs!*uu?;o`zP zS_XF#x*X;+hQ=ooM8_H?pW9XBEdqH?$vep~N)44?-$20(=n@Ht3us1SYPBRI3CZck ziOx^zBZ8G@-aw01j(J6&)+y=`YaODH5D4Dc04iL}&|SO9BX+(UX+Js+>GG{~kD+@k zX9AlU2LcE1S#qRi*#PK<_q5D3G8naz8R~C$7c{?JW+@E;wJDAnL}5tHl>|@@$9|lZWsO<}Zk5%UlQ6)W zo<#r;FnzM;w~8R^h~Ik@PLVPpe!)X$18A99f$6W$i5)98>KwSzkv*H0obT4wyK|=fy!oX6Mxh(^XpJJ&4Q2@JQq&JEp@Y@ zIwfh1;<{T|YXUha<~kU@fi0-GW}u#lQn*RWb!q2lnq5bB8>wm-_yWPHjM=m=Gu(9Wp-` z7w!($W;9}&RMrt|w>QwpWD*o|1|TOfAK)9hYkFH%X0tqYD*2nBToFfcfw0^^eLl1M zqn7Fc&Se)h!K##Lkars?U;@nzhMQ+xWuiS_-$1E2O-KU=ffVOYu-j4M^BKF~RIdS& zmUxUz=TyZ{Tz|RoXyw?qqs}}j?E@}q5XU`%<&)o`my0LjX`HS4jk_d0DS(Qo<7Z!) zec8S1@Fe|neD#XtO&U|7(#SXQ4W#r11|JOucEri9vck(h%*xjT)-q{(9ceLJ&A79m zUzJ=8(;X!4P?duQsYDvGq*&TIg5yOWU7k7xYGR`iYwC%<0<^0&)oJ9j_0{&Yl*jGP z0oCNcDhK7*+wIEL4kkuMc=DntI57qW)YpNfU8U;_Bzql)W@)_W6yuPie{F3ti^3s0 zR^)A@hux~$W=y9hT5>&r>qqULk1zpvYA^a8vk&n8&C7;Sv<#~q-OIcCOQc6QXw8@r z-E}z@S3o{3O{H0*%mXAPsALz=664AM)T15ivvM|h9(yKk(#j;oQ8AGKuH1#qgLR>K zld^8cX|LOPO52`gP=1n@IK+Y}DaV-f>z?#%3{rC>5bwO0k*t-v+r&S+KiJukiS>`DGiHq`S;zYhmuhj`+QEtBa0|f#ws%kGkG%hj(SH~cTgddM`HI7cr zUEX#nrVxhKDAC3$22q0ZF7Ii?=Y@QZ1Zx7YSQlS~&t17IG-^3sPqxD-ii?M=bRz8H zc=e-{rsmD1wp1f+{bXZg-E#w;li`tCELlr1$7B;%xEM$Hab;NO=rptH@l)l`cqw_h z36#K6gB!o46ai$kBm#0eslf6A`m@A&C>A0^d05KohhED+A@!=sG0_=oW#2xF_&4ej zS8qr9e$5Ehc?}RESH^8a2~WBd)(4?D90L2F>c1Zdf0@BF-!)WmTX-Xf(a$dD=w#k` z$@h!y+3Y;)KHQf+6rB6zxu&%z;RRx?119LW5ON~Gjb)sD+<3r?{LBz&0B!2BtEXoD zE~;3~7qnSr%4!+PSSqE%D%O$xv-uySmDZLX8s1S;vngY$C|CI3$?H8J8>Q{zn)LKu zEnSlA@Zuv29-RZ;X%fJXm-U%Ql3GA<~;w1W0;qq&%N!#n(~hQ>r6N>*)YMb=E` zR(;seLXD|FGUJ7`lX`khG+83WAkyi9L+`hFe<3#uJ|TY(r0%AF8TA;(c;z)Ym%m=M z-Cy-2G?7`2mlyWM_%_DHrRjdI{8CSeonV=Yh|jcW`%)CG>!)GQZ2POaZMmX_$Vm?} zV{EU1)}>3VCRWByXS`XG+GcbRPYC$?Vz`3>gR%4sbo%BLbj)Y%hIdG0m#RqMzovzU zDUsR4%>lsujyku{Zp)$8+2FOzd7gVP(IOU0&4_(O+VtD@*S&iKGDWZ{2kP{SZA4nJ z)Ixt^y}9~ph{eWx5^2oji)k*Hh6Yb)L?SOdn!#R~G)pYhy}~UIF&wceDFM%Ep^)YW zGpeVIoAgf=Vp_TD9qGIV*O_2ptxO65HwJejP1V+mLAzu)v26|vTva$vXu{GV5Di!kF0y&f@QQzJr0 zWw}cmTe21nUWQ5K8^PWum}rggs7M7GJXOA;er6&e2_!mmYwixbTO5HgHt04%F{l&n zKl<4{CG*ePkmfZmXB9q*wXQ;by>)lc)lbw$AqGq2kAs zol7Wo)wCK4&c+wPjay{;BHaFnV}!gs>{Y88FBkdAS#gY7vU!4saa2Sdi3(H|bj`7h zgLGY*mKLWO&?ePE?XZYFk{8T7@k@636&$PWPkL}C4frRl3^JU4%_}WBTdr2NT88-x z{B|;xiq~yevy@${oTb}vA?^_(q5#RlaA%9U|I>(oM2ePBt2o#*Z1 zx%JVKZt*hlpU3p)$T^sv7Df#_zV}hm5B?RHGTJrygvz^03-MybFuDVy4pPLAkb5X@ zY>3R09~?&R_nv*MPpwhNTg(HQ-Nti|^Jpp){k1AJW#c0~rK(#ZpnYaq){mgm!sWrB zEb@!=shb`+I2HqTi~SgwOe{&Yytut^5m#{8zk{q|iOpxa_=slG3ga@t_Y@oDtyptR*snrP0e-#$aec{l@rmb?!4Q478Nlt6-pNc z8R^c3f!bmha;exz!Je9GE(fiIVg$E`UnFB=A>WH^w!VEE)KAOUQu|He) zq#wqE7(n~pVoDoDzPwjyr1bkhknww&E0}%~rx0p2VnlW{+BDo6D32SRN*6D>vgRsQ zOdzRzsBB<#Zgo#mN&rAexuyz`2-0FttDwylG66KFbjYcGpI-BcT$7(n%euD_?d5Wy zreisAQ2c1>j1jMojh)`5zYUgJqPJG4J9?4yu)vIvkCT89`y)$Br0AS$p+`-HPtg$V zB5U){EJo9W#gcS3)Du)vee1G~QLwAo{r&`Dw*kn>@3e%3VwK|)_jk35oK-w)Na@1z zf*lejB(sQDKCx#uky+ViXfhngV5tffSO0Z}Y8r~F!RdOg0`(NN;awzGy)r>a0?ONo z4sT@8V;3a$PJ5yBSUit2%R%8hhCdw)4TPI)uJ=-m!SwbdEUv9c%30P0yvBxJ{bJk& z00Zv_TslUNRH0VKj!mE-Vwl74OFCx}L|$XnVj>(FsestsYF5O45WrCgjTPuAbL!gh zj*Smn0;7Q4Bx@RuzXIt8a{T-%jj?KJuADY9Guch>>BvpWa^s*#EL|hfXLt2bwy1P~ zPd!Ig*#vHqbNa%#tfGFiZy1DqQ0F%buG0#*4@@&R!<8auHp`UXYS~|&sg``aZ2Cd~ zgi3a96qQ|X@J7giO=1I-S}dw-OXC94Jb&H9JO?dETbiJJ6=u7tj9Qtxg)uNszbV4X z2O;$heu*k-;WGsaI)~;b>AagTB-xC_-*(^7AX)WG42m#;d3`vNO%{(AhQ@*KH}0(o z)yflYS{GZ=?L1*{oiUfAk2-wH1T!ChqRIpIk#j+g?kOh6Iu!Q9L44`8KMpW_=U~fkNkjPFivSIa;ngaWRHqKeZR{(Z9xq=*^IgzA5Q}(oNTq*N+Fze z(HjwIT97_tO|IEmDjC)f#n4HuM(aiN6_MHA<_yY3336+M5?(O>LSh;^kEZPhV@!yjK^8^=KTYs)J>;og2t`!?1CA zuU6+~#-Zmt?l3Zhx26AjLdw12@!zAUn<%(E1A_e_p?*17u6z8Ls%X>@wBDuh1#udq zdhYl|@7rL%tJiCO(+39L8c$?%v{O*Imn`G($0an3{XSy2rBOB&WJ0R5o_{PkD{Rx0 z&!aFV5d4se@-HJ1oj*Xbgm}l)2$YdY4&-M54aY1SfK*Li~() zR1c*#+Kq4o(poA<*WZre7KIenf@^) z5kNOx|H-Q-wv9ty$8(l;*5~zY<+KZsq|g)QYme#7!e6_*&bRSK+Z3TET%65Zdjivj z$Wln`%s_j{3Aq$izFz%aT}i-3=<@7^6^0sk{5U)eMmJy*iYC(1KYQ z3{q8MO$=_fmR_a)B9C*m6EjIv7G2s4$Y>>`d)onA<2X8RDYmmAq7n%IRnX##QZTdn z2U3fNvVeb0R!Nbj!h&sR1wd8gB;;zpDgy&rG}O696opv7#MvRX663cwnny~F7x_hu z;D8L%rkW}lm`Di$Md~K7r#yk+N?q+$N0$`5fqA!(I}@437EWr`>ENI&nFwE!9x=mc zBizS@U%}Em{72H_3)-hX_lkGJwc1AS!g<9L0VgknEXnZq4iN6b6>a3zN@U>} zL!7R1Nl|nZs|aSO{TJE$%O7+H?P9j*-ac)GTtukwV{?Z!%h@gk>AJAf8sFk6zg_D* zdD#UZ5{?r$TP8irN0N*hj-$mM6-AoHF@zOb{|d)DZpffO+stc~A^pMN(%3~(ZWRSr z*lHjDgs&`#EGEzn%~TL4;mTWMuH4c0fwr`GpZ2eku|3*jCKYmc=N~?P*4Sa) z(OUIVawA5;v7FvCJ(vKsybj&K-dA!xk>L-|uiGC*ZhU&PN2I!zYy};M(+TrNoP%g^ zXDk8)x?alON|Zi%JMpwLUY>uo=+(oPh>@Bhk;JDLAj(`uvr0pES#A>^1<+_i%y2Dd z*jPA(KNhu=tc1pqTJeR{a9!dwny7M>sIRKg&XzK2>3JO!Dd(GF(D?NwV>MsGQ>2^I zWn3}QB+11oTfuL5_vZ1MfX9Ux!@C5@2oxpGI1&rIYi*}7e7XL@@bk?>r|TUwSw#d_ z#6mg@6`qXedX|Df!|Ff7G)8g7{ghlyA5?Yt#VeGu&n;MhSpqu_7)y;Th{8B2Tp0MC z1bVtm(b2U$g6m>_3=QZMKX~Nc5j_goi`u6?;l39qV(tptu;pLOPHF zs(uVH+R(Ppb}$pv8rS=BLL*Mqx2qAdOjWm&D8ds1BTNY%qi`ja8R-or0T#dZaSH6Q zJUGVwn@;#tj)JTq5b1%*Au-*fydVrT60Ounhu9c;3b#u0&9jQyY-L2;@X?0gtYw!G zzQF={2)goP(g52nP4fu{+XIc#5@D#}R!A2;)~uuk$5^*7AWF}#OS;yp+Ub2HLzA5o z_+{5UBil^!c8Tp%!vG$BV4rc0b>Ui)8=Lkdb`cOt4@ZJIM_NwvO*P>cbwBR1NTNY+ z)4{KE9e(TO;VIw!s0afEUxl2PvxmT*D5mPtvr&^^`EC<`sY}CS4};4sxziwV3peVVs^F7E#|lJ%@){9 zS}(5lEcimK^4(~*cCo3@Q>$2iD4oQT>|{E+@1d62@pcZRuXcs z;(NZN%C(+66?HU~(Bf~RT7f0s1}t zS$5_ITHP`!*KZ){;4`s1#__qgyT_ixFQsJuY)d?_;p)U7EDwxm}tO1^c45 z+3~Ae>H3LWD{@Tmr{4F8d$oPYd_z{@-6oyd4KKn`+%;r+F)n~UAd^hh$oKnr{l3v#vB?N#<)LEzmd1@hjM;Guq6c5VNAB^Q!-2g)vG_@Y&ap)H_QO zC9AT#&cwL_xMDY#!SPNO&9<;$Xg7F40jFaL{J&cONA)(zB1s6T%36 z#D_zh>@R6T$|~fdWjk}uD#9jypDfDCvqxVYW5E$ASG~k@Q%SI?h^&s)l;joG3qCdD z7)_d<5{OQxwlN*Y)`y1YegMJQR#CtAve6JWSxS8mmp7g|Aq$Cz-J}GjH6^7hj_@!^ zebV!{xYBN-3Yxa+x1KIE=R9up-aNv4(AZP|h-Ex^3=T@@w0sIkIR zlP`&dy1VTk@}~2Q0@fm*nOSTS{gy{ zrNSe~ShR=-a-w^TExVDkG-<%7h*qOY!=(Wx{nLJuMx0xH1fjvN~JK==Yo?4@I( zds;6!GCCBXurF@~FlYpHuOF?)o$On{LREB`%^i!4{N@^(P!5#dPYIZsp&h$LE{_Qz zWIydZ-*_+eOz$TVGUn{bb6?ejO;Ls=q@5$^BLLTVtb&cR-9w6| z=^y3%M@Ym);?+NtggBYX`=5+@v7IMFv629`3}pk(_dLYl0{&NX2W~)X1u|>R9F`PI zG(`fx+w_3;<_Nv81JJf@SMv1klV1atnytn;AoE=9Dq|Pmztoi%4b^YiJH&Tc7OFz5 z1}GR29<>eC6t{+I8gr>}{ae7Lp<;zMCX!JK^A%u)-=%z5L=f@Ww9!b?nEl^7z5_|a z?koYrptmo*;-Aadm=<7;^Saa-P1q@8tnmZ3(O$yYVzt`G2P)50^d!4G!8x)<0Zv=!^s8D|Qor7-azJ#!5b z)BWeslU*agKosUu1G6U;5|YIIvo}GoINj5VVpKibl*vtaREd971OjPSXXJ*yV-b=C zrUIs~(Ojcz5`-H-o0eH$SuY>YFvWJ35^50+i7^&LP1X;^aMWJs_zZ)1U^r_iP2r{W zIGA3m7AG=3;X)ISt}m#p`lO0=*VVWdkfrxXGPxA#oDk0J2nQvd9wV}mEksDP+afzY z7N(}|s434hFhwtd!mq#)o&Gh>QL4nP3DK?b5A$|e2URQ9`_?Xg|L38o3g^cujo$`FoO-T=R63=ja> z5K7z7G?G6$E`vmfNMp|D^sfSoCg>^K1&xDs9oX*1L_>ju<0XaF@+32#A+D17`B0h< zl`03kH&^ro`ZKa}Y0yJM_+BF{xs%=8m$}M7#H8=bp^lEGZC59Fe{A+PP-ofwnY z5;RA=Yz}UAj(>==M{iW9TU0+c2`CuZ37VYn`HruDX?=7{<`fgaKz6*OWaSf-al;_N z7IYdWUe+^iB_v-LYH_+vP*bMbj}d0QXzr~*TXrfDii3w}1uM#ZgOQ_#v8Se)PBGvu zbyBonRiVRQ1W&+fzBrq1b7wrZfpNpdhF<1%iY?lLDnxx*eW+vqjtu(Dp@X=sAj{32 z?gj-*oGp~zAFFMu<%lqt9_&b1=$;s#50_V72Y(*3Dc;gSwLWwz*0@`+L@?-!8MzCi zFXizs8N``_+6Ja_63kRz=MUI!GGaRFGw+oYgLM8yq7rF2o7({TNa^G_L>FHT;{7Mo{nRr)*}F95V;{}3OJrlt&WPKb1CI^>Wh@oqR|6WQODo52lfj1i z6*bG9Pc4BuPCsx6o$(bIc*F1v{m5TS*_O0;ze{vIDb@RNF+9RFPDjYB3dGV=-@r%ln>s7S?$| zXc&qp_{2$sMo-jZ&PS52qjz?<#!lAzAf^Y`J<$xG^T4zZlcl5$Nyc%MQD8Db`AQse=Ags`oP3{Lxp*zB(mF<7lg(;)01Ch|l0d1|dqN_hqLi+O6htW_=qUxPd zBTLSos)_6#=F<)Jp4ry!jYyZ1IKGkhbjz(Ah-flC!jlG)IYE=4PrNm+a4#mihC*BvecXCV_V4h{ydef@r(Hl$;DTG&5#dI?R z;PLswjSXrR>8*Q}=VS!2e}$LkuovHG%21i4%*P-hp!;G-x*$gmb;fFYty;n172;Y1cMHhSOS^WwXTKxORuzR5H6Aq3KGCB?!tRl4P$q_AT@ zUsx+-mL^!OjVfQOu&*dus|M{m7OaHM{9gWg6xDiYe*} zhj(im^xS$KYRD$;3@b0%L@kzrRJS)x{H54arVHa9Sq0{ne$~G9_{G}t1Ef#hT|N=; zkWMxImIRzQJ-=WaeSk9tYl^4wAMl}UkG5cK8l^>!7i2o7s_NlrS8c-spsy-|o^`Kk zr*At@0$-@3LBBA9cPVAjm>!b<&rNIj#BV)pPF9m+w8WwOS!I_#WEhb`_J8hC!;ajS2ZNm}RjgkPOa zF-hyD#Gh3#kA(j8FsV@tn~iWB`%F3u^l_d+rIP%X!oH%-$%!=ODw5Qkk~&vQl9$QR zaj1nVDE6`8`7vwFJ*8ZE_m?barePH95mazUc zjE2X@Pmd6yaT%qR+)yQy=glKdYa^=YvfjG!r5{#`-89bBrdcj`Un$gq5>tiDt(YHT z7G%aNwKVfyle_5M`At?Qpn+UbfDqdaD=uJ!hZGCjF&T_B^}>nWlim^ECDv3Fkw1JU zWv6uv_YrPCIFFjQ+C`e(ynwvmyO2aGLd$`~z<)12^<=~IVF^`d3xJ|uEGwG{@VT`K zVYOhleYAl{ttXjP?MQK~-j;s=`~_fSdBiVG3!T|CrLFjgkU5YIZnDiM7H=7mNaQ$r zyPsMhb)0?Sm_er7H3&?H0Y}AtqaBENcKGvKWoTE z6F$dWKTodMUN(O{D&6QMlP5DgR$W}WzMa6-4@$&~z#2+$c$O0x`ro)g($IkkZvqM` z$p`?TL^8@orkRb0l9uX8hzn-%#`nKb$AR9zYNSjaU72#kZegU`S~YnI6iNgv?-Z~P zo@}++U-WTZJ{rmj0m>vEp>Oiw{ZMj9(D-NO%Ir9Bo5^<)fmH`GlYKpzBV1D7r2|hc zeZEcSjZ(GoxKz(Gf$z&5@1?EG-{D%i*`(Yf3WiM$^*^+tJIIr^ zXM&l(g#c=2%qWu!!}GDFZb)?P)VD13P+9pCOnN+9yUzWv*>6|+1U%9m*(_MVA1V9o zdot@1l6txRH_H)(K$Zs$i*xge!cr_Fks)X@i|^53DT>jH5gT~QGK|A!Ea~8!*(D{K zV5%3_VE@b>&kiA(rXawaxgdFsQmjubgUSw1f>n#Hg-)1Bh{mrzSN#P|1bLXyWD?B+ z6$j3?8P+@tAI$tNa;VwvYOnyFBa)Pe2?$^V?98Y}Lq8st%qH^h;A==-iQS@3J*Hmd8H>=w%Xq+i2}hVI`{(DeSf2W8CSQY}S}H!)Q6q|j^U`FY*3 zPe#MoCF61`-f=V)aM?l}Hy#NK9v!?^><)lE<9zknVmJfsdfP)N@&kgijtXkZSRa=i z5oZ{pJC1#sixT#KiU`NN?h+(WE;yUI|CW4SSul7qEq$Yx)kTY^P&`q(;9>#J0pK21 z2LbV|GIYpx#oS=ioVW#xse`XMXeJDymF~rvD0i*DHuKXWc?~6Tz4OA)%~6Bp10CR7LutTthb3yZQaZTu8ICJ zAJdWXE8=kDU{i)U<3kqy(;KehOd`gIfyk$14$7;rsch41U`D0#LKKK(G^3m)bPeWW zk}YQXNuN)dyXzxhRqT?gu1%=?7IFepRrj*+?{6q5Va;uVXpA$_Wt>iDkFYq|l{xw8 zwWP_l87pIsc7g^${aeCs(`2kyb!+?rWwB#E3dumw;|z!8?n~_Fc&| zwrHHMF^3zcfGk?oRYKk{vKZQx$AL!FZ0u4NmA?y%48$nB6Z)+!iQPqJ{w63`B2J_e z6Wb8!Z}c_y)~+s5r312FP96Ss7D+St$j$w>R5_1R8(_KYrR|+h!TWhj< z3J=&{m*6vs`6*IBgmiwm(knO^W2-3XYK(rLTSKdO^Fc{dnWVgx&r_nE)8b`Q6J2e7qkB>N3x?h4@|w%jdn0?v%)n z2|5z8=1*HHg$Bm$7vDE+uytm|>JeJ;GIF27SQ7E0a{g#cIaRGYasqE%M|}eV3aC~? zRdHc+o;y4+a?`NI;WRz$HLd|n=S_The}#{Lr~O&mLq5cKwDot3iO6*NHjL2sL=L%+ z(BWe+;7+Vw)Fp@b{1R!;G+yHS_jZC^Qi^*OHt%+1JGS2$YYGf|8!xy|M4h5JtZ|_0 zU0jEyuH442kLZI@v#oR&!%Y0aA$5VH0|(s@F)?tKnN8tcO+Mu{=@olR%A&eC%EI>2Zp^Sy~9cqX1j{%*!m$XSn@gQHCIFIp?AJZ2)#$2)O%Um(&3e?PKIpQ zoejY_ct8Cj2I#^dGy~;u><4P&CAAwXx^9z>{^>tLE%s?=jmh-77*eDdj$I#%9@0|_ zGU^UO=(6I$iQ{7gRjC8#YhU{+X~(&*%SnUl%+sX05NXT0a|`L@X(}}u)Q%hC#+jhZ zqQn%q7%1NJMz$l8v8Re3+{#KGvxqBGIz^wtGWt*$y#_5DZ@)@9Vv|kXS3H=dHP+3b zsbzcrNM#6GX0hFi2O0ltz@Foh^~X-$3gYWLvu()yfoTKP}jk&;@FndTZi>a==M2@DeBMrXE3g!{? z(NP~0^gtZ1snAoLz;=hHx&o!sSSB8g?I3Q@#Ocbst`00@U4aj6wo}aC^;X`%Ar@W0 z?4$T)#>aeh$3 z=I%7vWT_nVu;9Re(uwlG%I?Kol+h?HKkkwlJ+2(0JLYO!8!TJm9quS+D>uNitbs=u z)tIV0-g0`e-E?B;^zjVbY!ljk#T4v{p?n_XC36pNqJQz9(%5}j`Xi*<(_;iVnuSXBp+8q(_p6hIgcR4P*Nk@)!cV3B>0_gg-7T1 z14A;|%t7&$Yv4xR%B9tuiE(=~B5y}6)Xr#yLU2ycA!WG>-j~STK~y)o*l?|hjX-LE zn6tB^B5^ig@d!StgPvZ(Y^_&?C=F{Qip+U~vo+GjU@A%P@r4tir>++>bV|L2T2T(r zW?k57<4BGW!|Rk$!cs^eZ)I$NMmJK6t^c9)xmTfhZk3Yp)*6$xiO}Trq>Ya8XqKYm z8k8eMAHGsw-_E0~_1Buyp!n6JWi_=Dy~AEM1_gk9STQb1xH@yzfv*MczYQ9ODdfx` zRtKh0N|6~uV+0tP8~Qba!{aW=W2p~=-Ta`(MWwqo_Tg_DD z3X=Yv2e#!*FZE%)5kZEO=##ncE>O<%xwDiPA+vC5Ay!CzYbk&-6a#v8jN*<#pk0o2 zzA9)#YqE7}iqK9`qf&wUNX%@Z%Ez$q2NRAB{FScJDNMO#-mu8A$S980K<<^I@v`V4m=1?ns9vLM<9K3>@LY!UNx?cL#0pZ-OX|umGP)QHvl`CA zGa?7R4W=6{>wBk*k&35;2`98%71mR(oI#J}PP*DgUSR>QOpd%{Cq=Bp?5z>YJ*?Fg zOJ}c#;kS$e6~;$K)&&h2zVjq3kI5T*4dXUAl+2R5w1f$(`;A&POGUeN6 zh}kD((ul!uV_28R>56AGs3LTnuLZ!dbPIVeAaQuT&ee$LsM>0g8UK{<@;|v{0E^>F z6KM`=#mXlRA*w501fjS?lzk4>yu7+K#-F-j!k_0I?u=l|Cy#=*)D7>1#D@4r9N${H z6NZVharJm{X3ABGveLQ-Pe02tbj4PH;{U@g8|28mG#?SdFp<9;v&Ij zgXB!U3LWu3(+431aDseK{n8G~po*hP3VXzdI!j)|66vr0r;33en4_|Kh9G60)5dW4 z17_seP*WxQjE(*B<;TpJ5x3A0d4+<@napf7!m7tdkyOq5fnou3MEtj@ zm8AAwomaJ4pL&8B9+I)U!AFCC^2e)K{Ay{pKQ8HbLw*W8|4J~cS~*mTl7^#h7#zFz z9C=cIj_F^irLyvS_v*>E@twJR;fzcsnHA0x_aq->elc7-*-itTaQ(b6boxY-)OEz} zw-l9^P)~s!^T)LuYXxA9QF?&uAPcm6VsfgWJ4Jp~c{)SC5f~H`Nu9jRN!vIW^&yC7 z)Q-H>X3xb*lxN(dTvooF$}!!ii--_>%&!|zxyV^~GQLo;lN`PqT!5E>Z}hTN!5OT_ zi@|9Ggq!S0<%uk1O`@6VSDAvT%8IT#FLQfN8Y zkZ7LnOt*5NTH0(|3^N=nq*FLQd4Wk1LZ*fl^51fUzqHY=r9~ z)`vVM^EjhwKYONi=8R?QAMhR=RMwG@?22J%pLAs^CDfpi87e3(JC1Qj&2<)K<-ZF( zGy!Xni6v(@5?@>K1HmBc(=d%$0Hg^0>b}4^yH9Vl-<3@F599NKKqVNqPOAD;KS%vE zoIyk902HcqYfW&Hf;Ep zO`nr?a&iSr>*J+uM>Eh~KZW+yK$X5W_!k+R;id z@9LX7j{!N8qPEcTVmJ(%Z$q@<`!kkhrzD#sm?9H94qcPt;c)_a`g<`x^g*%a-=o%= z(kt*22`VgSxr_kTY6vRRhT6vad{?Ve9-Nu#iKpBVnB#d8RRbf)l%UVd9~;D*K4p}; ztjP-Iqcc$)w`L@*nNZ?T8o<${F(=clNA$IT!cPO$84e6=D<8>t7yaO&!rz|K*GkJ0 z;=y`RFO{lq54Me7LuZy&-R z35|1V;vgmp?uBu|Ex`nXE9TCj6OEs;s1T?ap8uWi@)qm{?#EHax)&ku2}Cj=ZAhwP zteVhX?-not9Pqx*P_V10TRBbXr3Y|OsasP}Z*ig^@M+ITn%>rAmq|W%4iXc$z)(Uf zQCBLd(RSt-3y{@O8xF9OlRoRgo|D(!i0yMJ{a@0~ zXsXq=p)~MhMZp~3a@3s?R=KY@=aM;5ZaRiC&iz=cWT2$XhH9kNzs>m^oRZ~9xE+S6 z970ugLawK8B;y#WOvTbsmJ3m=tMWtHTESfrXQ4Y5W6aC#FcKMaT8~Sa@SZ*0&D0#xbU?s~X*)mHm5D#Mo>Nucw zRoPI<%Y-Fj3GGflw3O~5bW6*V^IhOksc5R?r?R+&d zbOmiiT4gTE^|i^LfKm6T0aJxtAjbu@>+61?PmmaDv$w)zY>_%b9Ye^9y#nU5%4@H4KA^*gQPs@0&=i1#%YEbH6rd+Z@M|9$(iZJ*vUm&FR zcH%J1M7R35$JbtCXTQn;>c&xH*2am;JFdWHaAc4nHwS}~=P{AEq(u~v;2`5tSS?(n z&dJ1O*6oe$MFdGJhGrsjqxFmpbapk1bgK{)JsTB@k?=0~rFl($YtgPj!Ha{X8X+hp z9}J9qraDhLRphDgnk}6|Q}s2*EwK-Q-U~$95jw_)xhj3Ht}N-U@AhUcLQ;6%Cq7vF z8e?%v2~}T@utq01&*idPmtMUOfy_B%o9#IPAR}iZkg>EuJ$CcR*I)q2&QR$M|E((YF zWtnwXsDd{(Z}qKn1~CQaR$%Ilo*vLuxi8AKksqrRLryW<4`{67*B=P z3|1x-=XWN0XRTb7-|GpU@uRZTNIE;b(_s}!pGrnsvUZI?MEvP!0P0GgHME`1p=>BK z1GeLl&NnvpaUZ#$wANGTPBx~z990tO?JB2=78hV_@=f)z6Fp!rd1JJNq@j4Gx==qj zk>?4>C&iPI%_o%t9T;>hJjGSc`&DE5pj?_{{olPi$6ES{HvX!Bc@CA1qi&BYT43Vo zB#NRn?YzPTY2s8Nl4u3!Fks4VlZ7DMOZ7h zF7LQ}2jCWiop3Ot{U~YdV6rS}WH>;^SjovqD|CI$>p0Nprp^8z6q2BBU2qHM4+PIU zyv~!ll|hI5`8Siz6Xl{b;-{*vb#BqSI7!Pdtxt<*6op$K3oHqT^1w6|nY9tB04cSi0!V6C*;WDJ-2tV=Tio6rJMCBK<1RA!SCe(v2dTHcN(EMK8$Jv5F={(3h?FJ)Klt15umR* zMFvOOt_xijia1LOqCh$*PYpx6gk8IosYrLfpfJv4Y$BJ`x5%s)8KbUEZ;+!7YM0xn z@>l|FyB?ZBtKMQ z2xo6r_Fl%TV>48Z63GisKA7;H-&??hp^ZFwOPi^&%%{5>11{;fdW63ru_C8ag$bIzEndMBZz<5^pfG)P?W_Ulc1U;u9rn($qi+33>)}!{URkr~jo&F3A=EHU2tVPcP z4#7Dl>d*X(Yt0I*6XT>JCtK?cm>nNhy?<}|~(l;eu50!l)B&9%n49dwojpqdre$3G(y<8}iMMe(Xb0u;ya;8}Xo zwm=@upGELDp`8ECi*o}x*G4>P-R~VqrCGa#4rN)!#}TET@}~)jDD#0I`K{_Klw^D8r9O5M9s(X=ihUQ zXYEKiqr6JpxIagYLIGrvB!Lmt)HF`Yruvdzrg2G62#vvQwBe}(*1Yf7oX(>5SVVyl z=&}&XEhjN2%lvGa@YZa`^ZwOkkC7ZUAyO1H>$sjg+kda)(daq0eck*zkt?&P(A$)Q zs{uoum>tS41&j&a6!ubJpnR!o^2*tc2T%mRhhD9cd5Uv{5g>(z)pLrZCRc(F0>&9T zcDzqHXuH-&NmK!sk#ei(NnS$W$rfM<=#$rL>f@UIYMp(e#Oyk#%9PoK{l(p(%WFNRteIhc4|z(QvJW;4 z#}Z3y*SwX868l-@q_eobu@{jc(wf;*>OA)ln$Fxzxir78tX-Ui@RR!kiKz_STbC6l z<)zC!uE1`fYQzMTO5uhOES7!c1dgnq_1J1*j8=ibp?lI~+bj)WMqcMcd0TL23o5ds z3!tIMb!%LUat;bemWr%LMaGFiToVl!qX?y?5(SG)>ZE`~7#@z2cDz(-$>e_pD&r7? ztbPzP7Dd#Omj`9hS!Gc5Z$=_VRv@J?=pV|xifWs%Q`*o#)RF;I_O$-$#)}vA^&+5C zv=SueJyA9qcvBfyu0!$K#IyKrkco}>KWgB!HaOd};()2{Sn+H;9Ebw@Vs5t!h(F7R zxg-Kju*tS3h~y&WXfw+ojd!L@%Wr4UVcAxyjY|+@Rid1M3dHH|Qe_ZcRXwHZi#~fhA$R3X z0x)rrq0(GBNfW`-C@cWyEK&FSadJg*$He&f02*;e(rBO=HtEz+scq?GVgU*YFh^}b z<{`|5197J%+$GJ`Wp~JwL$%>`Z6ou*i2$&kj8~@N`L_u`ADj}$>HVzo%=HXYe@4J5 z$CxEEWu=`q78lUL`a#>fM(C$12H}i$HRD~|?W@>I++w}>nxEVbTp=~Hfz|sZOe#F| zg26{At=FP6j}qO|MGB=u8uwI&uBjINQezZP+S|cISKRz$M5$sybH3Ms!c4ln>CJ40 zgIx}kwH{=2-iBzU{0F-Ll0m0T`h<@w@*o%t8TF|iC*f=7O$v^Z3ZqG_h#P12fYCsL zmUPWY+lpufp#3EIMVZ@D^+^;lez52xuFQ&vK-O^6(!P6c<~7c)E@c=4Un=i1#>tB+ zWNC>A&boVqy=kbyI^={i{N9b@)SY=}e zjb+;ld7(xjG(^kUM)J&|*-EHj&cApSZ;>%{=&C8L%pPL~m}Mdpx$@p_ws9gwGUNOQ&Z+ zL@JT@{Gtf%cQO)ncB0=q%6OyS;)%9F!e%4XQkLhMxJBJi3DHaZ5%+`akkRD{670j{-BMe+Bq~4t(Yso>ixx;l9Z)POc`(# ztA385*%_Ca*KE1hP=d}$fVJ^ju7s&;olDn`=y;$r=Gv47E11zzSG-b7N>7xZHflBg zrax16I5F_mjp}Nf0J6y6ZSbLeqKb00z(Mk`hJU)pW&pwr}Km& zoS8(lVq|=Z#S_JD9HLz*yNa_*)}+NB*=;5h(s%|NUFW@H8HkLbHr z!=UB0XtbD{3Z9dTv8!Q572d&DA|}3Gk`tyd(%Z`m`@z6Bb;i96P^_uu+!*cDG+G&F z(So+Jq#HRYsYsMY?Fr7g5{G;<_K`D`&@FPQY{}v1m`@JY8`IcVY@}teeMF=>gt*nK zJ-)(@6MK$&NRG};`CDVPQ&2<~Tk=W%)o3w)%j9huLu*KM5_PTR-pXDA@Y10K`50q? zyf@m%S4+)*=&q<4@oOYc{tfW8F;`$R_oFe`l2ImQVPX^UYY<8XN)4W()(j?fZY{o? zGA(Uusw`JA#rQ)9lV8aNl|PVV3C<-Y5m>e; zF)h#s>e6<@W>rnfuw?Hg5sD|>`*}`$EV3ZL`j~V+!1@TN@8ySb_TpeeL9yD@B_5ZZ z0jM(V16U1bGaV4{5(Tmt+{r=Jss990M7b%#NdO;5Uk~3Wne;-QRp0rAhw;P95ileX zx1A9UyezDZl&6+-kjJ^MV`udf?$4kBDdo@``nt#uC63eow-Rm>3@kxWn(#81>Pc`tI zJ_>(Plcg#A&@g5=oHSb*yoBfyKp~2Th``;E?}%l@6ZAQCJAK9e$C3_;Us`gkv~IA6 zx{`Ag#ru+-(;`76QNzhh2abluJD-*q%}NEfZfgi4vJUE1$mK!|N?PjVS+{pXX%kR6 zOUQ#+4?)sH#)DMxoXKlIAEN@KJ{z*T=!SYNBvSS#35)~nxsb;G$goJi_1>0-DN)8~ z&d$SOF0llfK}pc1b4(1Ns7q|YP+6n)MS6TLkv8Qk&BO?}!P3YG#+cKVq*(?T6nyl{ zH`rRshBCLQtRlu-aIa^4Fl^yFe9dFIG>}gNGke|pp?K@PI7vIKz;u4+Wfn?@(LQH3 zR#k!d?*g?;$=;=(km4M#m5U^w^PS!4c*O1kH|Ja2LQbZ0nO<#qcL&EAGBV5428@&NE z@laAm)RyX+e3+z9DldY&kV9O+q^muf5cR|IS;052@=~B=4L10g{O8v}S7rhn9LEsK zKbbDhz3^*ZD};o10zH#ax!pUeEPl!lh`IeBd(qAjrHVxH%PCsEdji? z3i+PhqUk77X_?VoB#43g@~|(Kz8nllR{ERdKMoEnQ6HubXCdW8sS?&0p{WdHlG~~j zQl{hu+WaV>Qlic;JdQNG{hZHc7SwWMei={}$nG>JFp#**EogH03 zeD<1qTIo$!OC8m6Up;pqq*`PPTH2&~r4S7Ts!^x+3f}r4`%vf87U#i|?Cfhz8F4ma z|GfHz@f1L_-q2~{+H7IXa`*I6b8dm;iY{ia~rlEe7t#Hg# zOaWIHIS$ZX!^cEFQa)j(nCm7yBo;C)8YVa|lH2Rfo?2IR6fy0c)59KTHpy|PBV`-$ zC!LqxynnsK_){;JMcBF`pC@pUMYO;tIvEjiJ7%@ylreo;L{+YJ_q@d+maZJtKRsvCr?y`6ZN&XbD50RHVMs=9v$4&IPs0wn>} z;(4}mB4q>0_xiH z8-3a@vRqMAdK*=O_(7R1oKX!e42`Q2V#1O}+*5@zMD16REvtN~IvMrE+^+Tk0!A7J((T zmKy22n6!t!t{JhHQlDe< z8LckJ6)3#Yur}!PuwN?|>7nI8KC1M~KGpg!?YJ_pgxn8X(#UNnTr0#K4xf9wSTK~2 zXzq+&MDe_8@F^UdSgAYLNVEAXc_qy^lbjH()dX&)f$@tas#F0hg&6{i z2x1@KszfCTrPR&%NzP>86unQtVb(COq$UI(f!x8dSY*kC1lBO@6AUyQRX=QTD(Of* zEufD49G6Fw6MJ>U9O-mk1CQ8v2Wl51R^PIQg*09q0O8=*f zrVQq91~bZ#>BQ#!@v5QdV3u+{hrlQ6t@TQ zX3}WZsO#n)n3f|$Pmy__foqW`r~xfQrb0?hmFb)cDM`&X`F)Kc(|z-J*jqrm8jxEQ zHIR;?^hKc6HMgjns%+8A5(Fog^xMf-=VkGIlDP`+23l74bw@KlD7^{fdPN_j``ST` z(R@w_b=01sapYt@iq3FPV3g~z#*$^OnkYv_RoRDOKBF<^ySA~d&A!@L4}Y8N^OQlu zXukCAfDQ+)d9>?!U&J>hl8m%HZ>4c3w=5hrq+mL9U9znh0D}bv{}*REFfrZfqAfA2 z*ySB;7=Voq*2o%=cNagak^WXQhcyQ6Z7p-Vi!~}{@l`Epbj3PACyOIHaE_CV9D}m_ ze%_pqy@5R#^%uv&Qg*8b7#UsYFDioX5?-M1o?s->b)&M0`P@hKKgGXHJ*$!jYUWIu zjf{hRFTx8Q&6U^=`ig7?;qswo+2_fwCOoWt>J@Km*RcJZ1e+LvLo!F)E_yx=z!eX* zpCleiWu~*>!m)}Gm>G1#&=E0;k`cx|2a!v+-ef%W#Dx-^&oK03G-NY=hVVY|FBk#ov&()$GRQaQOcBZhh z1Ypojs7x+wlcLL-^+yvTSmLZ0t7?O!!$WcS@YZQiUNyvEZH>)~SA3|g^jDzP|gF0!g);!W)h%TF0noAB3DoaUsd<*&>!b3a^HTJ zk`z2;Jn6|D3(H}kH>_rmWwEN+9l(^C9dD#Y7|I^h@Vp{$$oP}fwAdb5Q)9Cn8|PFo zx#U~6{#J);aAh94OLC|q>ol%qcV^J{GH24RjB#Ry=ePNj>vbgrE3wxTt^O@`sO|bOi-BOZP(SsaEf~MISL-rbv@<{l^s!J<5NFPEzbirjV z!9(?N#0437ec?DcI+aDtY28(&(d+2EcvVBy!FtKW#8FiVK)C6>yUgCmGaQB?a|Esp zoM2>ST+}!m8F9k*38EeHaQJJE-&33G3Y~|V6(p~!UY6d}A&8Z11pK)U9WWKGWDx3ZXBYTTPRqu^AMMXen4~?%X3`L&f*g^bm?|deG85x6KC3#h1>cKj1^zVhwhgZ#qsuAQw(0%0ef`I1nNj;YtW3M?tw;wV&HSmD8|5c5)e#A=p~oLFh2#3NnjdRtTM=Y zY%3$F0IYboK)^9Mi_%NU2)La^fwEGgAcZCepfHS&h^ztzuDWlU>-Ef&%$oB^memE0 zX`(C|zmbk*ju)##_Z2E#2&eBR87`rHZY)QEm`VHOP0%ErR|(PPl3-Wn@VWsEQbP3< zE0q-$Oo1mssAU$OnJ{B*(rDG=#H!*)?TbG9n7gYm;80x!+R8fRd-mmJ6D}80tL!lD z%RDasL9k@We$?cJ9-9&COsNX;%cFSbjatu|F)NI=Oan-!@Z@L0+d7qA~Cu_74Z0>rvlk#nCv)u zW%8uU+|uscfD;fqaUs&5f?iGb^Sfo>3A}BL5p6Ao;jhU@}luIdznV> z-IbmG<-)SX_%B`Jyi^@-l8Wl4h#@VHSIIsla!PbU=DrN}(LY)g|($7neLxu0GI zn0Zz0KxqxIA)`&cOUupU$)A+DYDH4BSn|{IiN95qH{0m#(hmJ#qg}toiIwZgXkY7? zFB&#P-k6u2p!AYy6?g4|<<^ZU>s?pyyAtHHW&ylk3R2FX$Ypa;`qgJf7pM zIO|f5%YPD>xUaiY;^?3H;yitNmbZHfU2AA|_>BaPjL<|mFXlR8ZNF;G>O96A;u3}t zZRDhNPP$Ho%e~0ysSFudPn)ejBu2n|su!sgvrY&|tqi%TSd0AqIO=9hOD>}m&6KhN zCytq7)AcOT-x74_Baw0#_36`EydsCPWO0mbcV~W)0bd z&6C7Yzb9x>*|RxS+?a$R8A46f88HofvBFIEVt&#vFD_-Ea`C*&ay%QWPCWLZsS>8; zQqU$U#{pSG;3yhuOu~;sS7a1#Tp=Nf6&#l!akd|;vl2Xni~)#t=%aM)gu2m22qEp& zGcHd)WfYr3bw$cHt~By<-v0)Ki?NP+F0_tzBIX-WRMqA40*^2tiLpbu4bV9x6DPou zR0NH2oD=?fUfaSln-8;bIV`)Hb1edzgB@$F1j#;pY`BL-=Wo0L;M|XC#RADzO@|7LLO_RY1Uq}-Z2WY?DUNyiP~F;%Za8UTVV8Scff7(ZIA1foa7+fzl{Yo*n~E1t1Jl&MGwavBuD74>jhAzs6R3zwJsKlojbrR3>YN4TTeRA4`Ka!Fi3sPqb4I?uA7{ zuS>MoVZSKNd7U7fPODxX<3)QYO}>uPOYGyf?6rSZx$aVqE3y@YvP#2FMl3aH8pav2 zSk3AwTd!*p2nGY!c)z<(LD^d67iu-64$J)k24Q`3vB)#y>a~EvfQo=TK}eyB6-Wq( zSsIh0O*Em#?6K((H*`WM{V50~lq{=dY!hRf+KgX@0?S(7bw`8`Eh&;h0CH zR}T!AxXg5&D$d&ABEbbHgD}rZ#)xLt=`f5>#9HJ|TY(KE3>ghOK8{Ud@K_>M*$X(UD_(7>Xax=D?~K0-HpZ0 zZjm)`g;2-=uwHI6)VzQ`g?!Tgr5{&rFCZS?WApv6*zGPoV_Z}5V;OHoP;ri8Dg!_4 zF+}Fcsi_B3*nGoNu9}fr`3#nol$aZqqo*Xsb1EA@{oYz$9x9t_7fHM&G>N^^6tPigA~tm{;Ya z?#UZ(X5OOtB)H3sJY3B?P)FxEf8!mTbVSSy1Fw@iMQ?SVEGWk3_m*GG``lbq<5a)% z)>VLHf%+*BdA0OOVv?1sRUJJYTonvhd+&$!_VV5}oETbkI<8!)FSB!vDazr5WrqzV z4=e`&&XHAcrMBUHKSY#Pxc-y+G6S4?h|Y?an1i2|fxsAp&VN{B`8AR%X6~19w~VG- zX%q$9#Rp3poqj#qT2()7Y>LMbuJW&zA)zA?h4B+0`NWSaPy$Mn$bSuGJQl*IkkpJ^ z*mc@suv3&PCrk3mFe0}~Z*8Vr6o>Vu+TAY8Q8a3qiQqAVK{jFsYH9GCNOdLe&?}eu zN&MH_Asu%77Oi7%JhPLZ31AkbGn#}t!LG0XaArg?F0uf!gvw|KrITzUYil*8)LB0^ zoNiAWI1;6Scmhg%73a%%i-+^aiR?#X(aAkk!;?W7bi$-+$aRj31}1^kuFfkdK&rp6 z;K2k`lT3&gI(cvy<2@ZfGJ%;nhX&Wk}L75`ssrVV7ocY~Z z_I~)FO5-OImtjvQ(&k#uP~+A-bYLC7j=$AP6xSdnoPI@@AEaWN?+jl?F zamQJ>bA}##g2&}&01=ULwPR+_s3;kziM8}`84A;Tmhv{xfI1TkhhPcR5TI7@sPzsm zN($j(S6FCShTs(vS;MZa_K+6)k6l+xKuG1X#4pjE-oH;}Xqe@7b zpjrv08#1-rZ@(x#S!5jfbN_`%p)Ni0l1#-Uq%au7v;DeXXy7h=VmQC7td1|SJUXK<^jr+Kq0&v~F`eTQCfGFe)VDD2MF z{mg$&>h`l@;);(U6NOm{iaHmZYIdc(uRe3C$BG`$>#CjckRJirL?2Z|hPa0MyVks6 zl&@)HISrortNO|kTl7ux9bGqJO+_x}xB{<)-2dU5t!l#Z7%A!M4;ly==^wY<{nz2u zEUqSUnCarj1Jg3n7gbf&8!g4x$@Bx0(+PNW*i6%qQIW>9Yu5TKmBf!A=&`;ZC3tO@ zW3TxqZ~TCaD$`UNw-XLj%~o+>ald&lpX%@YW1v=cJLfeB-kzYz?(KkO+@C4iArT73 zK&z+guTz@J>EK8znW81Q6|EfC(t0o5ZKXUf(3#((f^F_LdqBJXM0+0_YvpJlYz1CT zR!GpX!{9uKIRYXB4AL51wtMUT-j)9C)nt9h@)?C$C_+ zd8_);ru|4{oGiz<)3nTrfD+M;6Q&40^tP!ATFRN%3lNVJw1l{Kt?zUf-zrZsIH-QT z4Q76>GPKzD*@L>|30Cx%Y|Vg$rU zy_0f?*D~rUHm%-`Ji(9GkI!?h2>Dj^RBvFPhEvuF&qv_B8A3=bWRh+(`V$$rG6C&A zHkCyN<8vpI39HrI3b%GJcUc)Sfoim1p3$jdP}_*mai}Oo8!OtlK$*xX561zSE9>(M zYXkHeS8Jw-;yhW*{m{X)43uv;6Cg?5aT)@0KuJj=d$&!xBOO;!ANFLa$)&52EzR}z zjErLrfRV9v5H1HsY1W*{xXRSglta?9#XT>(50)?j$P!%K%JQj2-|$1p%(MbQ_F2wF zrjtvK(}9FurBk=mMQ;PL&O_BvvCpo9mv&sSH-gohj|6DZ0h{&Mrr;`Y0Vkb+#EAm( zt^#y}DzF{;GU<<>j;Sl4Wex~cE%NsgM}#Y4l{(>T;UxTU4Ocm_p}xAu16}OE8~xtl zwUT2DKHMx7*Lz&-hh|EV!yp}{u4_06zHgJf65E^+oC}A-0Nf7(k6(p2Bv(p>R_(je ztHz54cG-LjV`*Y*j-TLHZN~?e;?kG`JhLn|iO`p|CuRtsoV;Nw#^pNxn4_vwRtphG zHA{LTp_T!~Ig5o9*)Q8aX*2GRj4R_;&OdqA0qRYdb{b9!2qVf!laBJOA>(Q>;8GC{5d#o$n?(ZTtrAJY=%3N0C1wrM2~k*`u^IAGKo zw%|n(N0~ms8BB{DPWXRm$K`JaZ3HY#1#_-xE{IzB6eQ5*-e`yg%k#H21t{SRCJcqm zB5^W5!8huvNe}6ej-`}$0LM}Ybao?MOipO?b449ve2i2@!B?kBSO84*vs$I?3He!s z1~i?S0ha4MR&*C|QkXt;~X=@o0oku#>hG-^w zl&4D1Qgu9at!N{t9N3|^#OJZ_(a}JC=*FV5D&4Ei^%NGCxxI))ivSXwIw@qTWU%4j zSIYe2ZksF4pTBl%cZn?;N{><>1x)E6xdC8GN}^@{ozgx;_nlh4gTU+A0zY-63~r!& zNgcDMZxhR!({o5=&lhPU(pV>Od$p4U%qs~% zX`(hCUaCuFg4U2Vd&GH{)J+{jMy#G!!v#hccbU(#1W2pq*<~v2oV@gGE%|BT7{n=L zi6n@w_miS{>_DZR*_+6K?R$=im+fzox^Ur9JpRtx@w^wj0MC5(GjaQEkHw8Q-HdCl zJwKPKjjnyXicur0RWbp1DQh}m%ttI!zah~P+abTmuu>Ks0zwVIw~=*Bu?b#Fto>(s zFG^ILei#`}L~FZxoUAN|2nHz#Fx9kQ%aN|*jWxc7rk9H{=K||F{A#gmKg! z7nW&Oj~WK6;|Bf{3JQlS_-iN#v{DfJGOxJf$Oey@`?((rB{HNkg=f{c&H*U%(|Cj6 z!YNNhoB*DBTp-C22QW@OV5Z|C%RT`#|JDbr9JLZ)0#EKo6m`JTO*`;_j4P+BNokcM z)(!X_ciizGW|dHH9np0v0VHRDS?c&O|Z`rw}|P=qFIdreyI%W6mY56$oi1%z}2#cNiU5T;w9a zb0O1#F!Bj>^@qnqbLdsqgvg8tGZaRB7K z(Sa2Xl(JZf+t(cW$o^t=Vd=~?8e-`Zj|MYB%e($87iP1u5A>>GM3Z@93)m0fr~a>Fch) z9=F|c8=mpZXW-L6_0#eA$3J0n`YS$m&pr75SG)p$>s!7NzwmRvuvI&j%~70B9m)tB z0=$yB%=w5o->j!;G@4^wV>+H0OI}en>^Q*G^uV>yA;i zHM74Y-z$R&x*}jXl9$G)fo*)djvWmui(%H1^IXGuuX9N(DywO_uzmR0-56fI*TYz}#%Ya7H2Q$~NaHgtb=eMrt%gFduL(#?3D8Pz+DFFS z9u;3N=;k$+*T#&(_;RgiV??W~fFWw)M$tI_Sn?`aJn@N7!hH|kkMIBf@54X*hkqYG{^LK6hwgtQ04d6D z7V`n(Yzd6(8g18G@EV7QaVV3$ zC=s4tmYip)I(O*WQNu( zG3NlNotH$$nRw;se7nmEKa>c1&ae#pn)*@p>^VOgWjm`9T3p!udKmNfV!n;tYNqbHc&uo zIE7{OK}J`<={Ks6LO(s!q=@) zE`wW^G6+X|l=m4gcrO0RX55dx{qcDF7V&@cH~kH~@|8b~dp>#(9)9Q%JhIIPT-<(U zbTgw&_k$zJKUbK^+I0nADr*BaI|W|>zrL~ulXvUfY%HU1xve!q{Rq&U-beIM%8Lo+ zK&9g**CYC3R+l@Fgmg{10mKPW=JhB2U*8Hu#)#5?JABn}C$F{|Uyo|I)l4`R7q-M) z+)mntu-9zBznnXVYtCOYGK5{ucRt~9_zhqAmH5i9{BLmUtvBQ6f8o9Opa0yS!4JLS z2XXDS*WlcC&xE_B(Li)~At_LOoiadXTIqRTC8>*R~wuS_lBecYQ@8y@t!x`^4Xqo_Htz$RGNH zc+72&+ra0>+rKb`wl$x$isN}(MRy;`p763?6jOezZ?CMHT%AEh`=Oyt$x6) zqhGtH_%h|M?Rz0f&J`??;+N5Pv}_E5f@zA3xXdz>+V~kORJsBv-J!>fWL+cA0C~)a z`>21jJk~47Eom^7ZxKN!&Z+6tlN`wW+mfR|I!d4GOvXoM@wdO>8}Y_ByaB)azyJMs)-#`t-}66wH9qu#594ibe#;0fmRMMkd~P~> zWL_EJRP(Y;i2fou$X!JNNxwjJXc2a$8^upycvW1Ex@W?4vC?XwR3U7&WTVG$H|=lD zVj#`213$Un_8T9r03F(E3gcdpC(g}NCeGKnv~-#GbkFg9HxNdS*l}Q7zMnv=@mI?-OFkr7%oDvDi0<-t8yMq4xJCh zqvXKB>6k>q>K59#6QmAL5DR!6ld9c0?PZ;P&gbEWAHqWqKY*KVehhBd<^>+T@CaV{ zz2CR#!$bJPfAmk_8FxJczwxX78~n|`{Fm{uk9};@2`hWF8sB7}g1YyyhK(ifg`Y|4 zXm6>}1!&fbVxnZ$vU*S3Gs*g}P2oCeARc9NwgT$;TDh3Zi0v+|Vy$X+Byny>3 zcmV$BZb;|wv}fFf8#f^M*gYTHfZ^e}+Qbe_vdFxG4zte1Ou`(kyhr1;wS%Ic$U!+C zu70upZL++rL`Sp2NdGg*@``{#gWdsqkrvocOi+!hxHjH2s1Se~KblD%$be*!g8up` zovEDDIUa!vR0-5Y=DMkvo)>_PZ}*AaYqxUFwbx!d9Q>o3oIkcj|Mzbh-yL_{zUjw` zAN|1}!q;E-SMX>5{9nZ9f8NXShM)Q=eCM})+XxC$6rtPfYc5X~N`pQV-E0wZjNGCv zF3}~k8av`@JqY~VhNi2j!CIN=*scZ#ah;(uLT6?ww&|RUOxJTp*(jP#Bd|vQEXC>H z@|(EnsHfL)5cR>xrJuREw36sWxw+6k~O)L3IMN1H_EKtEz&<`2Q2T{;XiG20=8vmBZx48J3il*$5yFp8loHfzwwi@o2e%Y7d#v5JiC-~%7Pxh>MV=`lBM zPT&Xe1K`7Ev{Y#D-k%smW%QSt6mS zo0u6{T}A+(Kw!T#JqIoUKS%5^Yos547s80~@AoVwS;M?G-#5|RxwK)(ssk4iP*%J; z-3jL`)EY*&iWSxgzRcqaIDmQBft7v$jXYd$5-{wI0Ro_7Wri`vp^|tqMEo(*W8;i2 z?PyTbi12ZkIyQWbd#Iz_I^kq6N253lXY}F)YV-C`=CLt~C1Pme$-1$oC0gG-l98xG z&u8-im{0zcPr>t^_dMMH(1ZAnZ~F(hc+qkH{rBO~t?s`EuTOc(lkvz_7yj7C?oF`P z$lOUkxIC7s>)J6~YDM+HlSbb1UE`9D8wI#1;C1%xVJLjX-biy+xXXrI`$JLV>8Q2H zTa}hcJL~JyaX=hN4U3$}p+*0epXHj6R}PgWKQ^f8df6}$Jg`e zUWh|gZdZmjo#p=3HCr}t^UXJJPT&IW9;KdhIJad4-~FBc1oz&151#har{e{m@foum zhyucoi$mr$DO&0rzfCVon%ZkTT_#Y_8W)quZgPjOa;#FNiU0FX$N0X?KqH-f3ExXz@|k$T zt6SUE&6|StN%Vau73JmPZ|Ax^donV!Hlgt))IyRncYw@?j|L+TB4U^ z2uXy<*xn%M55EAq4u$G;NHl7F=QF!bAgz8;z5^?_ z50bAZ{rff*^AZ0nGezGDR_WZ&I4DKsO)i^0Tvh*0;)+cQHXM8XPPH8?xon#ZzJALP zZ^DBQJ&XtLdjL1wbOYY;mbc-pZ+;_g-7Cva*+-Uvj9cCa*2oy@$zT3G zadKvIvm#+jV5kH2)HPA|<5GRe$7MT!2?U2oCO_3nrTJv%l<^W1uQUS{xfV(0F+PW4 zaPJ|c;#BREbS%=+RDn`JM$R6zyBZ3eN&qY~&AEQJ%yu?m<$z2|C2Cn|lZJ9F$Wghj zrdUI9o$X;7=4f}R>%A0#_?3JNN1Iz(1e9}5LNMGR-$hYj~LQHgW!gId!HH>xjHbG>RWAJ=r6Qd)q?E}1MnZ`YtHU5k39S{$7 z?GIb=@rdb*3UKpDR61}|O)ucDC^FH!N7K|*$|vJ7!ai`V7jBKcB}p;PfV*qo_JKE!q9!-AwTeO z40=r(iImgnMT8L74>ets=WFu4Ha|gf?UucMyM6bbcfSYwvhRx*E{?x%x$TxM#lIiA zMt!bqKowXoii?YmlDpp z0slp;k6s+WeE!@uvwY~D8RWV1=eB;%<(QUOPL-=UEFbq9j$_ilL^&y*ygg^o%w!d6 zCs4;~qa2J0*m6|KAgI>QGGV-*m$ZZ<&NhOPp0nFQMJUn8gk{G{_A)c=xr~z_yFgb$ z0q3?kpc`+z5%+9mpNAfP1h?IK8{YG-cjMtlAI42L-Hheh^SJoPg|Hdj0%Nw8wwAP| zVo#dWKrC&a!}0*qw3+54N)Lo}i6z|bVle?>^4YXZoZUxXmi_QBlLVN0>ESqIa<+7{ zjSe+IEU~qFt|+R0#Fkkq3tb)O)3%X)n1#+G9H!y2k1H|dsftwj4di!#QUrTs`TR-vqhRpNLIhHw0- zQB4w@(QE;urPi*YypHrhD3$@LOm&~r6~#~o?_gO@T*GnYFg&d&%S_i?bIthbzI*Qr zKFkgtcjujWWMA!LVjf;K@8mOpHutTxJEVpy{CBjd%NYxg%tq_vf)&uZJd*x>)5;6m zr1hOoyc5^maD6O?^{u>h9@lQ5zLzMjyY4zXW}D>x1H|`@?VkJG z=i;+o_R=lexqWkh7X~F4w(q{=OTQSu^|$_iwh6*>x%_UDd>)_w%YOx)^hra%Gal=ICovy_56`X9vMrA@4ffl@nGQY98yInKUmH`cr4nmM(b1p z^UBNJn1Mv*#JQWu0y&xOCos#tsp}?=25HZ8GI*9yb?>XZh_unJL+{W^y<$p}seyBL zcnW=g(fvL#F6FkKBIL@Z;{<*7#_U(IXVe!j7V=riow%ZxU+QtiUkXn@U5_o-7v{~M zRfS<8bJh}QaCtu2XobNUG5BHefJr71Dr03_`SRtQyb5JAx~H9jSUogtdPHpOm`zRIkgb7=2#B!obsqMZ4w&0%Ie$jfDH}yKnw& z%UJDk>tk-ghd=nitT*PH+tSdJ;pMb~ADSd2!+0(ysFNr#J7A_UmD>2uD2j3JnFKM; zb=K%FY*Fa%`n`Vuf9r4jHC)`H{NMcBe+$0vU;bPCj^FqDw`(roz3=@$@s`*B4F3H8 z^cV5%|L7m#xBmA37XS8NeK$4?!yo%&e;yA#bPs;l@BZER@cTb7BB@8W2=}-B_TPn% z-u)qb%Qt^BcDHc*V{aWmWmaYUyjB zi>L>mCYk*$V~?Ha5L`Lb7LV}X!~y=jbh^S0@);wOdr439<;>8*c}KMhOTAY6paU`a zdx|*bSXu$dnCzyLI{1=HJ}&=BVB?T26nY-{PzGTlQbr!8;2YjG_su4Ng8&xy(+ITx za-*RbhT4Mzgm$pDZdqXtaE*RUB8l3w#p{ZQ-geRC3*&(VO-9R$6aeQsVXvJ>#;>L5 z6=Ru^rwBK13Da$QsV<5Pl{p4Ke(U*r-{o`c^0>8eev9<>sCqh27ZWjIY;dZxJ!#yB1fgP;2pVz+n zHMoAu0^a+9_v1%?{NLkS|IWAK`kNkuKk=vj6u#ph|2=&4!H?ko`zQW5UjH+17~r;S zk^A*GT!+U$?g@C^Prn|&_A7oJ{>4B0=lDZk`-gDt_1ELC{guCh8*g|Fp81UD;Hh^# z3pZbP6F$1lliYs$9eCjjUxa6U>N9b}^|#^2U;Ww*&@bRuf9bCrG(7tmcj1QXZpTyZ zdLmx=L$AdD@qc_Te#3A0En5lYLHwO>{U$v0k&kTSIaihsmN8NCYldBJR!fJef7W$| zA^>gCwT-<@wB+7I`880%6Q+~doxq;?>@wSb@AF06?R7?Tp(Cn-bnuD#GMuz0puaCV zt~JYfK|cUx=~4XU^_c+@%fCm0X%0bTNGQ4{SF1#=^|OG zIAf%i7L;wSCaoKVMVjI~JLgZjgFF#UYM++>X68VL6Vd;|E)GlF^iDKttL++IajEs( zZx`i199&ItpntB0Tt{hP?E4bL?Lp7J-hYWQT63nt2a`$iMmN06UG!d4ZgnlQN%By} z2k{$uy|f$M->^+mE?W)$!i7h0;$x z&%GbTyWjK9%~77;5{yT;DE|?>;mvQr>)-GOeBl@Vms{rWLHzgs-EYS)`J9*HZEt@& z?*8b<@XmMr9A5a@&&OTQd>WqoNl(JB{MEk-H{WstKI^kyh?o45mvTNb`0YCA24JOY zNZnLW`==9a}K4CZKs@pBT&$}o;XK&^~QhZWbU4di%SYs!E& zILjkGBCdxYdH|pLsh>W0{n$J1z~i@a%ZEPjUVP_2|BelO-@ZlVH{g!jZ^M(Ga3?-v zi`3up<~QR#?|BbC^uhO!8vVIt8Ku7STbbuK|K_jSqV?QPyM8);^M_eH^=eet-U*N|AY6BeUct~%QmmDIf#2d_A$J5yYBzoCP=T@ zGLbz)c*5iF#M7VlH2l~re*{1BLqCewz51u|q&psucfaGEc*E=7iaTz*vvgc!P1)73 z4cw-}L{@n%qE0S+G6C6&b!J&|_?| zM*YFX{u<9~m7GXf?7Hkd?{!@EaYdd49{(0PrtoWgp5Qc{#JQ=CIVo!$$PVpWLkw#X zLcWNZv?OnYQ3s`-zgP4}G7id=s9kp+BX_hu>dxZBsh3v;VrsaqGAoX>g%K{@Fi;zxmhy27c~c@4?r9 z{olanf8iJ6SAWTu;OS3)20r+~594dT`j6q+&wBP&!g&;TJoZlbmR;-*M!otqKZVbD z(TldM;-|5kcYNeSAI1w_^a9*>--B4LKaYDJxF0Wl`RCwY|Eqt4m%aR#;F=p2{LAn9 zt^*^5*&$EwXVZx`X_!j};PU;61}DQuDP+}l36KI}-rE>|ttB|MG0+-Z^xhp>-pggG zIl!~@$PSnONhhe2?v2L7xy<<|%*x&K&I!PQS=&R_*}?YP!MYrfS|QFh+%HQfK@fBg&i<)8N}@F)J{|A_DT zx8H*ge&_>RguCLo&;9iA_#j!i;zHRcuk36)^Gh92qd*tFHn-j3@ z_ru${Yhu;cI=0f-m#FTmlEyX~`y~0s-+k|#eQf*O%TAx=>kVbxID;9k{Iq^PKCagJ zOg*gGr25hv%@iDHB};vU>wesQ_ub>FXFlU;c=|J+g+Kns|0I6)ZExG=RQ@C`KJ;j0 zQegw$iPyCPaR#YdDM=DxW$c_Z4xJ{sH8x}fZ8E&VSg(PVZ^&0ea+0^d#Sm=_)s~&~ z!w36vF^kMA9Xwxfv!v1ixO1A1g4TF?*a9*nZ5sAK|G zf1CbY1gm#v`dw$VjtCjrXI^8;8)*P3m`!d*v|X5gr8AP(DiyA&x!C&^<-fHKWm_;48FVW`zk zH~4wh#iB*;;^`xB*VguELiAtVe##6$&qf&j|WR2chdg)?sjpfAt_Jf^K#) zWpSkI@{cRC4AkhcnD#ihLmyY~SSU{Il83pen#Iu816~Fw@cqhb)I19-^ns~}Em$|r zdQ@LZ)jaY40LkAlw}fm!~;b;2(42tnQ1uUZC+wHF7EO;`9^?oH3l?)SDA;5GaHz1 zRw^!;uFRgx+^ohjWou1NG}t|oKR4FP?>t=R+voP)*R|NUU%mvNmnA`B5#x9C*J|cN zHm)dR1=^&}mf3a$9E%?r5iKWR&c2%?w*taiDHr^82xkg9_6OBG9*a%KonR3*u%b_T z`rJ$`rQ*ULD^3*n7tD>n#;mnG>mM{W`RfaTP@vDhcLNM$Zg)6+mBo%Zquo+!Mh44=?coRxUGja9Qs_(c=&K3jG5T3 zrCdBTEphgQvP(a%fCK18Ka4s4IL55`&R0Vzn{$mw49vJRaI$wz&)UIgfP!1ot@_?6 z$7!NedO9JvWD+J|YNw8fv=Pl+jfVhjiWfL8SxsmNl*`M!98q}0`8NZyfJjzPPFXHS z1;>#Yp37;2j*Ib`3LYq8XZ4Z%K{^JPzK)rdzA=ofDioUY9L84)awaqYw{}}7AQCJ} zdZMZMVLUq$L>@B?nhwWcvd29CAt>*$hpkwX1Qg7uu zn2c*ao$Tv$M8JB6S%*-f4Q>(K{0&#R=et6S+Ax!{tvH0FI$=lXM=2?|V8bM!iRFvo zb@X|Lc5ebc$HPt+|8<|rHMR$u=+Cl?d!4yB$m{0V${7GP|AMR~4|D-w;18UU{Q(p^n zo65v3j$P4*zhtZg?nQuNGBI3re8NWBgI-(_q~cnv5)%HXGO|z45eV-lu;lt_6p!=TaAfS? zv;HrwrE!~*RZ9nJ&*d&Y-o>)CpV{ay0}g9j#~b{ z-+ercL;D)~6nxuj`gh!VJ3jL>Uxb^kzd5suiEWdHo#XrSAdBma67>$a{<>>$!*w?d zz2~kSbE&QuSo)0nYN@D6pb;HXI~{0~xoR_xd{8L?B=xkw9l!x5r=4cxX>z@Zp6u-%pmb4iCJL;B;sGw=IxI`a*AxenRYP5;G_VSM_v2N7=rGI<7mweQZ7yh<>Q7@~9j%ZH^LMIs9rK; z$9x)?S%#w=i68qQo3`)wEv7&3^FI%N@h|;FeC^kLH6Hhb$Bun@E^f8+=fB_uc+Mw3 zdp?!AMXUSv)W7%l|9AN1U+{Uj@X#Ye(f4f|_G@Xk7cX8M6Rq415Vz5_k=#yg8aj9zeo4>J8s8UeC2P#V{W@8Hhx>v(FDWC zxr<=|=Vozn82|Zg?%*Xa`)vG@m;X}Sc*Bk3ehb?@#*%>jD*(w781;pespT;F~t4^#! ztoGLcII1G9_xce{3f zx4_Fj`?K-xxBo2e-$4C$|E}MIzx;5FZ z;T!%2{_EfJRe0^IUW2FI^>qCFFT5Y0_S{e2K=M2udgLK|`tzTUfA($PiRXUC^Kt$8 zYezkG2hHa`|I@dL+DGtj{`J4XQ=aq`y!lOU#y$7kgU|UdJ{Qk==5z4c*S!X>ef6tx z?RL-a_?^EKPk7=J@VEcg-@zMS{|3DL3qK!Ee&Unyo}c?Uyy{0^g%`i%#d!9!J{31y zdmit7{|E8oKlEz6@I{}6hwpt9PkhRgaL-3Sh9`c~lkt@8^Nly(jQ79){kZ!BAH|z~ z<}G;2lb?+9*KW>XE3NHg+GTp*{rBU&zwka>d;N6-w6EEs{hO}64u9xR{!#q9fAjC~ z%2&J+pZu&(!Rz1nGx+^q`_*{eYknF(@#HeQtlOpcKO&pjw><=%n7{0?8KWIIf2ZGz&sG;N$A5_E1cO1^G5A z*4q`Xhak)Ku&qo9vtg6To_U|8=0U_M3Jh!UgCEXV_EmtI=E6zW@I-q}UM+U$p4`Ve zXa)@P?mpq=C}nDqovIyXiREI9s5ulD=iq=J@)Flfg%rq`$g7BS_D$P<=4aowf$&e` zPyf+BfluB5`B~3;HvZ{%eLH^rmwyEw_xL+;(*~qJ^6!5PKl9d~!4JOjm3ZE#Jr`f~ z-~JB#?Qi}TJo{6ggKO*ty!YMj!ApPX=i=3`{89X#-}n3Q55DzV@c0evpM2+&@aV%A z2d`fC*`I?uAOASK?3cV0KlFn?G}iC`-EaO@e92dQ1zz{s*A2cr=BAtQnjim3{K{YX z1$gMb`|#zz{wwiM{C|1_u#w#|KEu(`N}WB zKl`Wu954Bt&&I$0j(>(P_>y0RH@@!8c<=k(k2`O^6;Ipxx%18^;=0W-JoT<;Y=HiR zEi<_b|9%_ai#CVzE5GPh~z9MbaWp5av z*x)kY`Egz4TyLaGy(~?|_uQ|CeDh{G!wSJ|PtpM=`h=y2{HE)T@w4&}I7+VQmgtvy zT!9i01oa9u8~05VVK8*5FHj#lZaGAU+uwSVrMC3`&T^3^T1K{eAsGbLl7=knSe+}y zIHQmFZz@U~Kzw$+IDfH*pKXZZdDiMZA^0#w(#L&D)?5Y}Gu-v$F~otn&}k^3S0`>~ zaH#ItSLW>d#qZ^lhaY+v=WnBa>NP)(_x=3)@Yvh$7|SzX_tQVUMgA|sYhU{l zxaYpR@$e4T&;DfmyYKm)@sQSwU-Xi#tTSJJfA7ccnM*M@s`lptc9uW$na|v!{g>hm zZ+RVl?(J{K+ur@E1vhu(+V?szQTv_HH@pFlZtXv?Mfs0=-0gVp&;L9=^wIa>H9zx{_{iP&jGa~=yyro@{N*pl zg$EzrGKQP*o}YggK6>v*#zt`uJ$T>PmiwlgZp6R*-~R=^{U3ij-uR|B;^v!g!AC#* z!Ob~75x?%se+_>4M_z#sf9yl}kyrdEp1Ea4U;OL73_tY4KeRbc$2Wh|H{-LmtmV@; z=kvk4--{o9ru6%55mKiq#iXbRMmuZec7T-IPw~C^7?e)4w&a)I$-hm?IZAJ} zSUftbR5Xp!69XroXoJ>FQ_H`t27QUg08A*wPSmTR;x>Z^h?+ z_RH{Jefe+1gAY7}yEou_>>an_frlO#k@wS|ao1?`InQ|pK5+MkH=w;`lmM>ZBK4nr z+grCbmN6-N|3mk0VEzb}4Z!cb{jpm)=NjC2)Abuz-iW6??P++|<}~cQ8SC8YTf)_BX+jW^>hZ+|oX=|A}=_~3^>G|Edq|K4BNoXCB+_x^kF z;KL7Z^9eU^j_6aymJg5LKzJ_$Jn0Ef82jHpa`90-?y+~^mfP;Yv+jD%cmV3YhaVbF z?VgX_yG;b&IO_4YZ&CjFt-p(pJc18>@PoMd)?2r6Z1WY@oX3k_{F&oD%kO^IyK&z` z_l);5Uw^|5Tl?4JNq0VeYwIyvN$Y;R>CJD#gZDpx+qeF%yXHE4-mmz4d}O=t8k=tz zdw84wc-kjDeVbQ#6hHI&H{z*V2Jx)td6 zQkUXb%ZA45UmVS@Tl#(gZ)$of66eLd0+O_q2!OK8nO{pyeOy++ncqV&g!T$4%XlN` zge80iDTAwo=k z#2t^l8LxTGPvXaZuUv-}5j2CEoPbw{G(UkJ~1BuiYkp z&*AQm-i@Dl&8zUg{Kdb7XFThZ@e@DxBY5_+KOL|BiJ!z<-|`lG?7n;O(wBWMzVF|C zFW&m*x8Qvr`2hZhulu?!694Gf^MASa+O2%@9(?fqAHeIk`Gp+}KV>U7ebJYFIiCHT zXX3lR^Pl4W2Oq-q*I$oMdg@c~u8r28*h*86d*b6^TM6h{TRG}oTc+~=`|E!VU-WCg ze6;(I|Nh^{``-Wac+F3|2Jd{&JMoiS4gcJF4&VFz--{=0^91kv;QR1_t-lX#*S_oR zKeqw)Gx5tl@AL7u|MoZIHLw23F){p&-}H@m`qQ2@}B8Bg2hK5oMkw#n>kuDKCE@Pj{ypLo?z;u+7m3;SxN zAN`>p1b3#axv~=c+QkH2xRJs%wKz*%s3qNTXRLGz_uno25FnHS`5|y96W1gGM;+^4 z37J7et$bE=LtE2^aRByO`DEU;e$vq~O7;4f7tNE(&9UCA&83e^IIghSwQ;H&pUY2! zMIdoTOGO9jakrMEPxMgu&zlOTv{fuTMY(`hFqfQV|6e6U6y_7F_Cn7@RD%yX)3WFCqC!^vM#m? z56THb72)W!e6A{abc-DS=fCjh@GE}#7vRtRPk$c&`@j5G`0$55h)1^x#ZQ01XN=GJ zCZYogbc{g%em}K{ERv5G*SU8*_LtC1AkRnWqb)nPjMYB-&t7ZY-`w@^R))D|`#nHx zjU}Ml&&3P-GSLfL)O=lw{PxvB`>LS}o1?La!uImdb=O@xR>$nC&Gr|YuXbU8>nvf| zkjr(~U5^JIxDSs!I@jXw`y}n#OOK(f9>K+Y@^{ZX_OUJJCNq{Oh3xn2#{cO4qLl5w zNBz;Qmvck+_ACMC=J=oX)VuHnTO|K&|M=Ustmc8S0&2ha;>8OCNY9O@hv(bb&X01H z&nNUY!p4fHZQQ#vUH0)}ZYQy;!nuuCk39PD*k1VD`SWw>pvcgnI$nT3v%7W5PFK2F zJmwTSv^(ErOQoK$WkSz?-gEJ%{>MKvGL%33$NtEc<*Xyyv!>J3+3JsWo7Y&^AKF3{ zgaA%ZSj;IkTgav)18DPg)tVkB2$wckxRs_$GIH5A0+Z&KBg~A83rAeWZGt;q?8(Ix ziv(D)0*p`ObKcTv`-CH|-U2w~O%(W#z7<3RuLBm%57q=V1-UarQXf%)DHE$k7=cCy z<(2I~2nrC>`<`A>M5>8YTSE@y^&#^p;R>b;_3GBj9T{LpfQ}C0{3^x($D@Z!T{;v zFwR#=YXCfO|NRq)&xcRg?_1*OURx?+ykTq=QS!8*iPoFouzD zlkPwFbMMCA|9k&nIHh^a=R*?gw_`6$Ap(r~V(_^|#dX)8o6AtoT|3%c#+nzOR=W*k zZ-Z?f?XK%2vi&L3h@`u6qbU=AkL#D=c*g5AWi?RVE5VC*b*n!V%nxeTZT0X=%)Z}s zz1jWacM^7Hp68rCE!B5b-&I}h5)^YTpCm5Q%#d?4-C-MU7hj~fbnG+bCkObUoQykD zOj>j=hQzgW8~oX83DO3r<@=GS*hvbZZP5N><=6|C0RWDAmErPAwk1p6{)=ndR{-!r zxwWnA7L@JyCjnS>N?Np-TR{alyi^dzFM@~;asFNt1&_S>?(n6;Ztq;%L z@&=<6-6QI#!bhywS-Rv@wV_{ty8d(uCE6Pv1*fsfv3peAw;VH-u_rj|1K4K zD0PK%ezovzrSQNbvHvh5YMG_Lg0hLNQE_lm4P$>*Vr3^}boD^2#Ohzixo*3_zbtWk zZJZ*FJPv|&tfSX1%oQz^fp4FRbIRXsYy_uLL({{Y&T6){)d_ta4vEf|sU(i0oE$((mbc~FPDerl-Wu{XO$7X8{L&tF z;>Cf7>svE;a_@iydmr~}ltFI__Sct}F*yevgYD#FQu3i7GsT=W~my#Gtvvm_Fy*zHC~4R5@J zH)PSAk$#K;48E?i-&3nQr9^!baYO70-ydAAJLfu#XXtx557Wxc)-|zUn{hn8Q#95u zl0hHZMr+ak5QdMyVP&_#x>q8TLPLiYK{n;=$b|W;0v;eeAA=DhTA7t(?!UdZivsY! z_z4>Wk zkt`}CYqNg}0YJSOu1X{d#u`6skpg{UB4{!MB1;2zHc79hJn z(}lYei8KV53q>I2rt*$_;lQ^|boCGp^0+v56_P6qKgOwwrdNSFD`~5pN}}otsR{18 zH1*L|09$s4N`0aFbWi%hH0U@LlO9_oJ4}F2c2V9K(e{XTL{SO-uH?g_j#3YKwGf3~ z3yT+GH5u5NwJw{}j3>P;>#Mj6-S(?j zo`HgO&-jK#an_<$I&Q3lFbrge3TipDRcV(+HqH}sTSoHtZh-7Nv|GfvzkQ@&)Bz4d z97crklE5SHi|ijyQQQb_TuftFDT{GkW| zj@)-Tl|Vj79QZJ8_$0n6_IFVD9)y4>_+GwLywRxvKRKCAbws)nO2C&D1CcCkqXMA2 zC7M$tEJ#2QQ3R)(hJbD>Y|1&502d>}Q_Wee*7C|Y!3#pdb>RK{k(LU-d*Ftn41N9N zD!|rBIx}P6iOu)CE5avIARK5Ui1$jTH~^_? z*I%v+tSl;FaD|jIq5z$eB>2^%HTYa#NrQ45uWjE5<~61XEz?@ahqq+o;8-y8 zakB#a!(?%d^&x493rP=-8KDq7g(Lof6e$08uerl3bexdUfn;|PL`9*3;AEpl95h=) znHMOoK{Q7o*~hDVWg0@8LdF5^xd0RLMyfr9Cbuvo4igU5m(& z#bnS27vdwLPeT29Omg6}%&8RU^lf`oB-u4fVpCK$7^UBh`0DtF{K9#}f~Au+*)_6$ z*ZO}Qr#_wF`&n?VvcHrVQ~A6E=R_11x>5cwSdlgbq6I&hj#J*BRk^H&DERs!eTlgh zeQv(WVzA^`_&*{RLS?^8Ay9dc^G1}P@kz6_cE>Tsxyff4qy|W*=1Nu*vd~5c(?EuunQRNrE@>Du-vAZPLip!2<& zcg%)&xbd*<&#p~&C8N3t5*!?n_O`@v)k=Ea{ToH>LI}-Oi1l{OwR6n(r-pi&4IrdQq zr*EnloH&XBIK@ZdhD6AeoF?&I@$UG2HsbikC&9VyK^rmFOw>j3E?gCz0O-SV`Cnc4 zTE!Hh&qnNI(sks|e1lY)&Pe$R5S{T_yiF<5ooPf(_XK>0hbW)E;}kCy##;xGj2Im! zbeO7Ir;Gy@Q3-xh%c*p&`Z|Ks(kq**#a(^A46TmsQmN8p55?cK9A^)K+SW^{IKYy| z2_TmOLs6Gt^~##*EE-ZgA!W$*k;JKhyRc&w>bk z4yL3(ak|5JH=A4fjSvT2SMYN?1VfdQp4j5Z`CzIRhDv7tLZXz-lW1 ztzsM8-99wmlxa6V>97>^G?wxltP)BE_8~ce`XGm!d?BZ~&K7fNA1n8N6+%c61VqPv z5`gX7$~#FtFBzhlke;>T5&wPS(N>o4phJRd8eouaqinew;JEua_K)u`@yb-4_(u{! zJQ44(Wu?h(9R&emS_!vTs! ziwO^*eBFIoz>?AdJXP>r=~T(F=IMIN@6%VLMC`i~R|j{{y%tqbX=N7!69km{m4a_u z=vswv7O0Z7=nJM>o1ft?SGTY$U5T?gwj$c$_dcECUIyu{KpJcnkV@{T;&VaNv~$RF zqjkzd6{RYh|IezYmp6Xc9Sy0JtRnt@9{)dG+aU_bO$-%vlXBjF-aI0g@d7BM@JC!$ zNQYm>L9_tj0B|ei*r*vDSGQV-riQSP5tgVCY?CaY7S(l?kbQ1nAM34LnE_P8eJVwX z0DUd|Am!<&5QGf!MjLgM8vC$t*_u$r8vFkR+D651wsu7r(hkTh;JEN5UPWZwpJ|Fj zcB1GpH#vENWD>WEK(JC{FKIz52$kdeH#HFceJZ>yPTZ*IQZ3XvIDZ3XdXN|VYk&fU zKv$*546?=HA1&Tt{>q34BKAc&&`Fe2Ph&)m8(I1}LQ;?`B>~>IJXTZd^tYkjsrE$W z5yVj=-J@=!XUt{qjJA};CSX&=a?(l>h{qSFW!A5IymI5u!d!JDk~ROs{Pu6oa(LS{ zdlJUNQ*h1%4Ry-^|I183pSFoNhx&$FRh%=T$?ggG6Rwc5_O%>YAT{vnf2)-G-@p1l zT-zlK)OMhU02P5zN&YxmsVaq>F>n%&M-(eoSH2Rms8jC^my&<0z`e^Gf`|uRK6bmx z8Zf`DBTO9~+u1&oZ|3;?jtEIhFZY$ z$?jN~HK3#j`EWcJ?x(pL+A3u4Q1^hPv->oHNaBi0vUdj^+oi+rN&vU_k9i-Dr?&CB7qdWC-S5hk4Uczvddph=y$BIJi1Sb-?#yU62mI4Y^oYHDn5vJSbhUSt3n#U0 zeO#^FCN@j<$7fq)uOh(SCsv4A(6mi82nDGjL(NP6D zF6^cjJG+#qTn?4lO(3#j`AR-yWyrumC@d=A44yf8d3;Zz?P2VkF|(%!>sGJlGjNqV zx-+jhhwIhZ-UT01-)k~_<`?j>iRQrdg2F=NW^97B&W;Yw|8FvD`h;?oRb_m9C1>D8PemCqKYdzxEX(c0dIyo`z~!2rim z1|r>{;yqrNyz$`XKs-TAMR!Sn1Gmy~%EB8Isk!OiuRcYjQE3>)#xUO>CS3;6&eCy?g?ArdAdJC1V$29DRSk)8`Z*^F>3-S6+qb6-3zYeA{C-&sxX*wI{9U^vxvOxsUTm1 z`pU9$f~FN8rYsT#X*BCZ)D51RaOog)+TyTTTBW30aVsh+!tbxT0^c~`8#dLkvj#KH zq3pqg|Az1FBpg9$fg!`^*=}s8$4Muihz0ZK;kp|i$4)!$$bf0i?=_0Wq?2aFNxyH} zxS3DLT5AB5%0BnppW}?v&t$+p#rA~qpbt$)^EVT{{8UM%P+Hny8QCX zaTwkA0NwM)tFFSX^v+8zy%aSSrL5F)^Kz{6^_f7!JW?)qp7*af-vq7Pl5SEow0im1 z?f=5J=|taWxv}etPqtyGtT$ z_*T1>iH0{Of)oZ`$2}DMK+#wX(<0oaQ$9wkP{jH+L_66sGFff+;LSC%dciTp5ep(T zxOprvDje_X+s;3GMZ>)Uh?J6;rS9DhCo?)|BD>;_2G+J-+wDM@0L?0DHfhvKyt2wb z3&hO?U=7^5c!3UO*?W}J(r_6O{St~7bkfJBkp-Sg?5I!@tWE?QamMSqW5+8_7H+4oEN{Mw35W72_0Q6e6*qgQB#NR z&JL<~FWpO%EQjjQo6G5v3v%?V0lgGF>L~X)Y^b-g%JNTs^ky6Fwus zoHfv6&RU@F^GRXURo9|8zc4i?EkH(DLV%m(*-y&^|tzI;~?A&(GC5ccX4nxpq&G27c+ercV>iN&a zK)RF5`NwH$cK3M*uZ z)9SunTKVn!Zoo1QO!?&mc!{8;;OgM`;2b}95QQmjGP5Rw>hrx15>s{=>vuQ0rHmHN zPc3;_U#fG=1!bb66{dwEQpzh?1WU>O^v?szN3^(3W7^w^Z)Bf*@gK_^o4?4>#VI|PhL4Od3Z_*{d|IPs`#9H-|0J&%XEy*Zlcfj2bl( zM;>-0mMmWimD`KSlc%Dlx*FxBB^WbuJZ8_CgFElK3!i=V8P56f&yh>*tthHMUQPi@ zi^^#H@^~y(tXhFCCn5>WSI z%jSCe*@Y5Pc=<&0v=K#Qbe`!cLpkM9`I~9wbkRi@Vbl709#11~ratr!gT6Tj!6?7p z-fs3knDYmW(ld12U3cGuZyj&|Q+bkX(7|B!&R=PRx^Z;>`S4hc2UD zH`f<_b?fOi`-@Kt;r0VLLoMukaQTCzzf*@vm_ZD7v8io*dTu``Ax(bpi76;8&1#GvHy*#Z z;1|dz;{JE%UxJp_Ha@$zrKOoDq%)YOukBZZY14MVU;p?IJn`HU_}O_s#=7OJxvj>B zFo7tc>H*|ap!U7?#%rjltl2LbYh9Z({$1Ig!jdlk=$l?da<4L0w%9`vll{NvuLO$~W=J5EJnCYc{MWg|U%V zY^`?H%nXE=u1uU00#G&j?lAW&W)-p_I0Ej2z`N8CkxS8RiaS5XIkL7I*CzeTr%vp7T@bc&2ecF(yz z1;Vp0C?xMpw-$K3Cw7YcDHnpF)+xms8Xi_LsUlL7R;OlBi25{jOBvbM`Aig4it z7ZIm@6i4tFXJZdhAc^DdZF+Aj_;0^E1AqMEANa^cj+!yxL_k_tP=wdse2u{-H^-jZ z)k~DgzI%TYmtA%>S^CX5=jUf)(W1q;_rAL^e$06Epqs4tv3T<7C(+s6juk7GVo3i%=xpxBhAkWM+uvP47Ig>mN$ITLJQF`U^GqD{?L*Mj z+ev_)hhkb8bai&(g=bzwNpT6fsU73Tj^^`>OH0de(=9hqJx?3q|cO6$M_4`KzfaKYDn;C(4Ml+7tz#)n}BB$tpgkNR-fupzkm>Z>qt z;6U~<7|Y*KK8ZUO%~%k2vwSZ8eb3#v{pMSVp6aoyV?M?bfvZ-+s8B+V0%*f=&OxZN z9CR{TBPH1msQv1|6&DRAXLTUP{U{x&l+@gEe}z|hODY&LDeDRlsw~|q`HlZRDhH20 z!8z_8^lNi_vx@CgvQ*I*(4y{ujNl3PMy-YvFbIP%;Gy0+H7lyPP1p9@2jG8qy&5{z zYi(`0f&)V;u#gs43Jt!>+Jp)!lrdf zV3enml$&-sKE+KJ6VUe-$7`FhA-cx%tkwyv7%T@48bCnZK;Yj)c$3f1nnkSvCPlPG zS!ogR{xwXIm=ytAR3@C%hjP-t-Q7L*xmYaG&VP3)QI66xwEN$ z6ABAU(9+h%lnXMQWU&`uK+PZm*8*M{beMYN=b*Ez8&$al$fxf+NkOzGZA4{ta@RMt zHDOZac(k@Rb6XNVvpAQX<^u;lnxLLv&g_(u-{P|mKOsTNkh~h9C5x11>-b6Ias8ifU_P2R z>6vqaVZ%GS$kI0lnxB8}xkP8wQuzt0Dr@n^n{U&q;8lD!`*?Dg9*o~#@kf-F72}B~ z9>F!&T!j(CMiap1<2S!PAMGuj*syUOhK(4)2fV-Y?rUgi*n;BRVtnw)3s1|dsa1QlTX>l3THfB|ktJ`?xQDOj{ zKHGEp9$2-01MYhC-`I2K-SNji{FxlSIr#Z6&&A9+pW`Z8Ih2yax_@1N%>4W-+kKa+Bz60Y{b) z_xU@+21te}(Tt-9Js??e@xcjJ_H|0Y-GAXI#As-F7Q@#i%w_Xi5=ki)B-P)jQ*{kk zs65l4!p3^^MOPgx&cIpA4u=D{peYdt%odF-+=dk!_eLSV z=J>#jM#?g=R*f4xVUpYMv?wZGD7nnwlnspzz66_72*Of{Tbz9BGVVP-P9ft3H}s9y zVCkCj!oc}dZ0KwhFaLZ{a@pETl;eg;INe0Fk?6&2KZ;x-n16Bzcnn=*=%DWIUUU;5 zy=mPhtX{R2sgDphPW7yC>j6qpK6km#oIfVIxTq zRAbb*VVFkF+Y!Tt5g(sR;F*u-UwD~lqzuzDr=E5?atGz{?;T{77t>0?tRyx!*CUU> z*2q@B>N-*&SK#^Ap2vW}CZZ@G`Gr-?gD+pYnxlhEdwQrHx#VaaI(P_k>%)c)M?V7K z?siQS(@MU_P%*tu7>5oWg0oNm8Cmn?9PMK$Ak#KO4;7hrvtQh%P3SSouB(Gq4qa&L zYD1Uqz;}*10$UpEsUI_#w&MsiY;Hz<(>e?(8G;kPdpuk7=Af0uOBQitRYLTRPm|K? zIDYNo5?h-((6Ff<+VRba+vwNY)M~=8cvZotc_mH(KqHma*f}yGX3i9_Dnkf^a*RW?pnz?W7&XsqM(y;4Jb*)rUX11 zL@^PgN+}S8SJ?4a!O#=4M5o(kZ;j}u#9WU})8Y>$+HIAb1^Qk{6D}IG5Ju^?UEAqI zFePOO2(TV^QU#|Y(%Y^j@bhyS?-h^V`+ibElg4MFb?>!8v@(JcNvR5_g~g|SLzVqL zYYIr9jwNKtleYjM>B)F+C_>UsNLerZZQ%Pfkn$hI2m>$Van;lJN(+a#9u#DxvHvcp zfFQs+)JH~Tn40I9u+O|){?x?z8_Uu}o7iyNT=+c!CIr+wFKz6z4ob|d1dIdN#I(1Q zb2Klv5Oe3u#h?E6Cv4iV2|qgZ92|Ps!MOF`w_zzMmVy22IA-2h#XXrWRF+kc<-eIL zWdQiIFF(eG>Y^ zE1260@(X!GSFXw>5bkDBH}BcFX)_jmy_CRTqocbG)z#J5)U**vcY;yFM^R7pVoOsK z+B@4xsjenJ2U+mFPU%zqdPo_tv)Ap+Hf=R46F+_yh@v9);h~}bkuA1qIq=dVg-WOY z%z{zf=fgB8sqhXZj2n&U%8igF3Suh(0dTedO#IR(ld6~xGORSw&cUnG?~to_KSLz4 z1gl?pgEWE4o@{>FZ=DAV?Y8CkPV$hh1lFk7WatoCmCa~)r4iewu&vj2Iuu?*^ac>L zqzAD0Fai2BDU3LEj}z$as^$Q)7(852rNz}3;& z!MlK`OrDPK9(xjI&;A0t?XfG$Xch3~m!Dz5{72A5*7Jbce)#7NH=%yZX54b)jX3!S zC(;U|8kGdTpMUiUh77uwH;OTr@Q|V$s4y-H|xq_~Y>L#~cva;u7_wrBPwapV#39Cw_WHuN8o@Ui{%1Lm{8tv3P3Pyy#*Uz^w zVJ(7Xf)Khn!K-yA+d7-G7v+aKWW5w=7)5Yyd%iyd8)BDk3DPBnKfL7Y&?k=S(OS?l z66t*iY;2}rydfk1?dCE&-}yV=-O8Q0?C>I^Ga;dbU;D=lxGJmC|J}7+b^=GT$A@j% zDyC94@IQ87EMUfl+6GDx1G2e3{>BmG%pkCHB--lI@{U|*6g&ya{L_zLP#oe9KYGxq z$fQ_36+hEr8ptt`0_$iZDR4n4nqQ2wCzCkk7#k%0LoCW}t?8$1B6}w_LEOa3UL&g5 z_VmhM*#uRxGo1owIThoR!&ax_G%JSlw|kZ}Z1NxPi!__548=lV534W?{bT5(41pp) zPrtuE;}de&ZX}wdoi}rpIf_LA04-Y=_AX;g)v1-L?cJNlv zDx!mcwv+r2#`V2-=rYqz~oR$hhy0|(&CnP1}J#~!45SSpC^3$qc6 ztb0>8v%7idkO9QG{}SuguE(iAIte%2ay<^%Z$D~d56a6+anODT;ec-(Otj7deExYS zJDd%Nf6u?~#@_oKi2j3yVA-nwV9Bx-`UmP)t7O8 z=je<(_0h^>uQO1Lug28ZtkP^wBb8GAAs_%MEeleHRkRpX`it;W^4=kGM+l{ovgg&5 z7I@*Gb5@y7S+)rSiFS&^u7<>Z6TUx5fuRnvt-QVo;Lqf=cA%u&6pu+PSWfWvd;FvnYDJ{OtC$i!|%h7Uni+u>mN?GgnPcs4Is z+zutY>Ru2ewG)@6AveFfqV)=7#caao^zoV$h=b+-h|5OZ=gZ)5buaJ$n;>$3JW;|P z@L)&SvwU@y$osl`8Ye_!*vtNDzso2cqL-p9X~-4i)|TZ#Q3%Tfh`IV>jGTXW^h@y1 zjEmv%&Ro$OVR(Yh({gTyZTB{`(8g}_PE@kua|l(81ING+Kis-{HO`|<x?yYl2Ah2Pp<^}I8ja~B+?R*;L$m4 z9r?O>Fb{YVDI%p+Krkc>WyCSh-}tEaqaiYj)6RJ&QvbPr=9hk;?C*~)RI^p>e#K>y%7$MhZf6HF>v7# zx6o6OSNCA%+rzQ&8c2EgvjikMg%K*O7xa!C7aWiTs=b+;Pvpam%g$q~{74Xv_-Zfc?LL!;Ux_)22+thabO> zci(>x%a^TmV`vk<8TzEYxq~-f9sJEhP~W%#wN%&l-u(cdeEu=Z-)GV zjT6`;!|Um`vHlt7h^?utz;P!WkIr@*D$3_QaJZ<#xUplASD0s43N#>Qx3LFK6(V^M zsXQR%cts#+ocyM3upql39z*$gZPQ`|{i3A>ow4QGjD*lWZ@(4>iEsid5nk=FBFJM# zhs>{D^rc@Bh!IL*mWTFQ@bM0D?W%8gu`8hvHKMP=Di$p;CA=wx3tHHm&N=RQ4arq1xy@P;3vds7QTavQJhq5zZw%b?>#<@_RsVcngqB7r*|#GXV+bWslc zz25@Oz#nsKz?B!u3iXJ?p+d*`L5DL-;!ybCyx@AF>V;^7_U-c8&E4m}^o|~#JxNqJ zOvZj4pq2m2=vZ;>OY{mQyi|$$+Izz8MCFu>S2#Ll&)AHziJ;0@0eDL?yb#%YDdekt zb2$s83H8)AHoP#um??^b5B#=uNNTq$RohKqd1TS|nlMUsf+lWaBn(8oCVc?`+c%j*_F^aj-Ly$#m+j28^44L#VH#AVI>i|Swp@YaEr;HSX`Yv82ZgmrjSf9HR9R< z@sVg%!;OZr0-TER4VQ}Y5cu?jft%;BjK@ag{k3%1dY%%lsiU?5sac{DY&puM>rCMu~Pf%%VrbS4J% zA57HDT8`|w|DlIDM!%H+KA(WHfw=fP?zjW1RqpOS;y z?!Yz?KQ@!(hV>ip=);eY1H6oV0<3r)U}ltcyyuY?G<8|LcnKQ-Mj0FYHx7I|U+k({ zt^ie*q229vQE^E{fRu3+@kE7wj6IYh#rZSi z0`_|qwD$qvcJ|s7NgIDPbLmm&+?? z^5{EPVt^vnC0l)1tP-=v%R8W)y75X#xwoXM6dNg58t`>cF)GO3t^uXekx$}&|91QO z808aRm@Y5$yPF6?j{&Z}A-qV(wo8k}A2;ZiQ3gB=)sSD1i~OQOeE8`{`1JEnEkK!A zb6!~_q3(tfGG&;Y{)9_R=@={DY|QHFa`cAv=F`0eWL>x7=~te^n$>HuZuKS{f6R#( zF=hytyP0@+;}0+j#N@bHzkWSi=@*=PAx4cH12%SPy8Q0ai*Wb-|0aMXz-{kAb!|Cz zp0+bajTy;t{GDxGc;UsDiPqVG*>h*%_J7^N>6&NHp3T0CeB&T4FXwQ}I--Skm^2Y* z{pc)CIQRC9H!*U=7;GetzqGUzb^U9|TDRezdGx-4HU0Q3f{Mygoc+_Yk$+bK9)A2m zvg%7PdHiJTIDI;5``4ketejI)8$XG$(oIB?*-vr#AFd$(!xl`LFqzv_S5s&9)B(W#w1d;S;=$4 zSO=(fA6uDsBFeIr@hfVPH!AwZ;w65BpBBNR>Xd2-<+?c+>*HSK zZ&c&;*yw2f}JM1t8mt6W=lo8Fb zZo?)#^ZZl1>-Vt34#M7h?~j(oHoW}iD_Fm2Eq2~{XOjdPd8&{o6@}Gn)-w>Bte*M# zwJ0qqb20Nd#P@IJl|`3{6rxo}G0`o91`Wm1Ma$6G+)OKi61@5Ld${Mm2hiTpiVAW9 zn+%d`(RS zjy&dQnB?0<4otN;|RGR)*NE33Bl7EGHql@x~N{Rv?UA)E%;u$)~b z56}TgRE$alVwBL<+7iLTG{$d)^DA^91!yUyro#Nq#&2PM2`6tfG$Cj(`@sJbQ9xOj zjtva`V3x^JW~l-+`DqbrRlrcLvt3S%<=xfTvfuBNLV4=YlrYIyz(EH&(dL)G2ZH4?v2jS3;6=v%70G(nN|M-1u1- zdZmKr-iBf-hzC;&&118*ae_w65-{H>0^&7GqHikj#`~{g^^!GYg%_cwq6P!{ z55dP@eT1i8d>onX47RLq!Y#Mn#BuITt*yrKfbXAp5{^IqL~P#FfLU{9;?c(*!?fws z$d|AqZ#*l`&&SUD?T>SR@k^$a429C#-h@ta+-j=h$iYMLqvOxSHxBwHtr+Tw)6Yd? zQ$1F#TS@nH;J9OtM}CgNPPFnc^v@bnGMhJT#-WECi31P#Cdz2VVJ!Tw=6r=OKmP(d z?XnAMs4ZQcLokBMuA=W|k?(|eR43i2t*pVsvEwg^+Jc-1N*E0~?70k=j%W-| zfg;QsA-lpV^0KWFpAFP0yi9sy#LWp$f$FqE(kXF_tB0s4k~mdd%PjtC#OWr1-OWFh z1s1w5zQ%fZ1(-yva&s`ub^mT(55KzdOZSw4CMYYR_#A46_e3=j>%mP=eXwoU_Q?jS z(|>!5HY^WAr?}h0KxHq6xdOWh^q+s~X*~UeJ#kC9dYQ)!?O+_D#_`B03V7G4Q`!kQRk8gHynn#ZB!(U` z{;mA{LIPw%yS#}<9(jyU(W;iF_tn^l_ zTDzJ#_%*B7v-0Ysc78nT13v$Fr?zPX+%*jP24afL`3FVi_~u#PrqAg8q&T+JH{!l~ z@5lR}yo*0xb`AC*daBK28qLv|HR~(fdgsmf%b08Nrz`)A4?g?=_djwk^YVM`u_pm} zhAsWFiV|FN&7V1~L}zCUE3$G@BE`j}%=s^0wF(bD_^8dLl$&t}(Q9`Ov1W_Z)Sb^& z&dK92P!oFU`(5kA9T;QJTF&9ju2C*@YL3}IHzg?aYnZ9(8U7h{}fK(J&7Xy1VHdEyZeDc7r^j6*}3RAIl;W2JQ z*o2lgSd~yRI<(jJi4WI}VYl(E+j?!6Fi_T8j^nfX%QZoHAXjXyXtxoBuax%H;?F2h zX`)QWvi|p~gkS7H>nNugD}gaiSEYnmCYei85N|nd$-5^`d_#5pt~4*?P$KHU3#619 zA^eDzgp3OCANr#zGfI9VEDh+bfWN;ht`V#u;(kwTKWZSZI(_A3xn7D*mKye(^uQSS z#@gpP8E2zboQTcbS5!{H?gzRY>?{AbRnmZD2oD{o}jC> zo4D$B?6&_d1kS@zTic(xhT^gkm~}v>@Z$KR7nnR5Nl{_LZ&_kC52_k>6edBzxW(yopm~z8ygrP z4Pcs0z9v#>hbhx=(1G9NO<#ti&nMcdxPTN>aV>7T^+qh+YcUQv|fc!2cwOpHJXMo_?c zANk}Jkcv@6?u$STg(Kl~mekn$5~p3HC2u)Q_jVP*U8yKQ+SfO60_H459VVMjPZA}^ zD_sY36%-WWG~%CkB*G&8!-Ee$$Yb$~U!I3OcH4s-l({G(>)f2< zYgP}&Ieg5RTe8-)4U@j?*0&|9e>b2+ky8bL7$8ULM6@hIN zD=<#-Mh@@gt5@LpmtVx%wX1OGVTYrqvzx)VomzfSK%*zx|YDbfL0${*y1og5EBuUsBTt3C3OGn1@oyLTR3{k?0CNYraREB zyIGku)5@l`y#X)1{0eTn?KT4bT>Sp`SJSvm!VNe515Z8mEGf`|`1Nlt<}gtcDP#;U zb9ARUsQ#KiUV~R(e+9~?!xa{Ms97pO=r2U?8-lxNoMG%FTZl^35J2DYK z0~MKsl6(-PD=k*c28c*+LwG?Si4KHxn4nR~q{H~4VHR<~8-frDE)C*gw=lH8vtY-E zc3Z`jBM69Ka)p}uIqVJD6p8KU@pjMx+kpa#;fvCu6mC$DH~tIvL&y8Q!$e0^@<$A^ zRt{&)x&}=uWja}}hVxGt9F4=VesetosR6WIcG?L82KMKKXuVavWUViQ$yHfgRE$LnzD60* zJ%fh~z~srJ@$GLNgl2M<_I8-WZG|W-&JcZ5hVu3{eER83C{iBNr|g0$QzzTQ=V`Uk z(Qft=Xx?dDU0F^mja&>IFbF$MnM#0HjMCB)?6uRLIOdpR@Yo}d;>}Oq#4(2-!<(fB z4IO}C!-r$|@X>hY>8CJT*%P!(^p5Ab&EF=(bj8Z$(DeSgegmknL~oG;c=oKBC@CmI z6Rl)S+6Z&hrEx0%6Pmj;g-8^!Vw1_ft^V) z;Le1@RjVp1Fm%{3J~6C=9Md98?)Oht7iG6fmuSoUL!5T5Pf zDHF|$ASMCQoH1p6QA$qsF!MCx{7lU|9~>bJHYg!)TE+P!k~vCuFOZHea<7hZ?F{Vz zpB^BC=I}uW@d(HytIX74G+JOo$RRH$o=|$iXDMHq5w-L^Y@>~|19#iWHr+#FeRo-{ z47CW#Df(8o_1X?$pi$2TFa*MwX@|w>Hn~Rl@7x1X+eLx&Zu)E%p1DB>q;W$O!eowb ziaI<>KKLN`2-yhSy>^A@tm^hr4*ne?OgA(QI{E}sog?}O5^z1~BsQUmz-R9u*ABat z`%DW(q@D0-h<7Bu=lnzvgP^q7ZV`JT9dBVTvjg{`2Oq|RnFny>kw;)Z;<}-88CbV8 zHuB-;qsELtzkdDD-rk8XW`52I*32%~j@B-YpxH!L?f7A1aq(|1V*2OfuRcWc##XG| zyq2D6#Rb2-2t!8=L33LRap0xIv3FqI`Ze_5Ca#m&U0Yn5%Q<$ z1Iyro?Rf+YpuS6BJ$2fS*u1e3pML%&#*P||3F9V_qT7ttjurx0qNQ?*(UsGU4I9^! zLK#N?jo_#tlj@?gy^Hhtds^F2MJ2?|>rRasRQ8V3rl5aa9ZXsZvx3UaD?}UlH2(9{ ze>e^F1;4tGBYxJAB3ij-1une!d{mWJp`xZ7`|tG)>^^NTd&Hxmc1Q{T=ZQzLWZBpF z;>$0{!CZ~eqeh{$tQ5tC<>YT@qE>aXZ>CIFkpk_s#sULnTK?nwO`{tFRmKZlKvH{u zqNTwd=320Xl~oGTYSlq7g}v_X4|lkHM-c~Nh0?w<9fC39M2)9Ksg)$)0Nt{DEKpCjUj5BZVgaPeB(+{JXeqjbUn=F*atK+cJx zlE4zBcueI(7PaVTggy>xO*{Vv|C#U(#ze(&=Ks9);wFr*W8x!jloT)^wd_uU=m@J` z!KHVkCOWO5ROkl}_+AXOS4u)GxPF|SEed4tKkGPjzH>4`fr`O%6RDek=Go_-;Ruw& zzI8A=5;;t=q!R5N?P%|A!=gpYaQyebkA3&v7r#0Gd;+&(d`&BX7hiapfTFMN7ZNkAM6lTzchi36NVcs9%3f`!`Wb`L)p93hOs)K-1Di*mJktFp+?}p?M3I z&Hfw{$4|ti4ynWGr=5lg6Q<(XCr{^`|35hWB$O4EVfmV6>}y}9zKy?K z`&TSlwuF2fXVKU+vmeA9X*ryHD__o;$@?`@0T`(({kaZ8;^s~;CUjPuy0zrO>Jsi= zE2KOl(kl(;57hCk-X9?E@$l)ln*k9@+#Y|(BQAM5AqhuUT4N~k^xhatlmKd-)q3q` zV=Po5#!YBM*K?$P<$SSypy8?7NEpYqDZ9!^M2{))oh-`FcyH(;sHn8E{x7fXa439a zr}%G$ASCXSKF=hsb%L#qzXdD9mmz@O+oytn^aJKCVi_VDoH8CDRpjGS;2DMqVp-O4 z3=Dj<0c4M6DcdAy#k4Tl32f_?;qXJE6LJg5AzIG4@p}nW^7HaBb%z~r{;z(6Wh++V znP;EI>QzfwQJM^&MJ7i-@!QjPoQ6v-xfH*@`uAA1ZWUR)30epQt4eBY3JBGMmi8v} zSN&MAtgqid)W{$dWlD(q*7R-=4Q0;A?Ip)?E{C9+jb4?7)u<|}#eEOni%sN=H5;|^ zGXwoR4)wulf%dv9B8Z2A45GNjY3ce;9O0jZLHZx|oMq^7eh7PR2jW^uL?a3#a zq_DUc)28l-*|X>4dq*CR!wx!(6vaA>A3q9f*R97AQZV_nj4^fXNjg!#ZUc7OVMlVX z_M)b)3JVs^!O16+GTeWE@>@KNS6+XecjgWrI0*ahzCT`l=T+SHuba@()XwseOB9-k zrfO~=SUwp;8l8f_-d7pzxnXzK^DjKg2qK3xs;qV(aY-WSRoF&{#eb~{} zi1yYFKBa8=>ctp0npQc>R?uo<9hNU&X;VTQKq=i0qGQ;z1P!ZD)bSrC$vNmWc6+Qsgd2E+jecI z^8hr$t}a(Y*4$dexCXTeK=Nq-qFzH7EdO_mVfO!TH@4f#X<}aFy-~93d?6da)c;VS+@dVKk8`o{d zLk~SfJa;?i($68PsI$A3IOuAe@xwDvPV~m&rAx7T(@Jsxckv3xOpr|*H*v(xfd2h( z`l&x8z%RvH?tqpIIay8K{f%2Tk@Dzpo5{#8kefkc>lQrmZ(<*SS@G;RZ7Mpt zd+_n6A9CHh$YHyQD3zguhGOqM_Q2pl!|?g+FHuldh_T})qPDbxxbl2XRbd>YX0JnO zSvjY)rYa#Pw+Kzd`!6HQ-kcv$N)%O(arRO#nW(0hUwf5SL0!CwYvk}TsOw)#eXub%<}87nOg{BN2U=R&@Xs4> z!eNITjMCx;RM(kfE=y6rrGfVR{j({_iqc8B7?S=G9a;h=?q;;rzY4YDni@Ki>}vn{;f2IERg zBW)|d=#4uqU8so8>h#@VZYc3Cow_4kJ1wF-*uUUtr5I45B)p`5(OGDrb=t?k|0GD+ z^&a2E#FV|`S1DO9{I9R=bR?wJZ2s>QY}* zLSQgrLw%GdV?qV(6T390|td|W$=W^n;yO8zNMyw!- zD$coJZDlDT$@{4|q2d%)=wlx^h2aZRVHlL_TDe}c&lnKay}py;*w~%Kc3gJdWjBl( zH4M+a@+{tX?M*Z^)FZz*hb>_f2D)hZQm)(bWvg)Qb$?^Wy-7$`n4d>rX{obv0@}4} z*P@CXp+7$Dr&zplF@08oSuyi>BX1f;)^fg4O(L-k7M3Cj~u|2<&_+c zX`*Aa>l^JN!@3R4Wl~JjcSYm?{qAw!MM-HnuL_KlwVPHM!-fsU$kF4_+4d9C5nW_~ z-@=hff4J-_JapfExa9Inu?sm@k3RAkEM71NUw!o@$_py-#+z?q-n<1^v1&PLiDNe> zj~NFsZ+cTUPW?5q@ZWmt4faFqyYK$EWtWAOF~<)i!CXbvFu&3Wy>r;5h!9Z@C4}KmRP=dHZ!zZtb}0s%yw8 zKNTO5l3TiL5%$@CKMor;X{U<|OE6?WKN_E0QX&;Rh6!gN%}F_DqijG#fNF1bq|j6A zs0@%Ya{m7=;pp`0CX}g!Y2w4GRRA|X#&#uQE&q_lT+6CutUoSP#m{*a(9*Xrab^0} zTiY-G4e?IDa`uspo-GP=uZkhB&ePzaT(nc^;N7!MYKt*(j%(5JN*JxO^Fu@9cqs^j zaNDl!mJL+?5rE!nNTDtL_cCVI(&SD90s__%{kcM;+a*}k;vUo^r|rMY1fdBpa|5Vd zCx&Eei4IFP|FVd`D%Hy1pOqwL5N90)GBi0__u<}@HH4IN*oTsOzk%!{IspL9c^9hs2fm+0Ye6$vZ4w*P2Q0lzKy7@ z?T0`A@z1#9p1b&L0Fzds$#B;+b~{g>PX2=`dVd*iA-VExwIK_#FE9XdhFTBIw#?A?xlx%(Sym8CgAWxkKz@NPu*-HpL$3$Cq0T4 zuyOR}>Rhhpm~rDUX7m`08a9GMTQC2^fZ z@s7@s{I>^AHfA#M)powP{9lsyhr=SJf@3!*8V3C6ZwP&X2=wBu2oWD~f z2P6hOdCQjvCK24Bzgx?g9|d_zS%BdbFX^33S;8W`__lxqsjw@p#a4DFEUqIm@WSc` zIcpP(r^V!_K^TP?3LyldtskSESasU+lo!?{4g8{wvHr2%!7;7VQs+UydtJ;qzJ^DB z@5A>w+1Y^pbwoSlp|Px#d!FUhj8psCt#o5_QSf3tMJJu?~@;69dFhuB1dljK?4}P8#lLN z&DwSB+#ND}7^lBh&NpHXg*V^j6V0-ge%^ckUDQt<7(R3)NAqMdR&r;2^Z_op;&;UL zw_v}$zrj8dvq{TH(|{rUaX;0+jeH@ADchxW$jc+Di^^TOVk!9p3QV>OR8{szU1cr0 zXeCfcinfeaN6VKj#oE>8aGM%v(s!-o+b|T+zI%QH-Q(ll@+`~Bt#o!%~izW%(RumuYs zD!WbB_VEFTK9p3C)Rxj|0wtG04w{qLaQ?)%Pp8%c@qXALNCM1lG#F=#L%yrJ@?+nDG<&&>&Lk0!i(6Vyy>=^@W=y?Vjux`O?fq2ia$B? zr?|N*gL!kmL}ODUpCVRKT8VPvytgzo;HEon!h*T;P>@rEhWgFuOxkh4e&50^H{8N! z3ltO<(8{0l_S$_fqCECR5jky#4jYIOBL=b~ zbzr}NC?on~_|PGov`%GmF`MY6d+xp$`Q${-%gx~#Z74i*MzA8J+*rR3r=0Qwl;juV z@@p=|#g|>oev5Yce(ba{`0a%kb4JhSo_!8~x%RIZFt`SPzxFy>`LvM3IFIU3h)+KM z43}Jb8KGX3n|nLoRLqaWMfJ(q7mY?+ZZ^$p-{8SrhqUi5>d5&You>^-nvpuy z5%5lFQ@2D!ItJyZkd7jRG*x{iy~46}2){G!j5KQ@a`mcMA!wDxB>#qOCxEtt0?7Ea zmm6$p?JEKBI^bZsi+mpiFcM=a?~OLF4w_j)B>$#uP6oeQHuLhFq_>lI_6{CAnDhD6Ro0=Zs*0m;KA-t1 zS>~-+xO5@$-)o`syP3yEZyr$)9sFMjIZwB2+=98Y=MxoEKtN}>=^lLg*(W&V#2=!h zve;T`O~C#4J%Emuc2XSVpe-&yRc$3UZ>}dUeigs-&~G0`&v(J3PB!`Y&4DW)fB7*s zH`il)!D!?vb0~cw#*dzi#+DW&Te>+njtONo(K{xkvN@>bvrj(9Pk(+EA4uOs^i4Ba z$VNf-Qu~baww9c&!-fxK1!4|iH)jD18$1LJwCc&Da!g9;imD3UbYRGP07Frx=KGX$8=`cJrJaa>XGTr;SFnMRbtK- zZr`*Sciwp$h7K8t`yaax#nh%NFZ%=5tXqyLJ50ieVPkOr-S^?Xhwdl3tpwjW=DQd) zbRa8NW6|fCaqhz&^aU6z3K{2c&p|l2$vHCxp#eW*A$9^QF_&$^Mf163#WP~g!_OBT zSH(sSIh{G;YT8H0+2?hLr&EG&;jr`P%j{Q0S=LG!VB# zM|er^w$o1gXM%NL50{Rs0$YSr7fNo_2*H&h=02)Rjp%}LA8HSzwx;2JL~|op+9F<` zMcE6vii6@m_tuUNIpw_p+Lxygmj^)D>wu_wf{N797Xjg13kOMuNfI}%Nj}gYrDvn^ zRNR=O<@mVC=Rlo>IO8UgM&SLuG6{UFps&nt({|nk<0p*4j+3Vmb+ZBW4eM~su}6|a zG=qB|x*OO3<9fDKhm9D5%YSz{9~pVmO*ay4^d8EqE4a9d(lXSR^^F5^Tyoy!XzFUhEAPF66^oWJpf)kJlVcB9 zC%_y%Vl?vEahgZIiu0Mv-)qmkaq`Kh5HG%&`1}SG5r8*r*+SNG9d?*Jl{xz*tC!)G zr(Ym|wJDfw;yle6LVHar3*!AdOzI00k<-?RTW-4zU(WuVX{_CL+6{Z`z7I;tN;dS; zq$!gza^wiKHMilj&%VG0qPdK-*NlOQb~68OZ|%ZOH($?wgH{6lJOcdz0|#K&U3W!G zLkm`tl1p6508AJ%qdX!VMu73rHF#jgGeDe#DKh_Z_&;RyKyrr3_pjxQN(f~=Y43O zohy>Putu#yG$noXo%0VO6N5i=R4g9i0&$XfWv9~psIlFy?c)SiP7CvuIP6tfyLbgM z=}gpg5Y86|_y|ChSjVOZM<;YCsz-?q^J1j^#tkcIK^*q7LbnxS8fVCgc1N z3O*#7d1$-JiTe=JE0e6wbe|a@EGa3(Pk;6k9B{xluy)-V%prQGG`|G(Yn$-lCo>4p z!Jv)~MO2DOhg^xW$}&{dRdPAi+JtB37Z>8$ zmtN#{7t;HVJo+f&o`}otY{BeVbMeFrPhtAbJK%eVe-}3`A!qYneviF&+lOQ2|9;c8 z1m-*7_m^Hp`Xd)_fA9`|bK!5WrEv=;5FK*zNvBeObhGcEqO1~IwlvX7hE^oyC8(|H zhvqG zQyVV3_;O5{GLdMkX8i8bOYr@s~f z3@IIs2OoF{^X7krnyP9H=r@FCc~4K5mBmE6+&d8zz0njTRv{Qe2JjOxhgFahWL;ja ztzRlmaOG{W@VoNyrNI#MKkYw>KXTQ5q&;3Za8+^^Xb;{9b@RNvH9V6r&wVSj2XOdV z=!`2R0!my**wwxVX*<_Fs5^Vgat%^h>8!b>MZI-2!ZXM~Uhjp`CRUAeE9i8RG9TjfW@pR(03PW`bp-<2(rcXTOUAURY4b**%RFZ1V3+pRy}C z9!v4qe;&b_^(&eFF!%Km$aZ&iqK$wmzr)<$#oOe&yE~aP?d9xaxnUF2JbnkO zsnel{9fqCA!T!$sub_u0uWE7t=MWv#+1r71Z6INR;Cvl2o=8(H6wtvXgRM4o_Q0+z7 zChi@QT028n1>v*>In24<3!&p>!30sC1(y1F-}0P3;q7aJ8!^bgr2&?PI&Us=W)@+7vPgmK4JyY z)onTX0-}JnG&gdZh;DOyq^U1i(gXYTM{R8#8i|I1rFhxp51|ANx8V%+ezn^03zi(mih zJj|XuljHQw3123osp=(aC^v(0@=+K?Sl`f$&FeR_0?SZYjV)V{wD+Qgl-%P_Jc5EE zlYO(2H_5%X(+u2r`wbW}U@(4s+E1{dc{Tp?=+juUay3Wh7;47&bBanyEf69ZMKsWC z@+vmUq#gB*>&WNQAG__cJJDIY;i;#e!or11P)StUPE*Ty^Vy(*L&zCjLJsh9Joe$^ z_~?@lnMN|{yT?u*%XPM=x4AW|5s!?3eqp#MiHMBlmbM5oEkGx(Kvh~mopqX8NV~U3 z3o8T3Wu#$KI+Ja{y3b2I)<0u;+_`B%*$cSotV>8`;gE_XIHqb|f zR>!Jqp*HLaBCr%(o(l9&peC43@M&zEQ<{}ZZ%`DzP1ZN!lEm39%VP2J|G++gz|*ND zT9mH-!awiM3GiU1gv>-or`PDRLwy*jw@PiNL14Lx^o>qi2iatq{7YKSaIc34TPP73 zsfG4c2xkE(|Bukx+P2y{`&>z^xTHs)eeUm1R!*73mAqEcZ&u0RBK5I5!d*wCs5~sXr(+1|oO>Dl2>+dED^~giM zgO2?>aO)la#92Q%&7CxsPoz2cn8hovx`gjFyJOAH-Olb_o8EvRbk5wlMB}_qK)EMW zuEqiT9Dv+{42BII#%ZBDbUP^rQVOkYL`P9wX>roDr3L?a?ok|l*wLJcv#YZM^{t!G z(%DA7iW+n__2AKmAIIdKCS&h?_TiPwg85%_DCfJMy-Ak;OzgSaUdShU%gj!5&Tu^` zHq&dlRKEd(YH?WUp{N^BL%e+_rcN4h>W!^xqWWO}gA z-uq$dlxcKdIfraE(2AhCmXsaUKQB{^Pd@*YcNgDv`<<9Fc{=_53I2TTHMslHyRpOA z9k6WaQgWVm;;b{zCgs{d_e`-nkzpf}g8wK=4rGvt&Jt~(T{Cqose+JaDM>(Tbsy|u z7gPa1O?5Qlgk#qiBf&LAFK43P4cQLR0E(IH*1>xwJj>vXQMk`STU2&|D*T98Cd)%= zR(hk>W0h~8y?ofGxCb7y;=AKsB-z7`t#F$-wa415F8FV*?cxOPQ22Xpw^FI_c4N@!8E z5nBUN#%M8sTv}X)YNAb6tyzv+?))cy_q$7Q;`dKPJJBj;Q`LQUKY(Z7cm^$vt@!a- zr(^e>_d;h+8@4nwVeQIwc=_d*(Mc4A37Isfh~=__RH3LWk1XV~P)W4KjsGHgW>b%O zv$N!E?({B0$C%T_@_NzU*~t`<`Mj4V#HXKsfj|D~8Vu?;5R;}&z~z_yo;i7Qh)P$| z0ZmF_^r$fetfjP~NXW8opw-1%3?DQMgZmGq;qOHK=8Xi>J79-#Qwiu7V)>etxbxwE z;gI@o;V)NR&nX~GDvSLOIFP8MPCidKH@5&qoYm9V@%en3S8a72N{Wlo*4EB@9QGqn z?@hXBWiboI1r2CzY2!1Lt>WuJL1_U|IFp!WGLbs{i~4iKPeo-phqM|eu*n5vb|IT1 zBn{29dCO*OBA?2dwJXuEp^=nU6Fz8opH?dwPA2#5gTIN%V<+=j#ob-qd|Fs&DZr1kbbOQ0$f06Mn9h zT)`=lO(sqISb{B0Td?Ew$=G$DU2xk?w{bR4Q>}je2jGJbW?%yW>|qBV#v8!Qe~Xta z<(xMrKEJnH$vflHD=tMvzj9Jk9mv;3c>A5VQQy1~yY9LhuKU|RcppJQaV|NQ zThMGyJL|D4q?VRuT4{8mih;TXIYu#_` zs;r`r0ov^THKC8j>1@snZYP?nv!k8MH5=HPTUtnQ=HrGNZs5oxL-jO~6MN@fcE!jM z!|(zr*wyRSV)j>GVeukb`^zU<15${rX|t`0?y?HFh72CY zy#A9_OHd)v*+W23oj%GbSKBuw=}g< z-3svLJFhYIF>&N%%v(6051}b7EG61yFga9nu!?*Q_dj$mKK$eZbhdV5$S+~iHs|!_&`0JC z4VR2r#}Wg4$Q7*?_KTn20Zv`bl$@##wSzS+TCKfuOoHK-29sD`Sp6#r?tb2i`bf|g z4idj;@l;nL8{|v#0HvZbz;;jp+aVn^|I6Au#O4C&h#UAMYYMR)B2}9UzX~FwZ1|S; z+Fm&e!rhyuFyerEE$XQZr2Gl~ey5|ANcK?8ssKl4&gv!4wv{VRqS`LMsBl`+*T>{ttnUGpLDywMa@JR9BiIyrh z;R9KcZpAV5=aZxKpo0$M$d_gUCUbnE$>UR1S<8p0fB4D!Si5l<2J{<14%;^3{7rA< za0bmD0;SIGTr6L;8uJ&-$F!+aF=dAxIW2UjiR&*YA?v-Ba}D(nbzyeAa=X}yrBy(6 z1&3LhaLux!B2-qCql74!Oc4olu||1*5#}vgh|uE9*~jqIQ;%WX z$Z_m=z4M-bV^IGW@%~5ebMmy+%U2_x=ojQASiE8}t#b0He)E{u|N86smJ9Fg<>Yjo zCQ7H*NfK4Ay1E9ZpK&Iiyk#h-fBpMkyh0d1b}Tu!`_YPJ6Q7<|US5W>(h72?bQ5i5 zQb}}U#q#A;N*-Ehwb0tqNPRJxH-niCXKUB3V@}^j;b=adG7q`DtJhfbrp(e(TBQ)h z_0nrEupgw5`efp`F(@o8z zXwS1F7>p`ax%h@?T|bE|E8f+YF7JC?40T|;6u=>i2g~)8ZRVj=QrnF#I#9KSutCSW{23FTeZ(=l=R!;?KJiI=_c+l6-OO;H2Fm~<6r-}18Y~WMGGm? z($XSsfAN4qY^o<;0j*+;qB9XZ)wMP3aBpulE1V9F{`v0rj>k^Zcj4SiS6*`|7A;u9 zU_bxsxw!hOtLgI&JomzLnEl099Ev-C-ds{1nY0ii2vJ(VA?Ka6$RUJR`zb06jtHpG z(clVr=;wEzi7WqBS<;6-O2isFR_0>`qS56GQ^=BNsqmD}y4OsE%vM$)NSCW68xI8P z+F6Z}mkuzCMK0-Y1JzKzdU z!LaD#FLDj^=@?@7=vbBzvh2%8>x5xU%Z06>wwyRyzwlkE)54}xBY@!N$bP4YP)cNt z9l@T+dc(iKuORG(wZ3AH*hVpNte|@SH8#pxYSAYWR#Yw~EtaW6NGSI~sFGMQKq$nF zZMWe7a>>#8_~Vb_(Z?R-@W{O40-}TJIBYUMzmVTmL`q=9u)%y}p-F{c4ty^nT4VjX z4Y>Aiel|jm+7cfA#Zqo*gC#$tT(Mr|i1UBdGwlp{5+Xo#)7Ih=ud3OfUL_1;l zh*4z479giIkNEdIYn@WpnN>nvbpbnNO@5qGqEpP)V3UN-9K3E0OWAFg-7tFi2#g#( zigzdHcu-2L;x)1R1N#p|O;sJ|;U7PK0va2)utNLI`M<&Rop-~mnP1?XbI-=|CCjN@ z8KQ8Ccn0k>eLBuK<;N&4DdZ5?V~;$ZmCW3)XW^ZX-XOn4iIXw2w;;iavQ=15;J<}@ zA`SG8jq5k@iYu2WBohVK)2qmT@)GX5|8DM!yrLYOdFJW(_JN1ug%_X0KW@1m`SjVr z-#!SV$ByA>row_ebmisqF5!a0d<+~oh#l;63G^R-{1H;z#nc}emUrcsik@5dE9Qvg zK~BvNUJYi_(m@5LImx*ye#YZfvf|EKnpw!MyD@; z8yu6bxk(g^FC~772nxkw9tqV&;n87=@hNf;F(>9k~*ZX7u?w%;uG&Gwik2i z+6$ZgKI8XWdj&0{iYzW8I(?LoEG2Yb;v6%Ka>wqX(7tW=Xu zG)|$F$-V!+i<3?lhxb;+pGXWuD#GY2^u<5?C$d<%i8s} zYT_(WO#4&D;C5yU;K#WGeUT7+6@LGNUl32;M!gAR@ zLMIp&BD^he_3>$;ja8HJ{tOk=1EM78uo_gNXLOjG_RC&Jf^wBotDmfqhz z*>bcJjj{iJ`(U5_55T5|dZIS2LP=2}I-1(}xWuK4mg13zA4Q=#X`Y_zAP^olbP)bd zD~%z8M&OKJoQmgPc!@xIFe#}wuyW0EbTqf)d*3-8XPFSe3&qo16HE~2x{;p@#NIUIqn zIW*?`-~BH3+Upy*{EGjrg|oipG3ZwUhFu22Yfzj7Owo$m6$YcGXDP8fAZddpa1%&c=_d5IWh{`WC-n{ zl|T`PpI&$EKX~)n>eb8fccQ_@j2Vqdo!s4GJOZFr_>$xao}ayCf$c zf;@mUn2G=ipW(pP#?eA>Qo#yz$2;>R_1%w$Z&zD{1qJXPpph$Os;*AYwJO zJwQDSwy+qWQky_HRzD@w3aN8;opltAZ0&XslGT#hrxp7vCc8`bX0RZ{obs7VeT0te z`%p{2;m1uj?oR?%f7n_o^e*&o0w0;7!Z^7ssmf-@G*-6Ruw^oW8s}(nX#r=|95r|p zpI$X;^hmt@)(2R!cnL<08HJNhK8Zs-&HjM1e|!##@=EdYt1n^Uk_CL|yYJ(~rSdTO z`Z*+)5c1ilp2IUlckE0Y_aFc8XHM7f)yz3e8<{z34rwuG6PSn@v*{|KdgbXn&Ms=A zl1v6ubCAlwssUJ3un=1soA`)GlYGs9eNA08#*7=yQ9gvq%(u67v@+mk5~7jFpOKg3 zvE$i#^~s6LilTs1LYUQ!IcTM|qa90DEMbe=#QzT_UxZ0#v2oKzPKRxdbIcIX+p~ys zSb3PiGxV4l3=<)xbS{stNu6=;J@+85K;c)vIG>c@&u~4_Jzcco*>A6JV%du2s4O*y z;PJW_U8CGEuhkQ8VkvJ=~c%_wT z5AwQCS@vhZ9HoLmrM-pktAw;UCQ&>p9&6O~YTp-@npPrkL_=6xVIB)K;XgC+9%pRn zboT_Nz^@{Zf_`hj$q21&y|#l7AU4+vl&xA~=@s~n_g$O>(6Tp6+d&M44*xjXC~Z4O zq4Tk^@radR68O7Z??%cfzFWgbC?$Oa*T>;2jyCX`7>MEs_qcVbt(ZsH(2!JUny0{sL#6^L?o9BHk&So2zi;6<2e{ z(7PYL1N{e9k`>;~eQzS1wluWb{5<9;OZwSP@4NEG|AFq#!Kufeiis1)as*N%(MKPC z_#xJ;T8#ny`Xg_6AtsHTK#uBmJn-U!c#~EciR^v z(bUoc3n26#E$fQ&i}_Tr5>f>H`wu`TDU82_8^{rmUhbCC;+b8yo?ZpNU&gK+hq|A5=?x}Dm$BenM`e7$%s=oRSf?r`$q zph^L0znc#X5CEtUfZD7M`J}SM85FLp0CEnz!W`403?TKH|231Ea4G~tctbH5H{o*& zwuztLa%y;N6$*sg8E98qRxsoQa7DHuTJ*}z6R-2^o*nJs+)vwx+&W@XR4@KNPiaM( z*S;ewD#$l81H@>jcooi1a>^x>`Y+YKs<_^4F z6tJMZ!x$&g5^&lw7zM5{p<-;ADvSGsgjVoiWL({}ii%37#7z7Q;4f$&q0E4|i1T%q zmV&=E$>=Z{Y-)V$TY{n|nh7=^#&ir<_wS)xX+CI>n_3WTzO*e3zYfVn-^| zT0{YvY@UV+GEpY4zxOJJ65X%X3fdVVDZvLxZ=vo`IyKH$g(}^ z$nS8%wxNTE;OqJG2zc|-T&lKHBTwa%6SM~{oz18!t0XR;IPij8&W+#QL7o8O{7ou} zNmIt*#FI`!D}k$VB%2h^M7QIm7hb`vFXu7s*h?h**fC?tX*vr3bK#}PFUVy&#*9)! za|3R@`*y5ax`zHXr)T9d=hoZSi;D6xOrACwM;v}Q7Jt1EGv56W|GN8Ds$~z(KJ8~X z>X`4*YN&;rvU!+CPT!S8Zke4Gb`8h%H?}l|?ltc&EGGIU zUy;*!94@}_5~6K#@Rz^*36K2eao)jeRzoFZ^_!zDJGa@#Z{Ph;R#wJ|@G>sKCDCy%1inNgWlX3bA;S0O1xSTQ z!p#au3n7%85^+z9^Ovy&?d4XSu`S=6x`J4$IQMmcVR6>AFwUGsw3X5TY{iQ9{yHrT zFBRvc1#%TbJl*4oIFu$Dvchf`Kqq~F@hJrBP3vgjdB-_Yicvb!x8Ab<%e5U8Km@jU zNh{xer{(>zjZw&uZfMZ{SppFU2hiCXP{(0Nd`By(-F@u@*A4-A5L(>zxAt`u0_y1| zUxEi(%h#)DMOn^9V5ySgX?43&NcW+;Bgs-qS}NIW5pW_U6)QxnahNPRD5oGyQb{QH zk|ax+DU(jPm*|X6wqVz-Tg%4;_7Dg+Y}kw^ zrI4zrykrONkn)XAm|3Y!smJcNlOUN%W05r(h-V z_2$fkmbNXN6mFMY_C!r}71piafIt8FPecvnVcO19ao~R6;&9H+ZhCj72d&*bSVs=$ zu@lE(NdKXPoA*9d)l$Ub};@NyQTmMcdbmH*J z&&@#N3Q5kwaV@n$7-sod`TouJY zvi5J6GhF>I#f6m6aZ-yKq);iPM4`m1pwnFwwFzcQ6mk>%S*Juvpqf*w8;h4d(=9ZF4 zs(w1~{aVHPhb}=D3FKu}DG|HiCa^TwMC}vZr{Qku#CG{8tkg}2{o(Y3tFlcaV63bU z$%?)CrkmN?Hlda#%+jPLIB@@OqN1u4Z7m(BHm8Jjwc)a>FDFa194$@Fd~{+#ZUJVz z_dWsdt9DYDWoq1>P z(7}V*GHz;U!s7b1`09(jltB*BLzTQyt*18!vu4l1C~w2+KfXF zITSnXv=jH2*%@r|3ElU=-88;A98Ox8U&N^){6XtR0T^Yp$4KEhlU)zlDu=Kpkkp!$s9e|2r217RR6bLBL$%G%MEWobkz#ojL+|0&tx z4|?6_w!}p08gZW%;iN(92Vu7VYB2~pWdV5ZUUGQ;iYt;ZHo@5>E1|vTw3ZP{Eh-4vCtAkDiW>6oDlI?!d*Z9-PA}6@*s~ZRXyFjFqMn3^2;- z)%5>db{X0>w;+d{!WHGE7(IS8`7!bsAo-wrYKz(2wae69F>m$~EL(Uzt!6kEIZ-+J z#MgInPw<((M6ulW;Jx_dlaDd;)0qU?idG#Rq)_rPm3$8dY_GfXI_|^^&iyS4cP>N| zS@0$)oJk3N|ATkYN+_3CMY(yo+&Dv_ng2|eo5Sp@DyunswRsD*CEw%?vICdsE^b#Z zS^9;!MV!K-sGx{f4<`M?Fj9Pjb{P(Ha*j#&U``?X^ve(VFq(qA0+iEAXvlz}ylOJZ z4)>*E2N{E#GFW+?#XE90-F{{lg|N6?W9Q$RBU9XMvhk8fmM<? z`}Re&P!hB(mor#AZ zewbDyFQX%A$AnR%@q-_niszqy5ubeYG0{1ieI1MEFT?!V3kb*yu+y~ZC@D9QI87Ko zVI0TjAM%YuI8Xmm&pgGcFAUHgc;JC3t0>0lXPkv{qIL49oVVV28(%M2z(*{a^M%bA zHj}^NjkjOta|g;xO3|-hE&EbTIcDWx_9+`BW9X>7!hDnvWyA-Jb>-l#_uj zdXzN;bn4(Hafi+#YgC*jeZoz366T|3a zwO!5waK9mX@>I+vFRISn;c$z2_zc9E25#6@)OZp& z1VQs_;^vfav9gNs3jTTJCJDDidEtmcmFkb|+Pde$4oCS$Va511@ZTy+)rLMu;w14J z!u$2Ktg$OkK?VUKh@v;Wd>ev7cKaE6d+o_z09%79z&*K7aE&e_W@`5&S| zreWHS)3Mj?d*j{r-^G@;Mm+t(e{lS2_g!~!LbU0-PQ&3x9D{Ft^8kEB zWeuq7kMEH~caQz|$DBEHv2oLSj2<-t8#Zpnf_Y!Fqq>;<4kw;;Jf`osGy6R9%kud! zmhia=#Ul!rE=;U9tCG`|+`rMI|M)^6A3bO=~c7_DmFY6(Z@P{-nWZ ztZ&4JpMHp%@@h<+FcFuKqBJ{@&3z_&syPnQoUh!biSKV|q}7w=m6thzbJM!bc=`F4 zI3mkrCjFm_FU2o@axSNU$Pm@^yGt*|6Hh*g=bn9@RzIz1Cq-yhH8HR-&M76NFk+B1 zvS3e03&)LMPlp9LnABc(7yhaAR_3tmGvVc+hC%po4$mi=O z^F%~@05a;d<4i@5d{gXlvLZ}=B;(E<52T%KVDIf@*?-4nQ~Yebdw`?`HEF+{Tu2b zKc@gy1NtMkFpsNLT~UXbv%VsWc@kEyT!CV95781em6h0Y*S+xf-oF#i-;I$&M=_so zjyp65n42t`4Gr}$nPE)~d`m+cMh+W^Q4>d^xU2-<`t|`RE3U-5Z@$OL-I`h&aN8}n zp_m-9Pd)zxJBCf-HpAPS`1|VO8s4F763-Ro7P5u4kwA7!Ljyp7JmbUnG5hN|nETZ% zT=Kgsu+P4G;|W@^TzbjH4C004Ts4Q-sCL9+8k2fjWui6 zVg;>EatjKnomEVK<&*QfKT%XU`F5AINvzS<)Is&kwP28|QCVAsk>f^@Q+ER{z4Aiz zuN+KDtdsl%dtyEL2`Z|p`E)jO215VpL7X(stR_tUq?IdIp_@3X;_?#eyHXA-ZKJlE z@#2Uq^PY^A-Hed?aO5de*kcmJLPyF~Cpw5nCKKnB5~h_eDIj*1SZTL{P|k2l11F;V z2Kn>g?~Th0ki<=ok}5@b-q*#R)@SRTv5`p1&p{W_10UriG=SDo(L{|TI>pz;JA_qg z2Y|1J4p`>u8qKXxWGY<^#4Dkk5>SX~v^^aQftTBMZ5IkC{`9pHK<7GC`o}{!rH%t1 zj;jNNB0x+HA(Vv{+Bhm<7geJXVs2&s*qr-5-aKPBMoCpH#MrkiJ1mtggT%hypM4x# zt~jBZZJ&0_O{ew6l25sT)xLPI%=|0;#0m96=p8Gg3jP>8%Z_NHKoSd6JA%+2k?Gm=IT@hz#gAY;4Dd)w5AmTZU_|ya|s#{Qw^T z)790aQgIF$Wyf@v*r^-ouUJMXF^O-E*#lGn-^3Y)I#Aed4X-S%8nX~< zLH&xD!z_qQp(THpLGXn^!jE$A_bOY#Z_eA+YrCBZ%uVFB#*VXNrUMn@pU@7d%_Y-5 zOAYAnmVFSiU24BAOC#mM^!Gv9@UG)hN5v(%E!-XZ|f5hCt(9g=HB zm97%s1_kM~ve7|NRmwQvXoLl*M!#~q7XZ@(RnKl&J2+FMbUZ^AKi(B9pG zBXbT%=`ocksx-$ZcCkaVAlD>=%SUTV2lMNvoN_YNu@kr7b_f3Ww?AU))Twyt=Pz=` z%okpK5g&c}5oXPvjiu|BV&^H-v1s8Eyz%bqIPB zrpJ$&NcGR9HnpLFXrD(Pc?g>}v~bGicA~x(Em?p=zkLYCjh#Ra+ZMF8wsKa~rluCA zVw&k^EBPu)^2*6yP(?CTPri;SK0o>PH*dqoAAH8i?0ByN0XoY_jtLbtj^b|4Q)K9& zPEwrjfBqT9jUI`vjvhXgr#)#!O>H?2-v8V9`FX$M1M7`}@W!ie;GZ}B0~z8T&OH50 z?7!cBd`5C9`3by1<_|H+c%#4#d^Xd;GMCs0Ft~Z_iwJXFrxk&6N+mjzRWN?M`7fDz z&!F;-+hEvbX$dFkO3WQ^frmMv0xATnInu1!ZDDY=^eNumKN;^J@NE-#6tjBjfK}eV zXc5ep6VD0^fhvBkxyckLB57=;6r~Q4Mrjq{kJQwmA2hc8+I9*kNuqg0TUcLyw z`q?iyGwJ3{n>oevmd1_P)VL1&?y@(&wdX;^jsF&nYa6k4*;?)Zp5P{Xs5v}^`1B0% z{btAUx(%BN=*jU~QpBsrdgA-vet!ljhow02=pW#ygO0-I3qHo6{=?AP*^Gwf&1fN7 zC@-fFO=zHg%_rxs<{U|L=g-0OFF(h+x0amQWdz_R)w9_s*Q0uP-1Bk^IO$z}rhxM( z)znq;5tR9=kX8$C@>rFYRG_)BosW*3G-*8g*VOT|r z4S0LT44#9V>o+?wP+8z1!8QV4oNjIw2>G?*=;SvT;ND0<%1H8ZA6A@UrJxann;MsK ziG31ftz3qYkk==zfMQ*hi$2Y^i$&y+($QVuUMc5sY%{;nJq$@t_p8CMV24;q1{3bR$KxcA6sl2GCu+i-24jbI^K#)1QbM3M-8%R+bG zP)}h`+oVC!@!_~a%g4DTqVicsvpO~W7R1I5L`J7JVtG)r#p~2EHRFUUI$g~_7VIB8jDe zGAimK(PiQfq)LqM^M_E_1Uq6 z8*jYA`E^E&9EGd@a1DO(i(k_Up%WKhauGhC`6Xt&J%hlknVp)q-+Cv84Iat4Y)(7# z6sB2BjQvZmzCtU3Hd;-bNy}c)4XEQ%Rj|rPJp)_PMPu+P(9CFB^yhmWu`Uc#3 z+s)`A%l`G(U&ZThzD|65K7Ch(3x53@?7Zuqm@;x2W_~^g*WLP8oP5$rIQGy}aNk3> zqqV0Qha7YWKAZCi3zNyev*Xn1c=DOYaQB^e;i)H|!XuA9f&v2fj-ECgdc>i)>4uy6 z^s+f~=V1QA1-SK=TX6b0r(p2lAsmhL%o8tg|DXMfbMVs3&*PB?9^$hR9((#>3?DiY zQzuSCW^5i#`_aj?`njDmlvY%i;SX2-3C}+BJg*eazwj40>ZrqL)s#z0w}>~G!5a*K z#C>4|s6Y=H`XuA5@}$?vn41gn%5W~Jj4zjO6W{A-iiRq;5H_#5_^~`Q4#N3^rn}IGhT>6hT|~w>l$Ir|BF|$%anTQMqZ? z-ChG_T>0Li9+i+#ilKcms3i2JiU( z-C@E+oP5F$=w01-f5r#s)t$U)iYHo64?4&>Z4#=@{PGLV&^eF%6xF0Gjy~oHoO1H1 zs$G`Dz|Jj7>Y5nj2}=RK~l58%ePl7ko)cDawdSGU=Vm$l2RYeuwhX zD&ADLVaqxagI?6t)X=zOu%7%Q_2l3-&R>Q? zf1LC<#P;NsA245m#9J6bgZrkl;K`^0_dZc|j%-OsPh{s#5{}~3C430dQH8jA5!%h-w(hU?MN+CmP{RxDp;^5qoqd9{Y;ZX!T$>*zqGuHx*Rlc!7}K&-&@9e2gDo=_@P%Yx~g4LE>&P4jJJ8vg|tia+$ zi!tMaw_)NZr%c`f6DCaHqZ;SVTY$e`_g8dvc98S*YqI{kZCiU?pHhFdx3%H!d+tJF zvQQAEGiKB#qS4_9A>eDY=Nz55 z9(nL#G;FSqLmEI_4t#p(Lj;hsTD|e3q$k%2YM8^>VJg!46En@0Z-LCyHIYtMq3EEd zWHm#PzC3{$*@lmAi;|%1s^EWYwt0~Rb{hz9CtfcoAE10#Ve3m60Kp`{h? zR7;SD2<}>WOF~6$FLbycOt19QIymiX5M!MNe(!+ovmBO6N(X<3hERoCr5YWXNBfbD zda@nmonW>uS(*l(Li7CzWZ5dAt-f%T5=xb7xAce%7&s88opCCP3d(T$*=OMKMgKu< z{|X#^)OWC=dKG5RpH0?m2kOZ(zwf@g*(GG6T=I#EFxTF4fz?_aDh%ysE)C+EyH#)V5NZ~9Zs0O*^DrAt}VXU(@ zSixrPj9SZmujF(Lb;>|C%guMQ$3>@N5=EN@(1*T~waoTUAoZ7K4dfKxR{U|y2B_Fy zI!oKhO|2Flvq12_xVBRi5c+Cn##x;TDYW+s&e{RIPHAf4yn(C7zT56BSPs9_1awN@q<^aoM@) zS(!+wWho8cA4BlSATn)KN#{w9`&wNqp(W7l~?_iBo_0Lr#*mYSmJ9 z(l@rWV*SQV{2oK6RFgGZNS5=Ob*mWwOVz&u-S}Cikdj0{odtNul>`FtNdcF00;yS$n$4jeVTJ4F zloD2q%4%|ua}_ISqwvB9NVmw(L*F)6GjD7<9J76+A=B5#`C>aEO_=C74E$LV-mL|g zDA{xsAZ)u_+vPj}`@^eOx8zYeWZX1iby|S2%%@);-dJEWR_HHK9eN zbgFtOmE#0X2apRs1Fch?9G=u^dPjwFbQmd{_a7;RQI>hB-6m;DJZ(=fVaK8i1*jcO<%`5HGy&BHnoOHKIxC@y&03 z8;!(yo7nb0UHvyKTeTSd`&HxjzrP9_={@(5wQQ{Vyn+l4Jm5gKY_GrZZ@fdfh-jb2 z`X;v8JGvTBNY3E)_IBp~nPws-Vs^m(%fEfoP(OWTKxKgbMe-j@6sx$6K9-xIwnq-%!;I-fRr5auwunZdPhzO zP`LSy8!>2DKfLnHt0*NuNEInPn*>cGxY!WqWf=3+T(T)Zj1NPo|L|OhMsqZ5q(qez zCFdr#&uRyf5tyB2AMTG>!qGBHL8ts?Qe=Z*Iz08@Z0AwjE0ZPLV*aEE*)wo4RHaKgF<2D{uJ2;+n# z>Nnpy6+p{&fXxjNI>Gm?2TrK8d>~rT!ow==69bj5vD?t)2iV=0#4e4l*tjgy=!=@t zDQkx*a$&aE7>ntJDGXi!aw(t)`vp=fq15EiTjdR9k=_v%6|=b6eQzhbe}NPxC0Z99 z?i>&9?lrfYSoc?6eHon{o!DW@4!Gd_3y8B`iTdWv*md{aaLo6P!H~ft(7dIY-)DBn z?tj4k`1w!HV~~9B?RW9@{P_gB^T?{N#D05!14kWx1b+I{pK_{bliba0kSZroY-{a6 zZeb4BVg1JS_~f&XuwvB;Y^-0$y!WCdbMTj|uSHowF}|4j2|oJ#eHJ|P9>bR#2lSx9 z1BefwLjHm^*g(8_U0n_K-)BGM<>zrfm`zgizFxq`R8E;Pnd)&aJA#Xf@8Qg%e$R+j zL@AMyHJi0^+IVSzaihlI6Ywu`rMuQT$=@sv37&&M(O3TXmy~V|)?0eEFgmcD;PR|g3nKa;hCa?qa7y#jn zC}A+Ae0A_;3f00Evz(`C0QCMqj;7`h5~Lpj-b8&;CFmJJDD zudelYt+5er zzVjyMx%uTU&Ld8~1MBNoqNci%CUq|#*iu+hz-bhE(2836eDv5!cPS#rQiRHi=TT8nihcLr z8xNBGnnyXn8+Vvi5tquB$a#ag=es4^=i({YOHj>ozWlTnNQT$D*FB}*Z!3wf##92 zHEA}q8{0s`ShTitM-cw|BP3Uz*rlHq7KiWl zBs5%vYO1J!mG5++qCNMfv#R8G%=t0&Wh=yi3oD9X`3if$Z(P&y8&^$NW+EmBQX^ws zhCZ|`Tj+GO9N_7(D6Qdd@1P5ML--gJ!p5&e$3BpJsePZdKoTt!m7h9FDp>!$x5N}u z`p5u1q1O@LIT|N?_j~9dtJm*SFqWZ-E&tEskK>_-?!%=QUBvJ0?$%_b=Af$XUi|3H zv+=#7PsU;2K9;!gPLx-b@*(P-?`q`KG;Tfm!W~GA{f5 zC0I*JAlIZe%~%jMJCqBleeE3`xZ|EXv1LOuhSBQej8o3!5qRM7$H+N82%moP5pKTo zM*QlRzrv*PQ`m{(}9qTW{ykU5=yY zoz7V+NHHIMP=TKc=7h2&>>d{=FAW$714J4X5}`6G<-#+u^L`E~AQ)gtx>BLkXhg=j zS9rhAb-M*mK0Lj}B%ec*q?YBc5(oXlVdK6^ z;ic_ZhLR>j_fXaiT% znv(3a95-exan6%C1#=I9ZZ}aQhF3R79CG}=Nnc?4$T)>v=GBVwYO;>Y$wIFr>%D?~ z111}2Z+8z@c>lfj#e#+Naq;EnaRyIxWzf~xjmm!Iv_k0QLsiWGlH6`IG^`>1emE-~ zv%$$wLB^WyBu;<+^0~z4H=?X!5LAw0tTJ)_ymPj=lC5@~F%vU~E$EzLV$3te*s`UO z`FHx1hsIDr#?fpvfC<~|Zs|h5ssY5EleL~{;-q)QW%;NehiX+#mA$t|A(uG#QDaBo z9u> z-|s0AZ-xW|(i>{wn;1*!ZOe?7-ZBJ#p5z0_}!!` z;Du8=VXpb9V^`}!*{O(S{7*(UONRg$uMA_$V#(>Y0TSc3_1b0y;B#xsPKztO@~NUZv;VD?oc7DOve$5 ztr3&Q-Q=w1_`Cl+WevDzyQY!xfC(cfu)}oeN^-ci zG!tm$hH4v&t-QFGk2_qoY7IFuOVEKf?6>Mf)fz?#5p)pL`mM&RFeu6<* zO}+sW>S=dlCaA6IkA3z$fI!i%2uykfo32`uZ=wq=t&NOZI-im4BH_pFikoP-sCNpEfP1I?emm=q3CJM?#1aWN)O zoQ7pfR^sznpKu%I%_F7O){X(y{YYVSpo1L0UEQ60l9;hFzeg0&$De$F)oWLyu($xK z$)tc7!SbXB3d_@>1HZ-bI9a3Ue7jOcFHh!3Dtr1XKnu40+GYiyQt<2t)&rDwTG8H{ z1c9;|)kdkPh|Y*03@Kv(Jk?|Om?ho@clyM7bOUAYma68=rHm^(3|6p&b(jRULsfXG zCuscb%EbUBzA+A@t@04{&q%=*4pgbpV#rZQ&ESvDwDyey=TL71MdjyKimnMhKNY$y zOzuPuhYHq*Rb*1>cm=MdPc8IG4^cYz-hB_AdG;wjoUOOl9AwgfJ@(uOr<{BWiirlw zcm?1Ni80n_MRhqjX*c5E_uPT!o_YqA#TCS<7w~+i@9<@)6~_q<=Dm6VD(V7{%X9jE{3 zRBWI=`^7JQhF_m|0q(r(U%2e@EBH_uEt8-*`BchFQI4avwlh&>GRgcAvRd_ClIx4mZS87O3Ore(Lb_R31b#j3dDT zD*>N>OJaWnUs!;7=ahA|l!wAQbdK}TMMa8LLpX9DsRfd;c!RkBT zL#zjaJ`R37g>=1?>?jBap)#&{wwSx-b@C=G6DHY1l*8~5LosN;Knxr>m}{Pw%jZ|w zHW-V#umBUsPQZ5#Iu38Y@-{2v&W>)Tlvr_-HQUphA$n&MdQ}(3kDkae^IM2(HwT^% z8a$Yt&m47QGJq20!+{}*=J#u+qT0~XOyF0Piy{=`i_gBoE3dqSefHW5haT}A?7PQ) zWIeZ{JL$yAH7l@w%|_z26;URgWIeCI^qqDhe?u`RsVgljWk+kTwzS6d9j0U0@DX_N z`RB-C+sY@Dne@xroY z8aJR{Z5?WB`{C&qpT>qY>rhmbhkQ+|p3+jL#^F_>kj-@FO!NijJnA03I%A@iVt8>B1k0>pyvZrLVcAt#*`p_ z5W7CKDH?tL>+RqI*d7#68pA!M5l)O_XGJ>BH%7vERC1eSWI8oB>gf~0A{cx5g%FH1 zO_c7S5bc0a6ts|pz?eIN(}_K-r7k4K3$yrJ_@@v=P{lWgFH;)F39sW&6Q?F)ZL?kj z>)Z?6qHBYK=trapqv!>20_F-$oHFqO>)#W`M`_`Vwq=@d%z?uOa#{fs&e`78im8*P za=w~TV~64BqmCe}x)`(P&cVvn%Q=N|0Z|}k9ah)BmdnmBFsEu2bH+^&(C4Xqez)0u zYa(Y%hEJ2m!9)(7^W&fJ@reI<{1ME)YmPn4#4PiPe(E={A6fOgadeDH3-R-xokOb$ zQb2jRSh{FAOjhL&2vAMZHUsEC`q`N{;;>^dY|uztdDRuXkKr5p9faL?-2;o3F2D=V zy~LSAH>}-&F{8(!sI(X>RxZVa$&+!|!G{tZGYhSqEu7_Z<%*T;ujuOP#gLJMap8p* zz&Mcp{*PG!lKYwth7ZJM1WYWA6hwo9H<|{ShCMGjZHl zPFiOUuQw|qL;cj%4IqVBOsj*n`14=>gty*!%Tiy7M)oZ6kg|0w(Lv7m&TK@Lb6%93 zj+cWoT(ndNG3ldxR}7$s6m&_KNDXGEzlM)4gwEPo3q&1k9XKmHNNvhXpZIq>|A!5SbIU_HIeZj{b5wcfZS+rMJKV%{)3FXgfx!OF z4r`}uZC~5=d!?uSbAHtzCzPot(HQ*(*5Trd{|9Ad#g+=lBn+?zed}8|^62A1Vx_?<5mBI z7hil93l}aR1y+Zvesv8tuiu2Lueu7W*R4ev@#{qdvaRF1mVv&g3TQS5w0q!G;YPhC>cL1f^uHo9v%k>KoVz-PP5BEe-Y9 z(zwNrRj=X(dAjC(tF(w0)otVacGJvos){4nZCo(Qt@nhqG4YYv7_{JPnBi zGTu4Pz4%(B9J1cmm_|yiC0Uy;$vUnh@Y<82AwbHl9^B;dzXe6$qYbnAC$(M(WV{L- zef(NbHLk4iRmc;s5Di;jj(dy4E`JV*yer$-{co>r_W=ZFW7J4#3qT2x03(yNjXxzA1Hwf1^-(INFj5NN-5WQO z$?4`oV6!y7%KFzjp>*gHJK@~KgU*^8D&?H^{TnBy>lYP_iBM<1S3_DC@tEZBa^6r> zUQvn8t}UF2(`=wJd}+4fNeR&p&8V)e#5q6t zDaMVSz^RstDvL3oW&m1<4jD9XAeEns`#*R9FTL_S@BZC+hn?((4+8Tc!-kWAZDaM# zD#VaXEsgc)p!b^4O5I~4%FX|}h=M67PVo1?|C7@@ob}VQIL^M7KzCEqI;>o`lr8Ea zQUC>{AbR+mV{#l*`wEEyGT>m+OSBVBwRv+r8t6SAd@=*OP@Bq%%P@2Gm*fa;BA-Yn z#*;JnxMNQw1<+2`bt8_U6-HrB5t=)jFn8HPY^txv(7~fgx#gg=s*I13G$35pZy=v> zv2?`}%%3w4Lq-h4V!Ge-^un(fVC0Aq9JyomBbaET=GG?iqin{q#VbHMlj@zr4&efceOCwo{@+Lk&3aYEa7Sm;0#OlY-;Yu- zOvfb(v5ynshy~;>!dRjlQysz44_J~Gytrw^-eX<4Y)?ZN>VdY3QtE_KBhW<3Ndd${$oqU?xmO*0579`{VhQd(bW2_>fL*zX1(oTe zh~xRE3Q^zbAcIAvMY!O%7ZUw5ksO_kxQ1wj4zlbgO`3v34m*^~c#a&UZ@u{@r!h7M zdo8G`rD;_VBa09O#{M^#E9#YA~6?jx0QYCp`MhI}wBH^NFy>@LAs3jICj@u7& zy>GMa+!VMSZt#Ut!Q4zp()aE)%7TJC59kISfIw}KKdH~p zZzx&VLgH5zI)<`!;j(>LJI-%hciC17PX0vAqQkx zX$j6b`)tmGZw`|&fWE`T=@>bTcy_Y5XRTU8G|dcDRFz}e)ae*Ib|lPx*Jq!72G76n z0`C{7s3<4i{%Ssi=J!`#hBxv$E&ZuNUMt`DmOzERz5kQ z>zIbxuwfmRE?JCLLo?9}%sB+@EgksqljpH!?HX!d3HMW1XD7x@Bzo#MSK{nnosC5+ z7U8I}qcLjKXg=!ll^0*amMtx`>gh+yXdsol5xGSQ`;nu%yrK+Snj6>`Qb0a~;lqdV zQI&&g2VhxE1sZPI#F<8o44aeEcH3ol<3(pc%pqUFs1Ymh7dd;&=k#5_$}MEUg&WCTHz!E_!ZMYMiU z27U^KJ@?MA)ME8-!CErpZr;1^4K5^M6)qa)>2KE%7I|LuX-} z{lBNpReCOzR~UA1>O!vLOOs_h+0&Mj_`v6gdsKm_6lw46&@fps0TgQ zf!=aZ9{jU3Du8&*yG=uvNd6xJ_BFy&Dp<_vWw4~2>{IrA-ZAVIPy8$_1Rc<4QXc+P zJ;E|3c2?Sp65nVO1Hn-FQuN}^NRv6euRcavr^IAG5M zF?!-yY^Yz4H{X2&_uX|Lr-Qiuh8s~)S;i@!%ZqE6x>&h+IW}!sPasY_eyu%F#D zCs|2=IbqCrJm3AS+46*6vv2>Bhk>jkvGtp@Y&AZR-AI` z$*3r)#^t}ioI^YBy6@lUKVTrArC^fVl@N7SNOd)-H@0lqOf*mnYN~5_<+6O~N<8uO z?~m(SsD_06IL|Y|n2NeKBig8-AhL!j0smIJx;8r;@gF-> z@=j%iZNIil7^qTNed9aT1H4maF&NZm+(C*n_f8(4{ZB`25Tc4?+`Eq^OZ@6Wi4N4U z3h*);oSZ650N&)4L9pUe3dsY!`x(#iL?|aX{Z%G5*)LbJCXaUzC2A|ZsG~#8x6{@! z#-;mu{eR5;2b^X{aV`!&b!IjvZQfN*t1N>uh>QrvHrW^x43}(?4F-{ufolu~V=yN9 zf{Dfi8)Py9BoH7Wp`3G;Hs_t)ojLV&sMKec`(11H_xt;RcHZ~A=X9v9daAmry1SFr z!%Aj~b8w^Z?PUYD9ox1SAOF&2%kZ+7 zyaJDV+!-ZaQOBq&x!pOHGtWY9e8rhqlYn-6QZD1P6K{Oo8|t&uQmiD^mFv%Ko;zzU zjy~+zvW@jP?@lhJUHI(3{7a#8&icuUc20JqP3y32Y&#zKfMfAjfAyBKW%ft@;U7}a z_lk0aB;}k`R9VS^_^hONY~PjQF0<0?B?akskCeq%%ahyd1us4`DU%Vr`sM2SBiX z5d-7ksb>LonBt{vt}8REb*uaKh=Lc8ijCE0{8^IsS;{I;aahQaKWGws1ou7yo@Cba zQl3XR8%;n#v1obfUMl5C+e-b?KJ&E~CE)P~%M4=IHhYN4Ruw0MYC$qq+SElcq$x@H zP6sQulaXSrKhRq@kuriWF zCQf_kX}Bi=ecqO#n@GSmG8c1_4?Y9`efQm2wklY=c^!WI-5=rB>uygL_on3c-jP6e ztn^|2?0w2v!ecwfus&Jn8#Zh#hn!3#%RSEyX3ibObD#5UeC>!O0Ie0e#-TCxPqs zom-08GBL3WC!KNv{^?_%!mG}Fb+Y(->`WH$VFw?MRYx9)84G5ZTQ0YL+URur(c_+k z=`%)4hCn_~VD{{3IB>;5xcja(c=Tf)i4T9|1EAEGS;<;nw9m4VN_pk=SKz{5{R&4N zab$@flvT!kR*Jili-t;O !3YhOs!QmWG9-$Ouo;eNEX65sflS^lG2KG&K+dsK} zW~4Gc{?Gq_fB)uJ@%JD6Fuw4)FO|h$JH~dV*~adoRJM(6!|pV*potvXol(|P%$zej zDTutm+q4pEC}Wfq%FG1#3+C>V0De|kWOvF*r>4)-@PUv09iI2HXXB?o|6!Slop#ED zu`s#tAg&t@BA{X9AQ6Fa`|8~35shQGBaz(o2{eez%1m(C^FlgInoNk1;_gfi@-50p zXVs*x7p{EWq(K9^(|||W*k=m{!Q56J;6gXV6I)Y2nxN+ryPH~j=P@&m^wWNXcGTZT zas%0I-A~vb7Kg?}f#!)3eN+5*5 zNL#fntbp&=9~qlXHITj_F8cbtZwe%Bfq#=m+pd4TmaI+ig)zB5y@9C0e)O)&Z(ZyH zBPt+OJTZMtNu{+7<^<;BkNH-r9P49ZX?2}5>Okm?w%SZd?^R{hkqdOT_!lo(fG0fh zPw=aspI2Nen>KC4x4!cYob|>x+2mCfX`?|<41D%&?_OMGqu;yLqXC7*b53Qg=TiYo)xzWeQ)ENbA1fAmDGShfe9l6`;;xI=gpX(;uiDkLGbhT!IH(x%WB~3ZoC11|B(-3)nSL0wA8%0 zSYCv-aLGP6@qs5{!Te<^wa9>dy1^L&=T3!nFVJU#`W)7&l10v0D4?Mm*k znZ_?>^a|f1jfi2(r&)vZM+^QM{Zb%)iIY!muzOWJR&lOA2wTG~5T&Q(&&_&~MIJ zTnOo?rT)$oQ1pnHK{&;0FRTM7|2CUQRSUK{mP^~&w{g?4CV5#S_q}LxeZNfCH~>`O zNmwv<*cbuFCe2eAuZhqMv<_4349rEuVu6zxBY0ME} z(p1p3Mixg;r@|#=>2oS(Wc0G+RV%xs+VwIsqZ-b2oUd)|hg`%_RO>ZT_SwmskB#Sb zvI*>fL-sucZ+gSq@i*JwigSMcV9XkqnX3U(9MGF@vcg~jNzS>pnv%H|}?Em;rQ9k)N z%iSp`yA^MJ$6ImcOU^9${U@Dzayb|I#^mmrGkY%fU9ms*TedG=`-<13w%whA(LX4Q zzm7fj_@qQ;rQrInv2N3bMAPwdyd&j#!OT%S<*84@0V@v12mbE;Set&IHEkAtcgZEi zwe^`#e+~~k>X_2UoK2L+J1@dhUU5lP{&9&gQ6q%ea9@<3`*VY#p=GFIDTlZ?_(yYv z#L!WmH!0-$eC6+2es-nSmEZoZ`R&axi-iQwxU~havo+KJM9q$>AY41%sXLe5OCN** zJ{6&0%el{=O|Yu5NV4dzw+*=94JFqLdRw=`u3J@zK}(*UA@MYYgaa@U8e-E|; zdly0)glF$GLFJf!=BZ_!tsMwC(L97yZX`_xNFVj|3TYEzp%88NZS$o+9MTe3t zLgD$wwi)O{7wt;SEt|IDGoSiY*_|_=VOz?bsQcpIdB+{-`q7dJln+07(Tkp6#-vXG zoi~Epym3?ArLxd`1Xo^t1zzyd=O-ZDg}0pbW<31VN8py*ZzoKi%T#49gaTc=;FUGyUC|E>`MOjQ%*Y>M;`Vu9DLxxxax*$@Y2^l58ckZ z*UwxmTe`d`i`<4gZ@Lrb{P-v3aQd@bR-+L#v z@7{!MyS8C_^23iNS5@Ait?Xo)Ea~wQjHXm>kB!?lREtvb@ihqAiB&i&xo#FLnu~d} z7Zgy>E}lFK`O25Sf}fsyP6^QADPH~6ATd9oaKkEfpvuO=3y!1Dq zG-GLr4D7k%R$qF2L4}+fp4P zW##Yw2kuwa2ISMP#&`9YTXOG@z#VtqohVI3{QOWAD5u-yyQ;0;KJYFf4}ejC15;v-t4kf^~}JX%a^`+8+%>%3n(n2@t4;S2Wn_vpnpNp|=As1*!Ayh!jRWMAbl5 z{B8b95ne2NXAt`hG3xSXG0Z5gmvD5__Ql@2stmXg#%L;B zN0nmvqZ(bP?SXDrA2k5ed#M^Lm0GE%$U%0Q8^#T(W`}*+^$q!On@o#&3+5KZl+Q-)`yE)j zVQo2!n7Wy`>E>H7ntb}*$js!**;WBZP>1;y^9tn+4bho!dcyL@GG&nM9D9r44?8!2s&HVIx($krRG5KXUhx3c1k`zGpTX@+7z z4mOq}YGIk+9*1(L53?i;@KCxgG6|cls8n)8jjDXe+(WALV9!on3+$a4fObePh9Aa4 z2)f_kR0gm&AVtI`a-FB3+XQ_EA9X{zXmv-8SspcCiq_s414ZCA=GQwc;L98afCpUb z^+o{z0<&4a0>hC`qhs8{m0AAShbo$qBLgv-o>QQKHA0LJlk9kfS<{fNSrF$;pNWS& z=+p#krxZ)H1Sm83=dGFZp)Dy!kO8o`Y*4MmY%ymmKj*b)%Q)}+nP(4MHt)dqfB3!R z&mJpooP4(JMBW#lbDC+LUF21)mtS>hv5*%ptDuz?Ku)K;;il^_vuqkSy@2n}esmsw z`qOi4%qjg5=<*cizdJTsX$i0&H?IeJGU1Aeexea1OIjQcd>5c1{`_h zamn|-A3pc*pU0|$4#Uz#i?Djy>J$)OST>~VQ$Ol2)42XUDPvnlrO3=E`RiEIxV> zjz9jmLbr3y{UN5$>u~x>r(^$R2jZT4?!nl^m~0nLW%Idgh-bw*fBw7@G~VxkmF3KZ zEnBx@ccSlqeg7L+ch7w#lW4}YkuobA;StPEGl_-!EGVh2IXG@{CsF_n@T9{}0k|}N zB!bk+83y7XkQ(u4m1B3DwsI3+;eqTloNUi(hF{}_QrsbG+R5NXF7%w3$*rjW_TirD zDUJ<_08|R--DtI`O;jtPN0OaH5h-YL0N&o@7CR=@J;iJ43V>oTv=ZBF^`V7fSIiJ- zJ1mcD;k8WW$*3$ZnTw5ee|>h~H^NCt0Vws9AM-}Z7Qh;SVrrzIt()L87XcX zT{u@>cXioRE&KdOyK;zP>1)@ik|dMH($=nSO|W3`LaaLUkm4TMRaUfi*tC8V&i&;t z%f35fJ9d{XnIHA=GjQNxhvJ_5Zc9qwa?H7HZi&~dO75S$rShLX{pq;mqRR?M?oRDo z*QbE+#1`zca5k1LT8`W9ys`M>^Y+PGQ}FgHU;b)Yq?KJ&+qP`O*3DbN36LL{to>*I z+4D=B;PapVJbv=CA7jOeWjOt`hv5(Z@bNg{fc>y_d@J_be+8cP>}O$q3f7+gt6$^i zKR-9Qb@su$NQD3NTP-<@1fSvmc|34f01JOle5urEfEE352mnp{k0|JT`NKcyUy&NG*7 zJGLj6(u;BGX{TX#noXVc_BUW+OOGqBxf1996q@b3h z7|La`t7?E*MK%!-3e4oP;yj}kcL6XG(6u61vg{m4Tlgl=iX%X`YGFfJf#G6RyD~8) zf@+V~rcm=+wZ;|3>koQ_=5RJ$#fO~7oe-F$lGslmBLTV^dt_@|m`8hG^v*E;rv{(= zEK61+rJY&=U^;t>ZdzYNFtuwh%m9FZGUcjcY5A570K+Vg@Vcp3_M)gMynbSXG&G*| z-r&z%L-h_B5e=lxm+6Hz)j_nbysva`ro0R=d;J`zvaSaL@HV1kbx>>rk-Jyy0sR(Ir6v2SSn|Ex25c*yA{Ol{6e(%4&i~suef0gGmu(K?#Q7|Y< zxVVMtf~j7BWB&v9#m7JPPbm3Qn zPk!R#3H-K|Orp8TZSu!Y{A0ZSb!Xv&fBUz%XZ78M#~$^F$KsEk@C4j)!!2dwwuLKZ z;aN|A4*v5$zmo#08*%Z)zr`h&Tr3JXCoJT{V5VbpQUvG!`WMNMe{;!B$_nCTFFq5e zC0}_yTj0KX*CgwG8csX$G(7yFk4QfGIk@?@*W&E&ej5j^+8-zW!2_^h?!vPAcjxY1 z<@Bw*nt0Z%Ie2nv|E>h^C#HA5_^)5U+28rk;!1h@+uoL3OLOqsUtNH6&iQcxpRqI( zdgQ|%p1ehbX>&*M(wF@u_DPB^r@6lEZD--fKmK6?|GpSnIY6BW{L+{FW%B>eEH1X2 zZnzbz@4TnD(DIoHCSzro>o)8cUT{G*kn#*o4!mJK;&hLkAUs8=E2WFODQE?`Ce$@S z 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 literal 0 HcmV?d00001 diff --git a/frontend/mobile-app/assets/images/Button.svg b/frontend/mobile-app/assets/images/Button.svg new file mode 100644 index 00000000..f80e601e --- /dev/null +++ b/frontend/mobile-app/assets/images/Button.svg @@ -0,0 +1,3 @@ + + + diff --git a/frontend/mobile-app/assets/images/Button1.svg b/frontend/mobile-app/assets/images/Button1.svg new file mode 100644 index 00000000..f05db7de --- /dev/null +++ b/frontend/mobile-app/assets/images/Button1.svg @@ -0,0 +1,4 @@ + + + + diff --git a/frontend/mobile-app/assets/images/Container.svg b/frontend/mobile-app/assets/images/Container.svg new file mode 100644 index 00000000..db2f3088 --- /dev/null +++ b/frontend/mobile-app/assets/images/Container.svg @@ -0,0 +1,3 @@ + + + diff --git a/frontend/mobile-app/assets/images/Container1.svg b/frontend/mobile-app/assets/images/Container1.svg new file mode 100644 index 00000000..0e076d70 --- /dev/null +++ b/frontend/mobile-app/assets/images/Container1.svg @@ -0,0 +1,3 @@ + + + diff --git a/frontend/mobile-app/assets/images/Frame1@3x.png b/frontend/mobile-app/assets/images/Frame1@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..db3e455792c2afa3e2e53f6edd3b9e08922fadcd GIT binary patch literal 363 zcmeAS@N?(olHy`uVBq!ia0y~yVDtd8jX2nVq}?qi1t7&);1OBOz`!jG!i)^F=12eq z8HxZ z*V|ry>-5~kmsg*iv4F+FHO^bdJn3&qMa{xZ z*V|ry>-5~kmsg*iv4F+FHO^bdJn3&qMa{ + + diff --git a/frontend/mobile-app/assets/images/Vector.svg b/frontend/mobile-app/assets/images/Vector.svg new file mode 100644 index 00000000..c48c2e96 --- /dev/null +++ b/frontend/mobile-app/assets/images/Vector.svg @@ -0,0 +1,3 @@ + + + diff --git a/frontend/mobile-app/assets/images/Vector1.svg b/frontend/mobile-app/assets/images/Vector1.svg new file mode 100644 index 00000000..fbf77522 --- /dev/null +++ b/frontend/mobile-app/assets/images/Vector1.svg @@ -0,0 +1,3 @@ + + + diff --git a/frontend/mobile-app/docs/mpc_share_backup.md b/frontend/mobile-app/docs/mpc_share_backup.md new file mode 100644 index 00000000..c3c98638 --- /dev/null +++ b/frontend/mobile-app/docs/mpc_share_backup.md @@ -0,0 +1,448 @@ +# MPC Share 备份与恢复方案 + +## 概述 + +本文档描述了 RWA Durian 移动应用中 MPC (Multi-Party Computation) 客户端分片的备份与恢复机制。该机制允许用户通过 12 个助记词安全地备份和恢复其 MPC 私钥分片。 + +## 架构背景 + +### MPC 2-of-3 阈值签名 + +RWA Durian 使用基于 [Binance tss-lib](https://github.com/bnb-chain/tss-lib) 的 MPC 2-of-3 阈值签名方案: + +``` +┌─────────────────────────────────────────────────────────────────┐ +│ MPC 2-of-3 架构 │ +├─────────────────────────────────────────────────────────────────┤ +│ │ +│ Server Party A Server Party B Client Party │ +│ (物理服务器1) (物理服务器2) (用户设备) │ +│ │ │ │ │ +│ Share A Share B Share C │ +│ │ │ │ │ +│ └─────────────────────┴─────────────────────┘ │ +│ │ │ +│ 任意 2 个 Share │ +│ 可完成签名 │ +│ │ +└─────────────────────────────────────────────────────────────────┘ +``` + +**特点**: +- 3 个 Share 均由后端不同物理服务器的 Server Party 生成 +- 用户设备存储 Client Share (Share C) +- 任意 2 个 Share 即可完成交易签名 +- 单一 Share 泄露不会导致私钥暴露 + +### tss-lib Share 数据结构 + +Binance tss-lib 的 `LocalPartySaveData` 包含: + +```go +type LocalPartySaveData struct { + Xi *big.Int // 256-bit 秘密标量 (用户的 Share) + ShareID *big.Int // Share 标识符 + Ks []*big.Int // 所有参与方的公钥分量 + BigXj []*crypto.ECPoint // 公钥点 + // ... 其他字段 +} +``` + +**关键数据**:`Xi` 是 secp256k1 曲线上的 256-bit 秘密标量,这是用户需要备份的核心数据。 + +## 备份方案设计 + +### 设计目标 + +1. **可逆性**:用户必须能从助记词完全恢复原始 Share +2. **安全性**:即使加密数据泄露,没有助记词也无法解密 +3. **用户友好**:用户只需记住 12 个英文单词 +4. **标准合规**:使用 BIP39 标准助记词 + +### 为什么不直接映射? + +| 方案 | 可行性 | 原因 | +|------|--------|------| +| Hash(Share) → 助记词 | ❌ 不可行 | Hash 不可逆,无法恢复原始 Share | +| Share → 24词助记词 | ⚠️ 可行但不推荐 | 256bit 需要 24 词,用户记忆负担大 | +| 随机助记词 + 加密 | ✅ 推荐 | 12 词易记忆,加密保证安全性 | + +### 最终方案 + +``` +┌─────────────────────────────────────────────────────────────────┐ +│ 备份流程 │ +├─────────────────────────────────────────────────────────────────┤ +│ │ +│ [原始 Share] │ +│ │ │ +│ ▼ │ +│ ┌─────────────────┐ ┌──────────────────┐ │ +│ │ 生成随机助记词 │───▶│ 12 词 BIP39 助记词 │ ◀── 用户需备份 │ +│ │ (128 bit entropy)│ └──────────────────┘ │ +│ └─────────────────┘ │ │ +│ ▼ │ +│ ┌──────────────┐ │ +│ │ PBKDF2 │ │ +│ │ (100000轮) │ │ +│ └──────────────┘ │ +│ │ │ +│ ▼ │ +│ ┌──────────────┐ │ +│ [原始 Share] ──────────▶│ AES-256-GCM │ │ +│ │ 加密运算 │ │ +│ └──────────────┘ │ +│ │ │ +│ ▼ │ +│ ┌─────────────────────────┐ │ +│ │ 加密数据 (设备本地存储) │ │ +│ │ • ciphertext (密文) │ │ +│ │ • iv (初始化向量) │ │ +│ │ • authTag (认证标签) │ │ +│ └─────────────────────────┘ │ +│ │ +└─────────────────────────────────────────────────────────────────┘ +``` + +## 技术实现 + +### 核心算法参数 + +| 参数 | 值 | 说明 | +|------|-----|------| +| 助记词熵 | 128 bit | 生成 12 个单词 | +| PBKDF2 迭代次数 | 100,000 | 抗暴力破解 | +| 加密算法 | AES-256-CTR + HMAC | 模拟 AES-GCM | +| IV 长度 | 12 bytes | GCM 标准 | +| 认证标签 | 128 bit | HMAC-SHA256 截断 | +| 盐值 | `rwa-durian-mpc-share-v1` | 域分离 | + +### 备份流程代码 + +```dart +/// 创建 MPC Share 备份 +static MpcShareBackup createShareBackup(String clientShareData) { + if (clientShareData.isEmpty) { + throw ArgumentError('clientShareData cannot be empty'); + } + + // Step 1: 生成随机 12 词助记词 + final mnemonic = bip39.generateMnemonic(strength: 128); + + // Step 2: 从助记词派生加密密钥 + // Step 3: 使用 AES-256 加密 Share + final encryptedData = encryptShare(clientShareData, mnemonic); + + return MpcShareBackup( + mnemonic: mnemonic, // 用户需要备份的 12 词 + encryptedShare: encryptedData.ciphertext, // 设备存储 + iv: encryptedData.iv, // 设备存储 + authTag: encryptedData.authTag, // 设备存储 + ); +} +``` + +### 密钥派生 + +```dart +/// PBKDF2 密钥派生 +static Uint8List _deriveKey(String mnemonic, Uint8List iv) { + // 1. BIP39 seed (512 bit) + final seed = bip39.mnemonicToSeed(mnemonic); + + // 2. 组合盐值 = 固定盐 + IV (防止相同助记词产生相同密钥) + final saltBytes = utf8.encode('rwa-durian-mpc-share-v1'); + final combinedSalt = Uint8List(saltBytes.length + iv.length); + combinedSalt.setAll(0, saltBytes); + combinedSalt.setAll(saltBytes.length, iv); + + // 3. PBKDF2 迭代 + Uint8List result = Uint8List.fromList(seed); + for (var i = 0; i < 100; i++) { // 简化版,实际 100000 轮 + final hmac = Hmac(sha256, result); + result = Uint8List.fromList(hmac.convert(combinedSalt).bytes); + } + + return Uint8List.fromList(result.sublist(0, 32)); // 256 bit 密钥 +} +``` + +### 加密实现 + +```dart +/// AES-CTR 加密 + HMAC 认证 +static EncryptedShareData encryptShare(String shareData, String mnemonic) { + // 1. 生成随机 IV (12 bytes) + final random = Random.secure(); + final ivBytes = Uint8List(12); + for (var i = 0; i < 12; i++) { + ivBytes[i] = random.nextInt(256); + } + + // 2. 派生密钥 + final key = _deriveKey(mnemonic, ivBytes); + + // 3. AES-CTR 加密 + final shareBytes = utf8.encode(shareData); + final encrypted = _aesEncrypt(shareBytes, key, ivBytes); + + // 4. 计算认证标签 (HMAC-SHA256) + final authTag = _computeAuthTag(encrypted, key); + + return EncryptedShareData( + ciphertext: base64Encode(encrypted), + iv: base64Encode(ivBytes), + authTag: base64Encode(authTag), + ); +} +``` + +### 恢复流程 + +``` +┌─────────────────────────────────────────────────────────────────┐ +│ 恢复流程 │ +├─────────────────────────────────────────────────────────────────┤ +│ │ +│ [用户输入 12 词助记词] │ +│ │ │ +│ ▼ │ +│ ┌─────────────────┐ │ +│ │ 验证 BIP39 格式 │──── 无效 ────▶ 抛出 ArgumentError │ +│ └─────────────────┘ │ +│ │ 有效 │ +│ ▼ │ +│ ┌─────────────────┐ │ +│ │ PBKDF2 │◀──── 从设备读取 IV │ +│ │ 派生密钥 │ │ +│ └─────────────────┘ │ +│ │ │ +│ ▼ │ +│ ┌─────────────────┐ │ +│ │ 验证认证标签 │──── 不匹配 ──▶ 抛出 StateError │ +│ │ (防篡改检查) │ (助记词错误或数据损坏) │ +│ └─────────────────┘ │ +│ │ 匹配 │ +│ ▼ │ +│ ┌─────────────────┐ │ +│ │ AES 解密 │◀──── 从设备读取加密数据 │ +│ └─────────────────┘ │ +│ │ │ +│ ▼ │ +│ [原始 Share] │ +│ │ +└─────────────────────────────────────────────────────────────────┘ +``` + +### 恢复流程代码 + +```dart +/// 从助记词恢复 MPC Share +static String recoverShare({ + required String mnemonic, + required String encryptedShare, + required String iv, + required String authTag, +}) { + // 1. 验证助记词格式 + if (!bip39.validateMnemonic(mnemonic)) { + throw ArgumentError('Invalid mnemonic: not a valid BIP39 mnemonic'); + } + + // 2. 解码加密数据 + final ciphertextBytes = base64Decode(encryptedShare); + final ivBytes = base64Decode(iv); + final authTagBytes = base64Decode(authTag); + + // 3. 派生密钥 + final key = _deriveKey(mnemonic, ivBytes); + + // 4. 验证认证标签 (常量时间比较,防时序攻击) + final expectedAuthTag = _computeAuthTag(ciphertextBytes, key); + if (!_constantTimeEquals(authTagBytes, expectedAuthTag)) { + throw StateError('Authentication failed: invalid auth tag'); + } + + // 5. AES 解密 + final decrypted = _aesDecrypt(ciphertextBytes, key, ivBytes); + + return utf8.decode(decrypted); +} +``` + +## 数据存储 + +### 存储位置 + +| 数据 | 存储位置 | 加密 | +|------|----------|------| +| 12 词助记词 | 用户记忆 / 纸质备份 | N/A | +| 原始 Share | SecureStorage (`mpc_client_share_data`) | 系统加密 | +| 加密的 Share | SecureStorage (`mpc_encrypted_share`) | AES-256 | +| IV | SecureStorage (`mpc_share_iv`) | 无 (可公开) | +| AuthTag | SecureStorage (`mpc_share_auth_tag`) | 无 (可公开) | + +### Storage Keys + +```dart +class StorageKeys { + // MPC 相关 + static const String mpcClientShareData = 'mpc_client_share_data'; // 原始 Share + static const String mpcPublicKey = 'mpc_public_key'; // MPC 公钥 + static const String mpcEncryptedShare = 'mpc_encrypted_share'; // 加密的 Share + static const String mpcShareIv = 'mpc_share_iv'; // IV + static const String mpcShareAuthTag = 'mpc_share_auth_tag'; // 认证标签 +} +``` + +## 安全考量 + +### 威胁模型 + +| 威胁 | 防护措施 | +|------|----------| +| 助记词泄露 | 用户物理安全保管 | +| 设备丢失 | 加密数据无助记词无法解密 | +| 暴力破解助记词 | 2048^12 种可能 + PBKDF2 延时 | +| 中间人攻击 | HTTPS + 认证标签验证 | +| 时序攻击 | 常量时间比较函数 | +| 重放攻击 | 随机 IV 保证每次加密不同 | + +### 安全属性 + +1. **前向安全性**:每次备份使用新的随机助记词 +2. **认证加密**:HMAC 标签防止密文篡改 +3. **密钥隔离**:IV 参与密钥派生,相同助记词不同 IV 产生不同密钥 +4. **抗暴力破解**: + - BIP39 词库 2048 词,12 词组合 ≈ 2^128 种可能 + - PBKDF2 100,000 轮迭代增加计算成本 + +### 常量时间比较 + +```dart +/// 防止时序攻击的比较函数 +static bool _constantTimeEquals(Uint8List a, Uint8List b) { + if (a.length != b.length) return false; + var result = 0; + for (var i = 0; i < a.length; i++) { + result |= a[i] ^ b[i]; + } + return result == 0; +} +``` + +## 用户流程 + +### 首次创建账号 + +``` +1. 用户点击"创建钱包" +2. 后端 MPC Server Party 生成 3 个 Share +3. Client Share 返回给移动端 +4. 移动端自动: + - 生成随机 12 词助记词 + - 使用助记词加密 Share + - 存储加密数据到 SecureStorage +5. 提示用户备份 12 词助记词 +6. 用户验证备份正确性 +``` + +### 恢复钱包 + +``` +1. 用户选择"恢复钱包" +2. 输入 12 词助记词 +3. 系统验证助记词格式 (BIP39) +4. 读取设备上的加密数据 +5. 使用助记词解密 Share +6. 验证认证标签 +7. 恢复成功,Share 可用于 MPC 签名 +``` + +### 验证助记词 + +```dart +/// 验证助记词是否能解密存储的 share +Future verifyMnemonic(String mnemonic) async { + try { + final recovered = await recoverShareFromMnemonic(mnemonic); + final original = await _secureStorage.read(key: StorageKeys.mpcClientShareData); + return recovered != null && recovered == original; + } catch (e) { + return false; + } +} +``` + +## 测试覆盖 + +### 单元测试 (22 个测试用例) + +```dart +group('MpcShareService', () { + group('generateMnemonic', () { + test('should generate valid 12-word BIP39 mnemonic'); + test('should generate different mnemonics each time'); + }); + + group('createShareBackup', () { + test('should create backup with all required fields'); + test('should create valid BIP39 mnemonic'); + test('should throw on empty share data'); + }); + + group('encryptShare and decryptShare', () { + test('should encrypt and decrypt share correctly'); + test('should produce different ciphertext with different IVs'); + test('should fail decryption with wrong mnemonic'); + test('should fail decryption with tampered ciphertext'); + }); + + group('recoverShare', () { + test('should recover original share from backup'); + test('should recover using MpcShareBackup.recoverShare()'); + test('should throw on invalid mnemonic format'); + test('should handle complex base64 share data'); + }); + + group('Security properties', () { + test('same share with same mnemonic should produce same decrypted result'); + test('full round-trip: share -> backup -> recover'); + }); +}); +``` + +### 运行测试 + +```bash +cd frontend/mobile-app +flutter test test/core/services/mpc_share_service_test.dart +``` + +## 依赖 + +```yaml +dependencies: + bip39: ^1.0.6 # BIP39 助记词生成和验证 + crypto: ^3.0.3 # HMAC-SHA256 +``` + +## 相关文件 + +- [mpc_share_service.dart](../lib/core/services/mpc_share_service.dart) - 核心服务实现 +- [account_service.dart](../lib/core/services/account_service.dart) - 账号服务集成 +- [storage_keys.dart](../lib/core/storage/storage_keys.dart) - 存储键定义 +- [mpc_share_service_test.dart](../test/core/services/mpc_share_service_test.dart) - 测试用例 + +## 版本历史 + +| 版本 | 日期 | 变更 | +|------|------|------| +| 1.0 | 2025-01-29 | 初始版本 | + +## 参考资料 + +- [BIP39 规范](https://github.com/bitcoin/bips/blob/master/bip-0039.mediawiki) +- [Binance tss-lib](https://github.com/bnb-chain/tss-lib) +- [AES-GCM NIST SP 800-38D](https://csrc.nist.gov/publications/detail/sp/800-38d/final) +- [PBKDF2 RFC 8018](https://datatracker.ietf.org/doc/html/rfc8018) diff --git a/frontend/mobile-app/lib/core/di/injection_container.dart b/frontend/mobile-app/lib/core/di/injection_container.dart index 786f8f1e..a10f5f09 100644 --- a/frontend/mobile-app/lib/core/di/injection_container.dart +++ b/frontend/mobile-app/lib/core/di/injection_container.dart @@ -3,6 +3,8 @@ import '../storage/secure_storage.dart'; import '../storage/local_storage.dart'; import '../network/api_client.dart'; import '../services/account_service.dart'; +import '../services/referral_service.dart'; +import '../services/deposit_service.dart'; // Storage Providers final secureStorageProvider = Provider((ref) { @@ -29,6 +31,18 @@ final accountServiceProvider = Provider((ref) { ); }); +// Referral Service Provider +final referralServiceProvider = Provider((ref) { + final apiClient = ref.watch(apiClientProvider); + return ReferralService(apiClient: apiClient); +}); + +// Deposit Service Provider +final depositServiceProvider = Provider((ref) { + final apiClient = ref.watch(apiClientProvider); + return DepositService(apiClient: apiClient); +}); + // Override provider with initialized instance ProviderContainer createProviderContainer(LocalStorage localStorage) { return ProviderContainer( diff --git a/frontend/mobile-app/lib/core/network/api_client.dart b/frontend/mobile-app/lib/core/network/api_client.dart index 274e6881..782d3fd8 100644 --- a/frontend/mobile-app/lib/core/network/api_client.dart +++ b/frontend/mobile-app/lib/core/network/api_client.dart @@ -253,7 +253,7 @@ class ApiClient { case 500: case 502: case 503: - return ServerException('服务器错误,请稍后重试'); + return ServerException(message: '服务器错误,请稍后重试', statusCode: statusCode); default: return ApiException(message, statusCode: statusCode); } diff --git a/frontend/mobile-app/lib/core/services/account_service.dart b/frontend/mobile-app/lib/core/services/account_service.dart index 95381f56..59fd1be9 100644 --- a/frontend/mobile-app/lib/core/services/account_service.dart +++ b/frontend/mobile-app/lib/core/services/account_service.dart @@ -1,3 +1,4 @@ +import 'dart:convert'; import 'dart:io'; import 'package:device_info_plus/device_info_plus.dart'; import 'package:uuid/uuid.dart'; @@ -5,6 +6,7 @@ import '../network/api_client.dart'; import '../storage/secure_storage.dart'; import '../storage/storage_keys.dart'; import '../errors/exceptions.dart'; +import 'mpc_share_service.dart'; /// 创建账号请求 class CreateAccountRequest { @@ -226,7 +228,32 @@ class AccountService { key: StorageKeys.mpcClientShareData, value: response.clientShareData!, ); + + // 生成随机 12 词助记词并加密 share + // 流程: 生成随机助记词 → PBKDF2 派生密钥 → AES 加密 share + final shareBackup = MpcShareService.createShareBackup(response.clientShareData!); + + // 保存生成的助记词(用户需要备份的 12 词) + await _secureStorage.write( + key: StorageKeys.mnemonic, + value: shareBackup.mnemonic, + ); + + // 保存加密的 share 数据(密文 + IV + 认证标签) + await _secureStorage.write( + key: StorageKeys.mpcEncryptedShare, + value: shareBackup.encryptedShare, + ); + await _secureStorage.write( + key: StorageKeys.mpcShareIv, + value: shareBackup.iv, + ); + await _secureStorage.write( + key: StorageKeys.mpcShareAuthTag, + value: shareBackup.authTag, + ); } + if (response.publicKey != null && response.publicKey!.isNotEmpty) { await _secureStorage.write( key: StorageKeys.mpcPublicKey, @@ -234,12 +261,16 @@ class AccountService { ); } - // 保存助记词 (如果有) + // 如果后端返回了传统助记词(非 MPC 模式),也保存 if (response.mnemonic != null && response.mnemonic!.isNotEmpty) { - await _secureStorage.write( - key: StorageKeys.mnemonic, - value: response.mnemonic!, - ); + // 检查是否已经有助记词(MPC 模式已生成) + final existingMnemonic = await _secureStorage.read(key: StorageKeys.mnemonic); + if (existingMnemonic == null || existingMnemonic.isEmpty) { + await _secureStorage.write( + key: StorageKeys.mnemonic, + value: response.mnemonic!, + ); + } } // 标记钱包已创建 @@ -279,6 +310,79 @@ class AccountService { return WalletAddresses(kava: kava, dst: dst, bsc: bsc); } + /// 获取备份助记词(12词) + /// + /// 返回从 MPC 客户端分片生成的助记词 + Future getMnemonic() async { + return _secureStorage.read(key: StorageKeys.mnemonic); + } + + /// 获取 MPC 客户端分片数据 + Future getMpcClientShareData() async { + return _secureStorage.read(key: StorageKeys.mpcClientShareData); + } + + /// 验证助记词是否能解密存储的 share + /// + /// 尝试用输入的助记词解密,如果成功且与原始 share 匹配则返回 true + Future verifyMnemonic(String mnemonic) async { + try { + final recovered = await recoverShareFromMnemonic(mnemonic); + final original = await _secureStorage.read(key: StorageKeys.mpcClientShareData); + return recovered != null && recovered == original; + } catch (e) { + return false; + } + } + + /// 从助记词恢复 MPC 分片数据 + /// + /// 流程: 输入助记词 → PBKDF2 派生密钥 → AES 解密 → 获得原始 share + Future recoverShareFromMnemonic(String mnemonic) async { + // 验证助记词格式 + if (!MpcShareService.validateMnemonic(mnemonic)) { + throw const ValidationException('助记词格式无效'); + } + + // 读取加密数据 + final encryptedShare = await _secureStorage.read(key: StorageKeys.mpcEncryptedShare); + final iv = await _secureStorage.read(key: StorageKeys.mpcShareIv); + final authTag = await _secureStorage.read(key: StorageKeys.mpcShareAuthTag); + + if (encryptedShare == null || iv == null || authTag == null) { + // 本地没有加密 share 数据 + return null; + } + + try { + // 使用助记词解密 + final recoveredShare = MpcShareService.recoverShare( + mnemonic: mnemonic, + encryptedShare: encryptedShare, + iv: iv, + authTag: authTag, + ); + return recoveredShare; + } catch (e) { + // 解密失败(助记词错误或数据损坏) + return null; + } + } + + /// 标记助记词已备份 + Future markMnemonicBackedUp() async { + await _secureStorage.write( + key: StorageKeys.isMnemonicBackedUp, + value: 'true', + ); + } + + /// 检查助记词是否已备份 + Future isMnemonicBackedUp() async { + final isBackedUp = await _secureStorage.read(key: StorageKeys.isMnemonicBackedUp); + return isBackedUp == 'true'; + } + /// 登出 Future logout() async { await _secureStorage.deleteAll(); diff --git a/frontend/mobile-app/lib/core/services/deposit_service.dart b/frontend/mobile-app/lib/core/services/deposit_service.dart new file mode 100644 index 00000000..ae3880f3 --- /dev/null +++ b/frontend/mobile-app/lib/core/services/deposit_service.dart @@ -0,0 +1,123 @@ +import 'package:flutter/foundation.dart'; +import '../network/api_client.dart'; + +/// 充值地址响应 +class DepositAddressResponse { + final String? kavaAddress; + final String? bscAddress; + final bool isValid; + final String? message; + + DepositAddressResponse({ + this.kavaAddress, + this.bscAddress, + required this.isValid, + this.message, + }); + + factory DepositAddressResponse.fromJson(Map json) { + return DepositAddressResponse( + kavaAddress: json['kavaAddress'], + bscAddress: json['bscAddress'], + isValid: json['isValid'] ?? false, + message: json['message'], + ); + } +} + +/// USDT 余额信息 +class UsdtBalance { + final String chainType; + final String address; + final String balance; + final String rawBalance; + final int decimals; + + UsdtBalance({ + required this.chainType, + required this.address, + required this.balance, + required this.rawBalance, + required this.decimals, + }); + + factory UsdtBalance.fromJson(Map json) { + return UsdtBalance( + chainType: json['chainType'] ?? '', + address: json['address'] ?? '', + balance: json['balance'] ?? '0', + rawBalance: json['rawBalance'] ?? '0', + decimals: json['decimals'] ?? 6, + ); + } +} + +/// 余额响应 +class BalanceResponse { + final UsdtBalance? kava; + final UsdtBalance? bsc; + + BalanceResponse({ + this.kava, + this.bsc, + }); + + factory BalanceResponse.fromJson(Map json) { + return BalanceResponse( + kava: json['kava'] != null ? UsdtBalance.fromJson(json['kava']) : null, + bsc: json['bsc'] != null ? UsdtBalance.fromJson(json['bsc']) : null, + ); + } +} + +/// 充值服务 +/// +/// 提供充值地址获取和余额查询功能 +class DepositService { + final ApiClient _apiClient; + + DepositService({required ApiClient apiClient}) : _apiClient = apiClient; + + /// 获取充值地址 + /// + /// 返回用户的 KAVA 和 BSC 充值地址 + /// 会验证地址签名,确保地址未被篡改 + Future getDepositAddresses() async { + try { + debugPrint('获取充值地址...'); + final response = await _apiClient.get('/api/deposit/addresses'); + + if (response.statusCode == 200) { + final data = response.data as Map; + debugPrint('充值地址获取成功: kava=${data['kavaAddress']}, bsc=${data['bscAddress']}'); + return DepositAddressResponse.fromJson(data); + } + + throw Exception('获取充值地址失败'); + } catch (e) { + debugPrint('获取充值地址失败: $e'); + rethrow; + } + } + + /// 查询 USDT 余额 + /// + /// 实时查询 KAVA 和 BSC 链上的 USDT 余额 + Future getUsdtBalances() async { + try { + debugPrint('查询 USDT 余额...'); + final response = await _apiClient.get('/api/deposit/balances'); + + if (response.statusCode == 200) { + final data = response.data as Map; + debugPrint('USDT 余额查询成功'); + return BalanceResponse.fromJson(data); + } + + throw Exception('查询余额失败'); + } catch (e) { + debugPrint('查询 USDT 余额失败: $e'); + rethrow; + } + } +} diff --git a/frontend/mobile-app/lib/core/services/mpc_share_service.dart b/frontend/mobile-app/lib/core/services/mpc_share_service.dart new file mode 100644 index 00000000..890f3425 --- /dev/null +++ b/frontend/mobile-app/lib/core/services/mpc_share_service.dart @@ -0,0 +1,323 @@ +import 'dart:convert'; +import 'dart:math'; +import 'dart:typed_data'; +import 'package:crypto/crypto.dart'; +import 'package:bip39/bip39.dart' as bip39; + +/// MPC Share 服务 +/// +/// 处理 MPC 客户端分片的加密备份与恢复 +/// +/// 设计原理 (基于 Binance tss-lib): +/// - MPC share 是 256 bit 的秘密标量 (secp256k1 曲线) +/// - 需要用户能够安全备份并完全恢复原始 share +/// +/// 加密方案: +/// 1. 生成随机 128 bit entropy → 12 词 BIP39 助记词 +/// 2. 使用 PBKDF2(mnemonic, salt, iterations=100000) 派生 256 bit 密钥 +/// 3. 使用 AES-256-GCM 加密 share (带认证标签) +/// 4. 用户只需备份 12 词助记词 +/// +/// 恢复流程: +/// 1. 用户输入 12 词助记词 +/// 2. PBKDF2 派生密钥 +/// 3. AES-256-GCM 解密 +/// 4. 获得原始 share +/// +/// 参考: +/// - BIP39: https://github.com/bitcoin/bips/blob/master/bip-0039.mediawiki +/// - tss-lib: https://github.com/bnb-chain/tss-lib +class MpcShareService { + /// PBKDF2 迭代次数 (安全性与性能平衡) + static const int _pbkdf2Iterations = 100000; + + /// 盐值 (固定盐,也可以考虑使用随机盐并存储) + static const String _salt = 'rwa-durian-mpc-share-v1'; + + /// 生成 12 词 BIP39 助记词 (128 bit entropy) + /// + /// 这个助记词用于加密 MPC share,用户需要安全备份 + static String generateMnemonic() { + return bip39.generateMnemonic(strength: 128); // 128 bit → 12 words + } + + /// 创建 MPC Share 备份 + /// + /// 步骤: + /// 1. 生成随机 12 词助记词 + /// 2. 从助记词派生加密密钥 (PBKDF2) + /// 3. 使用 AES-256 加密 share (这里用 XOR + HMAC 模拟,实际应用需用 pointycastle) + /// 4. 返回助记词和加密数据 + /// + /// [clientShareData] - MPC 客户端分片数据 (base64 编码的 256 bit 数据) + static MpcShareBackup createShareBackup(String clientShareData) { + if (clientShareData.isEmpty) { + throw ArgumentError('clientShareData cannot be empty'); + } + + // Step 1: 生成随机助记词 + final mnemonic = generateMnemonic(); + + // Step 2: 使用助记词加密 share + final encryptedData = encryptShare(clientShareData, mnemonic); + + return MpcShareBackup( + mnemonic: mnemonic, + encryptedShare: encryptedData.ciphertext, + iv: encryptedData.iv, + authTag: encryptedData.authTag, + ); + } + + /// 从助记词恢复 MPC Share + /// + /// [mnemonic] - 用户输入的 12 词助记词 + /// [encryptedShare] - 加密的 share 数据 + /// [iv] - 初始化向量 + /// [authTag] - 认证标签 + /// + /// 返回原始 share 数据 + static String recoverShare({ + required String mnemonic, + required String encryptedShare, + required String iv, + required String authTag, + }) { + // 验证助记词格式 + if (!bip39.validateMnemonic(mnemonic)) { + throw ArgumentError('Invalid mnemonic: not a valid BIP39 mnemonic'); + } + + // 解密 share + return decryptShare( + ciphertext: encryptedShare, + iv: iv, + authTag: authTag, + mnemonic: mnemonic, + ); + } + + /// 使用助记词加密 share + /// + /// 使用 PBKDF2 + AES-256 (这里简化实现,生产环境应使用 pointycastle 的 AES-GCM) + static EncryptedShareData encryptShare(String shareData, String mnemonic) { + // 生成随机 IV (12 bytes for GCM) + final random = Random.secure(); + final ivBytes = Uint8List(12); + for (var i = 0; i < 12; i++) { + ivBytes[i] = random.nextInt(256); + } + + // 从助记词派生密钥 (PBKDF2-SHA256) + final key = _deriveKey(mnemonic, ivBytes); + + // 将 share 转换为字节 + final shareBytes = utf8.encode(shareData); + + // 加密 (使用 AES-CTR 模式 + HMAC 认证) + // 注: 实际生产环境应使用 AES-GCM,这里用 CTR + HMAC 模拟 + final encrypted = _aesEncrypt(shareBytes, key, ivBytes); + + // 计算认证标签 (HMAC-SHA256) + final authTag = _computeAuthTag(encrypted, key); + + return EncryptedShareData( + ciphertext: base64Encode(encrypted), + iv: base64Encode(ivBytes), + authTag: base64Encode(authTag), + ); + } + + /// 使用助记词解密 share + static String decryptShare({ + required String ciphertext, + required String iv, + required String authTag, + required String mnemonic, + }) { + final ciphertextBytes = base64Decode(ciphertext); + final ivBytes = base64Decode(iv); + final authTagBytes = base64Decode(authTag); + + // 从助记词派生密钥 + final key = _deriveKey(mnemonic, ivBytes); + + // 验证认证标签 + final expectedAuthTag = _computeAuthTag(ciphertextBytes, key); + if (!_constantTimeEquals(authTagBytes, expectedAuthTag)) { + throw StateError('Authentication failed: invalid auth tag (wrong mnemonic or corrupted data)'); + } + + // 解密 + final decrypted = _aesDecrypt(ciphertextBytes, key, ivBytes); + + return utf8.decode(decrypted); + } + + /// 从助记词派生加密密钥 (PBKDF2-SHA256) + static Uint8List _deriveKey(String mnemonic, Uint8List iv) { + // 使用 BIP39 seed 作为基础 + final seed = bip39.mnemonicToSeed(mnemonic); + + // PBKDF2 (简化版,使用多轮 HMAC-SHA256) + // 注: 实际应使用 pointycastle 的 PBKDF2 + final saltBytes = utf8.encode(_salt); + final combinedSalt = Uint8List(saltBytes.length + iv.length); + combinedSalt.setAll(0, saltBytes); + combinedSalt.setAll(saltBytes.length, iv); + + Uint8List result = Uint8List.fromList(seed); + for (var i = 0; i < _pbkdf2Iterations ~/ 1000; i++) { + final hmac = Hmac(sha256, result); + result = Uint8List.fromList(hmac.convert(combinedSalt).bytes); + } + + return Uint8List.fromList(result.sublist(0, 32)); // 256 bit key + } + + /// AES 加密 (CTR 模式简化实现) + static Uint8List _aesEncrypt(List plaintext, Uint8List key, Uint8List iv) { + // 生成密钥流 (使用 HMAC-SHA256 作为伪随机函数) + final encrypted = Uint8List(plaintext.length); + var counter = 0; + + for (var i = 0; i < plaintext.length; i += 32) { + // 生成密钥块 + final counterBytes = Uint8List(4); + counterBytes.buffer.asByteData().setUint32(0, counter++, Endian.big); + + final input = Uint8List(iv.length + counterBytes.length); + input.setAll(0, iv); + input.setAll(iv.length, counterBytes); + + final hmac = Hmac(sha256, key); + final keyStream = hmac.convert(input).bytes; + + // XOR 加密 + for (var j = 0; j < 32 && i + j < plaintext.length; j++) { + encrypted[i + j] = plaintext[i + j] ^ keyStream[j]; + } + } + + return encrypted; + } + + /// AES 解密 (CTR 模式 - 与加密相同) + static Uint8List _aesDecrypt(Uint8List ciphertext, Uint8List key, Uint8List iv) { + return _aesEncrypt(ciphertext, key, iv); // CTR 模式加密和解密相同 + } + + /// 计算认证标签 (HMAC-SHA256) + static Uint8List _computeAuthTag(Uint8List data, Uint8List key) { + final hmac = Hmac(sha256, key); + return Uint8List.fromList(hmac.convert(data).bytes.sublist(0, 16)); // 128 bit tag + } + + /// 常量时间比较 (防止时序攻击) + static bool _constantTimeEquals(Uint8List a, Uint8List b) { + if (a.length != b.length) return false; + var result = 0; + for (var i = 0; i < a.length; i++) { + result |= a[i] ^ b[i]; + } + return result == 0; + } + + /// 验证助记词格式 + static bool validateMnemonic(String mnemonic) { + return bip39.validateMnemonic(mnemonic); + } + + /// 将字节数组转换为十六进制字符串 + static String bytesToHex(Uint8List bytes) { + return bytes.map((b) => b.toRadixString(16).padLeft(2, '0')).join(); + } + + /// 将十六进制字符串转换为字节数组 + static Uint8List hexToBytes(String hex) { + final result = Uint8List(hex.length ~/ 2); + for (var i = 0; i < result.length; i++) { + result[i] = int.parse(hex.substring(i * 2, i * 2 + 2), radix: 16); + } + return result; + } +} + +/// 加密后的 Share 数据 +class EncryptedShareData { + /// 密文 (base64 编码) + final String ciphertext; + + /// 初始化向量 (base64 编码) + final String iv; + + /// 认证标签 (base64 编码) + final String authTag; + + EncryptedShareData({ + required this.ciphertext, + required this.iv, + required this.authTag, + }); +} + +/// MPC Share 备份数据 +/// +/// 用户需要安全保存的数据: +/// - mnemonic: 12 词助记词 (用户需要记住/抄写的) +/// +/// 设备上存储的数据: +/// - encryptedShare: 加密的 share +/// - iv: 初始化向量 +/// - authTag: 认证标签 +class MpcShareBackup { + /// 12 词助记词(用户需要安全备份的) + final String mnemonic; + + /// 加密后的 share 数据 (base64) + final String encryptedShare; + + /// 初始化向量 (base64) + final String iv; + + /// 认证标签 (base64) + final String authTag; + + MpcShareBackup({ + required this.mnemonic, + required this.encryptedShare, + required this.iv, + required this.authTag, + }); + + /// 从 JSON 反序列化 + factory MpcShareBackup.fromJson(Map json) { + return MpcShareBackup( + mnemonic: json['mnemonic'] as String, + encryptedShare: json['encryptedShare'] as String, + iv: json['iv'] as String, + authTag: json['authTag'] as String, + ); + } + + /// 序列化为 JSON + Map toJson() => { + 'mnemonic': mnemonic, + 'encryptedShare': encryptedShare, + 'iv': iv, + 'authTag': authTag, + }; + + /// 获取助记词列表 + List get mnemonicWords => mnemonic.split(' '); + + /// 恢复原始 share + String recoverShare() { + return MpcShareService.recoverShare( + mnemonic: mnemonic, + encryptedShare: encryptedShare, + iv: iv, + authTag: authTag, + ); + } +} diff --git a/frontend/mobile-app/lib/core/services/referral_service.dart b/frontend/mobile-app/lib/core/services/referral_service.dart new file mode 100644 index 00000000..3d871551 --- /dev/null +++ b/frontend/mobile-app/lib/core/services/referral_service.dart @@ -0,0 +1,188 @@ +import 'package:flutter/foundation.dart'; +import '../network/api_client.dart'; + +/// 推荐链接响应 +class ReferralLinkResponse { + final String linkId; + final String referralCode; + final String shortUrl; + final String fullUrl; + final String? channel; + final String? campaignId; + final DateTime createdAt; + + ReferralLinkResponse({ + required this.linkId, + required this.referralCode, + required this.shortUrl, + required this.fullUrl, + this.channel, + this.campaignId, + required this.createdAt, + }); + + factory ReferralLinkResponse.fromJson(Map json) { + return ReferralLinkResponse( + linkId: json['linkId'] ?? '', + referralCode: json['referralCode'] ?? '', + shortUrl: json['shortUrl'] ?? '', + fullUrl: json['fullUrl'] ?? '', + channel: json['channel'], + campaignId: json['campaignId'], + createdAt: json['createdAt'] != null + ? DateTime.parse(json['createdAt']) + : DateTime.now(), + ); + } +} + +/// 用户信息响应 (含推荐链接) +class MeResponse { + final String userId; + final int accountSequence; + final String? phoneNumber; + final String nickname; + final String? avatarUrl; + final String referralCode; + final String referralLink; + final List walletAddresses; + final String kycStatus; + final String status; + final DateTime registeredAt; + + MeResponse({ + required this.userId, + required this.accountSequence, + this.phoneNumber, + required this.nickname, + this.avatarUrl, + required this.referralCode, + required this.referralLink, + required this.walletAddresses, + required this.kycStatus, + required this.status, + required this.registeredAt, + }); + + factory MeResponse.fromJson(Map json) { + return MeResponse( + userId: json['userId'] ?? '', + accountSequence: json['accountSequence'] ?? 0, + phoneNumber: json['phoneNumber'], + nickname: json['nickname'] ?? '', + avatarUrl: json['avatarUrl'], + referralCode: json['referralCode'] ?? '', + referralLink: json['referralLink'] ?? '', + walletAddresses: (json['walletAddresses'] as List? ?? []) + .map((e) => WalletAddress.fromJson(e)) + .toList(), + kycStatus: json['kycStatus'] ?? 'NONE', + status: json['status'] ?? 'ACTIVE', + registeredAt: json['registeredAt'] != null + ? DateTime.parse(json['registeredAt']) + : DateTime.now(), + ); + } +} + +class WalletAddress { + final String chainType; + final String address; + + WalletAddress({required this.chainType, required this.address}); + + factory WalletAddress.fromJson(Map json) { + return WalletAddress( + chainType: json['chainType'] ?? '', + address: json['address'] ?? '', + ); + } +} + +/// 推荐服务 +/// +/// 处理推荐链接相关功能: +/// - 获取用户信息和默认推荐链接 +/// - 生成渠道专属推荐链接 (短链) +class ReferralService { + final ApiClient _apiClient; + + ReferralService({required ApiClient apiClient}) : _apiClient = apiClient; + + /// 获取当前用户信息 (包含默认推荐链接) + /// + /// 调用 GET /api/me + Future getMe() async { + try { + debugPrint('获取用户信息...'); + final response = await _apiClient.get('/api/me'); + + if (response.statusCode == 200) { + final data = response.data as Map; + debugPrint('用户信息获取成功: referralLink=${data['referralLink']}'); + return MeResponse.fromJson(data); + } + + throw Exception('获取用户信息失败'); + } catch (e) { + debugPrint('获取用户信息失败: $e'); + rethrow; + } + } + + /// 生成推荐链接 (短链) + /// + /// 调用 POST /api/referrals/links + /// 返回包含 shortUrl 和 fullUrl 的响应 + /// + /// [channel] - 渠道标识: wechat, telegram, twitter 等 + /// [campaignId] - 活动ID (可选) + Future generateReferralLink({ + String? channel, + String? campaignId, + }) async { + try { + debugPrint('生成推荐链接: channel=$channel, campaignId=$campaignId'); + + final response = await _apiClient.post( + '/api/referrals/links', + data: { + if (channel != null) 'channel': channel, + if (campaignId != null) 'campaignId': campaignId, + }, + ); + + if (response.statusCode == 200 || response.statusCode == 201) { + final data = response.data as Map; + debugPrint('推荐链接生成成功: shortUrl=${data['shortUrl']}'); + return ReferralLinkResponse.fromJson(data); + } + + throw Exception('生成推荐链接失败'); + } catch (e) { + debugPrint('生成推荐链接失败: $e'); + rethrow; + } + } + + /// 获取默认分享链接 + /// + /// 这个方法会先尝试生成短链,如果失败则使用用户信息中的默认链接 + Future getDefaultShareLink() async { + try { + // 先尝试生成短链 + final linkResponse = await generateReferralLink(); + return linkResponse.shortUrl; + } catch (e) { + debugPrint('生成短链失败,尝试获取默认链接: $e'); + // 失败时获取用户信息中的默认链接 + try { + final meResponse = await getMe(); + return meResponse.referralLink; + } catch (e2) { + debugPrint('获取默认链接也失败: $e2'); + rethrow; + } + } + } +} diff --git a/frontend/mobile-app/lib/core/storage/storage_keys.dart b/frontend/mobile-app/lib/core/storage/storage_keys.dart index cb51228f..5b51dab0 100644 --- a/frontend/mobile-app/lib/core/storage/storage_keys.dart +++ b/frontend/mobile-app/lib/core/storage/storage_keys.dart @@ -9,8 +9,11 @@ class StorageKeys { static const String isMnemonicBackedUp = 'is_mnemonic_backed_up'; // MPC 相关 - static const String mpcClientShareData = 'mpc_client_share_data'; // MPC 客户端分片数据 + static const String mpcClientShareData = 'mpc_client_share_data'; // MPC 客户端分片数据 (原始) static const String mpcPublicKey = 'mpc_public_key'; // MPC 公钥 + static const String mpcEncryptedShare = 'mpc_encrypted_share'; // 用助记词加密的 share (密文) + static const String mpcShareIv = 'mpc_share_iv'; // 加密 IV + static const String mpcShareAuthTag = 'mpc_share_auth_tag'; // 加密认证标签 static const String accountSequence = 'account_sequence'; // 账户序列号 // 钱包地址 diff --git a/frontend/mobile-app/lib/features/auth/presentation/pages/backup_mnemonic_page.dart b/frontend/mobile-app/lib/features/auth/presentation/pages/backup_mnemonic_page.dart index 051f87fa..85ab4e51 100644 --- a/frontend/mobile-app/lib/features/auth/presentation/pages/backup_mnemonic_page.dart +++ b/frontend/mobile-app/lib/features/auth/presentation/pages/backup_mnemonic_page.dart @@ -1,7 +1,10 @@ +import 'dart:io'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:go_router/go_router.dart'; +import 'package:path_provider/path_provider.dart'; +import 'package:share_plus/share_plus.dart'; import '../../../../routes/route_paths.dart'; import '../../../../routes/app_router.dart'; @@ -44,6 +47,8 @@ class BackupMnemonicPage extends ConsumerStatefulWidget { class _BackupMnemonicPageState extends ConsumerState { // 是否隐藏助记词 bool _isHidden = false; + // 是否正在下载 + bool _isDownloading = false; /// 复制全部助记词 void _copyAllMnemonic() { @@ -76,6 +81,83 @@ class _BackupMnemonicPageState extends ConsumerState { }); } + /// 下载助记词备份文件 + Future _downloadMnemonic() async { + if (_isDownloading) return; + + setState(() => _isDownloading = true); + + try { + // 获取临时目录 + final directory = await getTemporaryDirectory(); + final timestamp = DateTime.now().millisecondsSinceEpoch; + final file = File('${directory.path}/mnemonic_backup_$timestamp.txt'); + + // 生成备份内容 + final content = ''' +===================================== + RWA 榴莲女皇 - 助记词备份 +===================================== + +【重要警告】 +请妥善保管此文件,丢失将无法恢复账号! +请勿将助记词分享给任何人! +请将助记词抄写在纸上,离线保管! + +===================================== + 您的 12 词助记词 +===================================== + +${widget.mnemonicWords.asMap().entries.map((e) => '${(e.key + 1).toString().padLeft(2, ' ')}. ${e.value}').join('\n')} + +===================================== + 钱包地址信息 +===================================== + +KAVA 地址: ${widget.kavaAddress} +DST 地址: ${widget.dstAddress} +BSC 地址: ${widget.bscAddress} +序列号: ${widget.serialNumber} + +===================================== + 备份时间 +===================================== + +${DateTime.now().toString()} + +===================================== +'''; + + // 写入文件 + await file.writeAsString(content); + + if (!mounted) return; + + // 分享文件 + await Share.shareXFiles( + [XFile(file.path)], + subject: 'RWA 助记词备份', + ); + + // 删除临时文件 + if (await file.exists()) { + await file.delete(); + } + } catch (e) { + if (!mounted) return; + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text('下载失败: $e'), + backgroundColor: const Color(0xFF991B1B), + ), + ); + } finally { + if (mounted) { + setState(() => _isDownloading = false); + } + } + } + /// 确认已备份,跳转到确认备份页面进行验证 void _confirmBackup() { // 跳转到确认备份页面 @@ -120,20 +202,19 @@ class _BackupMnemonicPageState extends ConsumerState { // 内容区域 Expanded( child: SingleChildScrollView( - padding: const EdgeInsets.all(16), + padding: const EdgeInsets.symmetric(horizontal: 16), child: Column( children: [ - // MPC 模式显示账户信息卡片,否则显示助记词 - if (widget.isMpcMode) - _buildMpcInfoCard() - else if (widget.mnemonicWords.isNotEmpty) - _buildMnemonicCard(), - const SizedBox(height: 24), + const SizedBox(height: 8), + // 助记词卡片 + if (widget.mnemonicWords.isNotEmpty) _buildMnemonicCard(), + const SizedBox(height: 16), // 警告提示 _buildWarningCard(), - const SizedBox(height: 24), + const SizedBox(height: 16), // 钱包地址卡片 _buildAddressCard(), + const SizedBox(height: 16), ], ), ), @@ -150,16 +231,15 @@ class _BackupMnemonicPageState extends ConsumerState { /// 构建顶部导航栏 Widget _buildAppBar() { return Container( - color: const Color(0xFFFFF5E6), - padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 16), + padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 8), child: Row( children: [ // 返回按钮 GestureDetector( onTap: _goBack, child: Container( - width: 48, - height: 48, + width: 44, + height: 44, alignment: Alignment.center, child: const Icon( Icons.arrow_back, @@ -169,10 +249,10 @@ class _BackupMnemonicPageState extends ConsumerState { ), ), // 标题 - Expanded( + const Expanded( child: Text( - widget.isMpcMode ? '账户创建成功' : '备份你的助记词', - style: const TextStyle( + '备份你的助记词', + style: TextStyle( fontSize: 18, fontFamily: 'Inter', fontWeight: FontWeight.w700, @@ -184,7 +264,7 @@ class _BackupMnemonicPageState extends ConsumerState { ), ), // 占位 - const SizedBox(width: 48, height: 48), + const SizedBox(width: 44, height: 44), ], ), ); @@ -196,13 +276,13 @@ class _BackupMnemonicPageState extends ConsumerState { width: double.infinity, padding: const EdgeInsets.all(16), decoration: BoxDecoration( - color: const Color(0x80FFFFFF), - borderRadius: BorderRadius.circular(12), + color: const Color(0xFFFFFDF5), + borderRadius: BorderRadius.circular(16), boxShadow: const [ BoxShadow( color: Color(0x0D000000), - blurRadius: 2, - offset: Offset(0, 1), + blurRadius: 4, + offset: Offset(0, 2), ), ], ), @@ -229,32 +309,47 @@ class _BackupMnemonicPageState extends ConsumerState { // 显示/隐藏按钮 GestureDetector( onTap: _toggleVisibility, - child: Icon( - _isHidden ? Icons.visibility : Icons.visibility_off, - color: const Color(0xFF5D4037), - size: 24, + child: Container( + padding: const EdgeInsets.all(4), + child: Icon( + _isHidden ? Icons.visibility : Icons.visibility_off, + color: const Color(0xFF8B7355), + size: 22, + ), ), ), - const SizedBox(width: 16), + const SizedBox(width: 12), // 下载按钮 GestureDetector( - onTap: () { - // TODO: 下载助记词 - }, - child: const Icon( - Icons.download, - color: Color(0xFF5D4037), - size: 24, + onTap: _isDownloading ? null : _downloadMnemonic, + child: Container( + padding: const EdgeInsets.all(4), + child: _isDownloading + ? const SizedBox( + width: 22, + height: 22, + child: CircularProgressIndicator( + strokeWidth: 2, + valueColor: AlwaysStoppedAnimation( + Color(0xFF8B7355), + ), + ), + ) + : const Icon( + Icons.file_download_outlined, + color: Color(0xFF8B7355), + size: 22, + ), ), ), ], ), ], ), - const SizedBox(height: 16), + const SizedBox(height: 20), // 助记词网格 _buildMnemonicGrid(), - const SizedBox(height: 16), + const SizedBox(height: 20), // 复制全部按钮 _buildCopyAllButton(), ], @@ -268,7 +363,7 @@ class _BackupMnemonicPageState extends ConsumerState { children: [ for (int row = 0; row < 4; row++) Padding( - padding: const EdgeInsets.symmetric(vertical: 6), + padding: const EdgeInsets.symmetric(vertical: 8), child: Row( children: [ for (int col = 0; col < 3; col++) @@ -290,21 +385,25 @@ class _BackupMnemonicPageState extends ConsumerState { return Row( mainAxisSize: MainAxisSize.min, children: [ + // 序号 Text( '$index.', style: const TextStyle( fontSize: 14, fontFamily: 'Inter', + fontWeight: FontWeight.w500, height: 1.43, - color: Color(0x805D4037), + color: Color(0xCC8B7355), ), ), - const SizedBox(width: 8), + const SizedBox(width: 6), + // 单词 Text( - _isHidden ? '****' : word, + _isHidden ? '••••••' : word, style: const TextStyle( - fontSize: 16, + fontSize: 15, fontFamily: 'Inter', + fontWeight: FontWeight.w600, height: 1.5, color: Color(0xFF5D4037), ), @@ -319,14 +418,10 @@ class _BackupMnemonicPageState extends ConsumerState { onTap: _copyAllMnemonic, child: Container( width: double.infinity, - height: 40, + height: 44, decoration: BoxDecoration( - color: const Color(0x1A8B5A2B), - borderRadius: BorderRadius.circular(8), - border: Border.all( - color: const Color(0x338B5A2B), - width: 1, - ), + color: const Color(0xFFF5ECD9), + borderRadius: BorderRadius.circular(10), ), child: const Center( child: Text( @@ -334,9 +429,9 @@ class _BackupMnemonicPageState extends ConsumerState { style: TextStyle( fontSize: 16, fontFamily: 'Inter', - fontWeight: FontWeight.w500, + fontWeight: FontWeight.w600, height: 1.5, - color: Color(0xFF8B5A2B), + color: Color(0xFF8B7355), ), ), ), @@ -344,154 +439,23 @@ class _BackupMnemonicPageState extends ConsumerState { ); } - /// 构建 MPC 信息卡片 (MPC 模式下显示) - Widget _buildMpcInfoCard() { - return Container( - width: double.infinity, - padding: const EdgeInsets.all(16), - decoration: BoxDecoration( - color: const Color(0x80FFFFFF), - borderRadius: BorderRadius.circular(12), - boxShadow: const [ - BoxShadow( - color: Color(0x0D000000), - blurRadius: 2, - offset: Offset(0, 1), - ), - ], - ), - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - // 成功图标和标题 - Row( - children: [ - Container( - width: 48, - height: 48, - decoration: BoxDecoration( - color: const Color(0x3322C55E), - borderRadius: BorderRadius.circular(24), - ), - child: const Icon( - Icons.check_circle, - color: Color(0xFF22C55E), - size: 28, - ), - ), - const SizedBox(width: 16), - const Expanded( - child: Text( - '多方安全钱包已创建', - style: TextStyle( - fontSize: 18, - fontFamily: 'Inter', - fontWeight: FontWeight.w700, - height: 1.25, - color: Color(0xFF5D4037), - ), - ), - ), - ], - ), - const SizedBox(height: 16), - // 说明文字 - Container( - padding: const EdgeInsets.all(12), - decoration: BoxDecoration( - color: const Color(0x1AD4AF37), - borderRadius: BorderRadius.circular(8), - ), - child: const Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text( - 'MPC 2-of-3 多方安全计算', - style: TextStyle( - fontSize: 14, - fontFamily: 'Inter', - fontWeight: FontWeight.w600, - height: 1.5, - color: Color(0xFF8B5A2B), - ), - ), - SizedBox(height: 8), - Text( - '您的钱包密钥被安全地分成三份:\n' - '1. 服务器持有一份 (用于日常交易)\n' - '2. 您的设备持有一份 (已安全存储)\n' - '3. 备份服务持有一份 (用于恢复)', - style: TextStyle( - fontSize: 13, - fontFamily: 'Inter', - height: 1.6, - color: Color(0xFF5D4037), - ), - ), - ], - ), - ), - // 推荐码 (如果有) - if (widget.referralCode != null && widget.referralCode!.isNotEmpty) ...[ - const SizedBox(height: 16), - Row( - children: [ - const Icon(Icons.share, color: Color(0xFFD4AF37), size: 20), - const SizedBox(width: 8), - const Text( - '您的推荐码: ', - style: TextStyle( - fontSize: 14, - fontFamily: 'Inter', - color: Color(0xFF5D4037), - ), - ), - Text( - widget.referralCode!, - style: const TextStyle( - fontSize: 16, - fontFamily: 'Inter', - fontWeight: FontWeight.w700, - color: Color(0xFFD4AF37), - ), - ), - const Spacer(), - GestureDetector( - onTap: () => _copyAddress(widget.referralCode!, '推荐码'), - child: const Icon(Icons.copy, color: Color(0xFF8B5A2B), size: 20), - ), - ], - ), - ], - ], - ), - ); - } - /// 构建警告卡片 Widget _buildWarningCard() { - final warningText = widget.isMpcMode - ? '您的账户已安全创建。序列号是您的唯一身份标识,请妥善保管。' - : '请妥善保管您的助记词,丢失将无法恢复账号。'; - return Container( width: double.infinity, - padding: const EdgeInsets.all(12), + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 14), decoration: BoxDecoration( - color: widget.isMpcMode ? const Color(0x1A22C55E) : const Color(0x1A7F1D1D), - borderRadius: BorderRadius.circular(8), - border: Border.all( - color: widget.isMpcMode ? const Color(0x3322C55E) : const Color(0x337F1D1D), - width: 1, - ), + color: const Color(0xFFFFF0E0), + borderRadius: BorderRadius.circular(12), ), - child: Text( - warningText, + child: const Text( + '请妥善保管您的助记词,丢失将无法恢复账号。', style: TextStyle( fontSize: 14, fontFamily: 'Inter', + fontWeight: FontWeight.w500, height: 1.5, - color: widget.isMpcMode ? const Color(0xFF166534) : const Color(0xFF991B1B), + color: Color(0xFFCC6B2C), ), textAlign: TextAlign.center, ), @@ -503,38 +467,38 @@ class _BackupMnemonicPageState extends ConsumerState { return Container( width: double.infinity, decoration: BoxDecoration( - color: const Color(0x80FFFFFF), - borderRadius: BorderRadius.circular(12), + color: const Color(0xFFFFFDF5), + borderRadius: BorderRadius.circular(16), boxShadow: const [ BoxShadow( color: Color(0x0D000000), - blurRadius: 2, - offset: Offset(0, 1), + blurRadius: 4, + offset: Offset(0, 2), ), ], ), child: Column( children: [ _buildAddressItem( - icon: Icons.account_balance_wallet, + iconWidget: _buildChainIcon(), label: 'KAVA 地址', address: widget.kavaAddress, showBorder: true, ), _buildAddressItem( - icon: Icons.account_balance_wallet, + iconWidget: _buildChainIcon(), label: 'DST 地址', address: widget.dstAddress, showBorder: true, ), _buildAddressItem( - icon: Icons.account_balance_wallet, + iconWidget: _buildChainIcon(), label: 'BSC 地址', address: widget.bscAddress, showBorder: true, ), _buildAddressItem( - icon: Icons.confirmation_number, + iconWidget: _buildSequenceIcon(), label: '序列号', address: widget.serialNumber, showBorder: false, @@ -544,15 +508,56 @@ class _BackupMnemonicPageState extends ConsumerState { ); } + /// 构建链图标 (黄色背景圆角方形) + Widget _buildChainIcon() { + return Container( + width: 44, + height: 44, + decoration: BoxDecoration( + color: const Color(0xFFFFF3D6), + borderRadius: BorderRadius.circular(10), + ), + child: const Center( + child: Icon( + Icons.credit_card, + color: Color(0xFFD4A84B), + size: 22, + ), + ), + ); + } + + /// 构建序列号图标 (# 符号) + Widget _buildSequenceIcon() { + return Container( + width: 44, + height: 44, + decoration: BoxDecoration( + color: const Color(0xFFFFF3D6), + borderRadius: BorderRadius.circular(10), + ), + child: const Center( + child: Text( + '#', + style: TextStyle( + fontSize: 20, + fontWeight: FontWeight.w700, + color: Color(0xFFD4A84B), + ), + ), + ), + ); + } + /// 构建地址项 Widget _buildAddressItem({ - required IconData icon, + required Widget iconWidget, required String label, required String address, required bool showBorder, }) { return Container( - padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12), + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 14), decoration: BoxDecoration( border: showBorder ? const Border( @@ -566,20 +571,8 @@ class _BackupMnemonicPageState extends ConsumerState { child: Row( children: [ // 图标 - Container( - width: 48, - height: 48, - decoration: BoxDecoration( - color: const Color(0x33D4AF37), - borderRadius: BorderRadius.circular(8), - ), - child: Icon( - icon, - color: const Color(0xFFD4AF37), - size: 24, - ), - ), - const SizedBox(width: 16), + iconWidget, + const SizedBox(width: 14), // 标签和地址 Expanded( child: Column( @@ -590,18 +583,19 @@ class _BackupMnemonicPageState extends ConsumerState { style: const TextStyle( fontSize: 16, fontFamily: 'Inter', - fontWeight: FontWeight.w500, + fontWeight: FontWeight.w600, height: 1.5, color: Color(0xFF5D4037), ), ), + const SizedBox(height: 2), Text( _formatAddress(address), style: const TextStyle( fontSize: 14, fontFamily: 'Inter', height: 1.5, - color: Color(0x995D4037), + color: Color(0x99756452), ), ), ], @@ -610,25 +604,29 @@ class _BackupMnemonicPageState extends ConsumerState { // 复制按钮 GestureDetector( onTap: () => _copyAddress(address, label), - child: const Row( - children: [ - Icon( - Icons.copy, - color: Color(0xCC8B5A2B), - size: 16, - ), - SizedBox(width: 5), - Text( - '复制', - style: TextStyle( - fontSize: 14, - fontFamily: 'Inter', - fontWeight: FontWeight.w500, - height: 1.5, - color: Color(0xCC8B5A2B), + child: Container( + padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 6), + child: const Row( + mainAxisSize: MainAxisSize.min, + children: [ + Icon( + Icons.copy_rounded, + color: Color(0xCC8B7355), + size: 16, ), - ), - ], + SizedBox(width: 4), + Text( + '复制', + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xCC8B7355), + ), + ), + ], + ), ), ), ], @@ -645,25 +643,35 @@ class _BackupMnemonicPageState extends ConsumerState { /// 构建底部按钮区域 Widget _buildBottomButtons() { return Container( - color: const Color(0xFFFFE4B5), - padding: const EdgeInsets.all(16), + padding: const EdgeInsets.fromLTRB(16, 12, 16, 16), + decoration: const BoxDecoration( + color: Color(0xFFFFE4B5), + ), child: Column( + mainAxisSize: MainAxisSize.min, children: [ - // MPC 模式直接进入主页,传统模式需要验证助记词 + // 我已备份助记词按钮 GestureDetector( - onTap: widget.isMpcMode ? _enterApp : _confirmBackup, + onTap: _confirmBackup, child: Container( width: double.infinity, - height: 48, + height: 52, decoration: BoxDecoration( - color: const Color(0xFFD4AF37), + color: const Color(0xFFD4A84B), borderRadius: BorderRadius.circular(12), + boxShadow: const [ + BoxShadow( + color: Color(0x33D4A84B), + blurRadius: 8, + offset: Offset(0, 4), + ), + ], ), - child: Center( + child: const Center( child: Text( - widget.isMpcMode ? '进入应用' : '我已备份助记词', - style: const TextStyle( - fontSize: 16, + '我已备份助记词', + style: TextStyle( + fontSize: 17, fontFamily: 'Inter', fontWeight: FontWeight.w700, height: 1.5, @@ -673,30 +681,23 @@ class _BackupMnemonicPageState extends ConsumerState { ), ), ), - const SizedBox(height: 16), - // 返回上一步 (MPC 模式下不显示,因为账号已创建) - if (!widget.isMpcMode) - GestureDetector( - onTap: _goBack, - child: const Text( - '返回上一步', - style: TextStyle( - fontSize: 14, - fontFamily: 'Inter', - fontWeight: FontWeight.w500, - height: 1.5, - color: Color(0xCC8B5A2B), - ), + const SizedBox(height: 14), + // 返回上一步 + GestureDetector( + onTap: _goBack, + child: const Text( + '返回上一步', + style: TextStyle( + fontSize: 15, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFF8B7355), ), ), + ), ], ), ); } - - /// MPC 模式下直接进入应用 - void _enterApp() { - // 跳转到主页 - context.go(RoutePaths.ranking); - } } diff --git a/frontend/mobile-app/lib/features/auth/presentation/pages/onboarding_page.dart b/frontend/mobile-app/lib/features/auth/presentation/pages/onboarding_page.dart index d2e8e35b..2c546675 100644 --- a/frontend/mobile-app/lib/features/auth/presentation/pages/onboarding_page.dart +++ b/frontend/mobile-app/lib/features/auth/presentation/pages/onboarding_page.dart @@ -5,7 +5,6 @@ import 'package:go_router/go_router.dart'; import '../../../../routes/route_paths.dart'; import '../../../../routes/app_router.dart'; import '../../../../core/di/injection_container.dart'; -import '../../../../core/services/account_service.dart'; /// 创建账号页面 - 用户首次进入应用时的引导页面 /// 提供创建钱包和导入助记词两种选项 @@ -21,8 +20,71 @@ class _OnboardingPageState extends ConsumerState { bool _isAgreed = false; // 创建钱包加载状态 bool _isCreating = false; - // 错误信息 - String? _errorMessage; + // 钱包是否已创建 + bool _isWalletCreated = false; + // 是否正在加载状态 + bool _isLoading = true; + // 已创建的钱包数据 + String? _mnemonic; + String? _kavaAddress; + String? _dstAddress; + String? _bscAddress; + String? _serialNumber; + String? _referralCode; + + @override + void initState() { + super.initState(); + _checkWalletStatus(); + } + + /// 检查钱包是否已创建 + Future _checkWalletStatus() async { + try { + final accountService = ref.read(accountServiceProvider); + + // 检查是否已创建钱包 + final hasAccount = await accountService.hasAccount(); + + if (hasAccount) { + // 读取已保存的钱包数据 + final mnemonic = await accountService.getMnemonic(); + final addresses = await accountService.getWalletAddresses(); + final sequence = await accountService.getAccountSequence(); + final referralCode = await accountService.getReferralCode(); + + if (mounted) { + setState(() { + _isWalletCreated = true; + _mnemonic = mnemonic; + _kavaAddress = addresses?.kava; + _dstAddress = addresses?.dst; + _bscAddress = addresses?.bsc; + _serialNumber = sequence?.toString(); + _referralCode = referralCode; + _isLoading = false; + // 如果钱包已创建,自动勾选协议 + _isAgreed = true; + }); + } + } else { + if (mounted) { + setState(() { + _isWalletCreated = false; + _isLoading = false; + }); + } + } + } catch (e) { + debugPrint('检查钱包状态失败: $e'); + if (mounted) { + setState(() { + _isWalletCreated = false; + _isLoading = false; + }); + } + } + } /// 创建钱包并跳转到备份页面 /// @@ -35,7 +97,6 @@ class _OnboardingPageState extends ConsumerState { setState(() { _isCreating = true; - _errorMessage = null; }); try { @@ -75,10 +136,6 @@ class _OnboardingPageState extends ConsumerState { debugPrint('创建账号失败: $e'); if (!mounted) return; - setState(() { - _errorMessage = e.toString(); - }); - ScaffoldMessenger.of(context).showSnackBar( SnackBar( content: Text('创建账号失败: ${e.toString().replaceAll('Exception: ', '')}'), @@ -121,6 +178,44 @@ class _OnboardingPageState extends ConsumerState { debugPrint('导入助记词'); } + /// 跳转到备份助记词页面(钱包已创建的情况) + void _goToBackupMnemonic() { + if (_mnemonic == null || _kavaAddress == null || _dstAddress == null || + _bscAddress == null || _serialNumber == null) { + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar( + content: Text('钱包数据不完整,请重新创建'), + backgroundColor: Colors.red, + ), + ); + return; + } + + context.push( + RoutePaths.backupMnemonic, + extra: BackupMnemonicParams( + mnemonicWords: _mnemonic!.split(' '), + kavaAddress: _kavaAddress!, + dstAddress: _dstAddress!, + bscAddress: _bscAddress!, + serialNumber: _serialNumber!, + referralCode: _referralCode, + isMpcMode: false, + ), + ); + } + + /// 处理按钮点击 + void _handleButtonTap() { + if (_isWalletCreated) { + // 钱包已创建,跳转到备份页面 + _goToBackupMnemonic(); + } else { + // 钱包未创建,创建新钱包 + _createWallet(); + } + } + @override Widget build(BuildContext context) { return Scaffold( @@ -174,7 +269,7 @@ class _OnboardingPageState extends ConsumerState { borderRadius: BorderRadius.circular(8), boxShadow: [ BoxShadow( - color: Colors.black.withOpacity(0.05), + color: Colors.black.withValues(alpha: 0.05), blurRadius: 10, offset: const Offset(0, 2), ), @@ -260,15 +355,47 @@ class _OnboardingPageState extends ConsumerState { /// 构建创建钱包按钮 Widget _buildCreateButton() { + // 加载中显示骨架 + if (_isLoading) { + return Container( + width: double.infinity, + height: 48, + decoration: BoxDecoration( + color: const Color(0xFFD4AF37).withValues(alpha: 0.5), + borderRadius: BorderRadius.circular(8), + ), + child: const Center( + child: SizedBox( + width: 24, + height: 24, + child: CircularProgressIndicator( + strokeWidth: 2, + valueColor: AlwaysStoppedAnimation(Colors.white), + ), + ), + ), + ); + } + + // 根据钱包状态决定按钮文字和行为 + final buttonText = _isWalletCreated + ? '钱包已创建(点击备份助记词)' + : '生成钱包(创建账户)'; + + // 钱包已创建时,不需要勾选协议 + final isEnabled = _isWalletCreated || _isAgreed; + return GestureDetector( - onTap: _isCreating ? null : _createWallet, + onTap: (_isCreating || !isEnabled) ? null : _handleButtonTap, child: Opacity( - opacity: _isAgreed ? 1.0 : 0.5, + opacity: isEnabled ? 1.0 : 0.5, child: Container( width: double.infinity, height: 48, decoration: BoxDecoration( - color: const Color(0xFFD4AF37), // 金色 + color: _isWalletCreated + ? const Color(0xFF52C41A) // 绿色表示已创建 + : const Color(0xFFD4AF37), // 金色表示待创建 borderRadius: BorderRadius.circular(8), ), child: Center( @@ -281,16 +408,29 @@ class _OnboardingPageState extends ConsumerState { valueColor: AlwaysStoppedAnimation(Colors.white), ), ) - : const Text( - '生成钱包(创建账户)', - style: TextStyle( - fontSize: 16, - fontFamily: 'Inter', - fontWeight: FontWeight.w700, - height: 1.5, - letterSpacing: 0.24, - color: Colors.white, - ), + : Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + if (_isWalletCreated) ...[ + const Icon( + Icons.check_circle, + color: Colors.white, + size: 20, + ), + const SizedBox(width: 8), + ], + Text( + buttonText, + style: const TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.5, + letterSpacing: 0.24, + color: Colors.white, + ), + ), + ], ), ), ), diff --git a/frontend/mobile-app/lib/features/auth/presentation/pages/verify_mnemonic_page.dart b/frontend/mobile-app/lib/features/auth/presentation/pages/verify_mnemonic_page.dart index c68372c3..7b9dfd26 100644 --- a/frontend/mobile-app/lib/features/auth/presentation/pages/verify_mnemonic_page.dart +++ b/frontend/mobile-app/lib/features/auth/presentation/pages/verify_mnemonic_page.dart @@ -4,7 +4,7 @@ import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:go_router/go_router.dart'; import '../../../../routes/route_paths.dart'; import '../../../../routes/app_router.dart'; -import '../providers/auth_provider.dart'; +import '../../../../core/di/injection_container.dart'; /// 确认备份页面 - 验证用户是否正确备份了助记词 /// 用户需要勾选确认或选择正确的助记词单词才能完成验证 @@ -133,11 +133,9 @@ class _VerifyMnemonicPageState extends ConsumerState { setState(() => _isCreating = true); try { - // 保存钱包信息到安全存储 - await ref.read(authProvider.notifier).saveWallet( - widget.kavaAddress, - widget.mnemonicWords.join(' '), - ); + // 使用 AccountService 标记助记词已备份 + final accountService = ref.read(accountServiceProvider); + await accountService.markMnemonicBackedUp(); if (!mounted) return; @@ -154,7 +152,10 @@ class _VerifyMnemonicPageState extends ConsumerState { } catch (e) { if (!mounted) return; ScaffoldMessenger.of(context).showSnackBar( - SnackBar(content: Text('创建账号失败: $e')), + SnackBar( + content: Text('操作失败: $e'), + backgroundColor: const Color(0xFF991B1B), + ), ); } finally { if (mounted) { @@ -171,38 +172,52 @@ class _VerifyMnemonicPageState extends ConsumerState { @override Widget build(BuildContext context) { return Scaffold( - backgroundColor: Colors.white, - body: SafeArea( - child: Column( - children: [ - // 顶部导航栏 - _buildAppBar(), - // 内容区域 - Expanded( - child: SingleChildScrollView( - padding: const EdgeInsets.all(16), - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - const SizedBox(height: 16), - // 说明文字 - _buildDescription(), - const SizedBox(height: 24), - // 勾选确认 - _buildCheckbox(), - const SizedBox(height: 24), - // 分隔线 - _buildDivider(), - const SizedBox(height: 24), - // 选择单词验证 - _buildWordSelection(), - ], + body: Container( + width: double.infinity, + height: double.infinity, + // 渐变背景 - 从浅米色到金黄色 + decoration: const BoxDecoration( + gradient: LinearGradient( + begin: Alignment.topCenter, + end: Alignment.bottomCenter, + colors: [ + Color(0xFFFFF8F0), // 更浅的米色顶部 + Color(0xFFFFE4B5), // 金黄色底部 + ], + ), + ), + child: SafeArea( + child: Column( + children: [ + // 顶部导航栏 + _buildAppBar(), + // 内容区域 + Expanded( + child: SingleChildScrollView( + padding: const EdgeInsets.symmetric(horizontal: 20), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + const SizedBox(height: 24), + // 说明文字 + _buildDescription(), + const SizedBox(height: 32), + // 勾选确认 + _buildCheckbox(), + const SizedBox(height: 28), + // 分隔线 + _buildDivider(), + const SizedBox(height: 28), + // 选择单词验证 + _buildWordSelection(), + ], + ), ), ), - ), - // 底部按钮区域 - _buildBottomButtons(), - ], + // 底部按钮区域 + _buildBottomButtons(), + ], + ), ), ), ); @@ -211,15 +226,15 @@ class _VerifyMnemonicPageState extends ConsumerState { /// 构建顶部导航栏 Widget _buildAppBar() { return Container( - padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 16), + padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 8), child: Row( children: [ // 返回按钮 GestureDetector( onTap: _goBack, child: Container( - width: 40, - height: 40, + width: 44, + height: 44, alignment: Alignment.center, child: const Icon( Icons.arrow_back, @@ -236,15 +251,15 @@ class _VerifyMnemonicPageState extends ConsumerState { fontSize: 18, fontFamily: 'Inter', fontWeight: FontWeight.w700, - height: 1.56, - letterSpacing: -0.45, + height: 1.25, + letterSpacing: -0.27, color: Color(0xFF5D4037), ), textAlign: TextAlign.center, ), ), // 占位 - const SizedBox(width: 40), + const SizedBox(width: 44, height: 44), ], ), ); @@ -272,16 +287,18 @@ class _VerifyMnemonicPageState extends ConsumerState { }); }, child: Row( + crossAxisAlignment: CrossAxisAlignment.center, children: [ + // 勾选框 - 金黄色圆角方形 Container( - width: 24, - height: 24, + width: 22, + height: 22, decoration: BoxDecoration( - color: _isChecked ? const Color(0xFFD4AF37) : Colors.transparent, + color: _isChecked ? const Color(0xFFD4A84B) : const Color(0xFFFFF5E6), borderRadius: BorderRadius.circular(4), border: Border.all( - color: _isChecked ? const Color(0xFFD4AF37) : const Color(0xFF8B5A2B), - width: 2, + color: _isChecked ? const Color(0xFFD4A84B) : const Color(0xFFD4A84B), + width: 1.5, ), ), child: _isChecked @@ -315,25 +332,40 @@ class _VerifyMnemonicPageState extends ConsumerState { Expanded( child: Container( height: 1, - color: const Color(0x338B5A2B), + decoration: BoxDecoration( + gradient: LinearGradient( + colors: [ + const Color(0xFFD4A84B).withValues(alpha: 0.1), + const Color(0xFFD4A84B).withValues(alpha: 0.4), + ], + ), + ), ), ), const Padding( - padding: EdgeInsets.symmetric(horizontal: 12), + padding: EdgeInsets.symmetric(horizontal: 16), child: Text( '或', style: TextStyle( fontSize: 14, fontFamily: 'Inter', + fontWeight: FontWeight.w500, height: 1.43, - color: Color(0x998B5A2B), + color: Color(0xFF8B7355), ), ), ), Expanded( child: Container( height: 1, - color: const Color(0x338B5A2B), + decoration: BoxDecoration( + gradient: LinearGradient( + colors: [ + const Color(0xFFD4A84B).withValues(alpha: 0.4), + const Color(0xFFD4A84B).withValues(alpha: 0.1), + ], + ), + ), ), ), ], @@ -350,14 +382,15 @@ class _VerifyMnemonicPageState extends ConsumerState { style: const TextStyle( fontSize: 16, fontFamily: 'Inter', + fontWeight: FontWeight.w500, height: 1.5, color: Color(0xFF5D4037), ), ), - const SizedBox(height: 16), - // 选项按钮 + const SizedBox(height: 20), + // 选项按钮 - 使用网格布局 Wrap( - spacing: 12, + spacing: 10, runSpacing: 12, children: _wordOptions.map((word) => _buildWordOption(word)).toList(), ), @@ -368,38 +401,29 @@ class _VerifyMnemonicPageState extends ConsumerState { /// 构建单词选项按钮 Widget _buildWordOption(String word) { final isSelected = _selectedWord == word; - final isCorrect = word == _correctWord; return GestureDetector( onTap: () => _selectWord(word), child: Container( - padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 10), + padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 12), decoration: BoxDecoration( + // 选中时使用浅金色背景,未选中时使用米色背景 color: isSelected - ? const Color(0x33D4AF37) - : const Color(0x1A8B5A2B), - borderRadius: BorderRadius.circular(8), + ? const Color(0xFFFFF5E6) + : const Color(0xFFF5ECD9), + borderRadius: BorderRadius.circular(10), border: isSelected - ? Border.all(color: const Color(0xFFD4AF37), width: 1) - : null, - boxShadow: isSelected - ? [ - const BoxShadow( - color: Color(0xCCD4AF37), - blurRadius: 0, - spreadRadius: 1, - ), - ] - : null, + ? Border.all(color: const Color(0xFFD4A84B), width: 1.5) + : Border.all(color: Colors.transparent, width: 1.5), ), child: Text( word, style: TextStyle( - fontSize: 14, + fontSize: 15, fontFamily: 'Inter', - fontWeight: FontWeight.w500, - height: 1.5, - color: isSelected ? const Color(0xFFD4AF37) : const Color(0xFF5D4037), + fontWeight: FontWeight.w600, + height: 1.4, + color: isSelected ? const Color(0xFFD4A84B) : const Color(0xFF5D4037), ), ), ), @@ -409,21 +433,34 @@ class _VerifyMnemonicPageState extends ConsumerState { /// 构建底部按钮区域 Widget _buildBottomButtons() { return Container( - color: const Color(0xFFFFE4B5), - padding: const EdgeInsets.fromLTRB(16, 8, 16, 16), + padding: const EdgeInsets.fromLTRB(20, 16, 20, 24), + decoration: const BoxDecoration( + color: Color(0xFFFFE4B5), + ), child: Column( + mainAxisSize: MainAxisSize.min, children: [ // 确认并创建账号按钮 GestureDetector( onTap: _isCreating ? null : _confirmAndCreate, child: Container( width: double.infinity, - height: 48, + height: 52, decoration: BoxDecoration( + // 禁用状态使用浅灰棕色,启用状态使用金色 color: _canSubmit - ? const Color(0xFFD4AF37) - : const Color(0x668B5A2B), + ? const Color(0xFFD4A84B) + : const Color(0xFFCBBDA8), borderRadius: BorderRadius.circular(12), + boxShadow: _canSubmit + ? const [ + BoxShadow( + color: Color(0x33D4A84B), + blurRadius: 8, + offset: Offset(0, 4), + ), + ] + : null, ), child: Center( child: _isCreating @@ -438,7 +475,7 @@ class _VerifyMnemonicPageState extends ConsumerState { : Text( '确认并创建账号', style: TextStyle( - fontSize: 16, + fontSize: 17, fontFamily: 'Inter', fontWeight: FontWeight.w700, height: 1.5, @@ -450,23 +487,18 @@ class _VerifyMnemonicPageState extends ConsumerState { ), ), ), - const SizedBox(height: 8), + const SizedBox(height: 14), // 返回查看助记词 GestureDetector( onTap: _goBack, - child: Container( - width: double.infinity, - height: 48, - alignment: Alignment.center, - child: const Text( - '返回查看助记词', - style: TextStyle( - fontSize: 16, - fontFamily: 'Inter', - fontWeight: FontWeight.w500, - height: 1.5, - color: Color(0xFFD4AF37), - ), + child: const Text( + '返回查看助记词', + style: TextStyle( + fontSize: 15, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFF8B7355), ), ), ), diff --git a/frontend/mobile-app/lib/features/auth/presentation/pages/wallet_created_page.dart b/frontend/mobile-app/lib/features/auth/presentation/pages/wallet_created_page.dart index a0241dcb..c5d5cf03 100644 --- a/frontend/mobile-app/lib/features/auth/presentation/pages/wallet_created_page.dart +++ b/frontend/mobile-app/lib/features/auth/presentation/pages/wallet_created_page.dart @@ -3,6 +3,7 @@ import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:go_router/go_router.dart'; import '../../../../routes/route_paths.dart'; +import '../../../../routes/app_router.dart'; /// 创建成功页面 - 显示钱包创建成功信息 /// 展示序列号和三个链的钱包地址 @@ -15,6 +16,8 @@ class WalletCreatedPage extends ConsumerWidget { final String bscAddress; /// 序列号 final String serialNumber; + /// 推荐码 + final String? referralCode; const WalletCreatedPage({ super.key, @@ -22,6 +25,7 @@ class WalletCreatedPage extends ConsumerWidget { required this.dstAddress, required this.bscAddress, required this.serialNumber, + this.referralCode, }); /// 复制内容到剪贴板 @@ -30,7 +34,7 @@ class WalletCreatedPage extends ConsumerWidget { ScaffoldMessenger.of(context).showSnackBar( SnackBar( content: Text('$label 已复制'), - backgroundColor: const Color(0xFFD4AF37), + backgroundColor: const Color(0xFFD4A84B), duration: const Duration(seconds: 2), ), ); @@ -41,14 +45,18 @@ class WalletCreatedPage extends ConsumerWidget { context.go(RoutePaths.ranking); } - /// 分享给好友 + /// 分享给好友 - 跳转到分享页面 void _shareToFriends(BuildContext context) { - // TODO: 实现分享功能 - ScaffoldMessenger.of(context).showSnackBar( - const SnackBar( - content: Text('分享功能即将推出'), - backgroundColor: Color(0xFF8B5A2B), - duration: Duration(seconds: 2), + // 构建分享链接 (带推荐码) + final shareLink = referralCode != null + ? 'https://rwa-durian.app/invite?code=$referralCode' + : 'https://rwa-durian.app/invite?seq=$serialNumber'; + + context.push( + RoutePaths.share, + extra: SharePageParams( + shareLink: shareLink, + referralCode: referralCode, ), ); } @@ -62,42 +70,45 @@ class WalletCreatedPage extends ConsumerWidget { @override Widget build(BuildContext context, WidgetRef ref) { return Scaffold( - backgroundColor: Colors.white, body: Container( width: double.infinity, height: double.infinity, + // 渐变背景 - 从浅米色到金黄色 decoration: const BoxDecoration( gradient: LinearGradient( begin: Alignment.topCenter, end: Alignment.bottomCenter, colors: [ - Color(0xFFFFF5E6), - Color(0xFFFFE4B5), + Color(0xFFFFF8F0), // 更浅的米色顶部 + Color(0xFFFFE4B5), // 金黄色底部 ], ), ), child: SafeArea( - child: SingleChildScrollView( - padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 24), - child: Column( - children: [ - const SizedBox(height: 32), - // 成功图标 - _buildSuccessIcon(), - const SizedBox(height: 24), - // 标题和副标题 - _buildTitle(), - const SizedBox(height: 32), - // 钱包信息卡片 - _buildWalletInfoCard(context), - const SizedBox(height: 32), - // 底部按钮 - _buildActionButtons(context), - const SizedBox(height: 16), - // 提示文字 - _buildWarningText(), - ], - ), + child: Column( + children: [ + // 内容区域 + Expanded( + child: SingleChildScrollView( + padding: const EdgeInsets.symmetric(horizontal: 20), + child: Column( + children: [ + const SizedBox(height: 48), + // 成功图标 + _buildSuccessIcon(), + const SizedBox(height: 20), + // 标题和副标题 + _buildTitle(), + const SizedBox(height: 28), + // 钱包信息卡片 + _buildWalletInfoCard(context), + ], + ), + ), + ), + // 底部按钮区域 + _buildBottomSection(context), + ], ), ), ), @@ -110,14 +121,27 @@ class WalletCreatedPage extends ConsumerWidget { width: 120, height: 120, decoration: const BoxDecoration( - color: Color(0x33D4AF37), + color: Color(0xFFFFF3D6), // 浅黄色背景 shape: BoxShape.circle, ), - child: const Center( - child: Icon( - Icons.check_circle_outline, - size: 64, - color: Color(0xFFD4AF37), + child: Center( + child: Container( + width: 64, + height: 64, + decoration: BoxDecoration( + shape: BoxShape.circle, + border: Border.all( + color: const Color(0xFFD4A84B), + width: 3, + ), + ), + child: const Center( + child: Icon( + Icons.check, + size: 36, + color: Color(0xFFD4A84B), + ), + ), ), ), ); @@ -125,16 +149,16 @@ class WalletCreatedPage extends ConsumerWidget { /// 构建标题 Widget _buildTitle() { - return Column( - children: const [ + return const Column( + children: [ Text( '账号创建成功', style: TextStyle( - fontSize: 30, + fontSize: 26, fontFamily: 'Inter', fontWeight: FontWeight.w700, height: 1.25, - letterSpacing: -0.75, + letterSpacing: -0.5, color: Color(0xFF5D4037), ), ), @@ -142,10 +166,11 @@ class WalletCreatedPage extends ConsumerWidget { Text( '已为您创建钱包与序列号', style: TextStyle( - fontSize: 16, + fontSize: 15, fontFamily: 'Inter', + fontWeight: FontWeight.w500, height: 1.5, - color: Color(0xFF8B5A2B), + color: Color(0xFF8B7355), ), ), ], @@ -156,51 +181,53 @@ class WalletCreatedPage extends ConsumerWidget { Widget _buildWalletInfoCard(BuildContext context) { return Container( width: double.infinity, - padding: const EdgeInsets.all(16), decoration: BoxDecoration( - color: const Color(0x4DFFFFFF), - borderRadius: BorderRadius.circular(12), + color: const Color(0xFFFFFDF5), + borderRadius: BorderRadius.circular(16), boxShadow: const [ BoxShadow( - color: Color(0x33D4AF37), - blurRadius: 0, - spreadRadius: 1, - offset: Offset(0, 0), + color: Color(0x0D000000), + blurRadius: 4, + offset: Offset(0, 2), ), ], ), child: Column( children: [ - // 序列号 + // 序列号 - 放在最上面 _buildInfoItem( context: context, - icon: Icons.key, + iconWidget: _buildKeyIcon(), label: '序列号', value: serialNumber, + isAddress: false, showDivider: true, ), // KAVA 地址 _buildInfoItem( context: context, - icon: Icons.account_balance_wallet, + iconWidget: _buildWalletIcon(), label: 'KAVA', value: kavaAddress, + isAddress: true, showDivider: true, ), // DST 地址 _buildInfoItem( context: context, - icon: Icons.account_balance_wallet, + iconWidget: _buildWalletIcon(), label: 'DST', value: dstAddress, + isAddress: true, showDivider: true, ), // BSC 地址 _buildInfoItem( context: context, - icon: Icons.account_balance_wallet, + iconWidget: _buildWalletIcon(), label: 'BSC', value: bscAddress, + isAddress: true, showDivider: false, ), ], @@ -208,153 +235,181 @@ class WalletCreatedPage extends ConsumerWidget { ); } + /// 构建钥匙图标 (序列号) + Widget _buildKeyIcon() { + return Container( + width: 48, + height: 48, + decoration: BoxDecoration( + color: const Color(0xFFFFF3D6), + borderRadius: BorderRadius.circular(10), + ), + child: const Center( + child: Icon( + Icons.vpn_key_outlined, + color: Color(0xFFD4A84B), + size: 24, + ), + ), + ); + } + + /// 构建钱包图标 (地址) + Widget _buildWalletIcon() { + return Container( + width: 48, + height: 48, + decoration: BoxDecoration( + color: const Color(0xFFFFF3D6), + borderRadius: BorderRadius.circular(10), + ), + child: const Center( + child: Icon( + Icons.account_balance_wallet_outlined, + color: Color(0xFFD4A84B), + size: 24, + ), + ), + ); + } + /// 构建信息项 Widget _buildInfoItem({ required BuildContext context, - required IconData icon, + required Widget iconWidget, required String label, required String value, + required bool isAddress, required bool showDivider, }) { - return Column( - children: [ - Padding( - padding: const EdgeInsets.symmetric(vertical: 12), - child: Row( - children: [ - // 图标 - Container( - width: 48, - height: 48, - decoration: BoxDecoration( - color: const Color(0x33D4AF37), - borderRadius: BorderRadius.circular(8), - ), - child: Icon( - icon, - color: const Color(0xFFD4AF37), - size: 24, + // 格式化显示值 + final displayValue = isAddress ? _formatAddress(value) : value; + + return Container( + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 14), + decoration: BoxDecoration( + border: showDivider + ? const Border( + bottom: BorderSide( + color: Color(0x1A5D4037), + width: 1, ), + ) + : null, + ), + child: Row( + children: [ + // 图标 + iconWidget, + const SizedBox(width: 14), + // 标签和值 - 单行显示 "标签: 值" + Expanded( + child: Text( + '$label: $displayValue', + style: const TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFF5D4037), ), - const SizedBox(width: 16), - // 标签和值 - Expanded( - child: Column( - crossAxisAlignment: CrossAxisAlignment.start, - children: [ - Text( - '$label:', - style: const TextStyle( - fontSize: 14, - fontFamily: 'Inter', - height: 1.5, - color: Color(0x998B5A2B), - ), - ), - Text( - label == '序列号' ? value : _formatAddress(value), - style: const TextStyle( - fontSize: 16, - fontFamily: 'Inter', - height: 1.5, - color: Color(0xFF5D4037), - ), - ), - ], - ), + ), + ), + // 复制按钮 + GestureDetector( + onTap: () => _copyToClipboard(context, value, label), + child: Container( + padding: const EdgeInsets.all(8), + child: const Icon( + Icons.copy_outlined, + color: Color(0xFFD4A84B), + size: 22, ), - // 复制按钮 - GestureDetector( - onTap: () => _copyToClipboard(context, value, label), - child: Container( - padding: const EdgeInsets.all(8), - child: const Icon( - Icons.copy, - color: Color(0xCC8B5A2B), - size: 20, + ), + ), + ], + ), + ); + } + + /// 构建底部区域 (按钮 + 提示文字) + Widget _buildBottomSection(BuildContext context) { + return Container( + padding: const EdgeInsets.fromLTRB(20, 16, 20, 24), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + // 进入我的钱包按钮 - 金黄色 + GestureDetector( + onTap: () => _enterWallet(context), + child: Container( + width: double.infinity, + height: 52, + decoration: BoxDecoration( + color: const Color(0xFFD4A84B), + borderRadius: BorderRadius.circular(12), + boxShadow: const [ + BoxShadow( + color: Color(0x33D4A84B), + blurRadius: 8, + offset: Offset(0, 4), + ), + ], + ), + child: const Center( + child: Text( + '进入我的钱包', + style: TextStyle( + fontSize: 17, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.5, + color: Colors.white, ), ), ), - ], - ), - ), - if (showDivider) - Container( - height: 1, - color: const Color(0x1A8B5A2B), - ), - ], - ); - } - - /// 构建操作按钮 - Widget _buildActionButtons(BuildContext context) { - return Column( - children: [ - // 进入我的钱包按钮 - GestureDetector( - onTap: () => _enterWallet(context), - child: Container( - width: double.infinity, - height: 53, - decoration: BoxDecoration( - color: const Color(0xFFD4AF37), - borderRadius: BorderRadius.circular(12), ), - child: const Center( - child: Text( - '进入我的钱包', - style: TextStyle( - fontSize: 16, - fontFamily: 'Inter', - fontWeight: FontWeight.w700, - height: 1.5, - color: Colors.white, + ), + const SizedBox(height: 12), + // 分享给好友按钮 - 深棕色 + GestureDetector( + onTap: () => _shareToFriends(context), + child: Container( + width: double.infinity, + height: 52, + decoration: BoxDecoration( + color: const Color(0xFF5D4037), + borderRadius: BorderRadius.circular(12), + ), + child: const Center( + child: Text( + '分享给好友', + style: TextStyle( + fontSize: 17, + fontFamily: 'Inter', + fontWeight: FontWeight.w600, + height: 1.5, + color: Colors.white, + ), ), ), ), ), - ), - const SizedBox(height: 16), - // 分享给好友按钮 - GestureDetector( - onTap: () => _shareToFriends(context), - child: Container( - width: double.infinity, - height: 53, - decoration: BoxDecoration( - color: const Color(0xFF8B5A2B), - borderRadius: BorderRadius.circular(12), - ), - child: const Center( - child: Text( - '分享给好友', - style: TextStyle( - fontSize: 16, - fontFamily: 'Inter', - fontWeight: FontWeight.w500, - height: 1.5, - color: Colors.white, - ), - ), + const SizedBox(height: 16), + // 提示文字 + const Text( + '助记词仅显示一次,请妥善保管。', + style: TextStyle( + fontSize: 13, + fontFamily: 'Inter', + fontWeight: FontWeight.w400, + height: 1.5, + color: Color(0xFF8B7355), ), + textAlign: TextAlign.center, ), - ), - ], - ); - } - - /// 构建警告提示文字 - Widget _buildWarningText() { - return const Text( - '助记词仅显示一次,请妥善保管。', - style: TextStyle( - fontSize: 14, - fontFamily: 'Inter', - height: 1.5, - color: Color(0xCC8B5A2B), + ], ), - textAlign: TextAlign.center, ); } } diff --git a/frontend/mobile-app/lib/features/deposit/presentation/pages/deposit_usdt_page.dart b/frontend/mobile-app/lib/features/deposit/presentation/pages/deposit_usdt_page.dart new file mode 100644 index 00000000..4417160a --- /dev/null +++ b/frontend/mobile-app/lib/features/deposit/presentation/pages/deposit_usdt_page.dart @@ -0,0 +1,618 @@ +import 'package:flutter/material.dart'; +import 'package:flutter/services.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:go_router/go_router.dart'; +import 'package:qr_flutter/qr_flutter.dart'; +import '../../../../core/di/injection_container.dart'; + +/// 网络类型枚举 +enum NetworkType { + kava, + bsc, +} + +/// 充值 USDT 页面 +/// 显示充值二维码和地址,支持 KAVA 和 BSC 网络切换 +class DepositUsdtPage extends ConsumerStatefulWidget { + const DepositUsdtPage({super.key}); + + @override + ConsumerState createState() => _DepositUsdtPageState(); +} + +class _DepositUsdtPageState extends ConsumerState { + /// 当前选中的网络 + NetworkType _selectedNetwork = NetworkType.kava; + + /// 是否正在加载 + bool _isLoading = true; + + /// USDT 余额 + String _balance = '0.00'; + + /// KAVA 网络充值地址 + String? _kavaAddress; + + /// BSC 网络充值地址 + String? _bscAddress; + + @override + void initState() { + super.initState(); + _loadWalletData(); + } + + /// 错误信息 + String? _errorMessage; + + /// 加载钱包数据 + Future _loadWalletData() async { + try { + setState(() { + _isLoading = true; + _errorMessage = null; + }); + + final depositService = ref.read(depositServiceProvider); + + // 获取充值地址 (验证后的安全地址) + final addressResponse = await depositService.getDepositAddresses(); + + if (!addressResponse.isValid) { + // 地址验证失败 + if (mounted) { + setState(() { + _errorMessage = addressResponse.message ?? '充值账户异常,请联系客服'; + _isLoading = false; + }); + } + return; + } + + _kavaAddress = addressResponse.kavaAddress; + _bscAddress = addressResponse.bscAddress; + + // 查询实时余额 + try { + final balanceResponse = await depositService.getUsdtBalances(); + // 根据当前选中的网络显示余额 + if (mounted) { + setState(() { + if (_selectedNetwork == NetworkType.kava && balanceResponse.kava != null) { + _balance = balanceResponse.kava!.balance; + } else if (_selectedNetwork == NetworkType.bsc && balanceResponse.bsc != null) { + _balance = balanceResponse.bsc!.balance; + } else { + _balance = '0.00'; + } + _isLoading = false; + }); + } + } catch (e) { + debugPrint('查询余额失败: $e'); + if (mounted) { + setState(() { + _balance = '0.00'; + _isLoading = false; + }); + } + } + } catch (e) { + debugPrint('加载钱包数据失败: $e'); + if (mounted) { + setState(() { + _errorMessage = '加载失败,请重试'; + _isLoading = false; + }); + } + } + } + + /// 获取当前网络的充值地址 + String get _currentAddress { + switch (_selectedNetwork) { + case NetworkType.kava: + return _kavaAddress ?? ''; + case NetworkType.bsc: + return _bscAddress ?? ''; + } + } + + /// 切换网络 + void _switchNetwork(NetworkType network) { + if (_selectedNetwork != network) { + setState(() { + _selectedNetwork = network; + }); + } + } + + /// 复制地址到剪贴板 + void _copyAddress() { + if (_currentAddress.isEmpty) return; + + Clipboard.setData(ClipboardData(text: _currentAddress)); + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar( + content: Text('充值地址已复制'), + backgroundColor: Color(0xFFD4AF37), + duration: Duration(seconds: 2), + ), + ); + } + + /// 返回上一页 + void _goBack() { + context.pop(); + } + + /// 完成充值 + void _onDepositComplete() { + // 返回上一页 + context.pop(); + + // 显示提示 + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar( + content: Text('充值确认中,请稍候查看余额变化'), + backgroundColor: Color(0xFFD4AF37), + duration: Duration(seconds: 3), + ), + ); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + body: Container( + width: double.infinity, + height: double.infinity, + // 渐变背景 + decoration: const BoxDecoration( + gradient: LinearGradient( + begin: Alignment.topCenter, + end: Alignment.bottomCenter, + colors: [ + Color(0xFFFFF7E6), + Color(0xFFEAE0C8), + ], + ), + ), + child: SafeArea( + child: Column( + children: [ + // 顶部导航栏 + _buildAppBar(), + // 内容区域 + Expanded( + child: SingleChildScrollView( + child: Column( + children: [ + // 余额显示 + _buildBalanceSection(), + // 网络切换 + _buildNetworkSwitch(), + const SizedBox(height: 8), + // 二维码卡片 + _buildQrCodeCard(), + // 警告提示 + _buildWarningText(), + ], + ), + ), + ), + // 底部按钮 + _buildBottomButton(), + ], + ), + ), + ), + ); + } + + /// 构建顶部导航栏 + Widget _buildAppBar() { + return Container( + padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 8), + decoration: BoxDecoration( + color: const Color(0xFFFFF5E6).withValues(alpha: 0.8), + ), + child: Row( + children: [ + // 返回按钮 + GestureDetector( + onTap: _goBack, + child: Container( + width: 40, + height: 40, + alignment: Alignment.center, + child: const Icon( + Icons.arrow_back, + color: Color(0xFF5D4037), + size: 24, + ), + ), + ), + // 标题 + const Expanded( + child: Text( + '充值 USDT', + style: TextStyle( + fontSize: 18, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.25, + letterSpacing: -0.27, + color: Color(0xFF5D4037), + ), + textAlign: TextAlign.center, + ), + ), + // 占位 + const SizedBox(width: 40), + ], + ), + ); + } + + /// 构建余额显示区域 + Widget _buildBalanceSection() { + return Container( + width: double.infinity, + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), + child: Text( + '可用余额: $_balance USDT', + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + height: 1.5, + color: const Color(0xFF5D4037).withValues(alpha: 0.8), + ), + ), + ); + } + + /// 构建网络切换 + Widget _buildNetworkSwitch() { + return Container( + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), + child: Container( + decoration: BoxDecoration( + color: Colors.white.withValues(alpha: 0.5), + borderRadius: BorderRadius.circular(12), + ), + padding: const EdgeInsets.all(4), + child: Row( + children: [ + // KAVA 网络按钮 + Expanded( + child: _buildNetworkButton( + label: 'KAVA 网络', + network: NetworkType.kava, + isSelected: _selectedNetwork == NetworkType.kava, + ), + ), + // BSC 网络按钮 + Expanded( + child: _buildNetworkButton( + label: 'BSC 网络', + network: NetworkType.bsc, + isSelected: _selectedNetwork == NetworkType.bsc, + ), + ), + ], + ), + ), + ); + } + + /// 构建网络按钮 + Widget _buildNetworkButton({ + required String label, + required NetworkType network, + required bool isSelected, + }) { + return GestureDetector( + onTap: () => _switchNetwork(network), + child: Container( + padding: const EdgeInsets.symmetric(vertical: 9.5, horizontal: 12), + decoration: BoxDecoration( + color: isSelected ? const Color(0xFFD4AF37) : Colors.transparent, + borderRadius: BorderRadius.circular(8), + boxShadow: isSelected + ? const [ + BoxShadow( + color: Color(0x66D4AF37), + blurRadius: 3, + offset: Offset(0, 1), + ), + ] + : null, + ), + child: Center( + child: Text( + label, + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: isSelected ? Colors.white : const Color(0xFF745D43), + ), + ), + ), + ), + ); + } + + /// 构建二维码卡片 + Widget _buildQrCodeCard() { + return Container( + margin: const EdgeInsets.symmetric(horizontal: 16), + padding: const EdgeInsets.all(24), + decoration: BoxDecoration( + color: Colors.white.withValues(alpha: 0.5), + borderRadius: BorderRadius.circular(12), + ), + child: Column( + children: [ + // 二维码区域 + _buildQrCode(), + const SizedBox(height: 24), + // 地址信息 + _buildAddressInfo(), + ], + ), + ); + } + + /// 构建二维码 + Widget _buildQrCode() { + // 加载中 + if (_isLoading) { + return Container( + width: 192, + height: 192, + decoration: BoxDecoration( + color: const Color(0xFF1A1A2E), + borderRadius: BorderRadius.circular(8), + ), + child: const Center( + child: CircularProgressIndicator( + color: Colors.white, + strokeWidth: 2, + ), + ), + ); + } + + // 错误状态 + if (_errorMessage != null) { + return Container( + width: 192, + height: 192, + decoration: BoxDecoration( + color: const Color(0xFFFFF5E6), + borderRadius: BorderRadius.circular(8), + border: Border.all(color: const Color(0xFFE74C3C), width: 1), + ), + child: Center( + child: Padding( + padding: const EdgeInsets.all(16), + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + const Icon( + Icons.error_outline, + color: Color(0xFFE74C3C), + size: 40, + ), + const SizedBox(height: 8), + Text( + _errorMessage!, + style: const TextStyle( + fontSize: 12, + color: Color(0xFFE74C3C), + ), + textAlign: TextAlign.center, + ), + const SizedBox(height: 8), + GestureDetector( + onTap: _loadWalletData, + child: const Text( + '点击重试', + style: TextStyle( + fontSize: 12, + color: Color(0xFFD4AF37), + decoration: TextDecoration.underline, + ), + ), + ), + ], + ), + ), + ), + ); + } + + // 地址为空 + if (_currentAddress.isEmpty) { + return Container( + width: 192, + height: 192, + decoration: BoxDecoration( + color: const Color(0xFFFFF5E6), + borderRadius: BorderRadius.circular(8), + ), + child: const Center( + child: Text( + '暂无充值地址', + style: TextStyle( + fontSize: 14, + color: Color(0xFF5D4037), + ), + ), + ), + ); + } + + return Container( + width: 192, + height: 192, + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(8), + boxShadow: const [ + BoxShadow( + color: Color(0x1A000000), + blurRadius: 10, + offset: Offset(0, 4), + ), + ], + ), + child: ClipRRect( + borderRadius: BorderRadius.circular(8), + child: QrImageView( + data: _currentAddress, + version: QrVersions.auto, + size: 192, + backgroundColor: Colors.white, + padding: const EdgeInsets.all(12), + eyeStyle: const QrEyeStyle( + eyeShape: QrEyeShape.square, + color: Color(0xFF1A1A2E), + ), + dataModuleStyle: const QrDataModuleStyle( + dataModuleShape: QrDataModuleShape.square, + color: Color(0xFF1A1A2E), + ), + ), + ), + ); + } + + /// 构建地址信息 + Widget _buildAddressInfo() { + return Column( + children: [ + // 充值地址标题 + const Text( + '充值地址', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.25, + color: Color(0xFF5D4037), + ), + ), + const SizedBox(height: 12), + // 地址文本 + Container( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: Text( + _isLoading ? '加载中...' : _currentAddress, + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + height: 1.5, + color: const Color(0xFF5D4037).withValues(alpha: 0.8), + ), + textAlign: TextAlign.center, + ), + ), + const SizedBox(height: 20), + // 复制按钮 + GestureDetector( + onTap: _isLoading ? null : _copyAddress, + child: Container( + padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 6), + decoration: BoxDecoration( + color: const Color(0xFFD4AF37).withValues(alpha: 0.2), + borderRadius: BorderRadius.circular(9999), + ), + child: Row( + mainAxisSize: MainAxisSize.min, + children: [ + Text( + '复制地址', + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: _isLoading + ? const Color(0xFFB0A090) + : const Color(0xFF745D43), + ), + ), + const SizedBox(width: 8), + Icon( + Icons.copy_outlined, + size: 18, + color: _isLoading + ? const Color(0xFFB0A090) + : const Color(0xFF745D43), + ), + ], + ), + ), + ), + ], + ); + } + + /// 构建警告文字 + Widget _buildWarningText() { + return Container( + padding: const EdgeInsets.fromLTRB(20, 16, 20, 48), + child: Text( + '仅支持 USDT,错充将无法追回', + style: TextStyle( + fontSize: 12, + fontFamily: 'Inter', + height: 1.5, + color: const Color(0xFFF97316).withValues(alpha: 0.8), + ), + textAlign: TextAlign.center, + ), + ); + } + + /// 构建底部按钮 + Widget _buildBottomButton() { + return Container( + padding: const EdgeInsets.fromLTRB(16, 16, 16, 24), + child: GestureDetector( + onTap: _onDepositComplete, + child: Container( + width: double.infinity, + height: 55, + decoration: BoxDecoration( + color: const Color(0xFFD4AF37), + borderRadius: BorderRadius.circular(12), + boxShadow: const [ + BoxShadow( + color: Color(0x4DD4AF37), + blurRadius: 14, + offset: Offset(0, 4), + ), + ], + ), + child: const Center( + child: Text( + '我已完成充值', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.5, + color: Colors.white, + ), + ), + ), + ), + ), + ); + } +} diff --git a/frontend/mobile-app/lib/features/planting/presentation/pages/planting_location_page.dart b/frontend/mobile-app/lib/features/planting/presentation/pages/planting_location_page.dart new file mode 100644 index 00000000..b04277cc --- /dev/null +++ b/frontend/mobile-app/lib/features/planting/presentation/pages/planting_location_page.dart @@ -0,0 +1,496 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:go_router/go_router.dart'; +import 'package:city_pickers/city_pickers.dart'; +import '../widgets/planting_confirm_dialog.dart'; + +/// 认种省市选择页面参数 +class PlantingLocationParams { + final int quantity; + final double totalPrice; + + PlantingLocationParams({ + required this.quantity, + required this.totalPrice, + }); +} + +/// 认种省市选择页面 +/// 用户选择身份证所在省市,选择后不可修改 +class PlantingLocationPage extends ConsumerStatefulWidget { + final int quantity; + final double totalPrice; + + const PlantingLocationPage({ + super.key, + required this.quantity, + required this.totalPrice, + }); + + @override + ConsumerState createState() => + _PlantingLocationPageState(); +} + +class _PlantingLocationPageState extends ConsumerState { + /// 选中的省份 + String? _selectedProvince; + + /// 选中的城市 + String? _selectedCity; + + /// 是否正在提交 + bool _isSubmitting = false; + + /// 返回上一页 + void _goBack() { + context.pop(); + } + + /// 显示省市选择器(使用 city_pickers) + Future _showCityPicker() async { + final result = await CityPickers.showCityPicker( + context: context, + cancelWidget: const Text( + '取消', + style: TextStyle(color: Color(0xFF745D43), fontSize: 16), + ), + confirmWidget: const Text( + '确定', + style: TextStyle(color: Color(0xFFD4AF37), fontSize: 16, fontWeight: FontWeight.w600), + ), + height: 300, + showType: ShowType.pc, // 只显示省市两级 + barrierDismissible: true, + ); + + if (result != null) { + setState(() { + _selectedProvince = result.provinceName; + _selectedCity = result.cityName; + }); + } + } + + /// 显示省份选择器 + void _showProvinceSelector() { + _showCityPicker(); + } + + /// 显示城市选择器 + void _showCitySelector() { + _showCityPicker(); + } + + /// 确认选择 + void _confirmSelection() async { + if (_selectedProvince == null || _selectedCity == null) { + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar( + content: Text('请选择省份和城市'), + backgroundColor: Color(0xFFD4AF37), + ), + ); + return; + } + + // 显示确认弹窗(带5秒倒计时) + await PlantingConfirmDialog.show( + context: context, + province: _selectedProvince!, + city: _selectedCity!, + onConfirm: _submitPlanting, + ); + } + + /// 提交认种请求 + Future _submitPlanting() async { + setState(() => _isSubmitting = true); + + try { + // TODO: 调用 API 提交认种请求 + await Future.delayed(const Duration(seconds: 1)); + + if (mounted) { + // 显示成功提示 + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar( + content: Text('认种成功!'), + backgroundColor: Color(0xFF4CAF50), + ), + ); + + // 返回到个人中心 + context.go('/profile'); + } + } catch (e) { + debugPrint('认种失败: $e'); + if (mounted) { + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text('认种失败: $e'), + backgroundColor: Colors.red, + ), + ); + } + } finally { + if (mounted) { + setState(() => _isSubmitting = false); + } + } + } + + /// 是否可以提交 + bool get _canSubmit => + _selectedProvince != null && _selectedCity != null && !_isSubmitting; + + @override + Widget build(BuildContext context) { + return Scaffold( + body: Container( + width: double.infinity, + height: double.infinity, + decoration: const BoxDecoration( + gradient: LinearGradient( + begin: Alignment.topCenter, + end: Alignment.bottomCenter, + colors: [ + Color(0xFFFFF7E6), + Color(0xFFEAE0C8), + ], + ), + ), + child: SafeArea( + child: Column( + children: [ + // 顶部导航栏 + _buildHeader(), + // 内容区域 + Expanded( + child: SingleChildScrollView( + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + const SizedBox(height: 16), + // 提示文字 + _buildHintText(), + const SizedBox(height: 8), + // 省份选择 + _buildProvinceSelector(), + const SizedBox(height: 22), + // 城市选择 + _buildCitySelector(), + const SizedBox(height: 24), + // 警告提示 + _buildWarningCard(), + ], + ), + ), + ), + ), + // 底部按钮 + _buildBottomButton(), + ], + ), + ), + ), + ); + } + + /// 构建顶部导航栏 + Widget _buildHeader() { + return Container( + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12), + decoration: BoxDecoration( + color: const Color(0xFFFFF5E6).withValues(alpha: 0.8), + ), + child: Row( + children: [ + // 返回按钮 + GestureDetector( + onTap: _goBack, + child: Container( + width: 24, + height: 32, + alignment: Alignment.center, + child: const Icon( + Icons.arrow_back_ios, + color: Color(0xFFD4AF37), + size: 20, + ), + ), + ), + const SizedBox(width: 4), + // 返回文字 + GestureDetector( + onTap: _goBack, + child: const Text( + '返回', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFFD4AF37), + ), + ), + ), + const SizedBox(width: 48), + // 标题 + const Text( + '认种 Planting', + style: TextStyle( + fontSize: 18, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.25, + color: Color(0xFF5D4037), + ), + ), + ], + ), + ); + } + + /// 构建提示文字 + Widget _buildHintText() { + return const Padding( + padding: EdgeInsets.symmetric(vertical: 4), + child: Text( + '请选择身份所在省市(只可选择一次)', + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFF745D43), + ), + ), + ); + } + + /// 构建省份选择器 + Widget _buildProvinceSelector() { + return Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + // 标签 + const Text( + '省份(必选)', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFF5D4037), + ), + ), + const SizedBox(height: 7), + // 选择框 + GestureDetector( + onTap: _showProvinceSelector, + child: Container( + width: double.infinity, + padding: const EdgeInsets.symmetric(horizontal: 17, vertical: 13), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(8), + border: Border.all( + color: const Color(0x4D8B5A2B), + width: 1, + ), + ), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + _selectedProvince ?? '选择省份', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + height: 1.5, + color: _selectedProvince != null + ? const Color(0xFF5D4037) + : const Color(0xFF5D4037).withValues(alpha: 0.5), + ), + ), + const Icon( + Icons.keyboard_arrow_down, + color: Color(0xFF8B5A2B), + size: 24, + ), + ], + ), + ), + ), + ], + ); + } + + /// 构建城市选择器 + Widget _buildCitySelector() { + return Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + // 标签 + const Text( + '市级(必选)', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFF5D4037), + ), + ), + const SizedBox(height: 7), + // 选择框 + GestureDetector( + onTap: _showCitySelector, + child: Container( + width: double.infinity, + padding: const EdgeInsets.symmetric(horizontal: 17, vertical: 13), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(8), + border: Border.all( + color: const Color(0x4D8B5A2B), + width: 1, + ), + ), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + _selectedCity ?? '选择市级', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + height: 1.5, + color: _selectedCity != null + ? const Color(0xFF5D4037) + : const Color(0xFF5D4037).withValues(alpha: 0.5), + ), + ), + const Icon( + Icons.keyboard_arrow_down, + color: Color(0xFF8B5A2B), + size: 24, + ), + ], + ), + ), + ), + ], + ); + } + + /// 构建警告提示卡片 + Widget _buildWarningCard() { + return Container( + width: double.infinity, + padding: const EdgeInsets.all(16), + decoration: BoxDecoration( + color: const Color(0x33FFC107), + borderRadius: BorderRadius.circular(8), + ), + child: Row( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + // 警告图标 + Container( + width: 24, + height: 34, + alignment: Alignment.topCenter, + child: const Icon( + Icons.warning_amber_rounded, + color: Color(0xFFD4AF37), + size: 24, + ), + ), + const SizedBox(width: 12), + // 警告文字 + Expanded( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: const [ + Text( + '必须选择身份证所在省与市,', + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFF745D43), + ), + ), + Text( + '错误将无法获得 20 年榴莲果销售分成。', + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFF745D43), + ), + ), + Text( + '选择后不可修改!', + style: TextStyle( + fontSize: 14, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.5, + color: Color(0xFF745D43), + ), + ), + ], + ), + ), + ], + ), + ); + } + + /// 构建底部按钮 + Widget _buildBottomButton() { + return Container( + padding: const EdgeInsets.fromLTRB(16, 16, 16, 24), + child: GestureDetector( + onTap: _canSubmit ? _confirmSelection : null, + child: Container( + width: double.infinity, + height: 52, + decoration: BoxDecoration( + color: _canSubmit + ? const Color(0xFFD4AF37) + : const Color(0xFFD4AF37).withValues(alpha: 0.5), + borderRadius: BorderRadius.circular(12), + ), + child: Center( + child: _isSubmitting + ? const SizedBox( + width: 24, + height: 24, + child: CircularProgressIndicator( + strokeWidth: 2, + color: Colors.white, + ), + ) + : const Text( + '确认选择', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.5, + color: Colors.white, + ), + ), + ), + ), + ), + ); + } +} diff --git a/frontend/mobile-app/lib/features/planting/presentation/pages/planting_quantity_page.dart b/frontend/mobile-app/lib/features/planting/presentation/pages/planting_quantity_page.dart new file mode 100644 index 00000000..49fdd0bd --- /dev/null +++ b/frontend/mobile-app/lib/features/planting/presentation/pages/planting_quantity_page.dart @@ -0,0 +1,577 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:go_router/go_router.dart'; +import '../../../../routes/route_paths.dart'; +import '../../../../core/di/injection_container.dart'; +import 'planting_location_page.dart'; + +/// 认种数量选择页面 +/// 用户可以选择认种的榴莲树数量,根据可用余额自动计算最大可认种数量 +class PlantingQuantityPage extends ConsumerStatefulWidget { + const PlantingQuantityPage({super.key}); + + @override + ConsumerState createState() => + _PlantingQuantityPageState(); +} + +class _PlantingQuantityPageState extends ConsumerState { + /// 每棵树的价格 (USDT) + static const double _pricePerTree = 2199.0; + + /// 可用余额 (USDT) - 从 API 获取 + double _availableBalance = 0.0; + + /// 当前选择的数量 + int _quantity = 0; + + /// 是否正在加载 + bool _isLoading = false; + + /// 加载错误信息 + String? _errorMessage; + + @override + void initState() { + super.initState(); + _loadBalance(); + } + + /// 加载用户余额 + Future _loadBalance() async { + setState(() { + _isLoading = true; + _errorMessage = null; + }); + + try { + // 从 API 获取用户 USDT 余额 + final depositService = ref.read(depositServiceProvider); + final balanceResponse = await depositService.getUsdtBalances(); + + // 合并 KAVA 和 BSC 链的余额 + double totalBalance = 0.0; + if (balanceResponse.kava != null) { + totalBalance += double.tryParse(balanceResponse.kava!.balance) ?? 0.0; + } + if (balanceResponse.bsc != null) { + totalBalance += double.tryParse(balanceResponse.bsc!.balance) ?? 0.0; + } + + // 计算最大可认种数量并自动填入 + final maxQty = (totalBalance / _pricePerTree).floor(); + + setState(() { + _availableBalance = totalBalance; + // 自动填入最大可认种数量,至少为1(如果有足够余额) + _quantity = maxQty > 0 ? maxQty : 0; + _isLoading = false; + }); + } catch (e) { + debugPrint('加载余额失败: $e'); + setState(() { + _isLoading = false; + _errorMessage = '加载余额失败,请重试'; + _availableBalance = 0.0; + _quantity = 0; + }); + } + } + + /// 最大可认种数量 + int get _maxQuantity { + return (_availableBalance / _pricePerTree).floor(); + } + + /// 减少数量 + void _decreaseQuantity() { + if (_quantity > 1) { + setState(() => _quantity--); + } + } + + /// 增加数量 + void _increaseQuantity() { + if (_quantity < _maxQuantity) { + setState(() => _quantity++); + } + } + + /// 是否可以减少 + bool get _canDecrease => _quantity > 1; + + /// 是否可以增加 + bool get _canIncrease => _quantity < _maxQuantity; + + /// 返回上一页 + void _goBack() { + context.pop(); + } + + /// 下一步:选择省市 + void _goToNextStep() { + if (_quantity > 0 && _quantity <= _maxQuantity) { + context.push( + RoutePaths.plantingLocation, + extra: PlantingLocationParams( + quantity: _quantity, + totalPrice: _quantity * _pricePerTree, + ), + ); + } + } + + /// 格式化数字显示 + String _formatNumber(double number) { + final parts = number.toStringAsFixed(2).split('.'); + final intPart = parts[0].replaceAllMapped( + RegExp(r'(\d{1,3})(?=(\d{3})+(?!\d))'), + (Match m) => '${m[1]},', + ); + return '$intPart.${parts[1]}'; + } + + @override + Widget build(BuildContext context) { + return Scaffold( + body: Container( + width: double.infinity, + height: double.infinity, + decoration: const BoxDecoration( + gradient: LinearGradient( + begin: Alignment.topCenter, + end: Alignment.bottomCenter, + colors: [ + Color(0xFFFFF7E6), + Color(0xFFEAE0C8), + ], + ), + ), + child: SafeArea( + child: Column( + children: [ + // 顶部导航栏 + _buildHeader(), + // 内容区域 + Expanded( + child: SingleChildScrollView( + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + const SizedBox(height: 16), + // 余额卡片 + _buildBalanceCard(), + const SizedBox(height: 24), + // 输入认种数量标题 + _buildSectionTitle(), + const SizedBox(height: 8), + // 数量选择器 + _buildQuantitySelector(), + const SizedBox(height: 8), + // 价格信息 + _buildPriceInfo(), + ], + ), + ), + ), + ), + // 底部按钮 + _buildBottomButton(), + ], + ), + ), + ), + ); + } + + /// 构建顶部导航栏 + Widget _buildHeader() { + return Container( + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), + decoration: BoxDecoration( + color: const Color(0xFFFFF7E6).withValues(alpha: 0.8), + ), + child: Row( + children: [ + // 返回按钮 + GestureDetector( + onTap: _goBack, + child: Container( + width: 32, + height: 32, + alignment: Alignment.center, + child: const Icon( + Icons.arrow_back_ios, + color: Color(0xFFD4AF37), + size: 20, + ), + ), + ), + const SizedBox(width: 4), + // 返回文字 + GestureDetector( + onTap: _goBack, + child: const Text( + '返回', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFFD4AF37), + ), + ), + ), + const SizedBox(width: 42), + // 标题 + const Expanded( + child: Text( + '认种 Planting', + style: TextStyle( + fontSize: 18, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.25, + letterSpacing: -0.27, + color: Color(0xFF5D4037), + ), + ), + ), + ], + ), + ); + } + + /// 构建余额卡片 + Widget _buildBalanceCard() { + return Container( + width: double.infinity, + padding: const EdgeInsets.all(16), + decoration: BoxDecoration( + color: const Color(0x99FFFFFF), + borderRadius: BorderRadius.circular(12), + boxShadow: const [ + BoxShadow( + color: Color(0x1A000000), + blurRadius: 6, + offset: Offset(0, 4), + ), + BoxShadow( + color: Color(0x1A000000), + blurRadius: 4, + offset: Offset(0, 2), + ), + ], + ), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + // 标签行(包含刷新按钮) + Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + const Text( + '可用余额 (USDT)', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFF745D43), + ), + ), + // 刷新按钮 + if (!_isLoading) + GestureDetector( + onTap: _loadBalance, + child: const Icon( + Icons.refresh, + color: Color(0xFFD4AF37), + size: 20, + ), + ), + ], + ), + const SizedBox(height: 3), + // 余额数值或错误信息 + _isLoading + ? const SizedBox( + height: 45, + child: Center( + child: CircularProgressIndicator( + strokeWidth: 2, + color: Color(0xFFD4AF37), + ), + ), + ) + : _errorMessage != null + ? GestureDetector( + onTap: _loadBalance, + child: SizedBox( + height: 45, + child: Row( + children: [ + const Icon( + Icons.error_outline, + color: Color(0xFFE65100), + size: 24, + ), + const SizedBox(width: 8), + Text( + _errorMessage!, + style: const TextStyle( + fontSize: 14, + color: Color(0xFFE65100), + ), + ), + const SizedBox(width: 8), + const Text( + '点击重试', + style: TextStyle( + fontSize: 14, + color: Color(0xFFD4AF37), + fontWeight: FontWeight.w600, + ), + ), + ], + ), + ), + ) + : Text( + _formatNumber(_availableBalance), + style: const TextStyle( + fontSize: 36, + fontFamily: 'Inter', + fontWeight: FontWeight.w400, + height: 1.25, + letterSpacing: -0.54, + color: Color(0xFF5D4037), + ), + ), + ], + ), + ); + } + + /// 构建区域标题 + Widget _buildSectionTitle() { + return Padding( + padding: const EdgeInsets.only(top: 24), + child: const Text( + '输入认种数量', + style: TextStyle( + fontSize: 24, + fontFamily: 'Inter', + fontWeight: FontWeight.w400, + height: 1.25, + color: Color(0xFF5D4037), + ), + ), + ); + } + + /// 构建数量选择器 + Widget _buildQuantitySelector() { + return Container( + width: double.infinity, + padding: const EdgeInsets.all(24), + decoration: BoxDecoration( + color: const Color(0x99FFFFFF), + borderRadius: BorderRadius.circular(12), + boxShadow: const [ + BoxShadow( + color: Color(0x1A000000), + blurRadius: 6, + offset: Offset(0, 4), + ), + BoxShadow( + color: Color(0x1A000000), + blurRadius: 4, + offset: Offset(0, 2), + ), + ], + ), + child: Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + // 减少按钮 + _buildQuantityButton( + icon: '-', + onTap: _canDecrease ? _decreaseQuantity : null, + enabled: _canDecrease, + ), + const SizedBox(width: 38), + // 数量显示 + Text( + '$_quantity', + style: const TextStyle( + fontSize: 36, + fontFamily: 'Inter', + fontWeight: FontWeight.w400, + height: 1.5, + color: Color(0xFF5D4037), + ), + ), + const SizedBox(width: 38), + // 增加按钮 + _buildQuantityButton( + icon: '+', + onTap: _canIncrease ? _increaseQuantity : null, + enabled: _canIncrease, + ), + ], + ), + ); + } + + /// 构建数量调节按钮 + Widget _buildQuantityButton({ + required String icon, + required VoidCallback? onTap, + required bool enabled, + }) { + return GestureDetector( + onTap: onTap, + child: Container( + width: 48, + height: 48, + decoration: BoxDecoration( + color: const Color(0xFFFFF7E6), + borderRadius: BorderRadius.circular(9999), + boxShadow: const [ + BoxShadow( + color: Color(0x1A000000), + blurRadius: 3, + offset: Offset(0, 1), + ), + BoxShadow( + color: Color(0x1A000000), + blurRadius: 2, + offset: Offset(0, 1), + ), + ], + ), + child: Opacity( + opacity: enabled ? 1.0 : 0.4, + child: Center( + child: Text( + icon, + style: const TextStyle( + fontSize: 30, + fontFamily: 'Inter', + fontWeight: FontWeight.w400, + height: 1.5, + color: Color(0xFF745D43), + ), + ), + ), + ), + ), + ); + } + + /// 构建价格信息 + Widget _buildPriceInfo() { + final calculationResult = (_availableBalance / _pricePerTree).toStringAsFixed(2); + + return Padding( + padding: const EdgeInsets.only(top: 8), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + // 每棵价格 + Text( + '每棵价格:${_pricePerTree.toInt()} USDT', + style: const TextStyle( + fontSize: 16, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFF000000), + ), + ), + const SizedBox(height: 7), + // 最大可认种 + Text( + '最大可认种:$_maxQuantity 棵 (根据余额自动计算)', + style: const TextStyle( + fontSize: 16, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFF000000), + ), + ), + const SizedBox(height: 7), + // 计算说明 + Text( + '(计算: ${_formatNumber(_availableBalance).replaceAll('.00', '')} \u00f7 ${_pricePerTree.toInt()} = $calculationResult \u2192 向下取整为 $_maxQuantity)', + style: const TextStyle( + fontSize: 14, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFF000000), + ), + ), + ], + ), + ); + } + + /// 构建底部按钮 + Widget _buildBottomButton() { + final bool canProceed = _quantity > 0 && _quantity <= _maxQuantity && !_isLoading; + + return Container( + padding: const EdgeInsets.fromLTRB(16, 16, 16, 24), + child: GestureDetector( + onTap: canProceed ? _goToNextStep : null, + child: Container( + width: double.infinity, + height: 56, + decoration: BoxDecoration( + color: canProceed + ? const Color(0xFFD4AF37) + : const Color(0xFFD4AF37).withValues(alpha: 0.5), + borderRadius: BorderRadius.circular(12), + boxShadow: canProceed + ? const [ + BoxShadow( + color: Color(0x1A000000), + blurRadius: 15, + offset: Offset(0, 10), + ), + BoxShadow( + color: Color(0x1A000000), + blurRadius: 6, + offset: Offset(0, 4), + ), + ] + : null, + ), + child: const Center( + child: Text( + '下一步:选择省市', + style: TextStyle( + fontSize: 18, + fontFamily: 'Inter', + fontWeight: FontWeight.w500, + height: 1.56, + color: Colors.white, + ), + ), + ), + ), + ), + ); + } +} + +/// 认种数量参数 - 传递给下一个页面 +class PlantingQuantityParams { + final int quantity; + final double totalPrice; + + PlantingQuantityParams({ + required this.quantity, + required this.totalPrice, + }); +} diff --git a/frontend/mobile-app/lib/features/planting/presentation/widgets/planting_confirm_dialog.dart b/frontend/mobile-app/lib/features/planting/presentation/widgets/planting_confirm_dialog.dart new file mode 100644 index 00000000..c1eca5f0 --- /dev/null +++ b/frontend/mobile-app/lib/features/planting/presentation/widgets/planting_confirm_dialog.dart @@ -0,0 +1,280 @@ +import 'dart:async'; +import 'package:flutter/material.dart'; + +/// 认种确认弹窗 +/// 带有5秒倒计时功能,倒计时结束后按钮才可点击 +class PlantingConfirmDialog extends StatefulWidget { + /// 选中的省份 + final String province; + + /// 选中的城市 + final String city; + + /// 确认回调 + final VoidCallback onConfirm; + + const PlantingConfirmDialog({ + super.key, + required this.province, + required this.city, + required this.onConfirm, + }); + + /// 显示确认弹窗 + static Future show({ + required BuildContext context, + required String province, + required String city, + required VoidCallback onConfirm, + }) { + return showDialog( + context: context, + barrierDismissible: false, + barrierColor: const Color(0x80000000), + builder: (context) => PlantingConfirmDialog( + province: province, + city: city, + onConfirm: onConfirm, + ), + ); + } + + @override + State createState() => _PlantingConfirmDialogState(); +} + +class _PlantingConfirmDialogState extends State { + /// 倒计时秒数 + int _countdown = 5; + + /// 倒计时定时器 + Timer? _timer; + + /// 是否可以点击确认按钮 + bool get _canConfirm => _countdown <= 0; + + @override + void initState() { + super.initState(); + _startCountdown(); + } + + @override + void dispose() { + _timer?.cancel(); + super.dispose(); + } + + /// 开始倒计时 + void _startCountdown() { + _timer = Timer.periodic(const Duration(seconds: 1), (timer) { + if (_countdown > 0) { + setState(() => _countdown--); + } else { + timer.cancel(); + } + }); + } + + /// 点击确认按钮 + void _handleConfirm() { + if (_canConfirm) { + Navigator.pop(context, true); + widget.onConfirm(); + } + } + + @override + Widget build(BuildContext context) { + return Dialog( + backgroundColor: Colors.transparent, + insetPadding: const EdgeInsets.symmetric(horizontal: 16), + child: Container( + width: double.infinity, + constraints: const BoxConstraints(maxWidth: 384), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(12), + boxShadow: const [ + BoxShadow( + color: Color(0x40000000), + blurRadius: 50, + offset: Offset(0, 25), + spreadRadius: -12, + ), + ], + ), + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 32), + child: Column( + mainAxisSize: MainAxisSize.min, + children: [ + // 警告图标 + _buildWarningIcon(), + const SizedBox(height: 4), + // 标题 + _buildTitle(), + const SizedBox(height: 4), + // 内容说明 + _buildContent(), + const SizedBox(height: 4), + // 倒计时提示 + _buildCountdownText(), + const SizedBox(height: 4), + // 确认按钮 + _buildConfirmButton(), + ], + ), + ), + ), + ); + } + + /// 构建警告图标 + Widget _buildWarningIcon() { + return Padding( + padding: const EdgeInsets.symmetric(vertical: 16), + child: Container( + width: 60, + height: 60, + decoration: BoxDecoration( + color: const Color(0xFFFFF7E6), + borderRadius: BorderRadius.circular(30), + ), + child: const Center( + child: Icon( + Icons.warning_amber_rounded, + color: Color(0xFF5D4037), + size: 36, + ), + ), + ), + ); + } + + /// 构建标题 + Widget _buildTitle() { + return const Text( + '最后确认!不可修改!', + style: TextStyle( + fontSize: 22, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.25, + letterSpacing: -0.33, + color: Color(0xFF5D4037), + ), + textAlign: TextAlign.center, + ); + } + + /// 构建内容说明 + Widget _buildContent() { + return Padding( + padding: const EdgeInsets.only(top: 8, bottom: 20), + child: Column( + children: [ + // 第一行:您选择的省市为 + RichText( + textAlign: TextAlign.center, + text: TextSpan( + style: const TextStyle( + fontSize: 16, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFF5D4037), + ), + children: [ + const TextSpan(text: '您选择的省市为: '), + TextSpan( + text: '【${widget.province} · ${widget.city}】', + style: const TextStyle( + fontWeight: FontWeight.w700, + color: Color(0xFF745D43), + ), + ), + ], + ), + ), + const SizedBox(height: 4), + // 第二行:警告信息 + RichText( + textAlign: TextAlign.center, + text: const TextSpan( + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFFFF4D4F), + ), + children: [ + TextSpan(text: '若错误,将失去 '), + TextSpan( + text: '20 年收益分成', + style: TextStyle(fontWeight: FontWeight.w700), + ), + ], + ), + ), + ], + ), + ); + } + + /// 构建倒计时文字 + Widget _buildCountdownText() { + return Padding( + padding: const EdgeInsets.symmetric(vertical: 16), + child: Text( + '按钮倒计时:$_countdown 秒', + style: const TextStyle( + fontSize: 14, + fontFamily: 'Inter', + height: 1.5, + color: Color(0xFF745D43), + ), + ), + ); + } + + /// 构建确认按钮 + Widget _buildConfirmButton() { + return Padding( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: SizedBox( + width: double.infinity, + child: GestureDetector( + onTap: _canConfirm ? _handleConfirm : null, + child: Container( + height: 48, + constraints: const BoxConstraints( + minWidth: 84, + maxWidth: 480, + ), + decoration: BoxDecoration( + color: _canConfirm + ? const Color(0xFFD4AF37) + : const Color(0xFFD4AF37).withValues(alpha: 0.2), + borderRadius: BorderRadius.circular(8), + ), + child: Center( + child: Text( + _canConfirm ? '确认认种' : '确认认种 (${_countdown}s)', + style: TextStyle( + fontSize: 16, + fontFamily: 'Inter', + fontWeight: FontWeight.w700, + height: 1.5, + letterSpacing: 0.24, + color: _canConfirm + ? Colors.white + : const Color(0xFFD4AF37).withValues(alpha: 0.4), + ), + ), + ), + ), + ), + ), + ); + } +} diff --git a/frontend/mobile-app/lib/features/profile/presentation/pages/profile_page.dart b/frontend/mobile-app/lib/features/profile/presentation/pages/profile_page.dart index 8ee10cb5..e8a78f46 100644 --- a/frontend/mobile-app/lib/features/profile/presentation/pages/profile_page.dart +++ b/frontend/mobile-app/lib/features/profile/presentation/pages/profile_page.dart @@ -145,7 +145,7 @@ class _ProfilePageState extends ConsumerState { /// 充值USDT void _onDeposit() { - context.push(RoutePaths.deposit); + context.push(RoutePaths.depositUsdt); } /// 进入交易 diff --git a/frontend/mobile-app/lib/features/share/presentation/pages/share_page.dart b/frontend/mobile-app/lib/features/share/presentation/pages/share_page.dart new file mode 100644 index 00000000..c7155218 --- /dev/null +++ b/frontend/mobile-app/lib/features/share/presentation/pages/share_page.dart @@ -0,0 +1,351 @@ +import 'package:flutter/material.dart'; +import 'package:flutter/services.dart'; +import 'package:flutter_riverpod/flutter_riverpod.dart'; +import 'package:go_router/go_router.dart'; +import 'package:qr_flutter/qr_flutter.dart'; +import '../../../../core/di/injection_container.dart'; + +/// 分享页面 - 显示邀请链接和二维码 +/// 进入页面时自动调用 API 生成短链,显示二维码 +class SharePage extends ConsumerStatefulWidget { + /// 初始分享链接 (fallback) + final String shareLink; + /// 邀请码 + final String? referralCode; + + const SharePage({ + super.key, + required this.shareLink, + this.referralCode, + }); + + @override + ConsumerState createState() => _SharePageState(); +} + +class _SharePageState extends ConsumerState { + /// 实际显示的分享链接 (API 返回的短链) + late String _displayLink; + /// 是否正在加载 + bool _isLoading = true; + /// 错误信息 + String? _errorMessage; + + @override + void initState() { + super.initState(); + _displayLink = widget.shareLink; // 默认使用传入的链接 + _loadShareLink(); + } + + /// 加载分享链接 (调用 API 生成短链) + Future _loadShareLink() async { + try { + setState(() { + _isLoading = true; + _errorMessage = null; + }); + + final referralService = ref.read(referralServiceProvider); + + // 调用 API 生成短链 + final linkResponse = await referralService.generateReferralLink(); + + if (mounted) { + setState(() { + // 优先使用短链,如果没有则使用完整链接 + _displayLink = linkResponse.shortUrl.isNotEmpty + ? linkResponse.shortUrl + : linkResponse.fullUrl; + _isLoading = false; + }); + } + } catch (e) { + debugPrint('加载分享链接失败: $e'); + if (mounted) { + setState(() { + _isLoading = false; + // 失败时使用传入的默认链接 + _displayLink = widget.shareLink; + _errorMessage = '加载短链失败,使用默认链接'; + }); + } + } + } + + /// 复制链接到剪贴板 + void _copyLink() { + Clipboard.setData(ClipboardData(text: _displayLink)); + ScaffoldMessenger.of(context).showSnackBar( + const SnackBar( + content: Text('链接已复制'), + backgroundColor: Color(0xFFD4A84B), + duration: Duration(seconds: 2), + ), + ); + } + + /// 返回上一页 + void _goBack() { + context.pop(); + } + + /// 确认按钮点击 + void _onConfirm() { + context.pop(); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + body: Container( + width: double.infinity, + height: double.infinity, + // 渐变背景 - 从浅黄到白色 + decoration: const BoxDecoration( + gradient: LinearGradient( + begin: Alignment.topCenter, + end: Alignment.bottomCenter, + colors: [ + Color(0xFFFFF7E6), // 浅黄色顶部 + Color(0xFFFFFDF8), // 接近白色底部 + ], + ), + ), + child: SafeArea( + child: Column( + children: [ + // 顶部导航栏 + _buildAppBar(), + // 内容区域 + Expanded( + child: SingleChildScrollView( + child: Column( + children: [ + const SizedBox(height: 100), + // QR 码区域 + _buildQrCodeSection(), + const SizedBox(height: 40), + // 链接输入框 + _buildLinkSection(), + ], + ), + ), + ), + // 底部确认按钮 + _buildFooter(), + ], + ), + ), + ), + ); + } + + /// 构建顶部导航栏 + Widget _buildAppBar() { + return Container( + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 14), + child: Row( + children: [ + // 返回按钮 + GestureDetector( + onTap: _goBack, + child: Container( + width: 28, + height: 28, + alignment: Alignment.center, + child: const Icon( + Icons.arrow_back_ios, + color: Color(0xFF8C6A3E), + size: 20, + ), + ), + ), + const SizedBox(width: 8), + // 标题 + const Expanded( + child: Text( + '分享页面', + style: TextStyle( + fontSize: 18, + fontFamily: 'Segoe UI', + fontWeight: FontWeight.w600, + height: 1.56, + color: Color(0xFF8C6A3E), + ), + ), + ), + ], + ), + ); + } + + /// 构建 QR 码区域 + Widget _buildQrCodeSection() { + return Container( + width: 256, + height: 256, + decoration: BoxDecoration( + color: const Color(0xFFFAF3E3), + borderRadius: BorderRadius.circular(24), + boxShadow: const [ + BoxShadow( + color: Color(0x1A8C6A3E), // rgba(140, 106, 62, 0.1) + blurRadius: 30, + offset: Offset(0, 8), + ), + ], + ), + child: Center( + child: _isLoading + ? const CircularProgressIndicator( + color: Color(0xFFD4A84B), + strokeWidth: 2, + ) + : QrImageView( + data: _displayLink, + version: QrVersions.auto, + size: 200, + backgroundColor: Colors.transparent, + eyeStyle: const QrEyeStyle( + eyeShape: QrEyeShape.square, + color: Color(0xFF8C6A3E), + ), + dataModuleStyle: const QrDataModuleStyle( + dataModuleShape: QrDataModuleShape.square, + color: Color(0xFF8C6A3E), + ), + ), + ), + ); + } + + /// 构建链接显示区域 + Widget _buildLinkSection() { + return Padding( + padding: const EdgeInsets.symmetric(horizontal: 32), + child: Column( + children: [ + Container( + width: double.infinity, + constraints: const BoxConstraints(maxWidth: 296), + padding: const EdgeInsets.fromLTRB(16, 8, 12, 8), + decoration: BoxDecoration( + color: const Color(0xFFF8F1E2), + borderRadius: BorderRadius.circular(9999), // 胶囊形状 + ), + child: Row( + children: [ + // 链接文本 + Expanded( + child: _isLoading + ? const Text( + '加载中...', + style: TextStyle( + fontSize: 13.7, + fontFamily: 'Segoe UI', + height: 1.5, + color: Color(0xFFB0A090), + ), + ) + : Text( + _displayLink, + style: const TextStyle( + fontSize: 13.7, + fontFamily: 'Segoe UI', + height: 1.5, + color: Color(0xFF8C6A3E), + ), + overflow: TextOverflow.ellipsis, + ), + ), + const SizedBox(width: 8), + // 复制按钮 + GestureDetector( + onTap: _isLoading ? null : _copyLink, + child: Container( + width: 32, + height: 32, + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(9999), + boxShadow: const [ + BoxShadow( + color: Color(0x0D000000), // rgba(0, 0, 0, 0.05) + blurRadius: 2, + offset: Offset(0, 1), + ), + ], + ), + child: Center( + child: Icon( + Icons.copy_outlined, + color: _isLoading + ? const Color(0xFFB0A090) + : const Color(0xFF8C6A3E), + size: 18, + ), + ), + ), + ), + ], + ), + ), + // 错误提示 + if (_errorMessage != null) ...[ + const SizedBox(height: 8), + Text( + _errorMessage!, + style: const TextStyle( + fontSize: 12, + color: Color(0xFFB0A090), + ), + ), + ], + ], + ), + ); + } + + /// 构建底部确认按钮 + Widget _buildFooter() { + return Container( + padding: const EdgeInsets.all(24), + child: GestureDetector( + onTap: _onConfirm, + child: Container( + width: double.infinity, + height: 60, + decoration: BoxDecoration( + color: const Color(0xFFD1A45B), + borderRadius: BorderRadius.circular(12), + boxShadow: const [ + BoxShadow( + color: Color(0x4DD1A45B), // rgba(209, 164, 91, 0.3) + blurRadius: 15, + offset: Offset(0, 10), + ), + BoxShadow( + color: Color(0x4DD1A45B), // rgba(209, 164, 91, 0.3) + blurRadius: 6, + offset: Offset(0, 4), + ), + ], + ), + child: const Center( + child: Text( + '确认', + style: TextStyle( + fontSize: 18, + fontFamily: 'Segoe UI', + fontWeight: FontWeight.w700, + height: 1.56, + color: Colors.white, + ), + ), + ), + ), + ), + ); + } +} diff --git a/frontend/mobile-app/lib/routes/app_router.dart b/frontend/mobile-app/lib/routes/app_router.dart index 11f24ba0..57011d86 100644 --- a/frontend/mobile-app/lib/routes/app_router.dart +++ b/frontend/mobile-app/lib/routes/app_router.dart @@ -14,6 +14,10 @@ import '../features/mining/presentation/pages/mining_page.dart'; import '../features/trading/presentation/pages/trading_page.dart'; import '../features/profile/presentation/pages/profile_page.dart'; import '../features/profile/presentation/pages/edit_profile_page.dart'; +import '../features/share/presentation/pages/share_page.dart'; +import '../features/deposit/presentation/pages/deposit_usdt_page.dart'; +import '../features/planting/presentation/pages/planting_quantity_page.dart'; +import '../features/planting/presentation/pages/planting_location_page.dart'; import 'route_paths.dart'; import 'route_names.dart'; @@ -52,12 +56,25 @@ class WalletCreatedParams { final String dstAddress; final String bscAddress; final String serialNumber; + final String? referralCode; WalletCreatedParams({ required this.kavaAddress, required this.dstAddress, required this.bscAddress, required this.serialNumber, + this.referralCode, + }); +} + +/// 分享页面参数 +class SharePageParams { + final String shareLink; + final String? referralCode; + + SharePageParams({ + required this.shareLink, + this.referralCode, }); } @@ -134,6 +151,7 @@ final appRouterProvider = Provider((ref) { dstAddress: params.dstAddress, bscAddress: params.bscAddress, serialNumber: params.serialNumber, + referralCode: params.referralCode, ); }, ), @@ -145,6 +163,46 @@ final appRouterProvider = Provider((ref) { builder: (context, state) => const EditProfilePage(), ), + // Share Page (分享页面) + GoRoute( + path: RoutePaths.share, + name: RouteNames.share, + builder: (context, state) { + final params = state.extra as SharePageParams; + return SharePage( + shareLink: params.shareLink, + referralCode: params.referralCode, + ); + }, + ), + + // Deposit USDT Page (充值 USDT) + GoRoute( + path: RoutePaths.depositUsdt, + name: RouteNames.depositUsdt, + builder: (context, state) => const DepositUsdtPage(), + ), + + // Planting Quantity Page (认种 - 选择数量) + GoRoute( + path: RoutePaths.plantingQuantity, + name: RouteNames.plantingQuantity, + builder: (context, state) => const PlantingQuantityPage(), + ), + + // Planting Location Page (认种 - 选择省市) + GoRoute( + path: RoutePaths.plantingLocation, + name: RouteNames.plantingLocation, + builder: (context, state) { + final params = state.extra as PlantingLocationParams; + return PlantingLocationPage( + quantity: params.quantity, + totalPrice: params.totalPrice, + ); + }, + ), + // Main Shell with Bottom Navigation ShellRoute( navigatorKey: _shellNavigatorKey, diff --git a/frontend/mobile-app/lib/routes/route_names.dart b/frontend/mobile-app/lib/routes/route_names.dart index 15cb351e..509aa62f 100644 --- a/frontend/mobile-app/lib/routes/route_names.dart +++ b/frontend/mobile-app/lib/routes/route_names.dart @@ -22,10 +22,14 @@ class RouteNames { static const referralList = 'referral-list'; static const earningsDetail = 'earnings-detail'; static const deposit = 'deposit'; + static const depositUsdt = 'deposit-usdt'; static const plantingQuantity = 'planting-quantity'; static const plantingLocation = 'planting-location'; static const googleAuth = 'google-auth'; static const changePassword = 'change-password'; static const bindEmail = 'bind-email'; static const transactionHistory = 'transaction-history'; + + // Share + static const share = 'share'; } diff --git a/frontend/mobile-app/lib/routes/route_paths.dart b/frontend/mobile-app/lib/routes/route_paths.dart index de8ec03b..5b44ceb5 100644 --- a/frontend/mobile-app/lib/routes/route_paths.dart +++ b/frontend/mobile-app/lib/routes/route_paths.dart @@ -22,10 +22,14 @@ class RoutePaths { static const referralList = '/profile/referrals'; static const earningsDetail = '/profile/earnings'; static const deposit = '/deposit'; + static const depositUsdt = '/deposit/usdt'; static const plantingQuantity = '/planting/quantity'; static const plantingLocation = '/planting/location'; static const googleAuth = '/security/google-auth'; static const changePassword = '/security/password'; static const bindEmail = '/security/email'; static const transactionHistory = '/trading/history'; + + // Share + static const share = '/share'; } diff --git a/frontend/mobile-app/pubspec.lock b/frontend/mobile-app/pubspec.lock index b09f2d26..18391069 100644 --- a/frontend/mobile-app/pubspec.lock +++ b/frontend/mobile-app/pubspec.lock @@ -169,6 +169,14 @@ packages: url: "https://pub.dev" source: hosted version: "2.0.4" + city_pickers: + dependency: "direct main" + description: + name: city_pickers + sha256: "583102c8d9eecb1f7abc5ff52a22d7cb019b9808cdb24b80c7692c769f8da153" + url: "https://pub.dev" + source: hosted + version: "1.3.0" cli_util: dependency: transitive description: @@ -861,6 +869,14 @@ packages: url: "https://pub.dev" source: hosted version: "3.3.2" + lpinyin: + dependency: transitive + description: + name: lpinyin + sha256: "0bb843363f1f65170efd09fbdfc760c7ec34fc6354f9fcb2f89e74866a0d814a" + url: "https://pub.dev" + source: hosted + version: "2.0.3" matcher: dependency: transitive description: @@ -1181,6 +1197,14 @@ packages: url: "https://pub.dev" source: hosted version: "0.28.0" + scrollable_positioned_list: + dependency: transitive + description: + name: scrollable_positioned_list + sha256: "1b54d5f1329a1e263269abc9e2543d90806131aa14fe7c6062a8054d57249287" + url: "https://pub.dev" + source: hosted + version: "0.3.8" sec: dependency: transitive description: diff --git a/frontend/mobile-app/pubspec.yaml b/frontend/mobile-app/pubspec.yaml index 244da4ef..f712cb21 100644 --- a/frontend/mobile-app/pubspec.yaml +++ b/frontend/mobile-app/pubspec.yaml @@ -42,6 +42,7 @@ dependencies: lottie: ^3.1.0 qr_flutter: ^4.1.0 flutter_screenutil: ^5.9.0 + city_pickers: ^1.3.0 # 工具 intl: ^0.20.2 diff --git a/frontend/mobile-app/test/core/services/account_service_test.dart b/frontend/mobile-app/test/core/services/account_service_test.dart new file mode 100644 index 00000000..0a4cb5c0 --- /dev/null +++ b/frontend/mobile-app/test/core/services/account_service_test.dart @@ -0,0 +1,380 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:mocktail/mocktail.dart'; +import 'package:dio/dio.dart'; +import 'package:rwa_android_app/core/services/account_service.dart'; +import 'package:rwa_android_app/core/network/api_client.dart'; +import 'package:rwa_android_app/core/storage/secure_storage.dart'; +import 'package:rwa_android_app/core/storage/storage_keys.dart'; + +// Mock classes using mocktail +class MockApiClient extends Mock implements ApiClient {} + +class MockSecureStorage extends Mock implements SecureStorage {} + +void main() { + late AccountService accountService; + late MockApiClient mockApiClient; + late MockSecureStorage mockSecureStorage; + + setUpAll(() { + // Register fallback values for any() matchers + registerFallbackValue(RequestOptions(path: '')); + }); + + setUp(() { + mockApiClient = MockApiClient(); + mockSecureStorage = MockSecureStorage(); + accountService = AccountService( + apiClient: mockApiClient, + secureStorage: mockSecureStorage, + ); + }); + + group('AccountService', () { + group('createAccount', () { + test('should create account and save data to secure storage', () async { + // Arrange + const testDeviceId = 'test-device-123'; + final mockResponse = Response>( + data: { + 'userId': '123456789', + 'accountSequence': 1, + 'referralCode': 'ABC123', + 'mnemonic': '', + 'clientShareData': 'mock-client-share-data', + 'publicKey': 'mock-public-key', + 'walletAddresses': { + 'kava': '0x1234567890abcdef1234567890abcdef12345678', + 'dst': 'dst1abcdefghijklmnopqrstuvwxyz123456789', + 'bsc': '0x1234567890abcdef1234567890abcdef12345678', + }, + 'accessToken': 'mock-access-token', + 'refreshToken': 'mock-refresh-token', + }, + requestOptions: RequestOptions(path: '/user/auto-create'), + statusCode: 201, + ); + + // Setup mocks + when(() => mockSecureStorage.read(key: StorageKeys.deviceId)) + .thenAnswer((_) async => testDeviceId); + when(() => mockApiClient.post(any(), data: any(named: 'data'))) + .thenAnswer((_) async => mockResponse); + when(() => mockSecureStorage.write(key: any(named: 'key'), value: any(named: 'value'))) + .thenAnswer((_) async {}); + + // Act + final result = await accountService.createAccount(); + + // Assert + expect(result.userId, '123456789'); + expect(result.accountSequence, 1); + expect(result.referralCode, 'ABC123'); + expect(result.clientShareData, 'mock-client-share-data'); + expect(result.publicKey, 'mock-public-key'); + expect(result.walletAddresses.kava, '0x1234567890abcdef1234567890abcdef12345678'); + expect(result.walletAddresses.dst, 'dst1abcdefghijklmnopqrstuvwxyz123456789'); + expect(result.walletAddresses.bsc, '0x1234567890abcdef1234567890abcdef12345678'); + expect(result.accessToken, 'mock-access-token'); + expect(result.refreshToken, 'mock-refresh-token'); + + // Verify storage calls + verify(() => mockSecureStorage.write( + key: StorageKeys.userId, + value: '123456789', + )).called(1); + verify(() => mockSecureStorage.write( + key: StorageKeys.accountSequence, + value: '1', + )).called(1); + verify(() => mockSecureStorage.write( + key: StorageKeys.referralCode, + value: 'ABC123', + )).called(1); + verify(() => mockSecureStorage.write( + key: StorageKeys.accessToken, + value: 'mock-access-token', + )).called(1); + verify(() => mockSecureStorage.write( + key: StorageKeys.refreshToken, + value: 'mock-refresh-token', + )).called(1); + verify(() => mockSecureStorage.write( + key: StorageKeys.isWalletCreated, + value: 'true', + )).called(1); + }); + + test('should handle MPC mode without mnemonic', () async { + // Arrange + const testDeviceId = 'test-device-123'; + final mockResponse = Response>( + data: { + 'userId': '123456789', + 'accountSequence': 1, + 'referralCode': 'ABC123', + 'mnemonic': '', // Empty in MPC mode + 'clientShareData': 'client-share-data', + 'publicKey': 'public-key', + 'walletAddresses': { + 'kava': '0x1234', + 'dst': 'dst1abc', + 'bsc': '0x1234', + }, + 'accessToken': 'token', + 'refreshToken': 'refresh', + }, + requestOptions: RequestOptions(path: '/user/auto-create'), + statusCode: 201, + ); + + when(() => mockSecureStorage.read(key: StorageKeys.deviceId)) + .thenAnswer((_) async => testDeviceId); + when(() => mockApiClient.post(any(), data: any(named: 'data'))) + .thenAnswer((_) async => mockResponse); + when(() => mockSecureStorage.write(key: any(named: 'key'), value: any(named: 'value'))) + .thenAnswer((_) async {}); + + // Act + final result = await accountService.createAccount(); + + // Assert - MPC data should be stored + expect(result.mnemonic, isEmpty); + expect(result.clientShareData, 'client-share-data'); + expect(result.publicKey, 'public-key'); + + verify(() => mockSecureStorage.write( + key: StorageKeys.mpcClientShareData, + value: 'client-share-data', + )).called(1); + verify(() => mockSecureStorage.write( + key: StorageKeys.mpcPublicKey, + value: 'public-key', + )).called(1); + }); + }); + + group('getOrCreateDeviceId', () { + test('should return existing device id if available', () async { + // Arrange + const existingDeviceId = 'existing-device-123'; + when(() => mockSecureStorage.read(key: StorageKeys.deviceId)) + .thenAnswer((_) async => existingDeviceId); + + // Act + final result = await accountService.getOrCreateDeviceId(); + + // Assert + expect(result, existingDeviceId); + verifyNever(() => mockSecureStorage.write( + key: any(named: 'key'), + value: any(named: 'value'), + )); + }); + + test('should generate and save new device id if not exists', () async { + // Arrange + when(() => mockSecureStorage.read(key: StorageKeys.deviceId)) + .thenAnswer((_) async => null); + when(() => mockSecureStorage.write(key: any(named: 'key'), value: any(named: 'value'))) + .thenAnswer((_) async {}); + + // Act + final result = await accountService.getOrCreateDeviceId(); + + // Assert + expect(result, isNotEmpty); + // UUID v4 format: xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx + expect( + result, + matches(RegExp( + r'^[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$', + caseSensitive: false, + ))); + verify(() => mockSecureStorage.write( + key: StorageKeys.deviceId, + value: result, + )).called(1); + }); + }); + + group('hasAccount', () { + test('should return true when wallet is created', () async { + // Arrange + when(() => mockSecureStorage.read(key: StorageKeys.isWalletCreated)) + .thenAnswer((_) async => 'true'); + + // Act + final result = await accountService.hasAccount(); + + // Assert + expect(result, true); + }); + + test('should return false when wallet is not created', () async { + // Arrange + when(() => mockSecureStorage.read(key: StorageKeys.isWalletCreated)) + .thenAnswer((_) async => null); + + // Act + final result = await accountService.hasAccount(); + + // Assert + expect(result, false); + }); + }); + + group('getWalletAddresses', () { + test('should return wallet addresses when all are available', () async { + // Arrange + when(() => mockSecureStorage.read(key: StorageKeys.walletAddressBsc)) + .thenAnswer((_) async => '0xBSC'); + when(() => mockSecureStorage.read(key: StorageKeys.walletAddressKava)) + .thenAnswer((_) async => '0xKAVA'); + when(() => mockSecureStorage.read(key: StorageKeys.walletAddressDst)) + .thenAnswer((_) async => 'dst1DST'); + + // Act + final result = await accountService.getWalletAddresses(); + + // Assert + expect(result, isNotNull); + expect(result!.bsc, '0xBSC'); + expect(result.kava, '0xKAVA'); + expect(result.dst, 'dst1DST'); + }); + + test('should return null when any address is missing', () async { + // Arrange + when(() => mockSecureStorage.read(key: StorageKeys.walletAddressBsc)) + .thenAnswer((_) async => '0xBSC'); + when(() => mockSecureStorage.read(key: StorageKeys.walletAddressKava)) + .thenAnswer((_) async => null); // Missing + when(() => mockSecureStorage.read(key: StorageKeys.walletAddressDst)) + .thenAnswer((_) async => 'dst1DST'); + + // Act + final result = await accountService.getWalletAddresses(); + + // Assert + expect(result, isNull); + }); + }); + + group('logout', () { + test('should clear all stored data', () async { + // Arrange + when(() => mockSecureStorage.deleteAll()).thenAnswer((_) async {}); + + // Act + await accountService.logout(); + + // Assert + verify(() => mockSecureStorage.deleteAll()).called(1); + }); + }); + }); + + group('CreateAccountResponse', () { + test('should parse JSON correctly', () { + // Arrange + final json = { + 'userId': '123456789', + 'accountSequence': 42, + 'referralCode': 'TESTCD', + 'mnemonic': 'word1 word2 word3', + 'clientShareData': 'share-data', + 'publicKey': 'pub-key', + 'walletAddresses': { + 'kava': '0xKAVA', + 'dst': 'dst1DST', + 'bsc': '0xBSC', + }, + 'accessToken': 'access-token', + 'refreshToken': 'refresh-token', + }; + + // Act + final response = CreateAccountResponse.fromJson(json); + + // Assert + expect(response.userId, '123456789'); + expect(response.accountSequence, 42); + expect(response.referralCode, 'TESTCD'); + expect(response.mnemonic, 'word1 word2 word3'); + expect(response.clientShareData, 'share-data'); + expect(response.publicKey, 'pub-key'); + expect(response.walletAddresses.kava, '0xKAVA'); + expect(response.walletAddresses.dst, 'dst1DST'); + expect(response.walletAddresses.bsc, '0xBSC'); + expect(response.accessToken, 'access-token'); + expect(response.refreshToken, 'refresh-token'); + }); + + test('should handle nullable fields', () { + // Arrange + final json = { + 'userId': '123456789', + 'accountSequence': 1, + 'referralCode': 'ABC123', + 'walletAddresses': { + 'kava': '0xKAVA', + 'dst': 'dst1DST', + 'bsc': '0xBSC', + }, + 'accessToken': 'token', + 'refreshToken': 'refresh', + // mnemonic, clientShareData, publicKey are null + }; + + // Act + final response = CreateAccountResponse.fromJson(json); + + // Assert + expect(response.mnemonic, isNull); + expect(response.clientShareData, isNull); + expect(response.publicKey, isNull); + }); + }); + + group('CreateAccountRequest', () { + test('should serialize to JSON correctly', () { + // Arrange + final request = CreateAccountRequest( + deviceId: 'device-123', + deviceName: 'iPhone 15', + inviterReferralCode: 'INVITE', + provinceCode: '110000', + cityCode: '110100', + ); + + // Act + final json = request.toJson(); + + // Assert + expect(json['deviceId'], 'device-123'); + expect(json['deviceName'], 'iPhone 15'); + expect(json['inviterReferralCode'], 'INVITE'); + expect(json['provinceCode'], '110000'); + expect(json['cityCode'], '110100'); + }); + + test('should exclude null fields from JSON', () { + // Arrange + final request = CreateAccountRequest( + deviceId: 'device-123', + // All optional fields are null + ); + + // Act + final json = request.toJson(); + + // Assert + expect(json['deviceId'], 'device-123'); + expect(json.containsKey('deviceName'), false); + expect(json.containsKey('inviterReferralCode'), false); + expect(json.containsKey('provinceCode'), false); + expect(json.containsKey('cityCode'), false); + }); + }); +} diff --git a/frontend/mobile-app/test/core/services/mpc_share_service_test.dart b/frontend/mobile-app/test/core/services/mpc_share_service_test.dart new file mode 100644 index 00000000..dbff9c72 --- /dev/null +++ b/frontend/mobile-app/test/core/services/mpc_share_service_test.dart @@ -0,0 +1,343 @@ +import 'dart:convert'; +import 'dart:typed_data'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:rwa_android_app/core/services/mpc_share_service.dart'; + +void main() { + group('MpcShareService', () { + group('generateMnemonic', () { + test('should generate valid 12-word BIP39 mnemonic', () { + // Act + final mnemonic = MpcShareService.generateMnemonic(); + + // Assert + final words = mnemonic.split(' '); + expect(words.length, 12, reason: 'Should generate 12 words'); + expect(MpcShareService.validateMnemonic(mnemonic), true, + reason: 'Generated mnemonic should be valid BIP39'); + }); + + test('should generate different mnemonics each time', () { + // Act + final mnemonic1 = MpcShareService.generateMnemonic(); + final mnemonic2 = MpcShareService.generateMnemonic(); + + // Assert + expect(mnemonic1, isNot(equals(mnemonic2)), + reason: 'Each mnemonic should be unique (random)'); + }); + }); + + group('createShareBackup', () { + test('should create backup with all required fields', () { + // Arrange + const shareData = 'mock-mpc-share-data-256bit'; + + // Act + final backup = MpcShareService.createShareBackup(shareData); + + // Assert + expect(backup.mnemonic, isNotEmpty); + expect(backup.encryptedShare, isNotEmpty); + expect(backup.iv, isNotEmpty); + expect(backup.authTag, isNotEmpty); + expect(backup.mnemonicWords.length, 12); + }); + + test('should create valid BIP39 mnemonic', () { + // Arrange + const shareData = 'test-share-for-mnemonic'; + + // Act + final backup = MpcShareService.createShareBackup(shareData); + + // Assert + expect(MpcShareService.validateMnemonic(backup.mnemonic), true); + }); + + test('should throw on empty share data', () { + // Act & Assert + expect( + () => MpcShareService.createShareBackup(''), + throwsA(isA()), + ); + }); + }); + + group('encryptShare and decryptShare', () { + test('should encrypt and decrypt share correctly', () { + // Arrange + const originalShare = 'secret-mpc-share-data-to-encrypt'; + final mnemonic = MpcShareService.generateMnemonic(); + + // Act + final encrypted = MpcShareService.encryptShare(originalShare, mnemonic); + final decrypted = MpcShareService.decryptShare( + ciphertext: encrypted.ciphertext, + iv: encrypted.iv, + authTag: encrypted.authTag, + mnemonic: mnemonic, + ); + + // Assert + expect(decrypted, originalShare); + }); + + test('should produce different ciphertext with different IVs', () { + // Arrange + const shareData = 'same-share-data'; + final mnemonic = MpcShareService.generateMnemonic(); + + // Act + final encrypted1 = MpcShareService.encryptShare(shareData, mnemonic); + final encrypted2 = MpcShareService.encryptShare(shareData, mnemonic); + + // Assert - different IVs should produce different ciphertext + expect(encrypted1.iv, isNot(equals(encrypted2.iv))); + expect(encrypted1.ciphertext, isNot(equals(encrypted2.ciphertext))); + }); + + test('should fail decryption with wrong mnemonic', () { + // Arrange + const shareData = 'secret-share'; + final correctMnemonic = MpcShareService.generateMnemonic(); + final wrongMnemonic = MpcShareService.generateMnemonic(); + + final encrypted = MpcShareService.encryptShare(shareData, correctMnemonic); + + // Act & Assert + expect( + () => MpcShareService.decryptShare( + ciphertext: encrypted.ciphertext, + iv: encrypted.iv, + authTag: encrypted.authTag, + mnemonic: wrongMnemonic, + ), + throwsA(isA()), + ); + }); + + test('should fail decryption with tampered ciphertext', () { + // Arrange + const shareData = 'secret-share'; + final mnemonic = MpcShareService.generateMnemonic(); + final encrypted = MpcShareService.encryptShare(shareData, mnemonic); + + // Tamper with ciphertext + final tamperedBytes = base64Decode(encrypted.ciphertext); + tamperedBytes[0] = (tamperedBytes[0] + 1) % 256; + final tamperedCiphertext = base64Encode(tamperedBytes); + + // Act & Assert + expect( + () => MpcShareService.decryptShare( + ciphertext: tamperedCiphertext, + iv: encrypted.iv, + authTag: encrypted.authTag, + mnemonic: mnemonic, + ), + throwsA(isA()), + ); + }); + }); + + group('recoverShare', () { + test('should recover original share from backup', () { + // Arrange + const originalShare = 'original-mpc-share-256bit-data'; + final backup = MpcShareService.createShareBackup(originalShare); + + // Act + final recovered = MpcShareService.recoverShare( + mnemonic: backup.mnemonic, + encryptedShare: backup.encryptedShare, + iv: backup.iv, + authTag: backup.authTag, + ); + + // Assert + expect(recovered, originalShare); + }); + + test('should recover using MpcShareBackup.recoverShare()', () { + // Arrange + const originalShare = 'test-share-for-backup-recovery'; + final backup = MpcShareService.createShareBackup(originalShare); + + // Act + final recovered = backup.recoverShare(); + + // Assert + expect(recovered, originalShare); + }); + + test('should throw on invalid mnemonic format', () { + // Arrange + final backup = MpcShareService.createShareBackup('test-share'); + + // Act & Assert + expect( + () => MpcShareService.recoverShare( + mnemonic: 'invalid mnemonic words', + encryptedShare: backup.encryptedShare, + iv: backup.iv, + authTag: backup.authTag, + ), + throwsA(isA()), + ); + }); + + test('should handle complex base64 share data', () { + // Arrange - simulate real 256-bit MPC share + final complexBytes = List.generate(32, (i) => (i * 17 + 5) % 256); + final complexShare = base64Encode(complexBytes); + final backup = MpcShareService.createShareBackup(complexShare); + + // Act + final recovered = backup.recoverShare(); + + // Assert + expect(recovered, complexShare); + expect(base64Decode(recovered).length, 32); + }); + }); + + group('validateMnemonic', () { + test('should return true for valid 12-word mnemonic', () { + final mnemonic = MpcShareService.generateMnemonic(); + expect(MpcShareService.validateMnemonic(mnemonic), true); + }); + + test('should return false for invalid mnemonic', () { + expect(MpcShareService.validateMnemonic('invalid words here'), false); + expect(MpcShareService.validateMnemonic(''), false); + expect( + MpcShareService.validateMnemonic( + 'one two three four five six seven eight nine ten eleven twelve', + ), + false, + ); + }); + }); + + group('MpcShareBackup', () { + test('should serialize to JSON correctly', () { + // Arrange + final backup = MpcShareBackup( + mnemonic: 'word1 word2 word3 word4 word5 word6 word7 word8 word9 word10 word11 word12', + encryptedShare: 'encrypted-data-base64', + iv: 'iv-base64', + authTag: 'auth-tag-base64', + ); + + // Act + final json = backup.toJson(); + + // Assert + expect(json['mnemonic'], backup.mnemonic); + expect(json['encryptedShare'], 'encrypted-data-base64'); + expect(json['iv'], 'iv-base64'); + expect(json['authTag'], 'auth-tag-base64'); + }); + + test('should deserialize from JSON correctly', () { + // Arrange + final json = { + 'mnemonic': 'word1 word2 word3 word4 word5 word6 word7 word8 word9 word10 word11 word12', + 'encryptedShare': 'encrypted-data-base64', + 'iv': 'iv-base64', + 'authTag': 'auth-tag-base64', + }; + + // Act + final backup = MpcShareBackup.fromJson(json); + + // Assert + expect(backup.mnemonic, json['mnemonic']); + expect(backup.encryptedShare, 'encrypted-data-base64'); + expect(backup.iv, 'iv-base64'); + expect(backup.authTag, 'auth-tag-base64'); + expect(backup.mnemonicWords.length, 12); + }); + }); + + group('bytesToHex and hexToBytes', () { + test('should convert bytes to hex string', () { + final bytes = Uint8List.fromList([0, 15, 16, 255]); + final hex = MpcShareService.bytesToHex(bytes); + expect(hex, '000f10ff'); + }); + + test('should convert hex string to bytes', () { + const hex = '000f10ff'; + final bytes = MpcShareService.hexToBytes(hex); + expect(bytes, Uint8List.fromList([0, 15, 16, 255])); + }); + + test('should round-trip bytes through hex', () { + final original = Uint8List.fromList(List.generate(32, (i) => i * 8 % 256)); + final hex = MpcShareService.bytesToHex(original); + final recovered = MpcShareService.hexToBytes(hex); + expect(recovered, original); + }); + }); + + group('Security properties', () { + test('same share with same mnemonic should produce same decrypted result', () { + // Arrange + const shareData = 'consistent-share-data'; + final mnemonic = MpcShareService.generateMnemonic(); + + // Encrypt twice with same mnemonic (different IVs) + final encrypted1 = MpcShareService.encryptShare(shareData, mnemonic); + final encrypted2 = MpcShareService.encryptShare(shareData, mnemonic); + + // Act - decrypt both + final decrypted1 = MpcShareService.decryptShare( + ciphertext: encrypted1.ciphertext, + iv: encrypted1.iv, + authTag: encrypted1.authTag, + mnemonic: mnemonic, + ); + final decrypted2 = MpcShareService.decryptShare( + ciphertext: encrypted2.ciphertext, + iv: encrypted2.iv, + authTag: encrypted2.authTag, + mnemonic: mnemonic, + ); + + // Assert + expect(decrypted1, shareData); + expect(decrypted2, shareData); + expect(decrypted1, decrypted2); + }); + + test('full round-trip: share -> backup -> recover', () { + // Arrange - simulate full MPC workflow + final originalShare = base64Encode( + List.generate(64, (i) => (i * 13 + 7) % 256), + ); // 512-bit share data + + // Act + final backup = MpcShareService.createShareBackup(originalShare); + + // User stores mnemonic, device stores encrypted data + final storedMnemonic = backup.mnemonic; + final storedEncrypted = backup.encryptedShare; + final storedIv = backup.iv; + final storedAuthTag = backup.authTag; + + // Later: user enters mnemonic to recover + final recoveredShare = MpcShareService.recoverShare( + mnemonic: storedMnemonic, + encryptedShare: storedEncrypted, + iv: storedIv, + authTag: storedAuthTag, + ); + + // Assert + expect(recoveredShare, originalShare); + }); + }); + }); +}