diff --git a/backend/services/referral-service/.dockerignore b/backend/services/referral-service/.dockerignore new file mode 100644 index 00000000..75ceeee8 --- /dev/null +++ b/backend/services/referral-service/.dockerignore @@ -0,0 +1,37 @@ +# Dependencies +node_modules +npm-debug.log + +# Build output +dist + +# Test coverage +coverage +.nyc_output + +# IDE +.idea +.vscode +*.swp +*.swo + +# OS +.DS_Store +Thumbs.db + +# Environment +.env +.env.local +.env.*.local + +# Git +.git +.gitignore + +# Docs +*.md +!README.md + +# Logs +logs +*.log diff --git a/backend/services/referral-service/.env.development b/backend/services/referral-service/.env.development new file mode 100644 index 00000000..499fa154 --- /dev/null +++ b/backend/services/referral-service/.env.development @@ -0,0 +1,26 @@ +# 应用配置 +NODE_ENV=development +PORT=3004 +APP_NAME=referral-service + +# 数据库 +DATABASE_URL="postgresql://postgres:postgres@localhost:5432/rwadurian_referral?schema=public" + +# JWT (与 identity-service 共享密钥) +JWT_SECRET=your-super-secret-jwt-key-change-in-production +JWT_ACCESS_EXPIRES_IN=2h +JWT_REFRESH_EXPIRES_IN=7d + +# Redis +REDIS_HOST=localhost +REDIS_PORT=6379 +REDIS_PASSWORD= + +# Kafka +KAFKA_BROKERS=localhost:9092 +KAFKA_GROUP_ID=referral-service-group +KAFKA_CLIENT_ID=referral-service + +# 外部服务 +IDENTITY_SERVICE_URL=http://localhost:3001 +PLANTING_SERVICE_URL=http://localhost:3003 diff --git a/backend/services/referral-service/.env.example b/backend/services/referral-service/.env.example new file mode 100644 index 00000000..499fa154 --- /dev/null +++ b/backend/services/referral-service/.env.example @@ -0,0 +1,26 @@ +# 应用配置 +NODE_ENV=development +PORT=3004 +APP_NAME=referral-service + +# 数据库 +DATABASE_URL="postgresql://postgres:postgres@localhost:5432/rwadurian_referral?schema=public" + +# JWT (与 identity-service 共享密钥) +JWT_SECRET=your-super-secret-jwt-key-change-in-production +JWT_ACCESS_EXPIRES_IN=2h +JWT_REFRESH_EXPIRES_IN=7d + +# Redis +REDIS_HOST=localhost +REDIS_PORT=6379 +REDIS_PASSWORD= + +# Kafka +KAFKA_BROKERS=localhost:9092 +KAFKA_GROUP_ID=referral-service-group +KAFKA_CLIENT_ID=referral-service + +# 外部服务 +IDENTITY_SERVICE_URL=http://localhost:3001 +PLANTING_SERVICE_URL=http://localhost:3003 diff --git a/backend/services/referral-service/.eslintrc.js b/backend/services/referral-service/.eslintrc.js new file mode 100644 index 00000000..259de13c --- /dev/null +++ b/backend/services/referral-service/.eslintrc.js @@ -0,0 +1,25 @@ +module.exports = { + parser: '@typescript-eslint/parser', + parserOptions: { + project: 'tsconfig.json', + tsconfigRootDir: __dirname, + sourceType: 'module', + }, + plugins: ['@typescript-eslint/eslint-plugin'], + extends: [ + 'plugin:@typescript-eslint/recommended', + 'plugin:prettier/recommended', + ], + root: true, + env: { + node: true, + jest: true, + }, + ignorePatterns: ['.eslintrc.js'], + rules: { + '@typescript-eslint/interface-name-prefix': 'off', + '@typescript-eslint/explicit-function-return-type': 'off', + '@typescript-eslint/explicit-module-boundary-types': 'off', + '@typescript-eslint/no-explicit-any': 'off', + }, +}; diff --git a/backend/services/referral-service/.gitignore b/backend/services/referral-service/.gitignore new file mode 100644 index 00000000..f538d126 --- /dev/null +++ b/backend/services/referral-service/.gitignore @@ -0,0 +1,35 @@ +# Dependencies +node_modules/ + +# Build output +dist/ + +# Coverage +coverage/ + +# IDE +.idea/ +.vscode/ +*.swp +*.swo + +# Environment +.env +.env.local +.env.*.local + +# Logs +logs/ +*.log +npm-debug.log* + +# OS +.DS_Store +Thumbs.db + +# Temporary files +*.tmp +*.temp + +# Claude Code local settings +.claude/ diff --git a/backend/services/referral-service/.prettierrc b/backend/services/referral-service/.prettierrc new file mode 100644 index 00000000..243b2659 --- /dev/null +++ b/backend/services/referral-service/.prettierrc @@ -0,0 +1,7 @@ +{ + "singleQuote": true, + "trailingComma": "all", + "tabWidth": 2, + "semi": true, + "printWidth": 100 +} diff --git a/backend/services/referral-service/DEVELOPMENT_GUIDE.md b/backend/services/referral-service/DEVELOPMENT_GUIDE.md index f1120db3..aaf7607c 100644 --- a/backend/services/referral-service/DEVELOPMENT_GUIDE.md +++ b/backend/services/referral-service/DEVELOPMENT_GUIDE.md @@ -4,78 +4,131 @@ Referral Service 是 RWA 榴莲女皇平台的推荐团队微服务,负责管理推荐关系树、团队统计维护、龙虎榜分值计算、省市团队占比统计等功能。 +### 核心职责 ✅ +- 推荐关系树构建(基于序列号/推荐码) +- 团队统计数据维护 +- 上级查找(最近的授权省/市/社区) +- 直推用户管理 +- 团队层级关系管理 +- 本省/本市认种占比计算 +- 龙虎榜分值计算 + +### 不负责 ❌ +- 用户账户管理(Identity Context) +- 奖励计算(Reward Context) +- 授权考核判定(Authorization Context) + +--- + ## 技术栈 -- **框架**: NestJS -- **数据库**: PostgreSQL + Prisma ORM -- **架构**: DDD + Hexagonal Architecture (六边形架构) -- **语言**: TypeScript +| 组件 | 技术选型 | +|------|----------| +| **框架** | NestJS 10.x | +| **数据库** | PostgreSQL + Prisma ORM | +| **架构** | DDD + Hexagonal Architecture (六边形架构) | +| **语言** | TypeScript 5.x | +| **消息队列** | Kafka (kafkajs) | +| **缓存** | Redis (ioredis) | +| **API文档** | Swagger (@nestjs/swagger) | -## 架构参考 +--- -请参考 `identity-service` 的架构模式,保持一致性: +## 架构设计 + +参考 `identity-service` 的架构模式,严格保持一致性: ``` referral-service/ ├── prisma/ -│ └── schema.prisma # 数据库模型 +│ ├── schema.prisma # 数据库模型定义 +│ └── migrations/ # 数据库迁移文件 +│ ├── src/ -│ ├── api/ # Presentation Layer (API层) +│ ├── api/ # 🔵 Presentation Layer (表现层/API层) │ │ ├── controllers/ +│ │ │ ├── health.controller.ts │ │ │ ├── referral.controller.ts │ │ │ ├── team.controller.ts │ │ │ └── leaderboard.controller.ts │ │ ├── dto/ -│ │ │ ├── referral-info.dto.ts -│ │ │ ├── team-statistics.dto.ts -│ │ │ ├── direct-referral.dto.ts -│ │ │ └── leaderboard.dto.ts +│ │ │ ├── request/ +│ │ │ │ └── create-referral.dto.ts +│ │ │ └── response/ +│ │ │ ├── referral-info.dto.ts +│ │ │ ├── team-statistics.dto.ts +│ │ │ ├── direct-referral.dto.ts +│ │ │ └── leaderboard.dto.ts │ │ └── api.module.ts │ │ -│ ├── application/ # Application Layer (应用层) +│ ├── application/ # 🟢 Application Layer (应用层) │ │ ├── commands/ -│ │ │ ├── create-referral-relationship.command.ts -│ │ │ ├── update-team-statistics.command.ts -│ │ │ └── recalculate-leaderboard-score.command.ts +│ │ │ ├── create-referral-relationship/ +│ │ │ │ ├── create-referral-relationship.command.ts +│ │ │ │ └── create-referral-relationship.handler.ts +│ │ │ ├── update-team-statistics/ +│ │ │ │ ├── update-team-statistics.command.ts +│ │ │ │ └── update-team-statistics.handler.ts +│ │ │ └── index.ts │ │ ├── queries/ -│ │ │ ├── get-my-referral-info.query.ts -│ │ │ ├── get-direct-referrals.query.ts -│ │ │ ├── get-team-statistics.query.ts -│ │ │ ├── get-referral-chain.query.ts -│ │ │ ├── get-province-team-ranking.query.ts -│ │ │ └── get-leaderboard.query.ts -│ │ ├── handlers/ -│ │ │ ├── create-referral-relationship.handler.ts -│ │ │ ├── update-team-statistics.handler.ts -│ │ │ └── on-planting-order-paid.handler.ts -│ │ └── services/ -│ │ ├── referral-application.service.ts -│ │ └── team-statistics.service.ts +│ │ │ ├── get-my-referral-info/ +│ │ │ │ ├── get-my-referral-info.query.ts +│ │ │ │ └── get-my-referral-info.handler.ts +│ │ │ ├── get-direct-referrals/ +│ │ │ │ ├── get-direct-referrals.query.ts +│ │ │ │ └── get-direct-referrals.handler.ts +│ │ │ ├── get-team-statistics/ +│ │ │ │ ├── get-team-statistics.query.ts +│ │ │ │ └── get-team-statistics.handler.ts +│ │ │ ├── get-referral-chain/ +│ │ │ │ ├── get-referral-chain.query.ts +│ │ │ │ └── get-referral-chain.handler.ts +│ │ │ ├── get-leaderboard/ +│ │ │ │ ├── get-leaderboard.query.ts +│ │ │ │ └── get-leaderboard.handler.ts +│ │ │ └── index.ts +│ │ ├── services/ +│ │ │ └── referral-application.service.ts +│ │ └── application.module.ts │ │ -│ ├── domain/ # Domain Layer (领域层) +│ ├── domain/ # 🟡 Domain Layer (领域层) - 核心业务逻辑 │ │ ├── aggregates/ -│ │ │ ├── referral-relationship.aggregate.ts -│ │ │ └── team-statistics.aggregate.ts +│ │ │ ├── referral-relationship/ +│ │ │ │ ├── referral-relationship.aggregate.ts +│ │ │ │ ├── referral-relationship.spec.ts +│ │ │ │ └── index.ts +│ │ │ └── team-statistics/ +│ │ │ ├── team-statistics.aggregate.ts +│ │ │ ├── team-statistics.spec.ts +│ │ │ └── index.ts │ │ ├── value-objects/ │ │ │ ├── referral-code.vo.ts │ │ │ ├── referral-chain.vo.ts -│ │ │ ├── team-planting-count.vo.ts │ │ │ ├── leaderboard-score.vo.ts -│ │ │ └── province-city-distribution.vo.ts +│ │ │ ├── province-city-distribution.vo.ts +│ │ │ ├── user-id.vo.ts +│ │ │ └── index.ts │ │ ├── events/ +│ │ │ ├── domain-event.base.ts │ │ │ ├── referral-relationship-created.event.ts │ │ │ ├── team-statistics-updated.event.ts -│ │ │ └── leaderboard-score-changed.event.ts +│ │ │ ├── leaderboard-score-changed.event.ts +│ │ │ └── index.ts │ │ ├── repositories/ │ │ │ ├── referral-relationship.repository.interface.ts -│ │ │ └── team-statistics.repository.interface.ts -│ │ └── services/ -│ │ ├── referral-chain.service.ts -│ │ ├── leaderboard-calculator.service.ts -│ │ └── team-aggregation.service.ts +│ │ │ ├── team-statistics.repository.interface.ts +│ │ │ └── index.ts +│ │ ├── services/ +│ │ │ ├── referral-chain.service.ts +│ │ │ ├── leaderboard-calculator.service.ts +│ │ │ ├── team-aggregation.service.ts +│ │ │ └── index.ts +│ │ └── domain.module.ts │ │ -│ ├── infrastructure/ # Infrastructure Layer (基础设施层) +│ ├── infrastructure/ # 🔴 Infrastructure Layer (基础设施层) │ │ ├── persistence/ +│ │ │ ├── prisma/ +│ │ │ │ └── prisma.service.ts │ │ │ ├── mappers/ │ │ │ │ ├── referral-relationship.mapper.ts │ │ │ │ └── team-statistics.mapper.ts @@ -83,19 +136,64 @@ referral-service/ │ │ │ ├── referral-relationship.repository.impl.ts │ │ │ └── team-statistics.repository.impl.ts │ │ ├── external/ -│ │ │ ├── identity-service.client.ts -│ │ │ └── planting-service.client.ts +│ │ │ ├── identity-service/ +│ │ │ │ └── identity-service.client.ts +│ │ │ └── planting-service/ +│ │ │ └── planting-service.client.ts │ │ ├── kafka/ │ │ │ ├── event-consumer.controller.ts -│ │ │ └── event-publisher.service.ts +│ │ │ ├── event-publisher.service.ts +│ │ │ └── kafka.module.ts +│ │ ├── redis/ +│ │ │ ├── redis.service.ts +│ │ │ └── redis.module.ts │ │ └── infrastructure.module.ts │ │ -│ ├── app.module.ts -│ └── main.ts -├── .env.development +│ ├── shared/ # 共享模块 +│ │ ├── decorators/ +│ │ │ ├── current-user.decorator.ts +│ │ │ ├── public.decorator.ts +│ │ │ └── index.ts +│ │ ├── exceptions/ +│ │ │ ├── domain.exception.ts +│ │ │ ├── application.exception.ts +│ │ │ └── index.ts +│ │ ├── filters/ +│ │ │ ├── global-exception.filter.ts +│ │ │ └── domain-exception.filter.ts +│ │ ├── guards/ +│ │ │ └── jwt-auth.guard.ts +│ │ ├── interceptors/ +│ │ │ └── transform.interceptor.ts +│ │ └── strategies/ +│ │ └── jwt.strategy.ts +│ │ +│ ├── config/ # 配置 +│ │ ├── app.config.ts +│ │ ├── database.config.ts +│ │ ├── jwt.config.ts +│ │ ├── redis.config.ts +│ │ ├── kafka.config.ts +│ │ └── index.ts +│ │ +│ ├── app.module.ts # 主模块 +│ └── main.ts # 入口文件 +│ +├── test/ # 测试 +│ ├── unit/ +│ ├── integration/ +│ └── e2e/ +│ ├── .env.example +├── .env.development +├── .env.production +├── .eslintrc.js +├── .prettierrc +├── nest-cli.json ├── package.json -└── tsconfig.json +├── tsconfig.json +├── tsconfig.build.json +└── Dockerfile ``` --- @@ -105,33 +203,178 @@ referral-service/ ### 1.1 创建 NestJS 项目 ```bash -cd backend/services -npx @nestjs/cli new referral-service --skip-git --package-manager npm -cd referral-service +cd backend/services/referral-service +npx @nestjs/cli new . --skip-git --package-manager npm ``` ### 1.2 安装依赖 ```bash -npm install @nestjs/config @prisma/client class-validator class-transformer uuid -npm install -D prisma @types/uuid +# 核心依赖 +npm install @nestjs/config @nestjs/swagger @nestjs/jwt @nestjs/passport @nestjs/microservices +npm install @prisma/client class-validator class-transformer uuid ioredis kafkajs +npm install passport passport-jwt + +# 开发依赖 +npm install -D prisma @types/uuid @types/passport-jwt +npm install -D @nestjs/testing jest ts-jest @types/jest supertest @types/supertest ``` -### 1.3 配置环境变量 +### 1.3 package.json 配置 + +```json +{ + "name": "referral-service", + "version": "1.0.0", + "description": "RWA Referral & Team Context Service", + "author": "RWA Team", + "private": true, + "license": "UNLICENSED", + "prisma": { + "schema": "prisma/schema.prisma", + "seed": "ts-node prisma/seed.ts" + }, + "scripts": { + "build": "nest build", + "format": "prettier --write \"src/**/*.ts\" \"test/**/*.ts\"", + "start": "nest start", + "start:dev": "nest start --watch", + "start:debug": "nest start --debug --watch", + "start:prod": "node dist/main", + "lint": "eslint \"{src,apps,libs,test}/**/*.ts\" --fix", + "test": "jest", + "test:watch": "jest --watch", + "test:cov": "jest --coverage", + "test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand", + "test:e2e": "jest --config ./test/jest-e2e.json", + "prisma:generate": "prisma generate", + "prisma:migrate": "prisma migrate dev", + "prisma:migrate:prod": "prisma migrate deploy", + "prisma:studio": "prisma studio" + }, + "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/swagger": "^7.1.17", + "@prisma/client": "^5.7.0", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.0", + "ioredis": "^5.3.2", + "kafkajs": "^2.2.4", + "passport": "^0.7.0", + "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/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" + }, + "jest": { + "moduleFileExtensions": ["js", "json", "ts"], + "rootDir": "src", + "testRegex": ".*\\.spec\\.ts$", + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "collectCoverageFrom": ["**/*.(t|j)s"], + "coverageDirectory": "../coverage", + "testEnvironment": "node", + "moduleNameMapper": { + "^@/(.*)$": "/$1" + } + } +} +``` + +### 1.4 tsconfig.json 配置 + +```json +{ + "compilerOptions": { + "module": "commonjs", + "declaration": true, + "removeComments": true, + "emitDecoratorMetadata": true, + "experimentalDecorators": true, + "allowSyntheticDefaultImports": true, + "target": "ES2021", + "sourceMap": true, + "outDir": "./dist", + "baseUrl": "./src", + "paths": { + "@/*": ["./*"] + }, + "incremental": true, + "skipLibCheck": true, + "strictNullChecks": true, + "noImplicitAny": true, + "strictBindCallApply": true, + "forceConsistentCasingInFileNames": true, + "noFallthroughCasesInSwitch": true + } +} +``` + +### 1.5 环境变量配置 创建 `.env.development`: ```env -DATABASE_URL="postgresql://postgres:postgres@localhost:5432/rwadurian_referral?schema=public" +# 应用配置 NODE_ENV=development PORT=3004 +APP_NAME=referral-service -# 外部服务 -IDENTITY_SERVICE_URL=http://localhost:3001 -PLANTING_SERVICE_URL=http://localhost:3003 +# 数据库 +DATABASE_URL="postgresql://postgres:postgres@localhost:5432/rwadurian_referral?schema=public" + +# JWT (与 identity-service 共享密钥) +JWT_SECRET=your-super-secret-jwt-key-change-in-production +JWT_ACCESS_EXPIRES_IN=2h +JWT_REFRESH_EXPIRES_IN=7d + +# Redis +REDIS_HOST=localhost +REDIS_PORT=6379 +REDIS_PASSWORD= # Kafka KAFKA_BROKERS=localhost:9092 KAFKA_GROUP_ID=referral-service-group +KAFKA_CLIENT_ID=referral-service + +# 外部服务 +IDENTITY_SERVICE_URL=http://localhost:3001 +PLANTING_SERVICE_URL=http://localhost:3003 ``` --- @@ -151,152 +394,157 @@ datasource db { } // ============================================ -// 推荐关系表 (状态表) -// 记录用户与推荐人的关系 +// 推荐关系表 (聚合根1) +// 记录用户与推荐人的关系,推荐关系一旦建立终生不可修改 // ============================================ model ReferralRelationship { - id BigInt @id @default(autoincrement()) @map("relationship_id") - userId BigInt @unique @map("user_id") - referrerId BigInt? @map("referrer_id") // 直接推荐人 (null = 系统推荐/无推荐人) + id BigInt @id @default(autoincrement()) @map("relationship_id") + userId BigInt @unique @map("user_id") + + // 推荐人信息 + referrerId BigInt? @map("referrer_id") // 直接推荐人 (null = 无推荐人/根节点) + rootUserId BigInt? @map("root_user_id") // 顶级上级用户ID // 推荐码 myReferralCode String @unique @map("my_referral_code") @db.VarChar(20) usedReferralCode String? @map("used_referral_code") @db.VarChar(20) - // 推荐链 (上级链,最多10层) - referralChain BigInt[] @map("referral_chain") // [直接上级, 上上级, ...] - depth Int @default(0) @map("depth") // 在推荐树中的深度 + // 推荐链 (使用PostgreSQL数组类型,最多存储10层上级) + ancestorPath BigInt[] @map("ancestor_path") // [父节点, 祖父节点, ...] 从根到父的路径 + depth Int @default(0) @map("depth") // 层级深度 (0=根节点) - // 直推统计 (快速查询用) + // 直推统计 (快速查询用,冗余存储) directReferralCount Int @default(0) @map("direct_referral_count") - activeDirectCount Int @default(0) @map("active_direct_count") // 已认种的直推 + activeDirectCount Int @default(0) @map("active_direct_count") // 已认种的直推人数 createdAt DateTime @default(now()) @map("created_at") updatedAt DateTime @updatedAt @map("updated_at") - // 自引用 (方便查询推荐人) - referrer ReferralRelationship? @relation("ReferrerToReferral", fields: [referrerId], references: [userId]) + // 自引用关系 (方便查询推荐人) + referrer ReferralRelationship? @relation("ReferrerToReferral", fields: [referrerId], references: [userId]) directReferrals ReferralRelationship[] @relation("ReferrerToReferral") // 关联团队统计 teamStatistics TeamStatistics? @@map("referral_relationships") - @@index([referrerId]) - @@index([myReferralCode]) - @@index([usedReferralCode]) - @@index([depth]) - @@index([createdAt]) + @@index([referrerId], name: "idx_referrer") + @@index([myReferralCode], name: "idx_my_referral_code") + @@index([usedReferralCode], name: "idx_used_referral_code") + @@index([rootUserId], name: "idx_root_user") + @@index([depth], name: "idx_depth") + @@index([createdAt], name: "idx_referral_created") } // ============================================ -// 团队统计表 (状态表) -// 每个用户的团队认种统计数据 +// 团队统计表 (聚合根2) +// 每个用户的团队认种统计数据,需要实时更新 // ============================================ model TeamStatistics { id BigInt @id @default(autoincrement()) @map("statistics_id") userId BigInt @unique @map("user_id") - // 个人认种 - selfPlantingCount Int @default(0) @map("self_planting_count") // 自己认种数量 + // === 注册统计 === + directReferralCount Int @default(0) @map("direct_referral_count") // 直推注册数 + totalTeamCount Int @default(0) @map("total_team_count") // 团队总注册数 + + // === 个人认种 === + selfPlantingCount Int @default(0) @map("self_planting_count") // 自己认种数量 selfPlantingAmount Decimal @default(0) @map("self_planting_amount") @db.Decimal(20, 8) - // 团队认种 (包含自己和所有下级) - teamPlantingCount Int @default(0) @map("team_planting_count") // 团队总认种数量 - teamPlantingAmount Decimal @default(0) @map("team_planting_amount") @db.Decimal(20, 8) + // === 团队认种 (包含自己和所有下级) === + directPlantingCount Int @default(0) @map("direct_planting_count") // 直推认种数 + totalTeamPlantingCount Int @default(0) @map("total_team_planting_count") // 团队总认种数 + totalTeamPlantingAmount Decimal @default(0) @map("total_team_planting_amount") @db.Decimal(20, 8) - // 直推团队认种 (按直推分组) + // === 直推团队数据 (JSON存储每个直推的团队认种量) === + // 格式: [{ userId: bigint, personalCount: int, teamCount: int, amount: decimal }, ...] directTeamPlantingData Json @default("[]") @map("direct_team_planting_data") - // 格式: [{ userId: bigint, count: int, amount: decimal }, ...] - // 龙虎榜分值 - // 计算公式: 团队总认种量 - 最大单个直推团队认种量 - leaderboardScore Int @default(0) @map("leaderboard_score") - maxDirectTeamCount Int @default(0) @map("max_direct_team_count") // 最大直推团队认种量 + // === 龙虎榜相关 === + // 龙虎榜分值 = 团队总认种量 - 最大单个直推团队认种量 + maxSingleTeamPlantingCount Int @default(0) @map("max_single_team_planting_count") + effectivePlantingCountForRanking Int @default(0) @map("effective_planting_count_for_ranking") - // 省市分布 (用于计算省/市团队占比) + // === 本省本市统计 (用于省市授权考核) === + ownProvinceTeamCount Int @default(0) @map("own_province_team_count") // 自有团队本省认种 + ownCityTeamCount Int @default(0) @map("own_city_team_count") // 自有团队本市认种 + provinceTeamPercentage Decimal @default(0) @map("province_team_percentage") @db.Decimal(5, 2) // 本省占比 + cityTeamPercentage Decimal @default(0) @map("city_team_percentage") @db.Decimal(5, 2) // 本市占比 + + // === 省市分布 (JSON存储详细分布) === + // 格式: { "provinceCode": { "cityCode": count, ... }, ... } provinceCityDistribution Json @default("{}") @map("province_city_distribution") - // 格式: { "province_code": { "city_code": count, ... }, ... } - // 团队层级统计 - teamMemberCount Int @default(0) @map("team_member_count") // 团队总人数 - directMemberCount Int @default(0) @map("direct_member_count") // 直推人数 - - createdAt DateTime @default(now()) @map("created_at") - updatedAt DateTime @updatedAt @map("updated_at") + // 时间戳 + lastCalcAt DateTime? @map("last_calc_at") // 最后计算时间 + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") // 关联 referralRelationship ReferralRelationship @relation(fields: [userId], references: [userId]) @@map("team_statistics") - @@index([leaderboardScore(sort: Desc)]) - @@index([teamPlantingCount(sort: Desc)]) - @@index([selfPlantingCount]) + @@index([effectivePlantingCountForRanking(sort: Desc)], name: "idx_leaderboard_score") + @@index([totalTeamPlantingCount(sort: Desc)], name: "idx_team_planting") + @@index([selfPlantingCount], name: "idx_self_planting") } // ============================================ -// 直推列表表 (便于分页查询) +// 直推用户列表 (冗余表,便于分页查询) // ============================================ model DirectReferral { id BigInt @id @default(autoincrement()) @map("direct_referral_id") - referrerId BigInt @map("referrer_id") - referralId BigInt @map("referral_id") + referrerId BigInt @map("referrer_id") // 推荐人ID + referralId BigInt @map("referral_id") // 被推荐人ID + referralSequence BigInt @map("referral_sequence") // 被推荐人序列号 // 被推荐人信息快照 (冗余存储,避免跨服务查询) referralNickname String? @map("referral_nickname") @db.VarChar(100) referralAvatar String? @map("referral_avatar") @db.VarChar(255) - referralAccountNo String? @map("referral_account_no") @db.VarChar(20) // 该直推的认种统计 - plantingCount Int @default(0) @map("planting_count") - teamPlantingCount Int @default(0) @map("team_planting_count") // 该直推的团队认种量 + personalPlantingCount Int @default(0) @map("personal_planting_count") // 个人认种数 + teamPlantingCount Int @default(0) @map("team_planting_count") // 团队认种数(含个人) // 是否已认种 (用于区分活跃/非活跃) - hasPlanted Boolean @default(false) @map("has_planted") + hasPlanted Boolean @default(false) @map("has_planted") firstPlantedAt DateTime? @map("first_planted_at") createdAt DateTime @default(now()) @map("created_at") updatedAt DateTime @updatedAt @map("updated_at") - @@unique([referrerId, referralId]) + @@unique([referrerId, referralId], name: "uk_referrer_referral") @@map("direct_referrals") - @@index([referrerId]) - @@index([referralId]) - @@index([hasPlanted]) - @@index([teamPlantingCount(sort: Desc)]) + @@index([referrerId], name: "idx_direct_referrer") + @@index([referralId], name: "idx_direct_referral") + @@index([hasPlanted], name: "idx_has_planted") + @@index([teamPlantingCount(sort: Desc)], name: "idx_direct_team_planting") } // ============================================ -// 省市团队排名表 (用于省/市权益分配) +// 团队省市分布表 (用于省市权益分配) // ============================================ -model ProvinceCityTeamRanking { - id BigInt @id @default(autoincrement()) @map("ranking_id") - provinceCode String @map("province_code") @db.VarChar(10) - cityCode String? @map("city_code") @db.VarChar(10) // null = 省级排名 +model TeamProvinceCityDetail { + id BigInt @id @default(autoincrement()) @map("detail_id") + userId BigInt @map("user_id") - userId BigInt @map("user_id") + provinceCode String @map("province_code") @db.VarChar(10) + cityCode String @map("city_code") @db.VarChar(10) - // 该用户在此省/市的团队认种量 - plantingCount Int @default(0) @map("planting_count") + teamPlantingCount Int @default(0) @map("team_planting_count") // 该省/市团队认种数 - // 占比 (百分比,如 25.50 表示 25.50%) - percentage Decimal @default(0) @map("percentage") @db.Decimal(10, 4) - - // 排名 - rank Int @default(0) @map("rank") - - createdAt DateTime @default(now()) @map("created_at") updatedAt DateTime @updatedAt @map("updated_at") - @@unique([provinceCode, cityCode, userId]) - @@map("province_city_team_rankings") - @@index([provinceCode, cityCode]) - @@index([provinceCode, plantingCount(sort: Desc)]) - @@index([userId]) + @@unique([userId, provinceCode, cityCode], name: "uk_user_province_city") + @@map("team_province_city_details") + @@index([userId], name: "idx_detail_user") + @@index([provinceCode], name: "idx_detail_province") + @@index([cityCode], name: "idx_detail_city") } // ============================================ -// 推荐事件表 (行为表, append-only) +// 推荐事件表 (行为表,append-only,用于审计和事件溯源) // ============================================ model ReferralEvent { id BigInt @id @default(autoincrement()) @map("event_id") @@ -311,31 +559,60 @@ model ReferralEvent { // 元数据 userId BigInt? @map("user_id") - occurredAt DateTime @default(now()) @map("occurred_at") + occurredAt DateTime @default(now()) @map("occurred_at") @db.Timestamp(6) version Int @default(1) @map("version") @@map("referral_events") - @@index([aggregateType, aggregateId]) - @@index([eventType]) - @@index([userId]) - @@index([occurredAt]) + @@index([aggregateType, aggregateId], name: "idx_event_aggregate") + @@index([eventType], name: "idx_event_type") + @@index([userId], name: "idx_event_user") + @@index([occurredAt], name: "idx_event_occurred") } ``` ### 2.2 初始化数据库 ```bash -npx prisma migrate dev --name init +# 生成 Prisma Client npx prisma generate + +# 创建并运行迁移 +npx prisma migrate dev --name init + +# 查看数据库 (可选) +npx prisma studio ``` --- -## 第三阶段:领域层实现 +## 第三阶段:领域层实现 (Domain Layer) ### 3.1 值对象 (Value Objects) -#### 3.1.1 referral-code.vo.ts +#### 3.1.1 src/domain/value-objects/user-id.vo.ts +```typescript +export class UserId { + private constructor(public readonly value: bigint) {} + + static create(value: bigint | string | number): UserId { + const bigIntValue = typeof value === 'bigint' ? value : BigInt(value); + if (bigIntValue <= 0n) { + throw new Error('用户ID必须大于0'); + } + return new UserId(bigIntValue); + } + + equals(other: UserId): boolean { + return this.value === other.value; + } + + toString(): string { + return this.value.toString(); + } +} +``` + +#### 3.1.2 src/domain/value-objects/referral-code.vo.ts ```typescript export class ReferralCode { private constructor(public readonly value: string) { @@ -354,7 +631,7 @@ export class ReferralCode { static generate(userId: bigint): ReferralCode { // 生成规则: 前缀 + 用户ID哈希 + 随机字符 const prefix = 'RWA'; - const userIdHash = userId.toString(36).toUpperCase().slice(-3); + const userIdHash = userId.toString(36).toUpperCase().slice(-3).padStart(3, '0'); const random = Math.random().toString(36).toUpperCase().slice(2, 6); return new ReferralCode(`${prefix}${userIdHash}${random}`); } @@ -365,8 +642,14 @@ export class ReferralCode { } ``` -#### 3.1.2 referral-chain.vo.ts +#### 3.1.3 src/domain/value-objects/referral-chain.vo.ts ```typescript +import { UserId } from './user-id.vo'; + +/** + * 推荐链值对象 + * 存储从直接推荐人到根节点的用户ID列表 + */ export class ReferralChain { private static readonly MAX_DEPTH = 10; @@ -389,7 +672,11 @@ export class ReferralChain { } static fromArray(chain: bigint[]): ReferralChain { - return new ReferralChain(chain); + return new ReferralChain([...chain]); + } + + static empty(): ReferralChain { + return new ReferralChain([]); } get depth(): number { @@ -421,10 +708,11 @@ export class ReferralChain { } ``` -#### 3.1.3 leaderboard-score.vo.ts +#### 3.1.4 src/domain/value-objects/leaderboard-score.vo.ts ```typescript /** - * 龙虎榜分值 + * 龙虎榜分值值对象 + * * 计算公式: 团队总认种量 - 最大单个直推团队认种量 * * 这个公式的设计目的: @@ -442,14 +730,16 @@ export class LeaderboardScore { totalTeamCount: number, directTeamCounts: number[], // 每个直推的团队认种量 ): LeaderboardScore { - const maxDirectTeamCount = Math.max(0, ...directTeamCounts); - const score = totalTeamCount - maxDirectTeamCount; + const maxDirectTeamCount = directTeamCounts.length > 0 + ? Math.max(...directTeamCounts) + : 0; + const score = Math.max(0, totalTeamCount - maxDirectTeamCount); - return new LeaderboardScore( - totalTeamCount, - maxDirectTeamCount, - Math.max(0, score), // 分值不能为负 - ); + return new LeaderboardScore(totalTeamCount, maxDirectTeamCount, score); + } + + static zero(): LeaderboardScore { + return new LeaderboardScore(0, 0, 0); } /** @@ -463,15 +753,15 @@ export class LeaderboardScore { } /** - * 比较排名 + * 比较排名 (降序排列) */ compareTo(other: LeaderboardScore): number { - return other.score - this.score; // 降序 + return other.score - this.score; } } ``` -#### 3.1.4 province-city-distribution.vo.ts +#### 3.1.5 src/domain/value-objects/province-city-distribution.vo.ts ```typescript export interface ProvinceCityCount { provinceCode: string; @@ -480,7 +770,7 @@ export interface ProvinceCityCount { } /** - * 省市分布统计 + * 省市分布统计值对象 * 用于计算用户团队在各省/市的认种分布 */ export class ProvinceCityDistribution { @@ -492,7 +782,11 @@ export class ProvinceCityDistribution { return new ProvinceCityDistribution(new Map()); } - static fromJson(json: Record>): ProvinceCityDistribution { + static fromJson(json: Record> | null): ProvinceCityDistribution { + if (!json) { + return ProvinceCityDistribution.empty(); + } + const map = new Map>(); for (const [province, cities] of Object.entries(json)) { const cityMap = new Map(); @@ -505,7 +799,7 @@ export class ProvinceCityDistribution { } /** - * 添加认种记录 + * 添加认种记录 (返回新实例,不可变) */ add(provinceCode: string, cityCode: string, count: number): ProvinceCityDistribution { const newDist = new Map(this.distribution); @@ -555,6 +849,19 @@ export class ProvinceCityDistribution { return result; } + /** + * 获取总数 + */ + getTotal(): number { + let total = 0; + for (const cities of this.distribution.values()) { + for (const count of cities.values()) { + total += count; + } + } + return total; + } + toJson(): Record> { const result: Record> = {}; for (const [province, cities] of this.distribution) { @@ -568,99 +875,264 @@ export class ProvinceCityDistribution { } ``` -### 3.2 聚合根 (Aggregates) - -#### 3.2.1 referral-relationship.aggregate.ts - +#### 3.1.6 src/domain/value-objects/index.ts ```typescript -import { ReferralCode } from '../value-objects/referral-code.vo'; -import { ReferralChain } from '../value-objects/referral-chain.vo'; +export * from './user-id.vo'; +export * from './referral-code.vo'; +export * from './referral-chain.vo'; +export * from './leaderboard-score.vo'; +export * from './province-city-distribution.vo'; +``` +### 3.2 领域事件 (Domain Events) + +#### 3.2.1 src/domain/events/domain-event.base.ts +```typescript +import { v4 as uuidv4 } from 'uuid'; + +export abstract class DomainEvent { + public readonly eventId: string; + public readonly occurredAt: Date; + public readonly version: number; + + protected constructor(version: number = 1) { + this.eventId = uuidv4(); + this.occurredAt = new Date(); + this.version = version; + } + + abstract get eventType(): string; + abstract get aggregateId(): string; + abstract get aggregateType(): string; + abstract toPayload(): Record; +} +``` + +#### 3.2.2 src/domain/events/referral-relationship-created.event.ts +```typescript +import { DomainEvent } from './domain-event.base'; + +export interface ReferralRelationshipCreatedPayload { + userId: string; + referrerId: string | null; + myReferralCode: string; + usedReferralCode: string | null; + depth: number; + ancestorPath: string[]; +} + +export class ReferralRelationshipCreatedEvent extends DomainEvent { + constructor(private readonly payload: ReferralRelationshipCreatedPayload) { + super(); + } + + get eventType(): string { + return 'ReferralRelationshipCreated'; + } + + get aggregateId(): string { + return this.payload.userId; + } + + get aggregateType(): string { + return 'ReferralRelationship'; + } + + toPayload(): ReferralRelationshipCreatedPayload { + return { ...this.payload }; + } +} +``` + +#### 3.2.3 src/domain/events/team-statistics-updated.event.ts +```typescript +import { DomainEvent } from './domain-event.base'; + +export interface TeamStatisticsUpdatedPayload { + userId: string; + selfPlantingCount: number; + teamPlantingCount: number; + leaderboardScore: number; + triggeredByUserId?: string; // 触发更新的用户ID +} + +export class TeamStatisticsUpdatedEvent extends DomainEvent { + constructor(private readonly payload: TeamStatisticsUpdatedPayload) { + super(); + } + + get eventType(): string { + return 'TeamStatisticsUpdated'; + } + + get aggregateId(): string { + return this.payload.userId; + } + + get aggregateType(): string { + return 'TeamStatistics'; + } + + toPayload(): TeamStatisticsUpdatedPayload { + return { ...this.payload }; + } +} +``` + +#### 3.2.4 src/domain/events/index.ts +```typescript +export * from './domain-event.base'; +export * from './referral-relationship-created.event'; +export * from './team-statistics-updated.event'; +``` + +### 3.3 聚合根 (Aggregates) + +#### 3.3.1 src/domain/aggregates/referral-relationship/referral-relationship.aggregate.ts +```typescript +import { ReferralCode } from '../../value-objects/referral-code.vo'; +import { ReferralChain } from '../../value-objects/referral-chain.vo'; +import { UserId } from '../../value-objects/user-id.vo'; +import { DomainEvent } from '../../events/domain-event.base'; +import { ReferralRelationshipCreatedEvent } from '../../events/referral-relationship-created.event'; + +/** + * 推荐关系聚合根 + * + * 不变式: + * 1. 推荐关系一旦建立,终生不可修改 + * 2. 不能推荐自己 + * 3. 祖先路径必须完整 (从根节点到当前节点的所有上级) + */ export class ReferralRelationship { - private _id: bigint | null; - private readonly _userId: bigint; + private _id: bigint | null = null; + private readonly _userId: UserId; private readonly _myReferralCode: ReferralCode; private readonly _usedReferralCode: ReferralCode | null; - private readonly _referrerId: bigint | null; + private readonly _referrerId: UserId | null; + private readonly _rootUserId: UserId | null; private readonly _referralChain: ReferralChain; private _directReferralCount: number; private _activeDirectCount: number; private readonly _createdAt: Date; - private _domainEvents: any[] = []; + private _domainEvents: DomainEvent[] = []; private constructor( - userId: bigint, + userId: UserId, myReferralCode: ReferralCode, usedReferralCode: ReferralCode | null, - referrerId: bigint | null, + referrerId: UserId | null, + rootUserId: UserId | null, referralChain: ReferralChain, ) { this._userId = userId; this._myReferralCode = myReferralCode; this._usedReferralCode = usedReferralCode; this._referrerId = referrerId; + this._rootUserId = rootUserId; this._referralChain = referralChain; this._directReferralCount = 0; this._activeDirectCount = 0; this._createdAt = new Date(); } - // Getters + // ============ Getters ============ get id(): bigint | null { return this._id; } - get userId(): bigint { return this._userId; } + get userId(): UserId { return this._userId; } get myReferralCode(): ReferralCode { return this._myReferralCode; } get usedReferralCode(): ReferralCode | null { return this._usedReferralCode; } - get referrerId(): bigint | null { return this._referrerId; } + get referrerId(): UserId | null { return this._referrerId; } + get rootUserId(): UserId | null { return this._rootUserId; } get referralChain(): ReferralChain { return this._referralChain; } get depth(): number { return this._referralChain.depth; } get directReferralCount(): number { return this._directReferralCount; } get activeDirectCount(): number { return this._activeDirectCount; } - get domainEvents(): any[] { return this._domainEvents; } + get createdAt(): Date { return this._createdAt; } + get domainEvents(): DomainEvent[] { return [...this._domainEvents]; } + + // ============ 工厂方法 ============ /** - * 工厂方法:创建推荐关系 - * @param userId 用户ID - * @param referrer 推荐人 (可选) - * @param usedReferralCode 使用的推荐码 (可选) + * 创建推荐关系 (有推荐人) */ - static create( + static createWithReferrer( userId: bigint, - referrer: ReferralRelationship | null, - usedReferralCode: string | null, + referrer: ReferralRelationship, + usedReferralCode: string, ): ReferralRelationship { + const userIdVo = UserId.create(userId); + + // 验证不能推荐自己 + if (userIdVo.equals(referrer.userId)) { + throw new Error('不能推荐自己'); + } + const myCode = ReferralCode.generate(userId); - const usedCode = usedReferralCode ? ReferralCode.create(usedReferralCode) : null; + const usedCode = ReferralCode.create(usedReferralCode); // 构建推荐链 const referralChain = ReferralChain.create( - referrer?.userId ?? null, - referrer?.referralChain.toArray() ?? [], + referrer.userId.value, + referrer.referralChain.toArray(), ); + // 确定根节点 + const rootUserId = referrer.rootUserId || referrer.userId; + const relationship = new ReferralRelationship( - userId, + userIdVo, myCode, usedCode, - referrer?.userId ?? null, + referrer.userId, + rootUserId, referralChain, ); // 发布领域事件 - relationship._domainEvents.push({ - type: 'ReferralRelationshipCreated', - data: { - userId: userId.toString(), - referrerId: referrer?.userId?.toString() ?? null, - myReferralCode: myCode.value, - usedReferralCode: usedCode?.value ?? null, - depth: referralChain.depth, - }, - }); + relationship._domainEvents.push(new ReferralRelationshipCreatedEvent({ + userId: userId.toString(), + referrerId: referrer.userId.toString(), + myReferralCode: myCode.value, + usedReferralCode: usedCode.value, + depth: referralChain.depth, + ancestorPath: referralChain.toArray().map(id => id.toString()), + })); return relationship; } + /** + * 创建推荐关系 (无推荐人,根节点) + */ + static createRoot(userId: bigint): ReferralRelationship { + const userIdVo = UserId.create(userId); + const myCode = ReferralCode.generate(userId); + + const relationship = new ReferralRelationship( + userIdVo, + myCode, + null, + null, + null, + ReferralChain.empty(), + ); + + // 发布领域事件 + relationship._domainEvents.push(new ReferralRelationshipCreatedEvent({ + userId: userId.toString(), + referrerId: null, + myReferralCode: myCode.value, + usedReferralCode: null, + depth: 0, + ancestorPath: [], + })); + + return relationship; + } + + // ============ 领域行为 ============ + /** * 增加直推人数 */ @@ -671,7 +1143,7 @@ export class ReferralRelationship { /** * 直推用户认种后,更新活跃直推数 */ - markDirectAsActive(directUserId: bigint): void { + markDirectAsActive(): void { this._activeDirectCount++; } @@ -682,50 +1154,95 @@ export class ReferralRelationship { return this._referralChain.getAllAncestors(); } + /** + * 获取直接推荐人ID + */ + getDirectReferrerId(): bigint | null { + return this._referrerId?.value ?? null; + } + + /** + * 设置ID (用于从数据库重建) + */ + setId(id: bigint): void { + this._id = id; + } + clearDomainEvents(): void { this._domainEvents = []; } - // 从数据库重建 - static reconstitute(data: any): ReferralRelationship { + // ============ 重建 ============ + + /** + * 从数据库重建聚合 + */ + static reconstitute(data: { + id: bigint; + userId: bigint; + myReferralCode: string; + usedReferralCode: string | null; + referrerId: bigint | null; + rootUserId: bigint | null; + ancestorPath: bigint[]; + directReferralCount: number; + activeDirectCount: number; + createdAt: Date; + }): ReferralRelationship { const relationship = new ReferralRelationship( - data.userId, + UserId.create(data.userId), ReferralCode.create(data.myReferralCode), data.usedReferralCode ? ReferralCode.create(data.usedReferralCode) : null, - data.referrerId, - ReferralChain.fromArray(data.referralChain || []), + data.referrerId ? UserId.create(data.referrerId) : null, + data.rootUserId ? UserId.create(data.rootUserId) : null, + ReferralChain.fromArray(data.ancestorPath), ); relationship._id = data.id; - relationship._directReferralCount = data.directReferralCount ?? 0; - relationship._activeDirectCount = data.activeDirectCount ?? 0; + relationship._directReferralCount = data.directReferralCount; + relationship._activeDirectCount = data.activeDirectCount; return relationship; } } ``` -#### 3.2.2 team-statistics.aggregate.ts - +#### 3.3.2 src/domain/aggregates/team-statistics/team-statistics.aggregate.ts ```typescript -import { LeaderboardScore } from '../value-objects/leaderboard-score.vo'; -import { ProvinceCityDistribution } from '../value-objects/province-city-distribution.vo'; +import { UserId } from '../../value-objects/user-id.vo'; +import { LeaderboardScore } from '../../value-objects/leaderboard-score.vo'; +import { ProvinceCityDistribution } from '../../value-objects/province-city-distribution.vo'; +import { DomainEvent } from '../../events/domain-event.base'; +import { TeamStatisticsUpdatedEvent } from '../../events/team-statistics-updated.event'; interface DirectTeamData { userId: bigint; - count: number; + personalCount: number; + teamCount: number; amount: number; } +/** + * 团队统计聚合根 + * + * 不变式: + * 1. 团队统计必须在认种后实时更新 + * 2. 龙虎榜分值 = 团队总认种量 - 最大单个直推团队认种量 + */ export class TeamStatistics { - private _id: bigint | null; - private readonly _userId: bigint; + private _id: bigint | null = null; + private readonly _userId: UserId; + + // 注册统计 + private _directReferralCount: number; + private _totalTeamCount: number; // 个人认种 private _selfPlantingCount: number; private _selfPlantingAmount: number; // 团队认种 - private _teamPlantingCount: number; - private _teamPlantingAmount: number; + private _directPlantingCount: number; + private _totalTeamPlantingCount: number; + private _totalTeamPlantingAmount: number; // 直推团队数据 private _directTeamPlantingData: DirectTeamData[]; @@ -733,47 +1250,97 @@ export class TeamStatistics { // 龙虎榜 private _leaderboardScore: LeaderboardScore; + // 本省本市统计 + private _ownProvinceTeamCount: number; + private _ownCityTeamCount: number; + private _provinceTeamPercentage: number; + private _cityTeamPercentage: number; + // 省市分布 private _provinceCityDistribution: ProvinceCityDistribution; - // 团队人数 - private _teamMemberCount: number; - private _directMemberCount: number; + private _lastCalcAt: Date | null; + private readonly _createdAt: Date; - private _domainEvents: any[] = []; + private _domainEvents: DomainEvent[] = []; - private constructor(userId: bigint) { + private constructor(userId: UserId) { this._userId = userId; + this._directReferralCount = 0; + this._totalTeamCount = 0; this._selfPlantingCount = 0; this._selfPlantingAmount = 0; - this._teamPlantingCount = 0; - this._teamPlantingAmount = 0; + this._directPlantingCount = 0; + this._totalTeamPlantingCount = 0; + this._totalTeamPlantingAmount = 0; this._directTeamPlantingData = []; - this._leaderboardScore = LeaderboardScore.calculate(0, []); + this._leaderboardScore = LeaderboardScore.zero(); + this._ownProvinceTeamCount = 0; + this._ownCityTeamCount = 0; + this._provinceTeamPercentage = 0; + this._cityTeamPercentage = 0; this._provinceCityDistribution = ProvinceCityDistribution.empty(); - this._teamMemberCount = 0; - this._directMemberCount = 0; + this._lastCalcAt = null; + this._createdAt = new Date(); } - // Getters + // ============ Getters ============ get id(): bigint | null { return this._id; } - get userId(): bigint { return this._userId; } + get userId(): UserId { return this._userId; } + get directReferralCount(): number { return this._directReferralCount; } + get totalTeamCount(): number { return this._totalTeamCount; } get selfPlantingCount(): number { return this._selfPlantingCount; } get selfPlantingAmount(): number { return this._selfPlantingAmount; } - get teamPlantingCount(): number { return this._teamPlantingCount; } - get teamPlantingAmount(): number { return this._teamPlantingAmount; } + get directPlantingCount(): number { return this._directPlantingCount; } + get totalTeamPlantingCount(): number { return this._totalTeamPlantingCount; } + get totalTeamPlantingAmount(): number { return this._totalTeamPlantingAmount; } get directTeamPlantingData(): ReadonlyArray { return this._directTeamPlantingData; } get leaderboardScore(): LeaderboardScore { return this._leaderboardScore; } + get maxSingleTeamPlantingCount(): number { return this._leaderboardScore.maxDirectTeamCount; } + get effectivePlantingCountForRanking(): number { return this._leaderboardScore.score; } + get ownProvinceTeamCount(): number { return this._ownProvinceTeamCount; } + get ownCityTeamCount(): number { return this._ownCityTeamCount; } + get provinceTeamPercentage(): number { return this._provinceTeamPercentage; } + get cityTeamPercentage(): number { return this._cityTeamPercentage; } get provinceCityDistribution(): ProvinceCityDistribution { return this._provinceCityDistribution; } - get teamMemberCount(): number { return this._teamMemberCount; } - get directMemberCount(): number { return this._directMemberCount; } - get domainEvents(): any[] { return this._domainEvents; } + get lastCalcAt(): Date | null { return this._lastCalcAt; } + get createdAt(): Date { return this._createdAt; } + get domainEvents(): DomainEvent[] { return [...this._domainEvents]; } + + // ============ 工厂方法 ============ + + static create(userId: bigint): TeamStatistics { + return new TeamStatistics(UserId.create(userId)); + } + + // ============ 领域行为 ============ /** - * 工厂方法:创建团队统计 + * 增加直推注册 */ - static create(userId: bigint): TeamStatistics { - return new TeamStatistics(userId); + addDirectReferral(referralUserId: bigint): void { + this._directReferralCount++; + this._totalTeamCount++; + + // 初始化直推团队数据 + if (!this._directTeamPlantingData.find(d => d.userId === referralUserId)) { + this._directTeamPlantingData.push({ + userId: referralUserId, + personalCount: 0, + teamCount: 0, + amount: 0, + }); + } + + this._lastCalcAt = new Date(); + } + + /** + * 增加团队成员 (非直推) + */ + addTeamMember(): void { + this._totalTeamCount++; + this._lastCalcAt = new Date(); } /** @@ -784,13 +1351,15 @@ export class TeamStatistics { amount: number, provinceCode: string, cityCode: string, + userProvince?: string, + userCity?: string, ): void { this._selfPlantingCount += treeCount; this._selfPlantingAmount += amount; // 自己的认种也计入团队 - this._teamPlantingCount += treeCount; - this._teamPlantingAmount += amount; + this._totalTeamPlantingCount += treeCount; + this._totalTeamPlantingAmount += amount; // 更新省市分布 this._provinceCityDistribution = this._provinceCityDistribution.add( @@ -799,141 +1368,242 @@ export class TeamStatistics { treeCount, ); - // 重新计算龙虎榜分值 - this._recalculateLeaderboardScore(); + // 更新本省本市统计 + if (userProvince && provinceCode === userProvince) { + this._ownProvinceTeamCount += treeCount; + } + if (userCity && cityCode === userCity) { + this._ownCityTeamCount += treeCount; + } - this._domainEvents.push({ - type: 'TeamStatisticsUpdated', - data: { - userId: this._userId.toString(), - selfPlantingCount: this._selfPlantingCount, - teamPlantingCount: this._teamPlantingCount, - leaderboardScore: this._leaderboardScore.score, - }, - }); + // 重新计算龙虎榜分值和占比 + this._recalculateLeaderboardScore(); + this._recalculatePercentages(); + this._lastCalcAt = new Date(); + + this._domainEvents.push(new TeamStatisticsUpdatedEvent({ + userId: this._userId.toString(), + selfPlantingCount: this._selfPlantingCount, + teamPlantingCount: this._totalTeamPlantingCount, + leaderboardScore: this._leaderboardScore.score, + })); } /** - * 下级用户认种,更新团队统计 - * @param directUserId 直推用户ID (第一层下级) - * @param treeCount 认种数量 - * @param amount 认种金额 - * @param provinceCode 省份代码 - * @param cityCode 城市代码 + * 更新团队认种数据 (下级用户认种时调用) */ - addTeamPlanting( - directUserId: bigint, - treeCount: number, - amount: number, - provinceCode: string, - cityCode: string, + updatePlanting(params: { + isDirectReferral: boolean; + directReferralId?: bigint; + treeCount: number; + amount: number; + province: string; + city: string; + userProvince?: string; + userCity?: string; + }): void { + // 1. 更新团队总认种 + this._totalTeamPlantingCount += params.treeCount; + this._totalTeamPlantingAmount += params.amount; + + // 2. 如果是直推 + if (params.isDirectReferral && params.directReferralId) { + this._directPlantingCount += params.treeCount; + + // 更新直推明细 + const directTeam = this._directTeamPlantingData.find( + d => d.userId === params.directReferralId + ); + if (directTeam) { + directTeam.personalCount += params.treeCount; + directTeam.teamCount += params.treeCount; + directTeam.amount += params.amount; + } + } else if (params.directReferralId) { + // 非直推但需要更新直推的团队数据 + const directTeam = this._directTeamPlantingData.find( + d => d.userId === params.directReferralId + ); + if (directTeam) { + directTeam.teamCount += params.treeCount; + directTeam.amount += params.amount; + } + } + + // 3. 更新省市分布 + this._provinceCityDistribution = this._provinceCityDistribution.add( + params.province, + params.city, + params.treeCount, + ); + + // 4. 更新本省本市统计 + if (params.userProvince && params.province === params.userProvince) { + this._ownProvinceTeamCount += params.treeCount; + } + if (params.userCity && params.city === params.userCity) { + this._ownCityTeamCount += params.treeCount; + } + + // 5. 重新计算龙虎榜分值和占比 + this._recalculateLeaderboardScore(); + this._recalculatePercentages(); + this._lastCalcAt = new Date(); + + this._domainEvents.push(new TeamStatisticsUpdatedEvent({ + userId: this._userId.toString(), + selfPlantingCount: this._selfPlantingCount, + teamPlantingCount: this._totalTeamPlantingCount, + leaderboardScore: this._leaderboardScore.score, + triggeredByUserId: params.directReferralId?.toString(), + })); + } + + /** + * 更新直推团队认种数 + */ + updateDirectReferralTeamPlanting( + directReferralId: bigint, + teamPlantingCount: number, ): void { - // 更新团队总量 - this._teamPlantingCount += treeCount; - this._teamPlantingAmount += amount; - - // 更新直推团队数据 - const directTeamIndex = this._directTeamPlantingData.findIndex( - d => d.userId === directUserId, + const directTeam = this._directTeamPlantingData.find( + d => d.userId === directReferralId ); - - if (directTeamIndex >= 0) { - this._directTeamPlantingData[directTeamIndex].count += treeCount; - this._directTeamPlantingData[directTeamIndex].amount += amount; - } else { - this._directTeamPlantingData.push({ - userId: directUserId, - count: treeCount, - amount, - }); + if (directTeam) { + directTeam.teamCount = teamPlantingCount; } - // 更新省市分布 - this._provinceCityDistribution = this._provinceCityDistribution.add( - provinceCode, - cityCode, - treeCount, - ); - - // 重新计算龙虎榜分值 this._recalculateLeaderboardScore(); - - this._domainEvents.push({ - type: 'TeamStatisticsUpdated', - data: { - userId: this._userId.toString(), - teamPlantingCount: this._teamPlantingCount, - leaderboardScore: this._leaderboardScore.score, - updatedByDirectUserId: directUserId.toString(), - }, - }); + this._lastCalcAt = new Date(); } - /** - * 添加团队成员 - * @param isDirect 是否是直推 - */ - addTeamMember(isDirect: boolean): void { - this._teamMemberCount++; - if (isDirect) { - this._directMemberCount++; - } - } + // ============ 私有方法 ============ - /** - * 重新计算龙虎榜分值 - */ private _recalculateLeaderboardScore(): void { - const directCounts = this._directTeamPlantingData.map(d => d.count); + const directCounts = this._directTeamPlantingData.map(d => d.teamCount); this._leaderboardScore = LeaderboardScore.calculate( - this._teamPlantingCount, + this._totalTeamPlantingCount, directCounts, ); } - /** - * 获取最大直推团队认种量 - */ - get maxDirectTeamCount(): number { - return this._leaderboardScore.maxDirectTeamCount; + private _recalculatePercentages(): void { + if (this._totalTeamPlantingCount > 0) { + this._provinceTeamPercentage = + (this._ownProvinceTeamCount / this._totalTeamPlantingCount) * 100; + this._cityTeamPercentage = + (this._ownCityTeamCount / this._totalTeamPlantingCount) * 100; + } else { + this._provinceTeamPercentage = 0; + this._cityTeamPercentage = 0; + } + } + + setId(id: bigint): void { + this._id = id; } clearDomainEvents(): void { this._domainEvents = []; } - // 从数据库重建 - static reconstitute(data: any): TeamStatistics { - const stats = new TeamStatistics(data.userId); + // ============ 重建 ============ + + static reconstitute(data: { + id: bigint; + userId: bigint; + directReferralCount: number; + totalTeamCount: number; + selfPlantingCount: number; + selfPlantingAmount: number; + directPlantingCount: number; + totalTeamPlantingCount: number; + totalTeamPlantingAmount: number; + directTeamPlantingData: DirectTeamData[]; + ownProvinceTeamCount: number; + ownCityTeamCount: number; + provinceCityDistribution: Record> | null; + lastCalcAt: Date | null; + createdAt: Date; + }): TeamStatistics { + const stats = new TeamStatistics(UserId.create(data.userId)); stats._id = data.id; - stats._selfPlantingCount = data.selfPlantingCount ?? 0; - stats._selfPlantingAmount = Number(data.selfPlantingAmount) ?? 0; - stats._teamPlantingCount = data.teamPlantingCount ?? 0; - stats._teamPlantingAmount = Number(data.teamPlantingAmount) ?? 0; - stats._directTeamPlantingData = data.directTeamPlantingData ?? []; - stats._teamMemberCount = data.teamMemberCount ?? 0; - stats._directMemberCount = data.directMemberCount ?? 0; - - // 重建省市分布 + stats._directReferralCount = data.directReferralCount; + stats._totalTeamCount = data.totalTeamCount; + stats._selfPlantingCount = data.selfPlantingCount; + stats._selfPlantingAmount = data.selfPlantingAmount; + stats._directPlantingCount = data.directPlantingCount; + stats._totalTeamPlantingCount = data.totalTeamPlantingCount; + stats._totalTeamPlantingAmount = data.totalTeamPlantingAmount; + stats._directTeamPlantingData = data.directTeamPlantingData || []; + stats._ownProvinceTeamCount = data.ownProvinceTeamCount; + stats._ownCityTeamCount = data.ownCityTeamCount; stats._provinceCityDistribution = ProvinceCityDistribution.fromJson( - data.provinceCityDistribution ?? {}, + data.provinceCityDistribution ); + stats._lastCalcAt = data.lastCalcAt; - // 重新计算龙虎榜分值 + // 重新计算龙虎榜分值和占比 stats._recalculateLeaderboardScore(); + stats._recalculatePercentages(); return stats; } } ``` -### 3.3 领域服务 +### 3.4 仓储接口 (Repository Interfaces) -#### 3.3.1 referral-chain.service.ts +#### 3.4.1 src/domain/repositories/referral-relationship.repository.interface.ts +```typescript +import { ReferralRelationship } from '../aggregates/referral-relationship/referral-relationship.aggregate'; +export interface IReferralRelationshipRepository { + save(relationship: ReferralRelationship): Promise; + findById(id: bigint): Promise; + findByUserId(userId: bigint): Promise; + findByReferralCode(code: string): Promise; + findDirectReferrals(userId: bigint, page?: number, pageSize?: number): Promise; + countDirectReferrals(userId: bigint): Promise; + findByReferrerId(referrerId: bigint): Promise; + findByAncestor(ancestorId: bigint): Promise; +} + +export const REFERRAL_RELATIONSHIP_REPOSITORY = Symbol('IReferralRelationshipRepository'); +``` + +#### 3.4.2 src/domain/repositories/team-statistics.repository.interface.ts +```typescript +import { TeamStatistics } from '../aggregates/team-statistics/team-statistics.aggregate'; + +export interface ITeamStatisticsRepository { + save(stats: TeamStatistics): Promise; + findByUserId(userId: bigint): Promise; + getOrCreate(userId: bigint): Promise; + findTopByLeaderboardScore(limit: number): Promise; + findByProvinceCityRanking( + provinceCode: string, + cityCode: string | null, + limit: number, + ): Promise; + findByUserIds(userIds: bigint[]): Promise>; +} + +export const TEAM_STATISTICS_REPOSITORY = Symbol('ITeamStatisticsRepository'); +``` + +#### 3.4.3 src/domain/repositories/index.ts +```typescript +export * from './referral-relationship.repository.interface'; +export * from './team-statistics.repository.interface'; +``` + +### 3.5 领域服务 (Domain Services) + +#### 3.5.1 src/domain/services/referral-chain.service.ts ```typescript import { Injectable } from '@nestjs/common'; -import { ReferralRelationship } from '../aggregates/referral-relationship.aggregate'; +import { ReferralRelationship } from '../aggregates/referral-relationship/referral-relationship.aggregate'; @Injectable() export class ReferralChainService { @@ -968,15 +1638,15 @@ export class ReferralChainService { relationship1: ReferralRelationship, relationship2: ReferralRelationship, ): number | null { - const chain1 = [relationship1.userId, ...relationship1.getAllAncestorIds()]; - const chain2 = [relationship2.userId, ...relationship2.getAllAncestorIds()]; + const chain1 = [relationship1.userId.value, ...relationship1.getAllAncestorIds()]; + const chain2 = [relationship2.userId.value, ...relationship2.getAllAncestorIds()]; // 检查 user2 是否在 user1 的链上 - const idx1 = chain1.findIndex(id => id === relationship2.userId); + const idx1 = chain1.findIndex(id => id === relationship2.userId.value); if (idx1 >= 0) return idx1; // 检查 user1 是否在 user2 的链上 - const idx2 = chain2.findIndex(id => id === relationship1.userId); + const idx2 = chain2.findIndex(id => id === relationship1.userId.value); if (idx2 >= 0) return idx2; // 寻找公共祖先 @@ -992,11 +1662,10 @@ export class ReferralChainService { } ``` -#### 3.3.2 leaderboard-calculator.service.ts - +#### 3.5.2 src/domain/services/leaderboard-calculator.service.ts ```typescript import { Injectable } from '@nestjs/common'; -import { TeamStatistics } from '../aggregates/team-statistics.aggregate'; +import { TeamStatistics } from '../aggregates/team-statistics/team-statistics.aggregate'; export interface LeaderboardEntry { userId: bigint; @@ -1019,15 +1688,15 @@ export class LeaderboardCalculatorService { ): LeaderboardEntry[] { // 按龙虎榜分值排序 const sorted = allStats - .filter(s => s.leaderboardScore.score > 0) - .sort((a, b) => b.leaderboardScore.score - a.leaderboardScore.score); + .filter(s => s.effectivePlantingCountForRanking > 0) + .sort((a, b) => b.effectivePlantingCountForRanking - a.effectivePlantingCountForRanking); // 取前 limit 名并添加排名 return sorted.slice(0, limit).map((stats, index) => ({ - userId: stats.userId, - score: stats.leaderboardScore.score, - teamPlantingCount: stats.teamPlantingCount, - maxDirectTeamCount: stats.maxDirectTeamCount, + userId: stats.userId.value, + score: stats.effectivePlantingCountForRanking, + teamPlantingCount: stats.totalTeamPlantingCount, + maxDirectTeamCount: stats.maxSingleTeamPlantingCount, rank: index + 1, })); } @@ -1040,17 +1709,16 @@ export class LeaderboardCalculatorService { userId: bigint, ): number | null { const sorted = allStats - .filter(s => s.leaderboardScore.score > 0) - .sort((a, b) => b.leaderboardScore.score - a.leaderboardScore.score); + .filter(s => s.effectivePlantingCountForRanking > 0) + .sort((a, b) => b.effectivePlantingCountForRanking - a.effectivePlantingCountForRanking); - const index = sorted.findIndex(s => s.userId === userId); + const index = sorted.findIndex(s => s.userId.value === userId); return index >= 0 ? index + 1 : null; } } ``` -#### 3.3.3 team-aggregation.service.ts - +#### 3.5.3 src/domain/services/team-aggregation.service.ts ```typescript import { Injectable, Inject } from '@nestjs/common'; import { @@ -1061,7 +1729,7 @@ import { IReferralRelationshipRepository, REFERRAL_RELATIONSHIP_REPOSITORY } from '../repositories/referral-relationship.repository.interface'; -import { TeamStatistics } from '../aggregates/team-statistics.aggregate'; +import { TeamStatistics } from '../aggregates/team-statistics/team-statistics.aggregate'; @Injectable() export class TeamAggregationService { @@ -1074,12 +1742,6 @@ export class TeamAggregationService { /** * 当用户认种时,更新所有上级的团队统计 - * - * @param userId 认种的用户ID - * @param treeCount 认种数量 - * @param amount 认种金额 - * @param provinceCode 省份代码 - * @param cityCode 城市代码 */ async updateAncestorTeamStats( userId: bigint, @@ -1087,6 +1749,8 @@ export class TeamAggregationService { amount: number, provinceCode: string, cityCode: string, + userProvince?: string, + userCity?: string, ): Promise { // 1. 获取用户的推荐关系 const relationship = await this.relationshipRepository.findByUserId(userId); @@ -1095,11 +1759,8 @@ export class TeamAggregationService { } // 2. 更新自己的统计 - let selfStats = await this.statsRepository.findByUserId(userId); - if (!selfStats) { - selfStats = TeamStatistics.create(userId); - } - selfStats.addSelfPlanting(treeCount, amount, provinceCode, cityCode); + const selfStats = await this.statsRepository.getOrCreate(userId); + selfStats.addSelfPlanting(treeCount, amount, provinceCode, cityCode, userProvince, userCity); await this.statsRepository.save(selfStats); // 3. 获取推荐链 (所有上级) @@ -1110,574 +1771,109 @@ export class TeamAggregationService { const directReferrerId = ancestors[0]; // 5. 更新所有上级的团队统计 - for (const ancestorId of ancestors) { - let ancestorStats = await this.statsRepository.findByUserId(ancestorId); - if (!ancestorStats) { - ancestorStats = TeamStatistics.create(ancestorId); - } + for (let i = 0; i < ancestors.length; i++) { + const ancestorId = ancestors[i]; + const ancestorStats = await this.statsRepository.getOrCreate(ancestorId); - // 对于所有上级,都传入"通过哪个直推"来的数据 - // 需要追溯:这个认种是哪个直推带来的 - const directUserIdForAncestor = this.findDirectReferralForAncestor( - userId, - ancestorId, - ancestors, - ); + // 对于每个祖先,确定这个认种是通过哪个直推传递来的 + const directUserIdForAncestor = i === 0 ? userId : ancestors[i - 1]; - ancestorStats.addTeamPlanting( - directUserIdForAncestor, + ancestorStats.updatePlanting({ + isDirectReferral: i === 0, // 只有第一层是直推 + directReferralId: directUserIdForAncestor, treeCount, amount, - provinceCode, - cityCode, - ); + province: provinceCode, + city: cityCode, + userProvince, + userCity, + }); + await this.statsRepository.save(ancestorStats); } } /** - * 找出对于某个祖先来说,这个认种是通过哪个直推传递上来的 + * 当新用户注册时,更新所有上级的团队人数 */ - private findDirectReferralForAncestor( - plantingUserId: bigint, - ancestorId: bigint, - fullChain: bigint[], - ): bigint { - // fullChain: [直接推荐人, 上上级, 上上上级, ...] - // 如果 ancestorId 是直接推荐人,那么直推就是 plantingUserId - // 如果 ancestorId 是上上级,那么直推就是直接推荐人 (fullChain[0]) - // 以此类推 - - const ancestorIndex = fullChain.findIndex(id => id === ancestorId); - if (ancestorIndex === 0) { - // ancestorId 是直接推荐人,所以直推就是认种的用户本身 - return plantingUserId; - } else if (ancestorIndex > 0) { - // ancestorId 在链的更高位置,直推是它的下一级 - return fullChain[ancestorIndex - 1]; - } - - // 不应该到这里 - return plantingUserId; - } -} -``` - -### 3.4 仓储接口 - -#### 3.4.1 referral-relationship.repository.interface.ts - -```typescript -import { ReferralRelationship } from '../aggregates/referral-relationship.aggregate'; - -export interface IReferralRelationshipRepository { - save(relationship: ReferralRelationship): Promise; - findById(id: bigint): Promise; - findByUserId(userId: bigint): Promise; - findByReferralCode(code: string): Promise; - findDirectReferrals(userId: bigint, page?: number, pageSize?: number): Promise; - countDirectReferrals(userId: bigint): Promise; - findByReferrerId(referrerId: bigint): Promise; -} - -export const REFERRAL_RELATIONSHIP_REPOSITORY = Symbol('IReferralRelationshipRepository'); -``` - -#### 3.4.2 team-statistics.repository.interface.ts - -```typescript -import { TeamStatistics } from '../aggregates/team-statistics.aggregate'; - -export interface ITeamStatisticsRepository { - save(stats: TeamStatistics): Promise; - findByUserId(userId: bigint): Promise; - findTopByLeaderboardScore(limit: number): Promise; - findByProvinceCityRanking( - provinceCode: string, - cityCode: string | null, - limit: number, - ): Promise; - getOrCreate(userId: bigint): Promise; -} - -export const TEAM_STATISTICS_REPOSITORY = Symbol('ITeamStatisticsRepository'); -``` - ---- - -## 第四阶段:应用层实现 - -### 4.1 应用服务 - -```typescript -// application/services/referral-application.service.ts - -import { Injectable, Inject } from '@nestjs/common'; -import { ReferralRelationship } from '../../domain/aggregates/referral-relationship.aggregate'; -import { TeamStatistics } from '../../domain/aggregates/team-statistics.aggregate'; -import { - IReferralRelationshipRepository, - REFERRAL_RELATIONSHIP_REPOSITORY -} from '../../domain/repositories/referral-relationship.repository.interface'; -import { - ITeamStatisticsRepository, - TEAM_STATISTICS_REPOSITORY -} from '../../domain/repositories/team-statistics.repository.interface'; -import { ReferralChainService } from '../../domain/services/referral-chain.service'; -import { LeaderboardCalculatorService } from '../../domain/services/leaderboard-calculator.service'; - -@Injectable() -export class ReferralApplicationService { - constructor( - @Inject(REFERRAL_RELATIONSHIP_REPOSITORY) - private readonly relationshipRepository: IReferralRelationshipRepository, - @Inject(TEAM_STATISTICS_REPOSITORY) - private readonly statsRepository: ITeamStatisticsRepository, - private readonly referralChainService: ReferralChainService, - private readonly leaderboardService: LeaderboardCalculatorService, - ) {} - - /** - * 创建推荐关系 (用户注册时调用) - */ - async createRelationship( + async updateAncestorTeamMemberCount( userId: bigint, - referralCode: string | null, - ) { - // 检查是否已存在 - const existing = await this.relationshipRepository.findByUserId(userId); - if (existing) { - throw new Error('用户推荐关系已存在'); + directReferrerId: bigint, + ): Promise { + // 1. 更新直接推荐人 + const directReferrerStats = await this.statsRepository.getOrCreate(directReferrerId); + directReferrerStats.addDirectReferral(userId); + await this.statsRepository.save(directReferrerStats); + + // 2. 获取直接推荐人的推荐关系 + const directReferrerRelationship = await this.relationshipRepository.findByUserId(directReferrerId); + if (!directReferrerRelationship) return; + + // 3. 更新所有上级的团队人数 + const ancestors = directReferrerRelationship.getAllAncestorIds(); + for (const ancestorId of ancestors) { + const ancestorStats = await this.statsRepository.getOrCreate(ancestorId); + ancestorStats.addTeamMember(); + await this.statsRepository.save(ancestorStats); } - - // 查找推荐人 - let referrer: ReferralRelationship | null = null; - if (referralCode) { - referrer = await this.relationshipRepository.findByReferralCode(referralCode); - if (!referrer) { - throw new Error('无效的推荐码'); - } - } - - // 创建推荐关系 - const relationship = ReferralRelationship.create(userId, referrer, referralCode); - await this.relationshipRepository.save(relationship); - - // 更新推荐人的直推计数 - if (referrer) { - referrer.incrementDirectReferralCount(); - await this.relationshipRepository.save(referrer); - - // 更新推荐人的团队人数 - let referrerStats = await this.statsRepository.findByUserId(referrer.userId); - if (!referrerStats) { - referrerStats = TeamStatistics.create(referrer.userId); - } - referrerStats.addTeamMember(true); - await this.statsRepository.save(referrerStats); - - // 更新所有上级的团队人数 - for (const ancestorId of referrer.getAllAncestorIds()) { - let ancestorStats = await this.statsRepository.findByUserId(ancestorId); - if (!ancestorStats) { - ancestorStats = TeamStatistics.create(ancestorId); - } - ancestorStats.addTeamMember(false); - await this.statsRepository.save(ancestorStats); - } - } - - // 创建用户自己的团队统计 - const userStats = TeamStatistics.create(userId); - await this.statsRepository.save(userStats); - - return { - myReferralCode: relationship.myReferralCode.value, - referrerId: referrer?.userId.toString() ?? null, - depth: relationship.depth, - }; - } - - /** - * 获取我的推荐信息 (分享页) - */ - async getMyReferralInfo(userId: bigint) { - const relationship = await this.relationshipRepository.findByUserId(userId); - if (!relationship) { - throw new Error('用户推荐关系不存在'); - } - - const stats = await this.statsRepository.findByUserId(userId); - - return { - myReferralCode: relationship.myReferralCode.value, - directReferralCount: relationship.directReferralCount, - activeDirectCount: relationship.activeDirectCount, - teamMemberCount: stats?.teamMemberCount ?? 0, - teamPlantingCount: stats?.teamPlantingCount ?? 0, - leaderboardScore: stats?.leaderboardScore.score ?? 0, - leaderboardRank: await this.getUserLeaderboardRank(userId), - }; - } - - /** - * 获取直推列表 - */ - async getDirectReferrals(userId: bigint, page = 1, pageSize = 20) { - const directReferrals = await this.relationshipRepository.findDirectReferrals( - userId, - page, - pageSize, - ); - - const result = []; - for (const referral of directReferrals) { - const stats = await this.statsRepository.findByUserId(referral.userId); - result.push({ - userId: referral.userId.toString(), - referralCode: referral.myReferralCode.value, - plantingCount: stats?.selfPlantingCount ?? 0, - teamPlantingCount: stats?.teamPlantingCount ?? 0, - hasPlanted: (stats?.selfPlantingCount ?? 0) > 0, - }); - } - - const total = await this.relationshipRepository.countDirectReferrals(userId); - - return { - list: result, - total, - page, - pageSize, - }; - } - - /** - * 获取推荐链 (用于分享权益分配) - */ - async getReferralChain(userId: bigint): Promise { - const relationship = await this.relationshipRepository.findByUserId(userId); - if (!relationship) { - return []; - } - return relationship.getAllAncestorIds().map(id => id.toString()); - } - - /** - * 获取团队统计 - */ - async getTeamStatistics(userId: bigint) { - const stats = await this.statsRepository.findByUserId(userId); - if (!stats) { - return { - selfPlantingCount: 0, - teamPlantingCount: 0, - teamMemberCount: 0, - directMemberCount: 0, - leaderboardScore: 0, - provinceCityDistribution: {}, - }; - } - - return { - selfPlantingCount: stats.selfPlantingCount, - teamPlantingCount: stats.teamPlantingCount, - teamMemberCount: stats.teamMemberCount, - directMemberCount: stats.directMemberCount, - leaderboardScore: stats.leaderboardScore.score, - provinceCityDistribution: stats.provinceCityDistribution.toJson(), - }; - } - - /** - * 获取龙虎榜 - */ - async getLeaderboard(limit = 100) { - const allStats = await this.statsRepository.findTopByLeaderboardScore(limit); - return this.leaderboardService.calculateLeaderboard(allStats, limit); - } - - /** - * 获取用户龙虎榜排名 - */ - private async getUserLeaderboardRank(userId: bigint): Promise { - const allStats = await this.statsRepository.findTopByLeaderboardScore(1000); - return this.leaderboardService.getUserRank(allStats, userId); - } - - /** - * 获取省/市团队排名 - */ - async getProvinceCityTeamRanking( - provinceCode: string, - cityCode: string | null, - limit = 50, - ) { - const stats = await this.statsRepository.findByProvinceCityRanking( - provinceCode, - cityCode, - limit, - ); - - // 计算总量 - let totalPlanting = 0; - for (const s of stats) { - totalPlanting += cityCode - ? s.provinceCityDistribution.getCityTotal(provinceCode, cityCode) - : s.provinceCityDistribution.getProvinceTotal(provinceCode); - } - - // 计算每个用户的占比 - return stats.map((s, index) => { - const userPlanting = cityCode - ? s.provinceCityDistribution.getCityTotal(provinceCode, cityCode) - : s.provinceCityDistribution.getProvinceTotal(provinceCode); - - return { - userId: s.userId.toString(), - plantingCount: userPlanting, - percentage: totalPlanting > 0 - ? ((userPlanting / totalPlanting) * 100).toFixed(2) - : '0.00', - rank: index + 1, - }; - }); } } ``` +#### 3.5.4 src/domain/services/index.ts +```typescript +export * from './referral-chain.service'; +export * from './leaderboard-calculator.service'; +export * from './team-aggregation.service'; +``` + --- -## 第五阶段:API层实现 - -### 5.1 DTO 定义 +## 领域不变式 (Domain Invariants) ```typescript -// api/dto/referral-info.dto.ts -export class ReferralInfoDto { - myReferralCode: string; - directReferralCount: number; - activeDirectCount: number; - teamMemberCount: number; - teamPlantingCount: number; - leaderboardScore: number; - leaderboardRank: number | null; -} +// 文档参考,不需要创建此文件 +class ReferralContextInvariants { + // 1. 推荐关系不可修改 + static REFERRAL_RELATIONSHIP_IMMUTABLE = + "推荐关系一旦建立,终生不可修改" -// api/dto/direct-referral.dto.ts -export class DirectReferralDto { - userId: string; - referralCode: string; - plantingCount: number; - teamPlantingCount: number; - hasPlanted: boolean; -} + // 2. 祖先路径必须完整 + static ANCESTOR_PATH_COMPLETE = + "祖先路径必须包含从根节点到当前节点的所有上级" -// api/dto/team-statistics.dto.ts -export class TeamStatisticsDto { - selfPlantingCount: number; - teamPlantingCount: number; - teamMemberCount: number; - directMemberCount: number; - leaderboardScore: number; - provinceCityDistribution: Record>; -} + // 3. 不能推荐自己 + static CANNOT_REFER_SELF = + "用户不能推荐自己" -// api/dto/leaderboard.dto.ts -export class LeaderboardEntryDto { - userId: string; - score: number; - teamPlantingCount: number; - maxDirectTeamCount: number; - rank: number; -} -``` + // 4. 团队统计必须实时更新 + static TEAM_STATS_REALTIME = + "团队统计数据必须在认种后实时更新" -### 5.2 控制器 + // 5. 龙虎榜分值计算规则 + static LEADERBOARD_SCORE_RULE = + "龙虎榜分值 = 团队总认种量 - 最大单个直推团队认种量" -```typescript -// api/controllers/referral.controller.ts - -import { Controller, Get, Post, Body, Query, UseGuards, Req } from '@nestjs/common'; -import { ReferralApplicationService } from '../../application/services/referral-application.service'; -import { JwtAuthGuard } from '../guards/jwt-auth.guard'; - -@Controller('referrals') -@UseGuards(JwtAuthGuard) -export class ReferralController { - constructor(private readonly referralService: ReferralApplicationService) {} - - /** - * 获取我的推荐信息 (分享页) - */ - @Get('me') - async getMyReferralInfo(@Req() req: any) { - const userId = BigInt(req.user.id); - return this.referralService.getMyReferralInfo(userId); - } - - /** - * 获取直推列表 - */ - @Get('direct') - async getDirectReferrals( - @Req() req: any, - @Query('page') page = 1, - @Query('pageSize') pageSize = 20, - ) { - const userId = BigInt(req.user.id); - return this.referralService.getDirectReferrals(userId, page, pageSize); - } - - /** - * 获取推荐链 (内部API,用于资金分配) - */ - @Get('chain') - async getReferralChain(@Req() req: any) { - const userId = BigInt(req.user.id); - return this.referralService.getReferralChain(userId); - } -} - -// api/controllers/team.controller.ts - -import { Controller, Get, Query, UseGuards, Req } from '@nestjs/common'; -import { ReferralApplicationService } from '../../application/services/referral-application.service'; -import { JwtAuthGuard } from '../guards/jwt-auth.guard'; - -@Controller('team') -@UseGuards(JwtAuthGuard) -export class TeamController { - constructor(private readonly referralService: ReferralApplicationService) {} - - /** - * 获取我的团队统计 - */ - @Get('statistics') - async getTeamStatistics(@Req() req: any) { - const userId = BigInt(req.user.id); - return this.referralService.getTeamStatistics(userId); - } - - /** - * 获取省/市团队排名 - */ - @Get('province-city-ranking') - async getProvinceCityRanking( - @Query('provinceCode') provinceCode: string, - @Query('cityCode') cityCode: string | null, - @Query('limit') limit = 50, - ) { - return this.referralService.getProvinceCityTeamRanking(provinceCode, cityCode, limit); - } -} - -// api/controllers/leaderboard.controller.ts - -import { Controller, Get, Query, UseGuards, Req } from '@nestjs/common'; -import { ReferralApplicationService } from '../../application/services/referral-application.service'; -import { JwtAuthGuard } from '../guards/jwt-auth.guard'; - -@Controller('leaderboard') -@UseGuards(JwtAuthGuard) -export class LeaderboardController { - constructor(private readonly referralService: ReferralApplicationService) {} - - /** - * 获取龙虎榜 - */ - @Get() - async getLeaderboard(@Query('limit') limit = 100) { - return this.referralService.getLeaderboard(limit); - } + // 6. 推荐链最多10层 + static MAX_REFERRAL_DEPTH = + "推荐链深度限制为10层,超过的不计入" } ``` --- -## 事件处理 (Kafka) - -### 当用户认种时更新团队统计 - -```typescript -// infrastructure/kafka/event-consumer.controller.ts - -import { Controller } from '@nestjs/common'; -import { MessagePattern, Payload } from '@nestjs/microservices'; -import { TeamAggregationService } from '../../domain/services/team-aggregation.service'; - -interface PlantingOrderPaidEvent { - userId: string; - orderNo: string; - treeCount: number; - totalAmount: number; - provinceCode: string; - cityCode: string; -} - -@Controller() -export class EventConsumerController { - constructor( - private readonly teamAggregationService: TeamAggregationService, - ) {} - - /** - * 监听认种订单支付事件 - * 更新认种用户及其所有上级的团队统计 - */ - @MessagePattern('planting.order.paid') - async handlePlantingOrderPaid(@Payload() event: PlantingOrderPaidEvent) { - console.log('收到认种订单支付事件:', event); - - await this.teamAggregationService.updateAncestorTeamStats( - BigInt(event.userId), - event.treeCount, - event.totalAmount, - event.provinceCode, - event.cityCode, - ); - - console.log('团队统计更新完成'); - } -} -``` - ---- - -## 关键业务规则 (不变式) - -1. **推荐关系不可修改**: 一旦建立推荐关系,终生不可修改 -2. **推荐链最多10层**: 推荐链深度限制为10层,超过的不计入 -3. **龙虎榜分值计算**: `团队总认种量 - 最大单个直推团队认种量` -4. **省市权益分配**: 根据用户团队在该省/市的认种占比分配 -5. **团队统计实时更新**: 任何下级认种都会实时更新所有上级的统计 - ---- - -## API 端点汇总 +## API 端点设计 | 方法 | 路径 | 描述 | 认证 | |------|------|------|------| -| GET | /referrals/me | 获取我的推荐信息 (分享页) | 需要 | -| GET | /referrals/direct | 获取直推列表 | 需要 | -| GET | /referrals/chain | 获取推荐链 (内部API) | 需要 | -| GET | /team/statistics | 获取我的团队统计 | 需要 | -| GET | /team/province-city-ranking | 获取省/市团队排名 | 需要 | -| GET | /leaderboard | 获取龙虎榜 | 需要 | - ---- - -## 开发顺序建议 - -1. 项目初始化和 Prisma Schema -2. 值对象和枚举实现 -3. 聚合根实现 (ReferralRelationship, TeamStatistics) -4. 领域服务实现 (ReferralChainService, LeaderboardCalculatorService, TeamAggregationService) -5. 仓储接口和实现 -6. 应用服务实现 -7. Kafka 事件消费者 -8. DTO 和控制器实现 -9. 模块配置和测试 +| GET | `/health` | 健康检查 | 否 | +| GET | `/referrals/me` | 获取我的推荐信息 (分享页) | JWT | +| GET | `/referrals/direct` | 获取我的直推列表 | JWT | +| GET | `/referrals/chain` | 获取推荐链 (内部API,用于资金分配) | JWT | +| GET | `/team/statistics` | 获取我的团队统计 | JWT | +| GET | `/team/province-city-ranking` | 获取省/市团队排名 | JWT | +| GET | `/leaderboard` | 获取龙虎榜 | JWT | +| POST | `/internal/referral` | 创建推荐关系 (内部调用) | API-KEY | --- @@ -1706,11 +1902,79 @@ export class EventConsumerController { --- +## 事件订阅 (Kafka Events) + +### 订阅的事件 + +| Topic | 事件类型 | 触发条件 | 处理逻辑 | +|-------|---------|---------|---------| +| `identity.user.created` | UserAccountCreatedEvent | 用户注册成功 | 创建推荐关系,更新上级团队人数 | +| `planting.order.paid` | PlantingOrderPaidEvent | 认种订单支付成功 | 更新认种用户及所有上级的团队统计 | + +### 发布的事件 + +| Topic | 事件类型 | 触发条件 | +|-------|---------|---------| +| `referral.relationship.created` | ReferralRelationshipCreatedEvent | 推荐关系创建成功 | +| `referral.statistics.updated` | TeamStatisticsUpdatedEvent | 团队统计更新 | + +--- + +## 开发顺序建议 + +1. **Phase 1: 项目初始化** + - 创建NestJS项目 + - 安装依赖 + - 配置环境变量和TypeScript + +2. **Phase 2: 数据库层** + - 创建Prisma Schema + - 运行迁移 + - 创建PrismaService + +3. **Phase 3: 领域层 (最重要)** + - 实现所有值对象 + - 实现聚合根 (ReferralRelationship, TeamStatistics) + - 实现领域事件 + - 实现领域服务 + - 编写单元测试 + +4. **Phase 4: 基础设施层** + - 实现仓储 (Repository Implementations) + - 实现Kafka消费者和发布者 + - 实现Redis缓存服务 + - 实现外部服务客户端 + +5. **Phase 5: 应用层** + - 实现应用服务 (ReferralApplicationService) + - 实现Command/Query handlers + +6. **Phase 6: API层** + - 实现DTO + - 实现Controllers + - 配置Swagger文档 + - 配置JWT认证 + +7. **Phase 7: 测试和部署** + - 集成测试 + - E2E测试 + - Docker配置 + - CI/CD配置 + +--- + ## 注意事项 -1. 推荐关系在用户注册时由 identity-service 触发创建 -2. 团队统计在认种支付完成时由 planting-service 发布事件触发更新 -3. 龙虎榜分值设计目的是鼓励均衡发展团队 -4. 省市权益分配需要根据省市团队占比计算 -5. 使用 PostgreSQL 数组类型存储推荐链,方便查询 -6. 直推团队数据使用 JSON 类型存储,便于灵活扩展 +1. **推荐关系在用户注册时创建**:由 identity-service 发布 `UserAccountCreatedEvent`,本服务订阅并创建推荐关系 + +2. **团队统计在认种时更新**:由 planting-service 发布 `PlantingOrderPaidEvent`,本服务订阅并更新所有上级的统计 + +3. **龙虎榜分值设计目的**:鼓励均衡发展团队,防止"单腿"发展 + +4. **省市权益分配**:根据省市团队占比计算,用于省/市代理权益分配 + +5. **使用 PostgreSQL 数组类型**:存储推荐链,方便查询 + +6. **直推团队数据使用 JSON 类型**:便于灵活扩展 + +7. **保持与 identity-service 架构一致**:确保代码风格、目录结构、命名规范统一 diff --git a/backend/services/referral-service/Dockerfile.test b/backend/services/referral-service/Dockerfile.test new file mode 100644 index 00000000..06944f9f --- /dev/null +++ b/backend/services/referral-service/Dockerfile.test @@ -0,0 +1,31 @@ +# Test Dockerfile for referral-service +FROM node:20-alpine + +WORKDIR /app + +# Install build dependencies +RUN apk add --no-cache python3 make g++ + +# Copy package files +COPY package*.json ./ + +# Install dependencies +RUN npm ci + +# Copy prisma schema +COPY prisma ./prisma/ + +# Generate Prisma client +RUN npx prisma generate + +# Copy source code and tests +COPY . . + +# Build the application +RUN npm run build + +# Set environment for testing +ENV NODE_ENV=test + +# Default command - run all tests +CMD ["npm", "test"] diff --git a/backend/services/referral-service/Makefile b/backend/services/referral-service/Makefile new file mode 100644 index 00000000..8f363508 --- /dev/null +++ b/backend/services/referral-service/Makefile @@ -0,0 +1,128 @@ +.PHONY: install build clean test test-unit test-integration test-e2e test-all test-cov \ + docker-build docker-up docker-down docker-test docker-test-all \ + prisma-generate prisma-migrate prisma-studio lint format + +# ============================================ +# 基础命令 +# ============================================ +install: + npm install + +build: + npm run build + +clean: + rm -rf dist node_modules coverage .nyc_output + +lint: + npm run lint + +format: + npm run format + +# ============================================ +# Prisma 命令 +# ============================================ +prisma-generate: + npx prisma generate + +prisma-migrate: + npx prisma migrate dev + +prisma-migrate-prod: + npx prisma migrate deploy + +prisma-studio: + npx prisma studio + +# ============================================ +# 测试命令 +# ============================================ +test: test-unit + +test-unit: + @echo "==========================================" + @echo "Running Unit Tests..." + @echo "==========================================" + npm test + +test-unit-cov: + @echo "==========================================" + @echo "Running Unit Tests with Coverage..." + @echo "==========================================" + npm run test:cov + +test-integration: + @echo "==========================================" + @echo "Running Integration Tests..." + @echo "==========================================" + npm run test:integration + +test-e2e: + @echo "==========================================" + @echo "Running E2E Tests..." + @echo "==========================================" + npm run test:e2e + +test-all: test-unit test-integration test-e2e + @echo "==========================================" + @echo "All Tests Completed!" + @echo "==========================================" + +test-cov: + @echo "==========================================" + @echo "Running All Tests with Coverage..." + @echo "==========================================" + npm run test:cov + +# ============================================ +# Docker 命令 +# ============================================ +docker-build: + docker build -t referral-service:test -f Dockerfile.test . + +docker-up: + docker-compose -f docker-compose.test.yml up -d + +docker-down: + docker-compose -f docker-compose.test.yml down -v + +docker-test-unit: + @echo "==========================================" + @echo "Running Unit Tests in Docker..." + @echo "==========================================" + docker-compose -f docker-compose.test.yml run --rm test npm test + +docker-test-integration: + @echo "==========================================" + @echo "Running Integration Tests in Docker..." + @echo "==========================================" + docker-compose -f docker-compose.test.yml run --rm test npm run test:integration + +docker-test-e2e: + @echo "==========================================" + @echo "Running E2E Tests in Docker..." + @echo "==========================================" + docker-compose -f docker-compose.test.yml run --rm test npm run test:e2e + +docker-test-all: docker-up + @echo "==========================================" + @echo "Running All Tests in Docker..." + @echo "==========================================" + docker-compose -f docker-compose.test.yml run --rm test make test-all + $(MAKE) docker-down + +# ============================================ +# 开发命令 +# ============================================ +dev: + npm run start:dev + +start: + npm run start:prod + +# ============================================ +# CI/CD 命令 +# ============================================ +ci: install lint build test-all + @echo "CI Pipeline Completed!" diff --git a/backend/services/referral-service/docker-compose.test.yml b/backend/services/referral-service/docker-compose.test.yml new file mode 100644 index 00000000..f6118eb5 --- /dev/null +++ b/backend/services/referral-service/docker-compose.test.yml @@ -0,0 +1,78 @@ +version: '3.8' + +services: + # PostgreSQL for testing + postgres: + image: postgres:15-alpine + environment: + POSTGRES_USER: postgres + POSTGRES_PASSWORD: postgres + POSTGRES_DB: referral_test + ports: + - "5433:5432" + volumes: + - postgres_test_data:/var/lib/postgresql/data + healthcheck: + test: ["CMD-SHELL", "pg_isready -U postgres"] + interval: 5s + timeout: 5s + retries: 5 + + # Redis for testing + redis: + image: redis:7-alpine + ports: + - "6380:6379" + healthcheck: + test: ["CMD", "redis-cli", "ping"] + interval: 5s + timeout: 5s + retries: 5 + + # Kafka for testing (using Redpanda for simplicity) + redpanda: + image: redpandadata/redpanda:latest + command: + - redpanda + - start + - --smp 1 + - --memory 512M + - --reserve-memory 0M + - --overprovisioned + - --node-id 0 + - --kafka-addr PLAINTEXT://0.0.0.0:29092,OUTSIDE://0.0.0.0:9093 + - --advertise-kafka-addr PLAINTEXT://redpanda:29092,OUTSIDE://localhost:9093 + ports: + - "9093:9093" + - "29092:29092" + healthcheck: + test: ["CMD", "rpk", "cluster", "health"] + interval: 10s + timeout: 5s + retries: 5 + + # Test runner service + test: + build: + context: . + dockerfile: Dockerfile.test + depends_on: + postgres: + condition: service_healthy + redis: + condition: service_healthy + redpanda: + condition: service_healthy + environment: + NODE_ENV: test + DATABASE_URL: postgresql://postgres:postgres@postgres:5432/referral_test?schema=public + REDIS_HOST: redis + REDIS_PORT: 6379 + KAFKA_BROKERS: redpanda:29092 + JWT_SECRET: test-jwt-secret-for-docker-tests + volumes: + - ./coverage:/app/coverage + command: sh -c "npx prisma migrate deploy && npm run test:cov" + +volumes: + postgres_test_data: diff --git a/backend/services/referral-service/nest-cli.json b/backend/services/referral-service/nest-cli.json new file mode 100644 index 00000000..f9aa683b --- /dev/null +++ b/backend/services/referral-service/nest-cli.json @@ -0,0 +1,8 @@ +{ + "$schema": "https://json.schemastore.org/nest-cli", + "collection": "@nestjs/schematics", + "sourceRoot": "src", + "compilerOptions": { + "deleteOutDir": true + } +} diff --git a/backend/services/referral-service/package-lock.json b/backend/services/referral-service/package-lock.json new file mode 100644 index 00000000..fcbeeaa8 --- /dev/null +++ b/backend/services/referral-service/package-lock.json @@ -0,0 +1,10255 @@ +{ + "name": "referral-service", + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "referral-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/swagger": "^7.1.17", + "@prisma/client": "^5.7.0", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.0", + "ioredis": "^5.3.2", + "kafkajs": "^2.2.4", + "passport": "^0.7.0", + "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/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/@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/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/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/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==", + "dev": true, + "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/@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.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/@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/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/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/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/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/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.30", + "resolved": "https://registry.npmjs.org/libphonenumber-js/-/libphonenumber-js-1.12.30.tgz", + "integrity": "sha512-KxH7uIJFD6+cR6nhdh+wY6prFiH26A3W/W1gTMXnng2PXSwVfi5MhYkdq3Z2Y7vhBVa1/5VJgpNtI76UM2njGA==", + "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/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/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/referral-service/package.json b/backend/services/referral-service/package.json index e69de29b..31ba2fe8 100644 --- a/backend/services/referral-service/package.json +++ b/backend/services/referral-service/package.json @@ -0,0 +1,93 @@ +{ + "name": "referral-service", + "version": "1.0.0", + "description": "RWA Referral & Team Context Service", + "author": "RWA Team", + "private": true, + "license": "UNLICENSED", + "prisma": { + "schema": "prisma/schema.prisma", + "seed": "ts-node prisma/seed.ts" + }, + "scripts": { + "build": "nest build", + "format": "prettier --write \"src/**/*.ts\" \"test/**/*.ts\"", + "start": "nest start", + "start:dev": "nest start --watch", + "start:debug": "nest start --debug --watch", + "start:prod": "node dist/main", + "lint": "eslint \"{src,apps,libs,test}/**/*.ts\" --fix", + "test": "jest", + "test:watch": "jest --watch", + "test:cov": "jest --coverage", + "test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand", + "test:e2e": "jest --config ./test/jest-e2e.json", + "test:integration": "jest --config ./test/jest-integration.json", + "prisma:generate": "prisma generate", + "prisma:migrate": "prisma migrate dev", + "prisma:migrate:prod": "prisma migrate deploy", + "prisma:studio": "prisma studio" + }, + "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/swagger": "^7.1.17", + "@prisma/client": "^5.7.0", + "class-transformer": "^0.5.1", + "class-validator": "^0.14.0", + "ioredis": "^5.3.2", + "kafkajs": "^2.2.4", + "passport": "^0.7.0", + "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/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" + }, + "jest": { + "moduleFileExtensions": ["js", "json", "ts"], + "rootDir": ".", + "roots": ["/src/", "/test/"], + "testRegex": ".*\\.spec\\.ts$", + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "collectCoverageFrom": ["src/**/*.(t|j)s", "!src/main.ts", "!src/**/*.module.ts"], + "coverageDirectory": "./coverage", + "testEnvironment": "node", + "moduleNameMapper": { + "^@/(.*)$": "/src/$1" + } + } +} diff --git a/backend/services/referral-service/prisma/schema.prisma b/backend/services/referral-service/prisma/schema.prisma new file mode 100644 index 00000000..dc5397d1 --- /dev/null +++ b/backend/services/referral-service/prisma/schema.prisma @@ -0,0 +1,184 @@ +generator client { + provider = "prisma-client-js" +} + +datasource db { + provider = "postgresql" + url = env("DATABASE_URL") +} + +// ============================================ +// 推荐关系表 (聚合根1) +// 记录用户与推荐人的关系,推荐关系一旦建立终生不可修改 +// ============================================ +model ReferralRelationship { + id BigInt @id @default(autoincrement()) @map("relationship_id") + userId BigInt @unique @map("user_id") + + // 推荐人信息 + referrerId BigInt? @map("referrer_id") // 直接推荐人 (null = 无推荐人/根节点) + rootUserId BigInt? @map("root_user_id") // 顶级上级用户ID + + // 推荐码 + myReferralCode String @unique @map("my_referral_code") @db.VarChar(20) + usedReferralCode String? @map("used_referral_code") @db.VarChar(20) + + // 推荐链 (使用PostgreSQL数组类型,最多存储10层上级) + ancestorPath BigInt[] @map("ancestor_path") // [父节点, 祖父节点, ...] 从根到父的路径 + depth Int @default(0) @map("depth") // 层级深度 (0=根节点) + + // 直推统计 (快速查询用,冗余存储) + directReferralCount Int @default(0) @map("direct_referral_count") + activeDirectCount Int @default(0) @map("active_direct_count") // 已认种的直推人数 + + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + // 自引用关系 (方便查询推荐人) + referrer ReferralRelationship? @relation("ReferrerToReferral", fields: [referrerId], references: [userId]) + directReferrals ReferralRelationship[] @relation("ReferrerToReferral") + + // 关联团队统计 + teamStatistics TeamStatistics? + + @@map("referral_relationships") + @@index([referrerId], name: "idx_referrer") + @@index([myReferralCode], name: "idx_my_referral_code") + @@index([usedReferralCode], name: "idx_used_referral_code") + @@index([rootUserId], name: "idx_root_user") + @@index([depth], name: "idx_depth") + @@index([createdAt], name: "idx_referral_created") +} + +// ============================================ +// 团队统计表 (聚合根2) +// 每个用户的团队认种统计数据,需要实时更新 +// ============================================ +model TeamStatistics { + id BigInt @id @default(autoincrement()) @map("statistics_id") + userId BigInt @unique @map("user_id") + + // === 注册统计 === + directReferralCount Int @default(0) @map("direct_referral_count") // 直推注册数 + totalTeamCount Int @default(0) @map("total_team_count") // 团队总注册数 + + // === 个人认种 === + selfPlantingCount Int @default(0) @map("self_planting_count") // 自己认种数量 + selfPlantingAmount Decimal @default(0) @map("self_planting_amount") @db.Decimal(20, 8) + + // === 团队认种 (包含自己和所有下级) === + directPlantingCount Int @default(0) @map("direct_planting_count") // 直推认种数 + totalTeamPlantingCount Int @default(0) @map("total_team_planting_count") // 团队总认种数 + totalTeamPlantingAmount Decimal @default(0) @map("total_team_planting_amount") @db.Decimal(20, 8) + + // === 直推团队数据 (JSON存储每个直推的团队认种量) === + // 格式: [{ userId: bigint, personalCount: int, teamCount: int, amount: decimal }, ...] + directTeamPlantingData Json @default("[]") @map("direct_team_planting_data") + + // === 龙虎榜相关 === + // 龙虎榜分值 = 团队总认种量 - 最大单个直推团队认种量 + maxSingleTeamPlantingCount Int @default(0) @map("max_single_team_planting_count") + effectivePlantingCountForRanking Int @default(0) @map("effective_planting_count_for_ranking") + + // === 本省本市统计 (用于省市授权考核) === + ownProvinceTeamCount Int @default(0) @map("own_province_team_count") // 自有团队本省认种 + ownCityTeamCount Int @default(0) @map("own_city_team_count") // 自有团队本市认种 + provinceTeamPercentage Decimal @default(0) @map("province_team_percentage") @db.Decimal(5, 2) // 本省占比 + cityTeamPercentage Decimal @default(0) @map("city_team_percentage") @db.Decimal(5, 2) // 本市占比 + + // === 省市分布 (JSON存储详细分布) === + // 格式: { "provinceCode": { "cityCode": count, ... }, ... } + provinceCityDistribution Json @default("{}") @map("province_city_distribution") + + // 时间戳 + lastCalcAt DateTime? @map("last_calc_at") // 最后计算时间 + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + // 关联 + referralRelationship ReferralRelationship @relation(fields: [userId], references: [userId]) + + @@map("team_statistics") + @@index([effectivePlantingCountForRanking(sort: Desc)], name: "idx_leaderboard_score") + @@index([totalTeamPlantingCount(sort: Desc)], name: "idx_team_planting") + @@index([selfPlantingCount], name: "idx_self_planting") +} + +// ============================================ +// 直推用户列表 (冗余表,便于分页查询) +// ============================================ +model DirectReferral { + id BigInt @id @default(autoincrement()) @map("direct_referral_id") + referrerId BigInt @map("referrer_id") // 推荐人ID + referralId BigInt @map("referral_id") // 被推荐人ID + referralSequence BigInt @map("referral_sequence") // 被推荐人序列号 + + // 被推荐人信息快照 (冗余存储,避免跨服务查询) + referralNickname String? @map("referral_nickname") @db.VarChar(100) + referralAvatar String? @map("referral_avatar") @db.VarChar(255) + + // 该直推的认种统计 + personalPlantingCount Int @default(0) @map("personal_planting_count") // 个人认种数 + teamPlantingCount Int @default(0) @map("team_planting_count") // 团队认种数(含个人) + + // 是否已认种 (用于区分活跃/非活跃) + hasPlanted Boolean @default(false) @map("has_planted") + firstPlantedAt DateTime? @map("first_planted_at") + + createdAt DateTime @default(now()) @map("created_at") + updatedAt DateTime @updatedAt @map("updated_at") + + @@unique([referrerId, referralId], name: "uk_referrer_referral") + @@map("direct_referrals") + @@index([referrerId], name: "idx_direct_referrer") + @@index([referralId], name: "idx_direct_referral") + @@index([hasPlanted], name: "idx_has_planted") + @@index([teamPlantingCount(sort: Desc)], name: "idx_direct_team_planting") +} + +// ============================================ +// 团队省市分布表 (用于省市权益分配) +// ============================================ +model TeamProvinceCityDetail { + id BigInt @id @default(autoincrement()) @map("detail_id") + userId BigInt @map("user_id") + + provinceCode String @map("province_code") @db.VarChar(10) + cityCode String @map("city_code") @db.VarChar(10) + + teamPlantingCount Int @default(0) @map("team_planting_count") // 该省/市团队认种数 + + updatedAt DateTime @updatedAt @map("updated_at") + + @@unique([userId, provinceCode, cityCode], name: "uk_user_province_city") + @@map("team_province_city_details") + @@index([userId], name: "idx_detail_user") + @@index([provinceCode], name: "idx_detail_province") + @@index([cityCode], name: "idx_detail_city") +} + +// ============================================ +// 推荐事件表 (行为表,append-only,用于审计和事件溯源) +// ============================================ +model ReferralEvent { + id BigInt @id @default(autoincrement()) @map("event_id") + eventType String @map("event_type") @db.VarChar(50) + + // 聚合根信息 + aggregateId String @map("aggregate_id") @db.VarChar(100) + aggregateType String @map("aggregate_type") @db.VarChar(50) + + // 事件数据 + eventData Json @map("event_data") + + // 元数据 + userId BigInt? @map("user_id") + occurredAt DateTime @default(now()) @map("occurred_at") @db.Timestamp(6) + version Int @default(1) @map("version") + + @@map("referral_events") + @@index([aggregateType, aggregateId], name: "idx_event_aggregate") + @@index([eventType], name: "idx_event_type") + @@index([userId], name: "idx_event_user") + @@index([occurredAt], name: "idx_event_occurred") +} diff --git a/backend/services/referral-service/scripts/run-all-tests.sh b/backend/services/referral-service/scripts/run-all-tests.sh new file mode 100644 index 00000000..4f39a896 --- /dev/null +++ b/backend/services/referral-service/scripts/run-all-tests.sh @@ -0,0 +1,157 @@ +#!/bin/bash + +# ============================================ +# 完整测试运行脚本 +# Run all tests for referral-service +# ============================================ + +set -e + +echo "============================================" +echo "Referral Service - Complete Test Suite" +echo "============================================" +echo "" + +# Colors for output +RED='\033[0;31m' +GREEN='\033[0;32m' +YELLOW='\033[1;33m' +NC='\033[0m' # No Color + +# Track test results +UNIT_RESULT=0 +INTEGRATION_RESULT=0 +E2E_RESULT=0 +DOCKER_RESULT=0 + +# Function to print status +print_status() { + if [ $1 -eq 0 ]; then + echo -e "${GREEN}✓ $2 PASSED${NC}" + else + echo -e "${RED}✗ $2 FAILED${NC}" + fi +} + +# Change to project directory +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" +cd "$SCRIPT_DIR/.." + +echo "Working directory: $(pwd)" +echo "" + +# ============================================ +# 1. Install dependencies +# ============================================ +echo -e "${YELLOW}Step 1: Installing dependencies...${NC}" +npm install +echo "" + +# ============================================ +# 2. Generate Prisma client +# ============================================ +echo -e "${YELLOW}Step 2: Generating Prisma client...${NC}" +npx prisma generate +echo "" + +# ============================================ +# 3. Build the project +# ============================================ +echo -e "${YELLOW}Step 3: Building the project...${NC}" +npm run build +echo "" + +# ============================================ +# 4. Run Unit Tests +# ============================================ +echo "============================================" +echo -e "${YELLOW}Step 4: Running Unit Tests...${NC}" +echo "============================================" +if npm test; then + UNIT_RESULT=0 +else + UNIT_RESULT=1 +fi +echo "" + +# ============================================ +# 5. Run Integration Tests +# ============================================ +echo "============================================" +echo -e "${YELLOW}Step 5: Running Integration Tests...${NC}" +echo "============================================" +if npm run test:integration 2>/dev/null || npm test -- --testRegex="integration.spec.ts$"; then + INTEGRATION_RESULT=0 +else + INTEGRATION_RESULT=1 +fi +echo "" + +# ============================================ +# 6. Run E2E Tests +# ============================================ +echo "============================================" +echo -e "${YELLOW}Step 6: Running E2E Tests...${NC}" +echo "============================================" +if npm run test:e2e 2>/dev/null || npm test -- --testRegex="e2e-spec.ts$"; then + E2E_RESULT=0 +else + E2E_RESULT=1 +fi +echo "" + +# ============================================ +# 7. Run Docker Tests (optional) +# ============================================ +echo "============================================" +echo -e "${YELLOW}Step 7: Running Docker Tests...${NC}" +echo "============================================" + +if command -v docker &> /dev/null && command -v docker-compose &> /dev/null; then + echo "Docker is available, running containerized tests..." + + # Build and run tests in Docker + if docker-compose -f docker-compose.test.yml build test && \ + docker-compose -f docker-compose.test.yml up --abort-on-container-exit test; then + DOCKER_RESULT=0 + else + DOCKER_RESULT=1 + fi + + # Cleanup + docker-compose -f docker-compose.test.yml down -v +else + echo -e "${YELLOW}Docker not available, skipping Docker tests${NC}" + DOCKER_RESULT=-1 +fi +echo "" + +# ============================================ +# Summary +# ============================================ +echo "============================================" +echo "TEST SUMMARY" +echo "============================================" +print_status $UNIT_RESULT "Unit Tests" +print_status $INTEGRATION_RESULT "Integration Tests" +print_status $E2E_RESULT "E2E Tests" +if [ $DOCKER_RESULT -eq -1 ]; then + echo -e "${YELLOW}○ Docker Tests SKIPPED${NC}" +else + print_status $DOCKER_RESULT "Docker Tests" +fi +echo "============================================" + +# Calculate overall result +TOTAL_FAILURES=$((UNIT_RESULT + INTEGRATION_RESULT + E2E_RESULT)) +if [ $DOCKER_RESULT -gt 0 ]; then + TOTAL_FAILURES=$((TOTAL_FAILURES + DOCKER_RESULT)) +fi + +if [ $TOTAL_FAILURES -eq 0 ]; then + echo -e "${GREEN}All tests passed!${NC}" + exit 0 +else + echo -e "${RED}Some tests failed!${NC}" + exit 1 +fi diff --git a/backend/services/referral-service/src/api/controllers/health.controller.ts b/backend/services/referral-service/src/api/controllers/health.controller.ts new file mode 100644 index 00000000..bb1c2f80 --- /dev/null +++ b/backend/services/referral-service/src/api/controllers/health.controller.ts @@ -0,0 +1,31 @@ +import { Controller, Get } from '@nestjs/common'; +import { ApiTags, ApiOperation, ApiResponse } from '@nestjs/swagger'; + +@ApiTags('Health') +@Controller('health') +export class HealthController { + @Get() + @ApiOperation({ summary: '健康检查' }) + @ApiResponse({ status: 200, description: '服务正常' }) + check(): { status: string; service: string; timestamp: string } { + return { + status: 'ok', + service: 'referral-service', + timestamp: new Date().toISOString(), + }; + } + + @Get('ready') + @ApiOperation({ summary: '就绪检查' }) + @ApiResponse({ status: 200, description: '服务就绪' }) + ready(): { status: string } { + return { status: 'ready' }; + } + + @Get('live') + @ApiOperation({ summary: '存活检查' }) + @ApiResponse({ status: 200, description: '服务存活' }) + live(): { status: string } { + return { status: 'alive' }; + } +} diff --git a/backend/services/referral-service/src/api/controllers/index.ts b/backend/services/referral-service/src/api/controllers/index.ts new file mode 100644 index 00000000..9b55c024 --- /dev/null +++ b/backend/services/referral-service/src/api/controllers/index.ts @@ -0,0 +1,4 @@ +export * from './referral.controller'; +export * from './leaderboard.controller'; +export * from './team-statistics.controller'; +export * from './health.controller'; diff --git a/backend/services/referral-service/src/api/controllers/leaderboard.controller.ts b/backend/services/referral-service/src/api/controllers/leaderboard.controller.ts new file mode 100644 index 00000000..c8e976d3 --- /dev/null +++ b/backend/services/referral-service/src/api/controllers/leaderboard.controller.ts @@ -0,0 +1,69 @@ +import { Controller, Get, Query, Param, UseGuards } from '@nestjs/common'; +import { + ApiTags, + ApiOperation, + ApiResponse, + ApiBearerAuth, + ApiParam, +} from '@nestjs/swagger'; +import { JwtAuthGuard } from '../guards'; +import { CurrentUser } from '../decorators'; +import { TeamStatisticsService } from '../../application/services'; +import { + GetLeaderboardDto, + LeaderboardResponseDto, + UserRankResponseDto, +} from '../dto'; +import { GetLeaderboardQuery } from '../../application/queries'; + +@ApiTags('Leaderboard') +@Controller('leaderboard') +export class LeaderboardController { + constructor(private readonly teamStatisticsService: TeamStatisticsService) {} + + @Get() + @ApiOperation({ summary: '获取龙虎榜排名' }) + @ApiResponse({ status: 200, type: LeaderboardResponseDto }) + async getLeaderboard(@Query() dto: GetLeaderboardDto): Promise { + const query = new GetLeaderboardQuery(dto.limit, dto.offset); + return this.teamStatisticsService.getLeaderboard(query); + } + + @Get('me') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: '获取当前用户龙虎榜排名' }) + @ApiResponse({ status: 200, type: UserRankResponseDto }) + async getMyRank(@CurrentUser('userId') userId: bigint): Promise { + const rank = await this.teamStatisticsService.getUserRank(userId); + const leaderboard = await this.teamStatisticsService.getLeaderboard( + new GetLeaderboardQuery(1000, 0), + ); + const userEntry = leaderboard.entries.find((e) => e.userId === userId.toString()); + + return { + userId: userId.toString(), + rank, + score: userEntry?.score ?? 0, + }; + } + + @Get('user/:userId') + @ApiOperation({ summary: '获取指定用户龙虎榜排名' }) + @ApiParam({ name: 'userId', description: '用户ID' }) + @ApiResponse({ status: 200, type: UserRankResponseDto }) + async getUserRank(@Param('userId') userId: string): Promise { + const userIdBigInt = BigInt(userId); + const rank = await this.teamStatisticsService.getUserRank(userIdBigInt); + const leaderboard = await this.teamStatisticsService.getLeaderboard( + new GetLeaderboardQuery(1000, 0), + ); + const userEntry = leaderboard.entries.find((e) => e.userId === userId); + + return { + userId, + rank, + score: userEntry?.score ?? 0, + }; + } +} diff --git a/backend/services/referral-service/src/api/controllers/referral.controller.ts b/backend/services/referral-service/src/api/controllers/referral.controller.ts new file mode 100644 index 00000000..16635ae0 --- /dev/null +++ b/backend/services/referral-service/src/api/controllers/referral.controller.ts @@ -0,0 +1,108 @@ +import { + Controller, + Get, + Post, + Query, + Param, + Body, + UseGuards, + HttpCode, + HttpStatus, +} from '@nestjs/common'; +import { + ApiTags, + ApiOperation, + ApiResponse, + ApiBearerAuth, + ApiParam, +} from '@nestjs/swagger'; +import { JwtAuthGuard } from '../guards'; +import { CurrentUser } from '../decorators'; +import { ReferralService } from '../../application/services'; +import { + ValidateReferralCodeDto, + GetDirectReferralsDto, + ReferralInfoResponseDto, + DirectReferralsResponseDto, + ValidateCodeResponseDto, + CreateReferralDto, +} from '../dto'; +import { CreateReferralRelationshipCommand } from '../../application/commands'; +import { GetUserReferralInfoQuery, GetDirectReferralsQuery } from '../../application/queries'; + +@ApiTags('Referral') +@Controller('referral') +export class ReferralController { + constructor(private readonly referralService: ReferralService) {} + + @Get('me') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: '获取当前用户推荐信息' }) + @ApiResponse({ status: 200, type: ReferralInfoResponseDto }) + async getMyReferralInfo(@CurrentUser('userId') userId: bigint): Promise { + const query = new GetUserReferralInfoQuery(userId); + return this.referralService.getUserReferralInfo(query); + } + + @Get('me/direct-referrals') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: '获取当前用户直推列表' }) + @ApiResponse({ status: 200, type: DirectReferralsResponseDto }) + async getMyDirectReferrals( + @CurrentUser('userId') userId: bigint, + @Query() dto: GetDirectReferralsDto, + ): Promise { + const query = new GetDirectReferralsQuery(userId, dto.limit, dto.offset); + return this.referralService.getDirectReferrals(query); + } + + @Get('validate/:code') + @ApiOperation({ summary: '验证推荐码是否有效' }) + @ApiParam({ name: 'code', description: '推荐码' }) + @ApiResponse({ status: 200, type: ValidateCodeResponseDto }) + async validateCode(@Param('code') code: string): Promise { + const referrer = await this.referralService.getReferrerByCode(code.toUpperCase()); + return { + valid: referrer !== null, + referrerId: referrer?.userId, + }; + } + + @Post('validate') + @HttpCode(HttpStatus.OK) + @ApiOperation({ summary: '验证推荐码 (POST方式)' }) + @ApiResponse({ status: 200, type: ValidateCodeResponseDto }) + async validateCodePost(@Body() dto: ValidateReferralCodeDto): Promise { + const referrer = await this.referralService.getReferrerByCode(dto.code.toUpperCase()); + return { + valid: referrer !== null, + referrerId: referrer?.userId, + }; + } + + @Post() + @ApiOperation({ summary: '创建推荐关系 (内部接口)' }) + @ApiResponse({ status: 201, description: '创建成功' }) + async createReferralRelationship( + @Body() dto: CreateReferralDto, + ): Promise<{ referralCode: string }> { + const command = new CreateReferralRelationshipCommand( + BigInt(dto.userId), + dto.referrerCode ?? null, + ); + return this.referralService.createReferralRelationship(command); + } + + @Get('user/:userId') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: '获取指定用户推荐信息' }) + @ApiParam({ name: 'userId', description: '用户ID' }) + @ApiResponse({ status: 200, type: ReferralInfoResponseDto }) + async getUserReferralInfo(@Param('userId') userId: string): Promise { + const query = new GetUserReferralInfoQuery(BigInt(userId)); + return this.referralService.getUserReferralInfo(query); + } +} diff --git a/backend/services/referral-service/src/api/controllers/team-statistics.controller.ts b/backend/services/referral-service/src/api/controllers/team-statistics.controller.ts new file mode 100644 index 00000000..6d795b54 --- /dev/null +++ b/backend/services/referral-service/src/api/controllers/team-statistics.controller.ts @@ -0,0 +1,30 @@ +import { Controller, Get, UseGuards } from '@nestjs/common'; +import { + ApiTags, + ApiOperation, + ApiResponse, + ApiBearerAuth, +} from '@nestjs/swagger'; +import { JwtAuthGuard } from '../guards'; +import { CurrentUser } from '../decorators'; +import { TeamStatisticsService } from '../../application/services'; +import { ProvinceCityDistributionResponseDto } from '../dto'; +import { GetProvinceCityDistributionQuery } from '../../application/queries'; + +@ApiTags('Team Statistics') +@Controller('team-statistics') +export class TeamStatisticsController { + constructor(private readonly teamStatisticsService: TeamStatisticsService) {} + + @Get('me/distribution') + @UseGuards(JwtAuthGuard) + @ApiBearerAuth() + @ApiOperation({ summary: '获取当前用户团队省市分布' }) + @ApiResponse({ status: 200, type: ProvinceCityDistributionResponseDto }) + async getMyDistribution( + @CurrentUser('userId') userId: bigint, + ): Promise { + const query = new GetProvinceCityDistributionQuery(userId); + return this.teamStatisticsService.getProvinceCityDistribution(query); + } +} diff --git a/backend/services/referral-service/src/api/decorators/current-user.decorator.ts b/backend/services/referral-service/src/api/decorators/current-user.decorator.ts new file mode 100644 index 00000000..a292e40f --- /dev/null +++ b/backend/services/referral-service/src/api/decorators/current-user.decorator.ts @@ -0,0 +1,11 @@ +import { createParamDecorator, ExecutionContext } from '@nestjs/common'; +import { AuthenticatedRequest } from '../guards/jwt-auth.guard'; + +export const CurrentUser = createParamDecorator( + (data: keyof AuthenticatedRequest['user'] | undefined, ctx: ExecutionContext) => { + const request = ctx.switchToHttp().getRequest(); + const user = request.user; + + return data ? user?.[data] : user; + }, +); diff --git a/backend/services/referral-service/src/api/decorators/index.ts b/backend/services/referral-service/src/api/decorators/index.ts new file mode 100644 index 00000000..67e74a52 --- /dev/null +++ b/backend/services/referral-service/src/api/decorators/index.ts @@ -0,0 +1 @@ +export * from './current-user.decorator'; diff --git a/backend/services/referral-service/src/api/dto/index.ts b/backend/services/referral-service/src/api/dto/index.ts new file mode 100644 index 00000000..7bae33b2 --- /dev/null +++ b/backend/services/referral-service/src/api/dto/index.ts @@ -0,0 +1,3 @@ +export * from './referral.dto'; +export * from './leaderboard.dto'; +export * from './team-statistics.dto'; diff --git a/backend/services/referral-service/src/api/dto/leaderboard.dto.ts b/backend/services/referral-service/src/api/dto/leaderboard.dto.ts new file mode 100644 index 00000000..ba9be476 --- /dev/null +++ b/backend/services/referral-service/src/api/dto/leaderboard.dto.ts @@ -0,0 +1,56 @@ +import { IsInt, Min, Max, IsOptional } from 'class-validator'; +import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; + +export class GetLeaderboardDto { + @ApiPropertyOptional({ description: '每页数量', default: 100 }) + @IsOptional() + @IsInt() + @Min(1) + @Max(100) + limit?: number = 100; + + @ApiPropertyOptional({ description: '偏移量', default: 0 }) + @IsOptional() + @IsInt() + @Min(0) + offset?: number = 0; +} + +export class LeaderboardEntryResponseDto { + @ApiProperty({ description: '排名' }) + rank: number; + + @ApiProperty({ description: '用户ID' }) + userId: string; + + @ApiProperty({ description: '龙虎榜分值' }) + score: number; + + @ApiProperty({ description: '团队总认种量' }) + totalTeamCount: number; + + @ApiProperty({ description: '直推人数' }) + directReferralCount: number; +} + +export class LeaderboardResponseDto { + @ApiProperty({ type: [LeaderboardEntryResponseDto] }) + entries: LeaderboardEntryResponseDto[]; + + @ApiProperty({ description: '总数' }) + total: number; + + @ApiProperty({ description: '是否有更多' }) + hasMore: boolean; +} + +export class UserRankResponseDto { + @ApiProperty({ description: '用户ID' }) + userId: string; + + @ApiProperty({ description: '排名', nullable: true }) + rank: number | null; + + @ApiProperty({ description: '龙虎榜分值' }) + score: number; +} diff --git a/backend/services/referral-service/src/api/dto/referral.dto.ts b/backend/services/referral-service/src/api/dto/referral.dto.ts new file mode 100644 index 00000000..1e06d0fe --- /dev/null +++ b/backend/services/referral-service/src/api/dto/referral.dto.ts @@ -0,0 +1,105 @@ +import { IsString, IsOptional, Length, Matches, IsInt, Min, Max } from 'class-validator'; +import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; + +export class ValidateReferralCodeDto { + @ApiProperty({ description: '推荐码', example: 'RWA123ABC' }) + @IsString() + @Length(6, 20) + @Matches(/^[A-Z0-9]+$/, { message: '推荐码只能包含大写字母和数字' }) + code: string; +} + +export class CreateReferralDto { + @ApiProperty({ description: '用户ID', example: '123456789' }) + @IsString() + userId: string; + + @ApiPropertyOptional({ description: '推荐码', example: 'RWA123ABC' }) + @IsOptional() + @IsString() + @Length(6, 20) + referrerCode?: string; +} + +export class GetDirectReferralsDto { + @ApiPropertyOptional({ description: '每页数量', default: 50 }) + @IsOptional() + @IsInt() + @Min(1) + @Max(100) + limit?: number = 50; + + @ApiPropertyOptional({ description: '偏移量', default: 0 }) + @IsOptional() + @IsInt() + @Min(0) + offset?: number = 0; +} + +export class ReferralInfoResponseDto { + @ApiProperty({ description: '用户ID' }) + userId: string; + + @ApiProperty({ description: '推荐码' }) + referralCode: string; + + @ApiProperty({ description: '推荐人ID', nullable: true }) + referrerId: string | null; + + @ApiProperty({ description: '推荐链深度' }) + referralChainDepth: number; + + @ApiProperty({ description: '直推人数' }) + directReferralCount: number; + + @ApiProperty({ description: '团队总认种量' }) + totalTeamCount: number; + + @ApiProperty({ description: '个人认种量' }) + personalPlantingCount: number; + + @ApiProperty({ description: '团队认种量' }) + teamPlantingCount: number; + + @ApiProperty({ description: '龙虎榜分值' }) + leaderboardScore: number; + + @ApiProperty({ description: '龙虎榜排名', nullable: true }) + leaderboardRank: number | null; + + @ApiProperty({ description: '创建时间' }) + createdAt: Date; +} + +export class DirectReferralResponseDto { + @ApiProperty({ description: '用户ID' }) + userId: string; + + @ApiProperty({ description: '推荐码' }) + referralCode: string; + + @ApiProperty({ description: '团队认种量' }) + teamCount: number; + + @ApiProperty({ description: '加入时间' }) + joinedAt: Date; +} + +export class DirectReferralsResponseDto { + @ApiProperty({ type: [DirectReferralResponseDto] }) + referrals: DirectReferralResponseDto[]; + + @ApiProperty({ description: '总数' }) + total: number; + + @ApiProperty({ description: '是否有更多' }) + hasMore: boolean; +} + +export class ValidateCodeResponseDto { + @ApiProperty({ description: '是否有效' }) + valid: boolean; + + @ApiPropertyOptional({ description: '推荐人ID' }) + referrerId?: string; +} diff --git a/backend/services/referral-service/src/api/dto/team-statistics.dto.ts b/backend/services/referral-service/src/api/dto/team-statistics.dto.ts new file mode 100644 index 00000000..6935e376 --- /dev/null +++ b/backend/services/referral-service/src/api/dto/team-statistics.dto.ts @@ -0,0 +1,51 @@ +import { ApiProperty } from '@nestjs/swagger'; + +export class CityDistributionDto { + @ApiProperty({ description: '城市代码' }) + cityCode: string; + + @ApiProperty({ description: '认种数量' }) + count: number; +} + +export class ProvinceDistributionDto { + @ApiProperty({ description: '省份代码' }) + provinceCode: string; + + @ApiProperty({ description: '省份总认种量' }) + total: number; + + @ApiProperty({ type: [CityDistributionDto] }) + cities: CityDistributionDto[]; +} + +export class ProvinceCityDistributionResponseDto { + @ApiProperty({ type: [ProvinceDistributionDto] }) + provinces: ProvinceDistributionDto[]; + + @ApiProperty({ description: '总认种量' }) + totalCount: number; +} + +export class TeamStatisticsResponseDto { + @ApiProperty({ description: '用户ID' }) + userId: string; + + @ApiProperty({ description: '直推人数' }) + directReferralCount: number; + + @ApiProperty({ description: '团队总认种量' }) + totalTeamCount: number; + + @ApiProperty({ description: '个人认种量' }) + personalPlantingCount: number; + + @ApiProperty({ description: '团队认种量' }) + teamPlantingCount: number; + + @ApiProperty({ description: '龙虎榜分值' }) + leaderboardScore: number; + + @ApiProperty({ description: '最大单直推团队认种量' }) + maxDirectTeamCount: number; +} diff --git a/backend/services/referral-service/src/api/guards/index.ts b/backend/services/referral-service/src/api/guards/index.ts new file mode 100644 index 00000000..3525e2d2 --- /dev/null +++ b/backend/services/referral-service/src/api/guards/index.ts @@ -0,0 +1 @@ +export * from './jwt-auth.guard'; diff --git a/backend/services/referral-service/src/api/guards/jwt-auth.guard.ts b/backend/services/referral-service/src/api/guards/jwt-auth.guard.ts new file mode 100644 index 00000000..88981dad --- /dev/null +++ b/backend/services/referral-service/src/api/guards/jwt-auth.guard.ts @@ -0,0 +1,69 @@ +import { + Injectable, + CanActivate, + ExecutionContext, + UnauthorizedException, + Logger, +} from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; +import * as jwt from 'jsonwebtoken'; + +export interface JwtPayload { + sub: string; + userId: string; + type: 'access' | 'refresh'; + iat: number; + exp: number; +} + +export interface AuthenticatedRequest { + user: { + userId: bigint; + sub: string; + }; +} + +@Injectable() +export class JwtAuthGuard implements CanActivate { + private readonly logger = new Logger(JwtAuthGuard.name); + private readonly jwtSecret: string; + + constructor(private readonly configService: ConfigService) { + this.jwtSecret = this.configService.get('JWT_SECRET', 'default-secret-change-in-production'); + } + + async canActivate(context: ExecutionContext): Promise { + const request = context.switchToHttp().getRequest(); + const token = this.extractTokenFromHeader(request); + + if (!token) { + throw new UnauthorizedException('Missing authentication token'); + } + + try { + const payload = jwt.verify(token, this.jwtSecret) as JwtPayload; + + if (payload.type !== 'access') { + throw new UnauthorizedException('Invalid token type'); + } + + request.user = { + userId: BigInt(payload.userId), + sub: payload.sub, + }; + + return true; + } catch (error) { + this.logger.warn(`JWT verification failed: ${error}`); + throw new UnauthorizedException('Invalid authentication token'); + } + } + + private extractTokenFromHeader(request: { headers: { authorization?: string } }): string | null { + const authHeader = request.headers.authorization; + if (!authHeader) return null; + + const [type, token] = authHeader.split(' '); + return type === 'Bearer' ? token : null; + } +} diff --git a/backend/services/referral-service/src/api/index.ts b/backend/services/referral-service/src/api/index.ts new file mode 100644 index 00000000..a4b6fb56 --- /dev/null +++ b/backend/services/referral-service/src/api/index.ts @@ -0,0 +1,4 @@ +export * from './controllers'; +export * from './dto'; +export * from './guards'; +export * from './decorators'; diff --git a/backend/services/referral-service/src/app.module.ts b/backend/services/referral-service/src/app.module.ts new file mode 100644 index 00000000..62602db9 --- /dev/null +++ b/backend/services/referral-service/src/app.module.ts @@ -0,0 +1,14 @@ +import { Module } from '@nestjs/common'; +import { ConfigModule } from '@nestjs/config'; +import { ApiModule } from './modules'; + +@Module({ + imports: [ + ConfigModule.forRoot({ + isGlobal: true, + envFilePath: ['.env.development', '.env'], + }), + ApiModule, + ], +}) +export class AppModule {} diff --git a/backend/services/referral-service/src/application/commands/create-referral-relationship.command.ts b/backend/services/referral-service/src/application/commands/create-referral-relationship.command.ts new file mode 100644 index 00000000..434a8335 --- /dev/null +++ b/backend/services/referral-service/src/application/commands/create-referral-relationship.command.ts @@ -0,0 +1,6 @@ +export class CreateReferralRelationshipCommand { + constructor( + public readonly userId: bigint, + public readonly referrerCode: string | null, + ) {} +} diff --git a/backend/services/referral-service/src/application/commands/index.ts b/backend/services/referral-service/src/application/commands/index.ts new file mode 100644 index 00000000..ef89939a --- /dev/null +++ b/backend/services/referral-service/src/application/commands/index.ts @@ -0,0 +1,2 @@ +export * from './create-referral-relationship.command'; +export * from './update-team-statistics.command'; diff --git a/backend/services/referral-service/src/application/commands/update-team-statistics.command.ts b/backend/services/referral-service/src/application/commands/update-team-statistics.command.ts new file mode 100644 index 00000000..e6939479 --- /dev/null +++ b/backend/services/referral-service/src/application/commands/update-team-statistics.command.ts @@ -0,0 +1,8 @@ +export class UpdateTeamStatisticsCommand { + constructor( + public readonly userId: bigint, + public readonly plantingCount: number, + public readonly provinceCode: string, + public readonly cityCode: string, + ) {} +} diff --git a/backend/services/referral-service/src/application/event-handlers/index.ts b/backend/services/referral-service/src/application/event-handlers/index.ts new file mode 100644 index 00000000..8fbb9202 --- /dev/null +++ b/backend/services/referral-service/src/application/event-handlers/index.ts @@ -0,0 +1,2 @@ +export * from './user-registered.handler'; +export * from './planting-created.handler'; diff --git a/backend/services/referral-service/src/application/event-handlers/planting-created.handler.ts b/backend/services/referral-service/src/application/event-handlers/planting-created.handler.ts new file mode 100644 index 00000000..2355cca3 --- /dev/null +++ b/backend/services/referral-service/src/application/event-handlers/planting-created.handler.ts @@ -0,0 +1,64 @@ +import { Injectable, Logger, OnModuleInit } from '@nestjs/common'; +import { KafkaService } from '../../infrastructure'; +import { TeamStatisticsService } from '../services'; +import { UpdateTeamStatisticsCommand } from '../commands'; + +interface PlantingCreatedEvent { + eventName: string; + data: { + userId: string; + treeCount: number; + provinceCode: string; + cityCode: string; + }; +} + +/** + * 认种创建事件处理器 + * 监听 planting-service 发出的认种事件 + */ +@Injectable() +export class PlantingCreatedHandler implements OnModuleInit { + private readonly logger = new Logger(PlantingCreatedHandler.name); + + constructor( + private readonly kafkaService: KafkaService, + private readonly teamStatisticsService: TeamStatisticsService, + ) {} + + async onModuleInit() { + await this.kafkaService.subscribe( + 'referral-service-planting-created', + ['planting.planting.created'], + this.handleMessage.bind(this), + ); + this.logger.log('Subscribed to planting.created events'); + } + + private async handleMessage(topic: string, message: Record): Promise { + const event = message as unknown as PlantingCreatedEvent; + + if (event.eventName !== 'planting.created') { + return; + } + + try { + const command = new UpdateTeamStatisticsCommand( + BigInt(event.data.userId), + event.data.treeCount, + event.data.provinceCode, + event.data.cityCode, + ); + + await this.teamStatisticsService.handlePlantingEvent(command); + this.logger.log( + `Updated team statistics for user ${event.data.userId}, count: ${event.data.treeCount}`, + ); + } catch (error) { + this.logger.error( + `Failed to update team statistics for user ${event.data.userId}:`, + error, + ); + } + } +} diff --git a/backend/services/referral-service/src/application/event-handlers/user-registered.handler.ts b/backend/services/referral-service/src/application/event-handlers/user-registered.handler.ts new file mode 100644 index 00000000..95913985 --- /dev/null +++ b/backend/services/referral-service/src/application/event-handlers/user-registered.handler.ts @@ -0,0 +1,57 @@ +import { Injectable, Logger, OnModuleInit } from '@nestjs/common'; +import { KafkaService } from '../../infrastructure'; +import { ReferralService } from '../services'; +import { CreateReferralRelationshipCommand } from '../commands'; + +interface UserRegisteredEvent { + eventName: string; + data: { + userId: string; + referralCode?: string; + }; +} + +/** + * 用户注册事件处理器 + * 监听 identity-service 发出的用户注册事件 + */ +@Injectable() +export class UserRegisteredHandler implements OnModuleInit { + private readonly logger = new Logger(UserRegisteredHandler.name); + + constructor( + private readonly kafkaService: KafkaService, + private readonly referralService: ReferralService, + ) {} + + async onModuleInit() { + await this.kafkaService.subscribe( + 'referral-service-user-registered', + ['identity.user.registered'], + this.handleMessage.bind(this), + ); + this.logger.log('Subscribed to user.registered events'); + } + + private async handleMessage(topic: string, message: Record): Promise { + const event = message as unknown as UserRegisteredEvent; + + if (event.eventName !== 'user.registered') { + return; + } + + try { + const command = new CreateReferralRelationshipCommand( + BigInt(event.data.userId), + event.data.referralCode ?? null, + ); + + const result = await this.referralService.createReferralRelationship(command); + this.logger.log( + `Created referral relationship for user ${event.data.userId}, code: ${result.referralCode}`, + ); + } catch (error) { + this.logger.error(`Failed to create referral relationship for user ${event.data.userId}:`, error); + } + } +} diff --git a/backend/services/referral-service/src/application/index.ts b/backend/services/referral-service/src/application/index.ts new file mode 100644 index 00000000..4d3b8ea8 --- /dev/null +++ b/backend/services/referral-service/src/application/index.ts @@ -0,0 +1,4 @@ +export * from './commands'; +export * from './queries'; +export * from './services'; +export * from './event-handlers'; diff --git a/backend/services/referral-service/src/application/queries/get-direct-referrals.query.ts b/backend/services/referral-service/src/application/queries/get-direct-referrals.query.ts new file mode 100644 index 00000000..1967d270 --- /dev/null +++ b/backend/services/referral-service/src/application/queries/get-direct-referrals.query.ts @@ -0,0 +1,20 @@ +export class GetDirectReferralsQuery { + constructor( + public readonly userId: bigint, + public readonly limit: number = 50, + public readonly offset: number = 0, + ) {} +} + +export interface DirectReferralResult { + userId: string; + referralCode: string; + teamCount: number; + joinedAt: Date; +} + +export interface DirectReferralsResult { + referrals: DirectReferralResult[]; + total: number; + hasMore: boolean; +} diff --git a/backend/services/referral-service/src/application/queries/get-leaderboard.query.ts b/backend/services/referral-service/src/application/queries/get-leaderboard.query.ts new file mode 100644 index 00000000..f1965c95 --- /dev/null +++ b/backend/services/referral-service/src/application/queries/get-leaderboard.query.ts @@ -0,0 +1,20 @@ +export class GetLeaderboardQuery { + constructor( + public readonly limit: number = 100, + public readonly offset: number = 0, + ) {} +} + +export interface LeaderboardEntryResult { + rank: number; + userId: string; + score: number; + totalTeamCount: number; + directReferralCount: number; +} + +export interface LeaderboardResult { + entries: LeaderboardEntryResult[]; + total: number; + hasMore: boolean; +} diff --git a/backend/services/referral-service/src/application/queries/get-province-city-distribution.query.ts b/backend/services/referral-service/src/application/queries/get-province-city-distribution.query.ts new file mode 100644 index 00000000..0e00bfc5 --- /dev/null +++ b/backend/services/referral-service/src/application/queries/get-province-city-distribution.query.ts @@ -0,0 +1,15 @@ +export class GetProvinceCityDistributionQuery { + constructor(public readonly userId: bigint) {} +} + +export interface ProvinceCityDistributionResult { + provinces: Array<{ + provinceCode: string; + total: number; + cities: Array<{ + cityCode: string; + count: number; + }>; + }>; + totalCount: number; +} diff --git a/backend/services/referral-service/src/application/queries/get-user-referral-info.query.ts b/backend/services/referral-service/src/application/queries/get-user-referral-info.query.ts new file mode 100644 index 00000000..1330f2c7 --- /dev/null +++ b/backend/services/referral-service/src/application/queries/get-user-referral-info.query.ts @@ -0,0 +1,17 @@ +export class GetUserReferralInfoQuery { + constructor(public readonly userId: bigint) {} +} + +export interface UserReferralInfoResult { + userId: string; + referralCode: string; + referrerId: string | null; + referralChainDepth: number; + directReferralCount: number; + totalTeamCount: number; + personalPlantingCount: number; + teamPlantingCount: number; + leaderboardScore: number; + leaderboardRank: number | null; + createdAt: Date; +} diff --git a/backend/services/referral-service/src/application/queries/index.ts b/backend/services/referral-service/src/application/queries/index.ts new file mode 100644 index 00000000..a5541c1f --- /dev/null +++ b/backend/services/referral-service/src/application/queries/index.ts @@ -0,0 +1,4 @@ +export * from './get-user-referral-info.query'; +export * from './get-leaderboard.query'; +export * from './get-direct-referrals.query'; +export * from './get-province-city-distribution.query'; diff --git a/backend/services/referral-service/src/application/services/index.ts b/backend/services/referral-service/src/application/services/index.ts new file mode 100644 index 00000000..2c053626 --- /dev/null +++ b/backend/services/referral-service/src/application/services/index.ts @@ -0,0 +1,2 @@ +export * from './referral.service'; +export * from './team-statistics.service'; diff --git a/backend/services/referral-service/src/application/services/referral.service.ts b/backend/services/referral-service/src/application/services/referral.service.ts new file mode 100644 index 00000000..6a2538ca --- /dev/null +++ b/backend/services/referral-service/src/application/services/referral.service.ts @@ -0,0 +1,160 @@ +import { Injectable, Inject, Logger, NotFoundException, BadRequestException } from '@nestjs/common'; +import { + REFERRAL_RELATIONSHIP_REPOSITORY, + IReferralRelationshipRepository, + TEAM_STATISTICS_REPOSITORY, + ITeamStatisticsRepository, + ReferralRelationship, + ReferralChainService, +} from '../../domain'; +import { EventPublisherService, LeaderboardCacheService } from '../../infrastructure'; +import { CreateReferralRelationshipCommand } from '../commands'; +import { + GetUserReferralInfoQuery, + UserReferralInfoResult, + GetDirectReferralsQuery, + DirectReferralsResult, +} from '../queries'; + +@Injectable() +export class ReferralService { + private readonly logger = new Logger(ReferralService.name); + + constructor( + @Inject(REFERRAL_RELATIONSHIP_REPOSITORY) + private readonly referralRepo: IReferralRelationshipRepository, + @Inject(TEAM_STATISTICS_REPOSITORY) + private readonly teamStatsRepo: ITeamStatisticsRepository, + private readonly referralChainService: ReferralChainService, + private readonly eventPublisher: EventPublisherService, + private readonly leaderboardCache: LeaderboardCacheService, + ) {} + + /** + * 创建推荐关系 (用户注册时调用) + */ + async createReferralRelationship( + command: CreateReferralRelationshipCommand, + ): Promise<{ referralCode: string }> { + // 检查用户是否已有推荐关系 + const exists = await this.referralRepo.existsByUserId(command.userId); + if (exists) { + throw new BadRequestException('用户已存在推荐关系'); + } + + let referrerId: bigint | null = null; + let parentChain: bigint[] = []; + + // 如果有推荐码,查找推荐人 + if (command.referrerCode) { + const referrer = await this.referralRepo.findByReferralCode(command.referrerCode); + if (!referrer) { + throw new NotFoundException('推荐码不存在'); + } + referrerId = referrer.userId; + parentChain = referrer.referralChain; + + // 验证推荐链 + if (!this.referralChainService.validateChain(parentChain, command.userId)) { + throw new BadRequestException('无效的推荐关系'); + } + } + + // 创建推荐关系 + const relationship = ReferralRelationship.create(command.userId, referrerId, parentChain); + const saved = await this.referralRepo.save(relationship); + + // 创建团队统计记录 + await this.teamStatsRepo.create(command.userId); + + // 如果有推荐人,更新推荐人的直推计数 + if (referrerId) { + const referrerStats = await this.teamStatsRepo.findByUserId(referrerId); + if (referrerStats) { + referrerStats.addDirectReferral(command.userId); + await this.teamStatsRepo.save(referrerStats); + } + } + + // 发布领域事件 + await this.eventPublisher.publishDomainEvents(saved.domainEvents); + saved.clearDomainEvents(); + + this.logger.log(`Created referral relationship for user ${command.userId}`); + + return { referralCode: saved.referralCode }; + } + + /** + * 获取用户推荐信息 + */ + async getUserReferralInfo(query: GetUserReferralInfoQuery): Promise { + const relationship = await this.referralRepo.findByUserId(query.userId); + if (!relationship) { + throw new NotFoundException('用户推荐关系不存在'); + } + + const teamStats = await this.teamStatsRepo.findByUserId(query.userId); + const rank = await this.leaderboardCache.getUserRank(query.userId); + + return { + userId: relationship.userId.toString(), + referralCode: relationship.referralCode, + referrerId: relationship.referrerId?.toString() ?? null, + referralChainDepth: relationship.getChainDepth(), + directReferralCount: teamStats?.directReferralCount ?? 0, + totalTeamCount: teamStats?.totalTeamCount ?? 0, + personalPlantingCount: teamStats?.personalPlantingCount ?? 0, + teamPlantingCount: teamStats?.teamPlantingCount ?? 0, + leaderboardScore: teamStats?.leaderboardScore ?? 0, + leaderboardRank: rank, + createdAt: relationship.createdAt, + }; + } + + /** + * 根据推荐码获取推荐人信息 + */ + async getReferrerByCode(code: string): Promise<{ userId: string; referralCode: string } | null> { + const relationship = await this.referralRepo.findByReferralCode(code); + if (!relationship) return null; + + return { + userId: relationship.userId.toString(), + referralCode: relationship.referralCode, + }; + } + + /** + * 获取直推列表 + */ + async getDirectReferrals(query: GetDirectReferralsQuery): Promise { + const relationships = await this.referralRepo.findDirectReferrals(query.userId); + const teamStats = await this.teamStatsRepo.findByUserId(query.userId); + const directStats = teamStats?.getDirectReferralStats() ?? new Map(); + + // 分页 + const total = relationships.length; + const paginated = relationships.slice(query.offset, query.offset + query.limit); + + const referrals = paginated.map((r) => ({ + userId: r.userId.toString(), + referralCode: r.referralCode, + teamCount: directStats.get(r.userId) ?? 0, + joinedAt: r.createdAt, + })); + + return { + referrals, + total, + hasMore: query.offset + query.limit < total, + }; + } + + /** + * 检查推荐码是否有效 + */ + async validateReferralCode(code: string): Promise { + return this.referralRepo.existsByReferralCode(code); + } +} diff --git a/backend/services/referral-service/src/application/services/team-statistics.service.ts b/backend/services/referral-service/src/application/services/team-statistics.service.ts new file mode 100644 index 00000000..18d5958e --- /dev/null +++ b/backend/services/referral-service/src/application/services/team-statistics.service.ts @@ -0,0 +1,210 @@ +import { Injectable, Inject, Logger } from '@nestjs/common'; +import { + REFERRAL_RELATIONSHIP_REPOSITORY, + IReferralRelationshipRepository, + TEAM_STATISTICS_REPOSITORY, + ITeamStatisticsRepository, + ReferralChainService, +} from '../../domain'; +import { EventPublisherService, LeaderboardCacheService } from '../../infrastructure'; +import { UpdateTeamStatisticsCommand } from '../commands'; +import { + GetLeaderboardQuery, + LeaderboardResult, + GetProvinceCityDistributionQuery, + ProvinceCityDistributionResult, +} from '../queries'; + +@Injectable() +export class TeamStatisticsService { + private readonly logger = new Logger(TeamStatisticsService.name); + + constructor( + @Inject(REFERRAL_RELATIONSHIP_REPOSITORY) + private readonly referralRepo: IReferralRelationshipRepository, + @Inject(TEAM_STATISTICS_REPOSITORY) + private readonly teamStatsRepo: ITeamStatisticsRepository, + private readonly referralChainService: ReferralChainService, + private readonly eventPublisher: EventPublisherService, + private readonly leaderboardCache: LeaderboardCacheService, + ) {} + + /** + * 处理认种事件 - 更新用户及其所有上级的团队统计 + */ + async handlePlantingEvent(command: UpdateTeamStatisticsCommand): Promise { + // 获取用户的推荐关系 + const relationship = await this.referralRepo.findByUserId(command.userId); + if (!relationship) { + this.logger.warn(`User ${command.userId} has no referral relationship`); + return; + } + + // 更新用户自己的个人认种统计 + const userStats = await this.teamStatsRepo.findByUserId(command.userId); + if (userStats) { + userStats.addPersonalPlanting(command.plantingCount, command.provinceCode, command.cityCode); + await this.teamStatsRepo.save(userStats); + await this.leaderboardCache.updateScore(command.userId, userStats.leaderboardScore); + await this.eventPublisher.publishDomainEvents(userStats.domainEvents); + userStats.clearDomainEvents(); + } + + // 获取所有需要更新的上级 + const ancestors = relationship.getAllAncestorIds(); + if (ancestors.length === 0) return; + + // 准备批量更新数据 + const updates: Array<{ + userId: bigint; + countDelta: number; + provinceCode: string; + cityCode: string; + fromDirectReferralId?: bigint; + }> = []; + + // 第一级上级 (直接推荐人) 的 fromDirectReferralId 是当前用户 + // 第二级及以上的 fromDirectReferralId 是第一级上级 + let currentSource = command.userId; + + for (let i = 0; i < ancestors.length; i++) { + const ancestorId = ancestors[i]; + updates.push({ + userId: ancestorId, + countDelta: command.plantingCount, + provinceCode: command.provinceCode, + cityCode: command.cityCode, + fromDirectReferralId: i === 0 ? command.userId : ancestors[0], + }); + } + + // 批量更新 + await this.teamStatsRepo.batchUpdateTeamCounts(updates); + + // 更新排行榜缓存 + for (const ancestorId of ancestors) { + const stats = await this.teamStatsRepo.findByUserId(ancestorId); + if (stats) { + await this.leaderboardCache.updateScore(ancestorId, stats.leaderboardScore); + } + } + + this.logger.log( + `Updated team statistics for ${ancestors.length} ancestors of user ${command.userId}`, + ); + } + + /** + * 获取龙虎榜 + */ + async getLeaderboard(query: GetLeaderboardQuery): Promise { + // 尝试从缓存获取 + const cachedEntries = await this.leaderboardCache.getTopN(query.limit); + if (cachedEntries.length > 0) { + // 补充完整数据 + const userIds = cachedEntries.map((e) => e.userId); + const statsMap = new Map(); + const stats = await this.teamStatsRepo.findByUserIds(userIds); + stats.forEach((s) => { + statsMap.set(s.userId.toString(), { + totalTeamCount: s.totalTeamCount, + directReferralCount: s.directReferralCount, + }); + }); + + const entries = cachedEntries.map((e) => { + const extra = statsMap.get(e.userId.toString()); + return { + rank: e.rank, + userId: e.userId.toString(), + score: e.score, + totalTeamCount: extra?.totalTeamCount ?? 0, + directReferralCount: extra?.directReferralCount ?? 0, + }; + }); + + return { + entries, + total: entries.length, + hasMore: false, // 缓存暂不支持分页 + }; + } + + // 从数据库获取 + const dbEntries = await this.teamStatsRepo.getLeaderboard({ + limit: query.limit, + offset: query.offset, + }); + + const entries = dbEntries.map((e) => ({ + rank: e.rank, + userId: e.userId.toString(), + score: e.score, + totalTeamCount: e.totalTeamCount, + directReferralCount: e.directReferralCount, + })); + + return { + entries, + total: entries.length, + hasMore: entries.length === query.limit, + }; + } + + /** + * 获取用户排名 + */ + async getUserRank(userId: bigint): Promise { + // 先尝试缓存 + const cachedRank = await this.leaderboardCache.getUserRank(userId); + if (cachedRank !== null) { + return cachedRank; + } + + // 从数据库获取 + return this.teamStatsRepo.getUserRank(userId); + } + + /** + * 获取省市分布统计 + */ + async getProvinceCityDistribution( + query: GetProvinceCityDistributionQuery, + ): Promise { + const stats = await this.teamStatsRepo.findByUserId(query.userId); + if (!stats) { + return { + provinces: [], + totalCount: 0, + }; + } + + const distribution = stats.provinceCityDistribution; + const allData = distribution.getAll(); + + // 按省分组 + const provinceMap = new Map }>(); + for (const item of allData) { + if (!provinceMap.has(item.provinceCode)) { + provinceMap.set(item.provinceCode, { total: 0, cities: new Map() }); + } + const province = provinceMap.get(item.provinceCode)!; + province.total += item.count; + province.cities.set(item.cityCode, item.count); + } + + const provinces = Array.from(provinceMap.entries()).map(([provinceCode, data]) => ({ + provinceCode, + total: data.total, + cities: Array.from(data.cities.entries()).map(([cityCode, count]) => ({ + cityCode, + count, + })), + })); + + return { + provinces, + totalCount: distribution.getTotal(), + }; + } +} diff --git a/backend/services/referral-service/src/domain/aggregates/index.ts b/backend/services/referral-service/src/domain/aggregates/index.ts new file mode 100644 index 00000000..7637cece --- /dev/null +++ b/backend/services/referral-service/src/domain/aggregates/index.ts @@ -0,0 +1,2 @@ +export * from './referral-relationship'; +export * from './team-statistics'; diff --git a/backend/services/referral-service/src/domain/aggregates/referral-relationship/index.ts b/backend/services/referral-service/src/domain/aggregates/referral-relationship/index.ts new file mode 100644 index 00000000..ae5922bd --- /dev/null +++ b/backend/services/referral-service/src/domain/aggregates/referral-relationship/index.ts @@ -0,0 +1 @@ +export * from './referral-relationship.aggregate'; diff --git a/backend/services/referral-service/src/domain/aggregates/referral-relationship/referral-relationship.aggregate.ts b/backend/services/referral-service/src/domain/aggregates/referral-relationship/referral-relationship.aggregate.ts new file mode 100644 index 00000000..81989734 --- /dev/null +++ b/backend/services/referral-service/src/domain/aggregates/referral-relationship/referral-relationship.aggregate.ts @@ -0,0 +1,162 @@ +import { ReferralCode, ReferralChain, UserId } from '../../value-objects'; +import { DomainEvent, ReferralRelationshipCreatedEvent } from '../../events'; + +export interface ReferralRelationshipProps { + id: bigint; + userId: bigint; + referrerId: bigint | null; + referralCode: string; + referralChain: bigint[]; + createdAt: Date; + updatedAt: Date; +} + +/** + * 推荐关系聚合根 + * + * 职责: + * - 管理用户之间的推荐关系 + * - 生成和验证推荐码 + * - 维护推荐链 + */ +export class ReferralRelationship { + private _domainEvents: DomainEvent[] = []; + + private constructor( + private readonly _id: bigint, + private readonly _userId: UserId, + private readonly _referrerId: UserId | null, + private readonly _referralCode: ReferralCode, + private readonly _referralChain: ReferralChain, + private readonly _createdAt: Date, + private _updatedAt: Date, + ) {} + + // Getters + get id(): bigint { + return this._id; + } + get userId(): bigint { + return this._userId.value; + } + get referrerId(): bigint | null { + return this._referrerId?.value ?? null; + } + get referralCode(): string { + return this._referralCode.value; + } + get referralChain(): bigint[] { + return this._referralChain.toArray(); + } + get createdAt(): Date { + return this._createdAt; + } + get updatedAt(): Date { + return this._updatedAt; + } + get domainEvents(): DomainEvent[] { + return [...this._domainEvents]; + } + + /** + * 创建新的推荐关系 (用户注册时) + */ + static create( + userId: bigint, + referrerId: bigint | null, + parentReferralChain: bigint[] = [], + ): ReferralRelationship { + const userIdVo = UserId.create(userId); + const referrerIdVo = referrerId ? UserId.create(referrerId) : null; + const referralCode = ReferralCode.generate(userId); + const referralChain = ReferralChain.create(referrerId, parentReferralChain); + const now = new Date(); + + const relationship = new ReferralRelationship( + 0n, // ID will be assigned by database + userIdVo, + referrerIdVo, + referralCode, + referralChain, + now, + now, + ); + + // 发布领域事件 + relationship._domainEvents.push( + new ReferralRelationshipCreatedEvent( + userId, + referrerId, + referralCode.value, + referralChain.toArray(), + ), + ); + + return relationship; + } + + /** + * 从持久化数据重建 + */ + static reconstitute(props: ReferralRelationshipProps): ReferralRelationship { + return new ReferralRelationship( + props.id, + UserId.create(props.userId), + props.referrerId ? UserId.create(props.referrerId) : null, + ReferralCode.create(props.referralCode), + ReferralChain.fromArray(props.referralChain), + props.createdAt, + props.updatedAt, + ); + } + + /** + * 获取直接推荐人 + */ + getDirectReferrer(): bigint | null { + return this._referralChain.directReferrer; + } + + /** + * 获取指定层级的上级 + */ + getReferrerAtLevel(level: number): bigint | null { + return this._referralChain.getReferrerAtLevel(level); + } + + /** + * 获取所有上级ID (用于团队统计更新) + */ + getAllAncestorIds(): bigint[] { + return this._referralChain.getAllAncestors(); + } + + /** + * 推荐链深度 + */ + getChainDepth(): number { + return this._referralChain.depth; + } + + /** + * 清除领域事件 + */ + clearDomainEvents(): void { + this._domainEvents = []; + } + + /** + * 转换为持久化格式 + */ + toPersistence(): ReferralRelationshipProps { + return { + id: this._id, + userId: this._userId.value, + referrerId: this._referrerId?.value ?? null, + referralCode: this._referralCode.value, + referralChain: this._referralChain.toArray(), + createdAt: this._createdAt, + updatedAt: this._updatedAt, + }; + } +} diff --git a/backend/services/referral-service/src/domain/aggregates/team-statistics/index.ts b/backend/services/referral-service/src/domain/aggregates/team-statistics/index.ts new file mode 100644 index 00000000..4dfb4d8a --- /dev/null +++ b/backend/services/referral-service/src/domain/aggregates/team-statistics/index.ts @@ -0,0 +1 @@ +export * from './team-statistics.aggregate'; diff --git a/backend/services/referral-service/src/domain/aggregates/team-statistics/team-statistics.aggregate.ts b/backend/services/referral-service/src/domain/aggregates/team-statistics/team-statistics.aggregate.ts new file mode 100644 index 00000000..7155d508 --- /dev/null +++ b/backend/services/referral-service/src/domain/aggregates/team-statistics/team-statistics.aggregate.ts @@ -0,0 +1,268 @@ +import { UserId, LeaderboardScore, ProvinceCityDistribution } from '../../value-objects'; +import { DomainEvent, TeamStatisticsUpdatedEvent } from '../../events'; + +export interface DirectReferralStats { + referralId: bigint; + teamCount: number; +} + +export interface TeamStatisticsProps { + id: bigint; + userId: bigint; + directReferralCount: number; + totalTeamCount: number; + personalPlantingCount: number; + teamPlantingCount: number; + leaderboardScore: number; + maxDirectTeamCount: number; + provinceCityDistribution: Record> | null; + lastCalculatedAt: Date; + createdAt: Date; + updatedAt: Date; + directReferrals?: DirectReferralStats[]; +} + +/** + * 团队统计聚合根 + * + * 职责: + * - 维护用户的团队统计数据 + * - 计算龙虎榜分值 + * - 跟踪省市认种分布 + */ +export class TeamStatistics { + private _domainEvents: DomainEvent[] = []; + + private constructor( + private readonly _id: bigint, + private readonly _userId: UserId, + private _directReferralCount: number, + private _totalTeamCount: number, + private _personalPlantingCount: number, + private _teamPlantingCount: number, + private _leaderboardScore: LeaderboardScore, + private _provinceCityDistribution: ProvinceCityDistribution, + private _lastCalculatedAt: Date, + private readonly _createdAt: Date, + private _updatedAt: Date, + private _directReferralStats: Map = new Map(), + ) {} + + // Getters + get id(): bigint { + return this._id; + } + get userId(): bigint { + return this._userId.value; + } + get directReferralCount(): number { + return this._directReferralCount; + } + get totalTeamCount(): number { + return this._totalTeamCount; + } + get personalPlantingCount(): number { + return this._personalPlantingCount; + } + get teamPlantingCount(): number { + return this._teamPlantingCount; + } + get leaderboardScore(): number { + return this._leaderboardScore.score; + } + get maxDirectTeamCount(): number { + return this._leaderboardScore.maxDirectTeamCount; + } + get provinceCityDistribution(): ProvinceCityDistribution { + return this._provinceCityDistribution; + } + get lastCalculatedAt(): Date { + return this._lastCalculatedAt; + } + get createdAt(): Date { + return this._createdAt; + } + get updatedAt(): Date { + return this._updatedAt; + } + get domainEvents(): DomainEvent[] { + return [...this._domainEvents]; + } + + /** + * 创建新的团队统计记录 (推荐关系创建时) + */ + static create(userId: bigint): TeamStatistics { + const now = new Date(); + return new TeamStatistics( + 0n, + UserId.create(userId), + 0, + 0, + 0, + 0, + LeaderboardScore.zero(), + ProvinceCityDistribution.empty(), + now, + now, + now, + ); + } + + /** + * 从持久化数据重建 + */ + static reconstitute(props: TeamStatisticsProps): TeamStatistics { + const directTeamCounts = props.directReferrals?.map((d) => d.teamCount) ?? []; + const leaderboardScore = LeaderboardScore.calculate(props.totalTeamCount, directTeamCounts); + + const directReferralStats = new Map(); + props.directReferrals?.forEach((d) => { + directReferralStats.set(d.referralId, d.teamCount); + }); + + return new TeamStatistics( + props.id, + UserId.create(props.userId), + props.directReferralCount, + props.totalTeamCount, + props.personalPlantingCount, + props.teamPlantingCount, + leaderboardScore, + ProvinceCityDistribution.fromJson(props.provinceCityDistribution), + props.lastCalculatedAt, + props.createdAt, + props.updatedAt, + directReferralStats, + ); + } + + /** + * 新增直推成员 + */ + addDirectReferral(referralUserId: bigint): void { + this._directReferralCount++; + this._directReferralStats.set(referralUserId, 0); + this._updatedAt = new Date(); + } + + /** + * 增加团队认种量 + * @param count 认种数量 + * @param provinceCode 省份代码 + * @param cityCode 城市代码 + * @param fromDirectReferralId 来自哪个直推 (如果是直推的下级) + */ + addTeamPlanting( + count: number, + provinceCode: string, + cityCode: string, + fromDirectReferralId?: bigint, + ): void { + this._teamPlantingCount += count; + this._totalTeamCount += count; + this._provinceCityDistribution = this._provinceCityDistribution.add( + provinceCode, + cityCode, + count, + ); + + // 更新直推团队统计 + if (fromDirectReferralId) { + const currentCount = this._directReferralStats.get(fromDirectReferralId) ?? 0; + this._directReferralStats.set(fromDirectReferralId, currentCount + count); + } + + // 重新计算龙虎榜分值 + this.recalculateLeaderboardScore(); + this._lastCalculatedAt = new Date(); + this._updatedAt = new Date(); + + this._domainEvents.push( + new TeamStatisticsUpdatedEvent( + this._userId.value, + this._totalTeamCount, + this._directReferralCount, + this._leaderboardScore.score, + 'planting_added', + ), + ); + } + + /** + * 增加个人认种量 + */ + addPersonalPlanting(count: number, provinceCode: string, cityCode: string): void { + this._personalPlantingCount += count; + this._totalTeamCount += count; + this._provinceCityDistribution = this._provinceCityDistribution.add( + provinceCode, + cityCode, + count, + ); + this.recalculateLeaderboardScore(); + this._lastCalculatedAt = new Date(); + this._updatedAt = new Date(); + + this._domainEvents.push( + new TeamStatisticsUpdatedEvent( + this._userId.value, + this._totalTeamCount, + this._directReferralCount, + this._leaderboardScore.score, + 'planting_added', + ), + ); + } + + /** + * 重新计算龙虎榜分值 + */ + private recalculateLeaderboardScore(): void { + const directTeamCounts = Array.from(this._directReferralStats.values()); + this._leaderboardScore = this._leaderboardScore.recalculate( + this._totalTeamCount, + directTeamCounts, + ); + } + + /** + * 获取直推团队统计 + */ + getDirectReferralStats(): Map { + return new Map(this._directReferralStats); + } + + /** + * 清除领域事件 + */ + clearDomainEvents(): void { + this._domainEvents = []; + } + + /** + * 转换为持久化格式 + */ + toPersistence(): TeamStatisticsProps { + const directReferrals: DirectReferralStats[] = []; + this._directReferralStats.forEach((teamCount, referralId) => { + directReferrals.push({ referralId, teamCount }); + }); + + return { + id: this._id, + userId: this._userId.value, + directReferralCount: this._directReferralCount, + totalTeamCount: this._totalTeamCount, + personalPlantingCount: this._personalPlantingCount, + teamPlantingCount: this._teamPlantingCount, + leaderboardScore: this._leaderboardScore.score, + maxDirectTeamCount: this._leaderboardScore.maxDirectTeamCount, + provinceCityDistribution: this._provinceCityDistribution.toJson(), + lastCalculatedAt: this._lastCalculatedAt, + createdAt: this._createdAt, + updatedAt: this._updatedAt, + directReferrals, + }; + } +} diff --git a/backend/services/referral-service/src/domain/events/domain-event.base.ts b/backend/services/referral-service/src/domain/events/domain-event.base.ts new file mode 100644 index 00000000..493e1f1c --- /dev/null +++ b/backend/services/referral-service/src/domain/events/domain-event.base.ts @@ -0,0 +1,19 @@ +/** + * 领域事件基类 + */ +export abstract class DomainEvent { + public readonly occurredAt: Date; + public readonly eventId: string; + + constructor() { + this.occurredAt = new Date(); + this.eventId = this.generateEventId(); + } + + private generateEventId(): string { + return `${Date.now()}-${Math.random().toString(36).slice(2, 11)}`; + } + + abstract get eventName(): string; + abstract toPayload(): Record; +} diff --git a/backend/services/referral-service/src/domain/events/index.ts b/backend/services/referral-service/src/domain/events/index.ts new file mode 100644 index 00000000..435746a5 --- /dev/null +++ b/backend/services/referral-service/src/domain/events/index.ts @@ -0,0 +1,3 @@ +export * from './domain-event.base'; +export * from './referral-relationship-created.event'; +export * from './team-statistics-updated.event'; diff --git a/backend/services/referral-service/src/domain/events/referral-relationship-created.event.ts b/backend/services/referral-service/src/domain/events/referral-relationship-created.event.ts new file mode 100644 index 00000000..f98cfd96 --- /dev/null +++ b/backend/services/referral-service/src/domain/events/referral-relationship-created.event.ts @@ -0,0 +1,33 @@ +import { DomainEvent } from './domain-event.base'; + +/** + * 推荐关系创建事件 + */ +export class ReferralRelationshipCreatedEvent extends DomainEvent { + constructor( + public readonly userId: bigint, + public readonly referrerId: bigint | null, + public readonly referralCode: string, + public readonly referralChain: bigint[], + ) { + super(); + } + + get eventName(): string { + return 'referral.relationship.created'; + } + + toPayload(): Record { + return { + eventId: this.eventId, + eventName: this.eventName, + occurredAt: this.occurredAt.toISOString(), + data: { + userId: this.userId.toString(), + referrerId: this.referrerId?.toString() ?? null, + referralCode: this.referralCode, + referralChain: this.referralChain.map((id) => id.toString()), + }, + }; + } +} diff --git a/backend/services/referral-service/src/domain/events/team-statistics-updated.event.ts b/backend/services/referral-service/src/domain/events/team-statistics-updated.event.ts new file mode 100644 index 00000000..04f3c6ff --- /dev/null +++ b/backend/services/referral-service/src/domain/events/team-statistics-updated.event.ts @@ -0,0 +1,35 @@ +import { DomainEvent } from './domain-event.base'; + +/** + * 团队统计更新事件 + */ +export class TeamStatisticsUpdatedEvent extends DomainEvent { + constructor( + public readonly userId: bigint, + public readonly totalTeamCount: number, + public readonly directReferralCount: number, + public readonly leaderboardScore: number, + public readonly updateReason: 'planting_added' | 'planting_removed' | 'member_joined' | 'recalculation', + ) { + super(); + } + + get eventName(): string { + return 'referral.team_statistics.updated'; + } + + toPayload(): Record { + return { + eventId: this.eventId, + eventName: this.eventName, + occurredAt: this.occurredAt.toISOString(), + data: { + userId: this.userId.toString(), + totalTeamCount: this.totalTeamCount, + directReferralCount: this.directReferralCount, + leaderboardScore: this.leaderboardScore, + updateReason: this.updateReason, + }, + }; + } +} diff --git a/backend/services/referral-service/src/domain/index.ts b/backend/services/referral-service/src/domain/index.ts new file mode 100644 index 00000000..1fdc47a8 --- /dev/null +++ b/backend/services/referral-service/src/domain/index.ts @@ -0,0 +1,5 @@ +export * from './value-objects'; +export * from './events'; +export * from './aggregates'; +export * from './repositories'; +export * from './services'; diff --git a/backend/services/referral-service/src/domain/repositories/index.ts b/backend/services/referral-service/src/domain/repositories/index.ts new file mode 100644 index 00000000..aff4bd48 --- /dev/null +++ b/backend/services/referral-service/src/domain/repositories/index.ts @@ -0,0 +1,2 @@ +export * from './referral-relationship.repository.interface'; +export * from './team-statistics.repository.interface'; diff --git a/backend/services/referral-service/src/domain/repositories/referral-relationship.repository.interface.ts b/backend/services/referral-service/src/domain/repositories/referral-relationship.repository.interface.ts new file mode 100644 index 00000000..52b9414c --- /dev/null +++ b/backend/services/referral-service/src/domain/repositories/referral-relationship.repository.interface.ts @@ -0,0 +1,43 @@ +import { ReferralRelationship } from '../aggregates'; + +/** + * 推荐关系仓储接口 + */ +export interface IReferralRelationshipRepository { + /** + * 保存推荐关系 + */ + save(relationship: ReferralRelationship): Promise; + + /** + * 根据用户ID查找 + */ + findByUserId(userId: bigint): Promise; + + /** + * 根据推荐码查找 + */ + findByReferralCode(code: string): Promise; + + /** + * 查找用户的直推列表 + */ + findDirectReferrals(userId: bigint): Promise; + + /** + * 检查推荐码是否存在 + */ + existsByReferralCode(code: string): Promise; + + /** + * 检查用户是否已有推荐关系 + */ + existsByUserId(userId: bigint): Promise; + + /** + * 获取推荐链 (用于创建新用户时) + */ + getReferralChain(userId: bigint): Promise; +} + +export const REFERRAL_RELATIONSHIP_REPOSITORY = Symbol('IReferralRelationshipRepository'); diff --git a/backend/services/referral-service/src/domain/repositories/team-statistics.repository.interface.ts b/backend/services/referral-service/src/domain/repositories/team-statistics.repository.interface.ts new file mode 100644 index 00000000..78362f23 --- /dev/null +++ b/backend/services/referral-service/src/domain/repositories/team-statistics.repository.interface.ts @@ -0,0 +1,64 @@ +import { TeamStatistics } from '../aggregates'; + +export interface LeaderboardEntry { + userId: bigint; + score: number; + rank: number; + totalTeamCount: number; + directReferralCount: number; +} + +export interface LeaderboardQueryOptions { + limit?: number; + offset?: number; +} + +/** + * 团队统计仓储接口 + */ +export interface ITeamStatisticsRepository { + /** + * 保存团队统计 + */ + save(statistics: TeamStatistics): Promise; + + /** + * 根据用户ID查找 + */ + findByUserId(userId: bigint): Promise; + + /** + * 批量获取用户的团队统计 + */ + findByUserIds(userIds: bigint[]): Promise; + + /** + * 获取龙虎榜排名 + */ + getLeaderboard(options?: LeaderboardQueryOptions): Promise; + + /** + * 获取用户在龙虎榜的排名 + */ + getUserRank(userId: bigint): Promise; + + /** + * 批量更新团队统计 (用于认种事件处理) + */ + batchUpdateTeamCounts( + updates: Array<{ + userId: bigint; + countDelta: number; + provinceCode: string; + cityCode: string; + fromDirectReferralId?: bigint; + }>, + ): Promise; + + /** + * 创建初始团队统计记录 + */ + create(userId: bigint): Promise; +} + +export const TEAM_STATISTICS_REPOSITORY = Symbol('ITeamStatisticsRepository'); diff --git a/backend/services/referral-service/src/domain/services/index.ts b/backend/services/referral-service/src/domain/services/index.ts new file mode 100644 index 00000000..0bc79165 --- /dev/null +++ b/backend/services/referral-service/src/domain/services/index.ts @@ -0,0 +1,2 @@ +export * from './referral-chain.service'; +export * from './leaderboard-calculation.service'; diff --git a/backend/services/referral-service/src/domain/services/leaderboard-calculation.service.ts b/backend/services/referral-service/src/domain/services/leaderboard-calculation.service.ts new file mode 100644 index 00000000..14cf8083 --- /dev/null +++ b/backend/services/referral-service/src/domain/services/leaderboard-calculation.service.ts @@ -0,0 +1,68 @@ +import { Injectable } from '@nestjs/common'; +import { LeaderboardScore } from '../value-objects'; + +export interface DirectTeamStats { + referralId: bigint; + teamCount: number; +} + +/** + * 龙虎榜计算领域服务 + * + * 计算公式: 团队总认种量 - 最大单个直推团队认种量 + * + * 设计目的: + * - 鼓励均衡发展团队 + * - 防止\"烧伤\"现象(单腿发展) + */ +@Injectable() +export class LeaderboardCalculationService { + /** + * 计算龙虎榜分值 + */ + calculateScore(totalTeamCount: number, directTeamStats: DirectTeamStats[]): LeaderboardScore { + const directTeamCounts = directTeamStats.map((s) => s.teamCount); + return LeaderboardScore.calculate(totalTeamCount, directTeamCounts); + } + + /** + * 增量更新分值 (当有新认种时) + * @param currentScore 当前分值 + * @param newPlantingCount 新增认种数量 + * @param fromDirectReferralId 来自哪个直推 (如果有) + * @param directTeamStats 所有直推团队统计 + */ + updateScoreOnPlanting( + currentScore: LeaderboardScore, + newPlantingCount: number, + fromDirectReferralId: bigint | undefined, + directTeamStats: DirectTeamStats[], + ): LeaderboardScore { + // 更新直推团队统计 + const updatedStats = directTeamStats.map((s) => { + if (fromDirectReferralId && s.referralId === fromDirectReferralId) { + return { ...s, teamCount: s.teamCount + newPlantingCount }; + } + return s; + }); + + const newTotalTeamCount = currentScore.totalTeamCount + newPlantingCount; + return this.calculateScore(newTotalTeamCount, updatedStats); + } + + /** + * 比较两个用户的排名 + * @returns 负数表示a排在b前面, 正数表示b排在a前面 + */ + compareRank(scoreA: LeaderboardScore, scoreB: LeaderboardScore): number { + return scoreA.compareTo(scoreB); + } + + /** + * 检查分值是否有效 (用于数据验证) + */ + validateScore(score: LeaderboardScore, totalTeamCount: number, maxDirectTeamCount: number): boolean { + const expectedScore = Math.max(0, totalTeamCount - maxDirectTeamCount); + return score.score === expectedScore; + } +} diff --git a/backend/services/referral-service/src/domain/services/referral-chain.service.ts b/backend/services/referral-service/src/domain/services/referral-chain.service.ts new file mode 100644 index 00000000..ce21bbf2 --- /dev/null +++ b/backend/services/referral-service/src/domain/services/referral-chain.service.ts @@ -0,0 +1,62 @@ +import { Injectable } from '@nestjs/common'; +import { ReferralChain } from '../value-objects'; + +/** + * 推荐链领域服务 + * 处理推荐链的构建和验证逻辑 + */ +@Injectable() +export class ReferralChainService { + private static readonly MAX_DEPTH = 10; + + /** + * 构建新用户的推荐链 + * @param referrerId 推荐人ID + * @param parentChain 推荐人的推荐链 + */ + buildChain(referrerId: bigint | null, parentChain: bigint[] = []): ReferralChain { + return ReferralChain.create(referrerId, parentChain); + } + + /** + * 验证推荐链是否有效 + * - 不能形成循环 + * - 深度不能超过限制 + */ + validateChain(chain: bigint[], newUserId: bigint): boolean { + // 检查是否有循环 + if (chain.includes(newUserId)) { + return false; + } + + // 检查深度 + if (chain.length >= ReferralChainService.MAX_DEPTH) { + return false; + } + + return true; + } + + /** + * 获取需要更新团队统计的所有上级 + * @param chain 推荐链 + * @param maxLevels 最大层级 (默认全部) + */ + getAncestorsForUpdate(chain: bigint[], maxLevels?: number): bigint[] { + if (!maxLevels) { + return [...chain]; + } + return chain.slice(0, maxLevels); + } + + /** + * 计算用户在推荐链中的层级 + * @param chain 推荐链 + * @param ancestorId 上级ID + * @returns 层级 (0 = 直推, 1 = 二级, ...), 如果不在链中返回 -1 + */ + getLevelInChain(chain: bigint[], ancestorId: bigint): number { + const index = chain.findIndex((id) => id === ancestorId); + return index; + } +} diff --git a/backend/services/referral-service/src/domain/value-objects/index.ts b/backend/services/referral-service/src/domain/value-objects/index.ts new file mode 100644 index 00000000..b587d86c --- /dev/null +++ b/backend/services/referral-service/src/domain/value-objects/index.ts @@ -0,0 +1,5 @@ +export * from './user-id.vo'; +export * from './referral-code.vo'; +export * from './referral-chain.vo'; +export * from './leaderboard-score.vo'; +export * from './province-city-distribution.vo'; diff --git a/backend/services/referral-service/src/domain/value-objects/leaderboard-score.vo.ts b/backend/services/referral-service/src/domain/value-objects/leaderboard-score.vo.ts new file mode 100644 index 00000000..704dd213 --- /dev/null +++ b/backend/services/referral-service/src/domain/value-objects/leaderboard-score.vo.ts @@ -0,0 +1,41 @@ +/** + * 龙虎榜分值值对象 + * + * 计算公式: 团队总认种量 - 最大单个直推团队认种量 + * + * 这个公式的设计目的: + * - 鼓励均衡发展团队,而不是只依赖单个大团队 + * - 防止"烧伤"现象(单腿发展) + */ +export class LeaderboardScore { + private constructor( + public readonly totalTeamCount: number, + public readonly maxDirectTeamCount: number, + public readonly score: number, + ) {} + + static calculate(totalTeamCount: number, directTeamCounts: number[]): LeaderboardScore { + const maxDirectTeamCount = directTeamCounts.length > 0 ? Math.max(...directTeamCounts) : 0; + const score = Math.max(0, totalTeamCount - maxDirectTeamCount); + + return new LeaderboardScore(totalTeamCount, maxDirectTeamCount, score); + } + + static zero(): LeaderboardScore { + return new LeaderboardScore(0, 0, 0); + } + + /** + * 当团队认种发生变化时重新计算 + */ + recalculate(newTotalTeamCount: number, newDirectTeamCounts: number[]): LeaderboardScore { + return LeaderboardScore.calculate(newTotalTeamCount, newDirectTeamCounts); + } + + /** + * 比较排名 (降序排列) + */ + compareTo(other: LeaderboardScore): number { + return other.score - this.score; + } +} diff --git a/backend/services/referral-service/src/domain/value-objects/province-city-distribution.vo.ts b/backend/services/referral-service/src/domain/value-objects/province-city-distribution.vo.ts new file mode 100644 index 00000000..9d5301d9 --- /dev/null +++ b/backend/services/referral-service/src/domain/value-objects/province-city-distribution.vo.ts @@ -0,0 +1,108 @@ +export interface ProvinceCityCount { + provinceCode: string; + cityCode: string; + count: number; +} + +/** + * 省市分布统计值对象 + * 用于计算用户团队在各省/市的认种分布 + */ +export class ProvinceCityDistribution { + private constructor(private readonly distribution: Map>) {} + + static empty(): ProvinceCityDistribution { + return new ProvinceCityDistribution(new Map()); + } + + static fromJson(json: Record> | null): ProvinceCityDistribution { + if (!json) { + return ProvinceCityDistribution.empty(); + } + + const map = new Map>(); + for (const [province, cities] of Object.entries(json)) { + const cityMap = new Map(); + for (const [city, count] of Object.entries(cities)) { + cityMap.set(city, count); + } + map.set(province, cityMap); + } + return new ProvinceCityDistribution(map); + } + + /** + * 添加认种记录 (返回新实例,不可变) + */ + add(provinceCode: string, cityCode: string, count: number): ProvinceCityDistribution { + const newDist = new Map(this.distribution); + + if (!newDist.has(provinceCode)) { + newDist.set(provinceCode, new Map()); + } + + const cityMap = new Map(newDist.get(provinceCode)!); + cityMap.set(cityCode, (cityMap.get(cityCode) ?? 0) + count); + newDist.set(provinceCode, cityMap); + + return new ProvinceCityDistribution(newDist); + } + + /** + * 获取某省的总认种量 + */ + getProvinceTotal(provinceCode: string): number { + const cities = this.distribution.get(provinceCode); + if (!cities) return 0; + + let total = 0; + for (const count of cities.values()) { + total += count; + } + return total; + } + + /** + * 获取某市的总认种量 + */ + getCityTotal(provinceCode: string, cityCode: string): number { + return this.distribution.get(provinceCode)?.get(cityCode) ?? 0; + } + + /** + * 获取所有省市的统计 + */ + getAll(): ProvinceCityCount[] { + const result: ProvinceCityCount[] = []; + for (const [provinceCode, cities] of this.distribution) { + for (const [cityCode, count] of cities) { + result.push({ provinceCode, cityCode, count }); + } + } + return result; + } + + /** + * 获取总数 + */ + getTotal(): number { + let total = 0; + for (const cities of this.distribution.values()) { + for (const count of cities.values()) { + total += count; + } + } + return total; + } + + toJson(): Record> { + const result: Record> = {}; + for (const [province, cities] of this.distribution) { + result[province] = {}; + for (const [city, count] of cities) { + result[province][city] = count; + } + } + return result; + } +} diff --git a/backend/services/referral-service/src/domain/value-objects/referral-chain.vo.ts b/backend/services/referral-service/src/domain/value-objects/referral-chain.vo.ts new file mode 100644 index 00000000..a3c06e4e --- /dev/null +++ b/backend/services/referral-service/src/domain/value-objects/referral-chain.vo.ts @@ -0,0 +1,58 @@ +/** + * 推荐链值对象 + * 存储从直接推荐人到根节点的用户ID列表 + */ +export class ReferralChain { + private static readonly MAX_DEPTH = 10; + + private constructor(public readonly chain: bigint[]) { + if (chain.length > ReferralChain.MAX_DEPTH) { + throw new Error(`推荐链深度不能超过 ${ReferralChain.MAX_DEPTH}`); + } + } + + static create(referrerId: bigint | null, parentChain: bigint[] = []): ReferralChain { + if (!referrerId) { + return new ReferralChain([]); + } + + // 新链 = [直接推荐人] + 父链的前 MAX_DEPTH - 1 个元素 + const newChain = [referrerId, ...parentChain.slice(0, this.MAX_DEPTH - 1)]; + return new ReferralChain(newChain); + } + + static fromArray(chain: bigint[]): ReferralChain { + return new ReferralChain([...chain]); + } + + static empty(): ReferralChain { + return new ReferralChain([]); + } + + get depth(): number { + return this.chain.length; + } + + get directReferrer(): bigint | null { + return this.chain[0] ?? null; + } + + /** + * 获取指定层级的推荐人 + * @param level 层级 (0 = 直接推荐人, 1 = 推荐人的推荐人, ...) + */ + getReferrerAtLevel(level: number): bigint | null { + return this.chain[level] ?? null; + } + + /** + * 获取所有上级 (用于团队统计更新) + */ + getAllAncestors(): bigint[] { + return [...this.chain]; + } + + toArray(): bigint[] { + return [...this.chain]; + } +} diff --git a/backend/services/referral-service/src/domain/value-objects/referral-code.vo.ts b/backend/services/referral-service/src/domain/value-objects/referral-code.vo.ts new file mode 100644 index 00000000..ecba369c --- /dev/null +++ b/backend/services/referral-service/src/domain/value-objects/referral-code.vo.ts @@ -0,0 +1,26 @@ +export class ReferralCode { + private constructor(public readonly value: string) { + if (!value || value.length < 6 || value.length > 20) { + throw new Error('推荐码长度必须在6-20个字符之间'); + } + if (!/^[A-Z0-9]+$/.test(value)) { + throw new Error('推荐码只能包含大写字母和数字'); + } + } + + static create(value: string): ReferralCode { + return new ReferralCode(value.toUpperCase()); + } + + static generate(userId: bigint): ReferralCode { + // 生成规则: 前缀 + 用户ID哈希 + 随机字符 + const prefix = 'RWA'; + const userIdHash = userId.toString(36).toUpperCase().slice(-3).padStart(3, '0'); + const random = Math.random().toString(36).toUpperCase().slice(2, 6).padEnd(4, '0'); + return new ReferralCode(`${prefix}${userIdHash}${random}`); + } + + equals(other: ReferralCode): boolean { + return this.value === other.value; + } +} diff --git a/backend/services/referral-service/src/domain/value-objects/user-id.vo.ts b/backend/services/referral-service/src/domain/value-objects/user-id.vo.ts new file mode 100644 index 00000000..0c39395c --- /dev/null +++ b/backend/services/referral-service/src/domain/value-objects/user-id.vo.ts @@ -0,0 +1,19 @@ +export class UserId { + private constructor(public readonly value: bigint) {} + + static create(value: bigint | string | number): UserId { + const bigIntValue = typeof value === 'bigint' ? value : BigInt(value); + if (bigIntValue <= 0n) { + throw new Error('用户ID必须大于0'); + } + return new UserId(bigIntValue); + } + + equals(other: UserId): boolean { + return this.value === other.value; + } + + toString(): string { + return this.value.toString(); + } +} diff --git a/backend/services/referral-service/src/infrastructure/cache/index.ts b/backend/services/referral-service/src/infrastructure/cache/index.ts new file mode 100644 index 00000000..85417e0e --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/cache/index.ts @@ -0,0 +1,2 @@ +export * from './redis.service'; +export * from './leaderboard-cache.service'; diff --git a/backend/services/referral-service/src/infrastructure/cache/leaderboard-cache.service.ts b/backend/services/referral-service/src/infrastructure/cache/leaderboard-cache.service.ts new file mode 100644 index 00000000..88e13c8f --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/cache/leaderboard-cache.service.ts @@ -0,0 +1,85 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { RedisService } from './redis.service'; +import { LeaderboardEntry } from '../../domain'; + +const LEADERBOARD_KEY = 'leaderboard:scores'; +const LEADERBOARD_CACHE_KEY = 'leaderboard:cache'; +const CACHE_TTL_SECONDS = 300; // 5分钟 + +@Injectable() +export class LeaderboardCacheService { + private readonly logger = new Logger(LeaderboardCacheService.name); + + constructor(private readonly redisService: RedisService) {} + + /** + * 更新用户的龙虎榜分值 + */ + async updateScore(userId: bigint, score: number): Promise { + await this.redisService.zadd(LEADERBOARD_KEY, score, userId.toString()); + this.logger.debug(`Updated leaderboard score for user ${userId}: ${score}`); + } + + /** + * 获取用户排名 (0-based) + */ + async getUserRank(userId: bigint): Promise { + const rank = await this.redisService.zrevrank(LEADERBOARD_KEY, userId.toString()); + return rank !== null ? rank + 1 : null; + } + + /** + * 获取排行榜前N名 + */ + async getTopN(n: number): Promise { + // 尝试从缓存获取 + const cached = await this.redisService.get(`${LEADERBOARD_CACHE_KEY}:top${n}`); + if (cached) { + return cached; + } + + // 从有序集合获取 + const items = await this.redisService.zrevrangeWithScores(LEADERBOARD_KEY, 0, n - 1); + const entries: LeaderboardEntry[] = items.map((item, index) => ({ + userId: BigInt(item.member), + score: item.score, + rank: index + 1, + totalTeamCount: 0, // 需要从数据库补充 + directReferralCount: 0, + })); + + // 缓存结果 + await this.redisService.set(`${LEADERBOARD_CACHE_KEY}:top${n}`, entries, CACHE_TTL_SECONDS); + + return entries; + } + + /** + * 增量更新分值 + */ + async incrementScore(userId: bigint, delta: number): Promise { + return this.redisService.zincrby(LEADERBOARD_KEY, delta, userId.toString()); + } + + /** + * 批量更新分值 + */ + async batchUpdateScores(updates: Array<{ userId: bigint; score: number }>): Promise { + for (const update of updates) { + await this.updateScore(update.userId, update.score); + } + // 清除缓存 + await this.invalidateCache(); + } + + /** + * 清除排行榜缓存 + */ + async invalidateCache(): Promise { + // 清除所有排行榜缓存 + for (const n of [10, 50, 100]) { + await this.redisService.del(`${LEADERBOARD_CACHE_KEY}:top${n}`); + } + this.logger.debug('Leaderboard cache invalidated'); + } +} diff --git a/backend/services/referral-service/src/infrastructure/cache/redis.service.ts b/backend/services/referral-service/src/infrastructure/cache/redis.service.ts new file mode 100644 index 00000000..05fa072e --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/cache/redis.service.ts @@ -0,0 +1,120 @@ +import { Injectable, OnModuleInit, OnModuleDestroy, Logger } from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; +import Redis from 'ioredis'; + +@Injectable() +export class RedisService implements OnModuleInit, OnModuleDestroy { + private readonly logger = new Logger(RedisService.name); + private client: Redis; + + constructor(private readonly configService: ConfigService) { + this.client = new Redis({ + host: this.configService.get('REDIS_HOST', 'localhost'), + port: this.configService.get('REDIS_PORT', 6379), + password: this.configService.get('REDIS_PASSWORD'), + db: this.configService.get('REDIS_DB', 0), + keyPrefix: 'referral:', + }); + } + + async onModuleInit() { + this.logger.log('Connecting to Redis...'); + await this.client.ping(); + this.logger.log('Redis connected'); + } + + async onModuleDestroy() { + this.logger.log('Disconnecting from Redis...'); + await this.client.quit(); + this.logger.log('Redis disconnected'); + } + + async get(key: string): Promise { + const value = await this.client.get(key); + if (!value) return null; + return JSON.parse(value) as T; + } + + async set(key: string, value: unknown, ttlSeconds?: number): Promise { + const serialized = JSON.stringify(value); + if (ttlSeconds) { + await this.client.setex(key, ttlSeconds, serialized); + } else { + await this.client.set(key, serialized); + } + } + + async del(key: string): Promise { + await this.client.del(key); + } + + async exists(key: string): Promise { + return (await this.client.exists(key)) === 1; + } + + // 排行榜相关操作 + async zadd(key: string, score: number, member: string): Promise { + await this.client.zadd(key, score, member); + } + + async zrevrank(key: string, member: string): Promise { + return this.client.zrevrank(key, member); + } + + async zrevrange(key: string, start: number, stop: number): Promise { + return this.client.zrevrange(key, start, stop); + } + + async zrevrangeWithScores( + key: string, + start: number, + stop: number, + ): Promise> { + const result = await this.client.zrevrange(key, start, stop, 'WITHSCORES'); + const items: Array<{ member: string; score: number }> = []; + for (let i = 0; i < result.length; i += 2) { + items.push({ + member: result[i], + score: parseFloat(result[i + 1]), + }); + } + return items; + } + + async zincrby(key: string, increment: number, member: string): Promise { + const result = await this.client.zincrby(key, increment, member); + return parseFloat(result); + } + + // 哈希操作 + async hget(key: string, field: string): Promise { + const value = await this.client.hget(key, field); + if (!value) return null; + return JSON.parse(value) as T; + } + + async hset(key: string, field: string, value: unknown): Promise { + await this.client.hset(key, field, JSON.stringify(value)); + } + + async hmset(key: string, data: Record): Promise { + const serialized: Record = {}; + for (const [k, v] of Object.entries(data)) { + serialized[k] = JSON.stringify(v); + } + await this.client.hmset(key, serialized); + } + + async hgetall(key: string): Promise> { + const data = await this.client.hgetall(key); + const result: Record = {}; + for (const [k, v] of Object.entries(data)) { + result[k] = JSON.parse(v) as T; + } + return result; + } + + async hdel(key: string, ...fields: string[]): Promise { + await this.client.hdel(key, ...fields); + } +} diff --git a/backend/services/referral-service/src/infrastructure/database/index.ts b/backend/services/referral-service/src/infrastructure/database/index.ts new file mode 100644 index 00000000..cb6bbc36 --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/database/index.ts @@ -0,0 +1 @@ +export * from './prisma.service'; diff --git a/backend/services/referral-service/src/infrastructure/database/prisma.service.ts b/backend/services/referral-service/src/infrastructure/database/prisma.service.ts new file mode 100644 index 00000000..eeb2afb8 --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/database/prisma.service.ts @@ -0,0 +1,30 @@ +import { Injectable, OnModuleInit, OnModuleDestroy, Logger } from '@nestjs/common'; +import { PrismaClient } from '@prisma/client'; + +@Injectable() +export class PrismaService extends PrismaClient implements OnModuleInit, OnModuleDestroy { + private readonly logger = new Logger(PrismaService.name); + + constructor() { + super({ + log: [ + { emit: 'event', level: 'query' }, + { emit: 'stdout', level: 'info' }, + { emit: 'stdout', level: 'warn' }, + { emit: 'stdout', level: 'error' }, + ], + }); + } + + async onModuleInit() { + this.logger.log('Connecting to database...'); + await this.$connect(); + this.logger.log('Database connected successfully'); + } + + async onModuleDestroy() { + this.logger.log('Disconnecting from database...'); + await this.$disconnect(); + this.logger.log('Database disconnected'); + } +} diff --git a/backend/services/referral-service/src/infrastructure/index.ts b/backend/services/referral-service/src/infrastructure/index.ts new file mode 100644 index 00000000..62ddf87b --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/index.ts @@ -0,0 +1,4 @@ +export * from './database'; +export * from './repositories'; +export * from './messaging'; +export * from './cache'; diff --git a/backend/services/referral-service/src/infrastructure/messaging/event-publisher.service.ts b/backend/services/referral-service/src/infrastructure/messaging/event-publisher.service.ts new file mode 100644 index 00000000..428bfbec --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/messaging/event-publisher.service.ts @@ -0,0 +1,44 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { KafkaService } from './kafka.service'; +import { DomainEvent } from '../../domain'; + +export const KAFKA_TOPICS = { + REFERRAL_EVENTS: 'referral.events', + TEAM_STATISTICS_EVENTS: 'referral.team-statistics.events', +} as const; + +@Injectable() +export class EventPublisherService { + private readonly logger = new Logger(EventPublisherService.name); + + constructor(private readonly kafkaService: KafkaService) {} + + async publishDomainEvents(events: DomainEvent[]): Promise { + if (events.length === 0) return; + + const messages = events.map((event) => ({ + topic: this.getTopicForEvent(event), + key: event.eventId, + value: event.toPayload(), + })); + + await this.kafkaService.publishBatch(messages); + this.logger.log(`Published ${events.length} domain events`); + } + + async publishEvent(event: DomainEvent): Promise { + await this.kafkaService.publish({ + topic: this.getTopicForEvent(event), + key: event.eventId, + value: event.toPayload(), + }); + this.logger.debug(`Published event: ${event.eventName}`); + } + + private getTopicForEvent(event: DomainEvent): string { + if (event.eventName.includes('team_statistics')) { + return KAFKA_TOPICS.TEAM_STATISTICS_EVENTS; + } + return KAFKA_TOPICS.REFERRAL_EVENTS; + } +} diff --git a/backend/services/referral-service/src/infrastructure/messaging/index.ts b/backend/services/referral-service/src/infrastructure/messaging/index.ts new file mode 100644 index 00000000..62ff5b11 --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/messaging/index.ts @@ -0,0 +1,2 @@ +export * from './kafka.service'; +export * from './event-publisher.service'; diff --git a/backend/services/referral-service/src/infrastructure/messaging/kafka.service.ts b/backend/services/referral-service/src/infrastructure/messaging/kafka.service.ts new file mode 100644 index 00000000..3afdeff7 --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/messaging/kafka.service.ts @@ -0,0 +1,110 @@ +import { Injectable, OnModuleInit, OnModuleDestroy, Logger } from '@nestjs/common'; +import { ConfigService } from '@nestjs/config'; +import { Kafka, Producer, Consumer, logLevel } from 'kafkajs'; + +export interface KafkaMessage { + topic: string; + key?: string; + value: Record; + headers?: Record; +} + +@Injectable() +export class KafkaService implements OnModuleInit, OnModuleDestroy { + private readonly logger = new Logger(KafkaService.name); + private kafka: Kafka; + private producer: Producer; + private consumers: Map = new Map(); + + constructor(private readonly configService: ConfigService) { + this.kafka = new Kafka({ + clientId: 'referral-service', + brokers: this.configService.get('KAFKA_BROKERS', 'localhost:9092').split(','), + logLevel: logLevel.WARN, + }); + this.producer = this.kafka.producer(); + } + + async onModuleInit() { + this.logger.log('Connecting to Kafka...'); + await this.producer.connect(); + this.logger.log('Kafka producer connected'); + } + + async onModuleDestroy() { + this.logger.log('Disconnecting from Kafka...'); + await this.producer.disconnect(); + for (const [groupId, consumer] of this.consumers) { + await consumer.disconnect(); + this.logger.log(`Consumer ${groupId} disconnected`); + } + this.logger.log('Kafka disconnected'); + } + + async publish(message: KafkaMessage): Promise { + await this.producer.send({ + topic: message.topic, + messages: [ + { + key: message.key, + value: JSON.stringify(message.value), + headers: message.headers, + }, + ], + }); + this.logger.debug(`Published message to ${message.topic}`); + } + + async publishBatch(messages: KafkaMessage[]): Promise { + const topicMessages = new Map }>>(); + + for (const msg of messages) { + if (!topicMessages.has(msg.topic)) { + topicMessages.set(msg.topic, []); + } + topicMessages.get(msg.topic)!.push({ + key: msg.key, + value: JSON.stringify(msg.value), + headers: msg.headers, + }); + } + + await this.producer.sendBatch({ + topicMessages: Array.from(topicMessages.entries()).map(([topic, messages]) => ({ + topic, + messages, + })), + }); + + this.logger.debug(`Published ${messages.length} messages in batch`); + } + + async subscribe( + groupId: string, + topics: string[], + handler: (topic: string, message: Record) => Promise, + ): Promise { + const consumer = this.kafka.consumer({ groupId }); + await consumer.connect(); + + for (const topic of topics) { + await consumer.subscribe({ topic, fromBeginning: false }); + } + + await consumer.run({ + eachMessage: async ({ topic, message }) => { + try { + const value = message.value ? JSON.parse(message.value.toString()) : null; + if (value) { + await handler(topic, value); + } + } catch (error) { + this.logger.error(`Error processing message from ${topic}:`, error); + } + }, + }); + + this.consumers.set(groupId, consumer); + this.logger.log(`Consumer ${groupId} subscribed to topics: ${topics.join(', ')}`); + } +} diff --git a/backend/services/referral-service/src/infrastructure/repositories/index.ts b/backend/services/referral-service/src/infrastructure/repositories/index.ts new file mode 100644 index 00000000..53f501bc --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/repositories/index.ts @@ -0,0 +1,2 @@ +export * from './referral-relationship.repository'; +export * from './team-statistics.repository'; diff --git a/backend/services/referral-service/src/infrastructure/repositories/referral-relationship.repository.ts b/backend/services/referral-service/src/infrastructure/repositories/referral-relationship.repository.ts new file mode 100644 index 00000000..a01e2e62 --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/repositories/referral-relationship.repository.ts @@ -0,0 +1,109 @@ +import { Injectable } from '@nestjs/common'; +import { PrismaService } from '../database/prisma.service'; +import { + IReferralRelationshipRepository, + ReferralRelationship, + ReferralRelationshipProps, +} from '../../domain'; + +@Injectable() +export class ReferralRelationshipRepository implements IReferralRelationshipRepository { + constructor(private readonly prisma: PrismaService) {} + + async save(relationship: ReferralRelationship): Promise { + const data = relationship.toPersistence(); + + const saved = await this.prisma.referralRelationship.upsert({ + where: { userId: data.userId }, + update: { + referrerId: data.referrerId, + myReferralCode: data.referralCode, + ancestorPath: data.referralChain, + depth: data.referralChain.length, + updatedAt: new Date(), + }, + create: { + userId: data.userId, + referrerId: data.referrerId, + myReferralCode: data.referralCode, + usedReferralCode: data.referrerId ? data.referralCode : null, + ancestorPath: data.referralChain, + depth: data.referralChain.length, + rootUserId: data.referralChain.length > 0 ? data.referralChain[data.referralChain.length - 1] : null, + }, + }); + + return ReferralRelationship.reconstitute(this.mapToProps(saved)); + } + + async findByUserId(userId: bigint): Promise { + const record = await this.prisma.referralRelationship.findUnique({ + where: { userId }, + }); + + if (!record) return null; + return ReferralRelationship.reconstitute(this.mapToProps(record)); + } + + async findByReferralCode(code: string): Promise { + const record = await this.prisma.referralRelationship.findUnique({ + where: { myReferralCode: code }, + }); + + if (!record) return null; + return ReferralRelationship.reconstitute(this.mapToProps(record)); + } + + async findDirectReferrals(userId: bigint): Promise { + const records = await this.prisma.referralRelationship.findMany({ + where: { referrerId: userId }, + orderBy: { createdAt: 'desc' }, + }); + + return records.map((r) => ReferralRelationship.reconstitute(this.mapToProps(r))); + } + + async existsByReferralCode(code: string): Promise { + const count = await this.prisma.referralRelationship.count({ + where: { myReferralCode: code }, + }); + return count > 0; + } + + async existsByUserId(userId: bigint): Promise { + const count = await this.prisma.referralRelationship.count({ + where: { userId }, + }); + return count > 0; + } + + async getReferralChain(userId: bigint): Promise { + const record = await this.prisma.referralRelationship.findUnique({ + where: { userId }, + select: { ancestorPath: true }, + }); + + if (!record) return []; + return record.ancestorPath; + } + + private mapToProps(record: { + id: bigint; + userId: bigint; + referrerId: bigint | null; + myReferralCode: string; + ancestorPath: bigint[]; + createdAt: Date; + updatedAt: Date; + }): ReferralRelationshipProps { + return { + id: record.id, + userId: record.userId, + referrerId: record.referrerId, + referralCode: record.myReferralCode, + referralChain: record.ancestorPath, + createdAt: record.createdAt, + updatedAt: record.updatedAt, + }; + } +} diff --git a/backend/services/referral-service/src/infrastructure/repositories/team-statistics.repository.ts b/backend/services/referral-service/src/infrastructure/repositories/team-statistics.repository.ts new file mode 100644 index 00000000..9d14823b --- /dev/null +++ b/backend/services/referral-service/src/infrastructure/repositories/team-statistics.repository.ts @@ -0,0 +1,293 @@ +import { Injectable } from '@nestjs/common'; +import { PrismaService } from '../database/prisma.service'; +import { + ITeamStatisticsRepository, + TeamStatistics, + TeamStatisticsProps, + LeaderboardEntry, + LeaderboardQueryOptions, + DirectReferralStats, +} from '../../domain'; + +@Injectable() +export class TeamStatisticsRepository implements ITeamStatisticsRepository { + constructor(private readonly prisma: PrismaService) {} + + async save(statistics: TeamStatistics): Promise { + const data = statistics.toPersistence(); + + await this.prisma.$transaction(async (tx) => { + // 保存主表 + await tx.teamStatistics.upsert({ + where: { userId: data.userId }, + update: { + directReferralCount: data.directReferralCount, + totalTeamCount: data.totalTeamCount, + selfPlantingCount: data.personalPlantingCount, + totalTeamPlantingCount: data.teamPlantingCount, + effectivePlantingCountForRanking: data.leaderboardScore, + maxSingleTeamPlantingCount: data.maxDirectTeamCount, + provinceCityDistribution: data.provinceCityDistribution ?? {}, + lastCalcAt: data.lastCalculatedAt, + updatedAt: new Date(), + }, + create: { + userId: data.userId, + directReferralCount: data.directReferralCount, + totalTeamCount: data.totalTeamCount, + selfPlantingCount: data.personalPlantingCount, + totalTeamPlantingCount: data.teamPlantingCount, + effectivePlantingCountForRanking: data.leaderboardScore, + maxSingleTeamPlantingCount: data.maxDirectTeamCount, + provinceCityDistribution: data.provinceCityDistribution ?? {}, + lastCalcAt: data.lastCalculatedAt, + }, + }); + + // 更新直推统计明细 + if (data.directReferrals && data.directReferrals.length > 0) { + for (const dr of data.directReferrals) { + await tx.directReferral.upsert({ + where: { + uk_referrer_referral: { + referrerId: data.userId, + referralId: dr.referralId, + }, + }, + update: { + teamPlantingCount: dr.teamCount, + updatedAt: new Date(), + }, + create: { + referrerId: data.userId, + referralId: dr.referralId, + referralSequence: dr.referralId, + teamPlantingCount: dr.teamCount, + }, + }); + } + } + }); + + // 重新查询完整数据 + return this.findByUserId(data.userId) as Promise; + } + + async findByUserId(userId: bigint): Promise { + const record = await this.prisma.teamStatistics.findUnique({ + where: { userId }, + }); + + if (!record) return null; + + // 查询直推统计 + const directReferrals = await this.prisma.directReferral.findMany({ + where: { referrerId: userId }, + select: { referralId: true, teamPlantingCount: true }, + }); + + return TeamStatistics.reconstitute(this.mapToProps(record, directReferrals)); + } + + async findByUserIds(userIds: bigint[]): Promise { + const records = await this.prisma.teamStatistics.findMany({ + where: { userId: { in: userIds } }, + }); + + const result: TeamStatistics[] = []; + for (const record of records) { + const directReferrals = await this.prisma.directReferral.findMany({ + where: { referrerId: record.userId }, + select: { referralId: true, teamPlantingCount: true }, + }); + result.push(TeamStatistics.reconstitute(this.mapToProps(record, directReferrals))); + } + + return result; + } + + async getLeaderboard(options: LeaderboardQueryOptions = {}): Promise { + const { limit = 100, offset = 0 } = options; + + const records = await this.prisma.teamStatistics.findMany({ + where: { + effectivePlantingCountForRanking: { gt: 0 }, + }, + orderBy: [ + { effectivePlantingCountForRanking: 'desc' }, + { totalTeamPlantingCount: 'desc' }, + ], + skip: offset, + take: limit, + select: { + userId: true, + effectivePlantingCountForRanking: true, + totalTeamPlantingCount: true, + directReferralCount: true, + }, + }); + + return records.map((r, index) => ({ + userId: r.userId, + score: r.effectivePlantingCountForRanking, + rank: offset + index + 1, + totalTeamCount: r.totalTeamPlantingCount, + directReferralCount: r.directReferralCount, + })); + } + + async getUserRank(userId: bigint): Promise { + const userStats = await this.prisma.teamStatistics.findUnique({ + where: { userId }, + select: { effectivePlantingCountForRanking: true }, + }); + + if (!userStats) return null; + + const higherRanks = await this.prisma.teamStatistics.count({ + where: { + effectivePlantingCountForRanking: { gt: userStats.effectivePlantingCountForRanking }, + }, + }); + + return higherRanks + 1; + } + + async batchUpdateTeamCounts( + updates: Array<{ + userId: bigint; + countDelta: number; + provinceCode: string; + cityCode: string; + fromDirectReferralId?: bigint; + }>, + ): Promise { + await this.prisma.$transaction(async (tx) => { + for (const update of updates) { + // 获取当前统计 + const current = await tx.teamStatistics.findUnique({ + where: { userId: update.userId }, + }); + + if (!current) continue; + + // 更新省市分布 + const distribution = (current.provinceCityDistribution as Record< + string, + Record + >) ?? {}; + if (!distribution[update.provinceCode]) { + distribution[update.provinceCode] = {}; + } + distribution[update.provinceCode][update.cityCode] = + (distribution[update.provinceCode][update.cityCode] ?? 0) + update.countDelta; + + // 更新直推团队统计 + let newMaxDirectTeamCount = current.maxSingleTeamPlantingCount; + if (update.fromDirectReferralId) { + const directRef = await tx.directReferral.upsert({ + where: { + uk_referrer_referral: { + referrerId: update.userId, + referralId: update.fromDirectReferralId, + }, + }, + update: { + teamPlantingCount: { increment: update.countDelta }, + }, + create: { + referrerId: update.userId, + referralId: update.fromDirectReferralId, + referralSequence: update.fromDirectReferralId, + teamPlantingCount: update.countDelta, + }, + }); + + if (directRef.teamPlantingCount > newMaxDirectTeamCount) { + newMaxDirectTeamCount = directRef.teamPlantingCount; + } + } + + // 重新计算龙虎榜分值 + const newTotalTeamPlantingCount = current.totalTeamPlantingCount + update.countDelta; + const newEffectiveScore = Math.max(0, newTotalTeamPlantingCount - newMaxDirectTeamCount); + + // 更新主表 + await tx.teamStatistics.update({ + where: { userId: update.userId }, + data: { + totalTeamPlantingCount: newTotalTeamPlantingCount, + effectivePlantingCountForRanking: newEffectiveScore, + maxSingleTeamPlantingCount: newMaxDirectTeamCount, + provinceCityDistribution: distribution, + lastCalcAt: new Date(), + updatedAt: new Date(), + }, + }); + } + }); + } + + async create(userId: bigint): Promise { + const created = await this.prisma.teamStatistics.create({ + data: { + userId, + directReferralCount: 0, + totalTeamCount: 0, + selfPlantingCount: 0, + totalTeamPlantingCount: 0, + effectivePlantingCountForRanking: 0, + maxSingleTeamPlantingCount: 0, + provinceCityDistribution: {}, + lastCalcAt: new Date(), + }, + }); + + return TeamStatistics.reconstitute(this.mapToProps(created, [])); + } + + private mapToProps( + record: { + id: bigint; + userId: bigint; + directReferralCount: number; + totalTeamCount: number; + selfPlantingCount: number; + totalTeamPlantingCount: number; + effectivePlantingCountForRanking: number; + maxSingleTeamPlantingCount: number; + provinceCityDistribution: unknown; + lastCalcAt: Date | null; + createdAt: Date; + updatedAt: Date; + }, + directReferrals: Array<{ + referralId: bigint; + teamPlantingCount: number; + }>, + ): TeamStatisticsProps { + const directReferralStats: DirectReferralStats[] = directReferrals.map((dr) => ({ + referralId: dr.referralId, + teamCount: dr.teamPlantingCount, + })); + + return { + id: record.id, + userId: record.userId, + directReferralCount: record.directReferralCount, + totalTeamCount: record.totalTeamCount, + personalPlantingCount: record.selfPlantingCount, + teamPlantingCount: record.totalTeamPlantingCount, + leaderboardScore: record.effectivePlantingCountForRanking, + maxDirectTeamCount: record.maxSingleTeamPlantingCount, + provinceCityDistribution: record.provinceCityDistribution as Record< + string, + Record + > | null, + lastCalculatedAt: record.lastCalcAt ?? new Date(), + createdAt: record.createdAt, + updatedAt: record.updatedAt, + directReferrals: directReferralStats, + }; + } +} diff --git a/backend/services/referral-service/src/main.ts b/backend/services/referral-service/src/main.ts new file mode 100644 index 00000000..e915a2e7 --- /dev/null +++ b/backend/services/referral-service/src/main.ts @@ -0,0 +1,55 @@ +import { NestFactory } from '@nestjs/core'; +import { ValidationPipe, Logger } from '@nestjs/common'; +import { SwaggerModule, DocumentBuilder } from '@nestjs/swagger'; +import { AppModule } from './app.module'; + +async function bootstrap() { + const logger = new Logger('Bootstrap'); + const app = await NestFactory.create(AppModule); + + // 全局验证管道 + app.useGlobalPipes( + new ValidationPipe({ + whitelist: true, + forbidNonWhitelisted: true, + transform: true, + transformOptions: { + enableImplicitConversion: true, + }, + }), + ); + + // CORS配置 + app.enableCors({ + origin: process.env.CORS_ORIGIN || '*', + methods: ['GET', 'POST', 'PUT', 'DELETE', 'PATCH', 'OPTIONS'], + credentials: true, + }); + + // API前缀 + app.setGlobalPrefix('api/v1'); + + // Swagger文档 + if (process.env.NODE_ENV !== 'production') { + const config = new DocumentBuilder() + .setTitle('Referral Service API') + .setDescription('推荐团队服务 API 文档') + .setVersion('1.0') + .addBearerAuth() + .addTag('Referral', '推荐关系管理') + .addTag('Leaderboard', '龙虎榜') + .addTag('Team Statistics', '团队统计') + .addTag('Health', '健康检查') + .build(); + + const document = SwaggerModule.createDocument(app, config); + SwaggerModule.setup('api/docs', app, document); + logger.log('Swagger documentation available at /api/docs'); + } + + const port = process.env.PORT || 3002; + await app.listen(port); + logger.log(`Referral Service is running on port ${port}`); +} + +bootstrap(); diff --git a/backend/services/referral-service/src/modules/api.module.ts b/backend/services/referral-service/src/modules/api.module.ts new file mode 100644 index 00000000..98f56783 --- /dev/null +++ b/backend/services/referral-service/src/modules/api.module.ts @@ -0,0 +1,20 @@ +import { Module } from '@nestjs/common'; +import { ConfigModule } from '@nestjs/config'; +import { ApplicationModule } from './application.module'; +import { + ReferralController, + LeaderboardController, + TeamStatisticsController, + HealthController, +} from '../api'; + +@Module({ + imports: [ConfigModule, ApplicationModule], + controllers: [ + ReferralController, + LeaderboardController, + TeamStatisticsController, + HealthController, + ], +}) +export class ApiModule {} diff --git a/backend/services/referral-service/src/modules/application.module.ts b/backend/services/referral-service/src/modules/application.module.ts new file mode 100644 index 00000000..74f68327 --- /dev/null +++ b/backend/services/referral-service/src/modules/application.module.ts @@ -0,0 +1,21 @@ +import { Module } from '@nestjs/common'; +import { DomainModule } from './domain.module'; +import { InfrastructureModule } from './infrastructure.module'; +import { + ReferralService, + TeamStatisticsService, + UserRegisteredHandler, + PlantingCreatedHandler, +} from '../application'; + +@Module({ + imports: [DomainModule, InfrastructureModule], + providers: [ + ReferralService, + TeamStatisticsService, + UserRegisteredHandler, + PlantingCreatedHandler, + ], + exports: [ReferralService, TeamStatisticsService], +}) +export class ApplicationModule {} diff --git a/backend/services/referral-service/src/modules/domain.module.ts b/backend/services/referral-service/src/modules/domain.module.ts new file mode 100644 index 00000000..821c050b --- /dev/null +++ b/backend/services/referral-service/src/modules/domain.module.ts @@ -0,0 +1,8 @@ +import { Module } from '@nestjs/common'; +import { ReferralChainService, LeaderboardCalculationService } from '../domain'; + +@Module({ + providers: [ReferralChainService, LeaderboardCalculationService], + exports: [ReferralChainService, LeaderboardCalculationService], +}) +export class DomainModule {} diff --git a/backend/services/referral-service/src/modules/index.ts b/backend/services/referral-service/src/modules/index.ts new file mode 100644 index 00000000..685446c7 --- /dev/null +++ b/backend/services/referral-service/src/modules/index.ts @@ -0,0 +1,4 @@ +export * from './domain.module'; +export * from './infrastructure.module'; +export * from './application.module'; +export * from './api.module'; diff --git a/backend/services/referral-service/src/modules/infrastructure.module.ts b/backend/services/referral-service/src/modules/infrastructure.module.ts new file mode 100644 index 00000000..34fc57be --- /dev/null +++ b/backend/services/referral-service/src/modules/infrastructure.module.ts @@ -0,0 +1,45 @@ +import { Module, Global } from '@nestjs/common'; +import { ConfigModule } from '@nestjs/config'; +import { + PrismaService, + ReferralRelationshipRepository, + TeamStatisticsRepository, + KafkaService, + EventPublisherService, + RedisService, + LeaderboardCacheService, +} from '../infrastructure'; +import { + REFERRAL_RELATIONSHIP_REPOSITORY, + TEAM_STATISTICS_REPOSITORY, +} from '../domain'; + +@Global() +@Module({ + imports: [ConfigModule], + providers: [ + PrismaService, + KafkaService, + RedisService, + EventPublisherService, + LeaderboardCacheService, + { + provide: REFERRAL_RELATIONSHIP_REPOSITORY, + useClass: ReferralRelationshipRepository, + }, + { + provide: TEAM_STATISTICS_REPOSITORY, + useClass: TeamStatisticsRepository, + }, + ], + exports: [ + PrismaService, + KafkaService, + RedisService, + EventPublisherService, + LeaderboardCacheService, + REFERRAL_RELATIONSHIP_REPOSITORY, + TEAM_STATISTICS_REPOSITORY, + ], +}) +export class InfrastructureModule {} diff --git a/backend/services/referral-service/test/domain/aggregates/referral-relationship.aggregate.spec.ts b/backend/services/referral-service/test/domain/aggregates/referral-relationship.aggregate.spec.ts new file mode 100644 index 00000000..1f237f34 --- /dev/null +++ b/backend/services/referral-service/test/domain/aggregates/referral-relationship.aggregate.spec.ts @@ -0,0 +1,115 @@ +import { ReferralRelationship } from '../../../src/domain/aggregates/referral-relationship/referral-relationship.aggregate'; +import { ReferralRelationshipCreatedEvent } from '../../../src/domain/events'; + +describe('ReferralRelationship Aggregate', () => { + describe('create', () => { + it('should create referral relationship without referrer', () => { + const relationship = ReferralRelationship.create(100n, null); + + expect(relationship.userId).toBe(100n); + expect(relationship.referrerId).toBeNull(); + expect(relationship.referralCode).toMatch(/^RWA/); + expect(relationship.referralChain).toEqual([]); + }); + + it('should create referral relationship with referrer', () => { + const parentChain = [200n, 300n]; + const relationship = ReferralRelationship.create(100n, 50n, parentChain); + + expect(relationship.userId).toBe(100n); + expect(relationship.referrerId).toBe(50n); + expect(relationship.referralChain).toEqual([50n, 200n, 300n]); + }); + + it('should emit ReferralRelationshipCreatedEvent', () => { + const relationship = ReferralRelationship.create(100n, 50n); + + expect(relationship.domainEvents.length).toBe(1); + expect(relationship.domainEvents[0]).toBeInstanceOf(ReferralRelationshipCreatedEvent); + + const event = relationship.domainEvents[0] as ReferralRelationshipCreatedEvent; + expect(event.userId).toBe(100n); + expect(event.referrerId).toBe(50n); + }); + }); + + describe('reconstitute', () => { + it('should reconstitute from persistence data', () => { + const props = { + id: 1n, + userId: 100n, + referrerId: 50n, + referralCode: 'RWATEST123', + referralChain: [50n, 200n], + createdAt: new Date('2024-01-01'), + updatedAt: new Date('2024-01-02'), + }; + + const relationship = ReferralRelationship.reconstitute(props); + + expect(relationship.id).toBe(1n); + expect(relationship.userId).toBe(100n); + expect(relationship.referrerId).toBe(50n); + expect(relationship.referralCode).toBe('RWATEST123'); + expect(relationship.referralChain).toEqual([50n, 200n]); + }); + }); + + describe('getDirectReferrer', () => { + it('should return direct referrer', () => { + const relationship = ReferralRelationship.create(100n, 50n, [200n]); + expect(relationship.getDirectReferrer()).toBe(50n); + }); + + it('should return null when no referrer', () => { + const relationship = ReferralRelationship.create(100n, null); + expect(relationship.getDirectReferrer()).toBeNull(); + }); + }); + + describe('getReferrerAtLevel', () => { + it('should return referrer at specific level', () => { + const relationship = ReferralRelationship.create(100n, 50n, [200n, 300n]); + + expect(relationship.getReferrerAtLevel(0)).toBe(50n); + expect(relationship.getReferrerAtLevel(1)).toBe(200n); + expect(relationship.getReferrerAtLevel(2)).toBe(300n); + }); + }); + + describe('getAllAncestorIds', () => { + it('should return all ancestor IDs', () => { + const relationship = ReferralRelationship.create(100n, 50n, [200n, 300n]); + expect(relationship.getAllAncestorIds()).toEqual([50n, 200n, 300n]); + }); + }); + + describe('getChainDepth', () => { + it('should return chain depth', () => { + const relationship = ReferralRelationship.create(100n, 50n, [200n, 300n]); + expect(relationship.getChainDepth()).toBe(3); + }); + }); + + describe('clearDomainEvents', () => { + it('should clear domain events', () => { + const relationship = ReferralRelationship.create(100n, 50n); + expect(relationship.domainEvents.length).toBe(1); + + relationship.clearDomainEvents(); + expect(relationship.domainEvents.length).toBe(0); + }); + }); + + describe('toPersistence', () => { + it('should convert to persistence format', () => { + const relationship = ReferralRelationship.create(100n, 50n, [200n]); + const data = relationship.toPersistence(); + + expect(data.userId).toBe(100n); + expect(data.referrerId).toBe(50n); + expect(data.referralCode).toMatch(/^RWA/); + expect(data.referralChain).toEqual([50n, 200n]); + }); + }); +}); diff --git a/backend/services/referral-service/test/domain/aggregates/team-statistics.aggregate.spec.ts b/backend/services/referral-service/test/domain/aggregates/team-statistics.aggregate.spec.ts new file mode 100644 index 00000000..c29cf1cb --- /dev/null +++ b/backend/services/referral-service/test/domain/aggregates/team-statistics.aggregate.spec.ts @@ -0,0 +1,165 @@ +import { TeamStatistics } from '../../../src/domain/aggregates/team-statistics/team-statistics.aggregate'; +import { TeamStatisticsUpdatedEvent } from '../../../src/domain/events'; + +describe('TeamStatistics Aggregate', () => { + describe('create', () => { + it('should create empty team statistics', () => { + const stats = TeamStatistics.create(100n); + + expect(stats.userId).toBe(100n); + expect(stats.directReferralCount).toBe(0); + expect(stats.totalTeamCount).toBe(0); + expect(stats.personalPlantingCount).toBe(0); + expect(stats.teamPlantingCount).toBe(0); + expect(stats.leaderboardScore).toBe(0); + }); + }); + + describe('reconstitute', () => { + it('should reconstitute from persistence data', () => { + const props = { + id: 1n, + userId: 100n, + directReferralCount: 5, + totalTeamCount: 100, + personalPlantingCount: 10, + teamPlantingCount: 90, + leaderboardScore: 60, + maxDirectTeamCount: 40, + provinceCityDistribution: { '110000': { '110100': 50 } }, + lastCalculatedAt: new Date(), + createdAt: new Date(), + updatedAt: new Date(), + directReferrals: [ + { referralId: 200n, teamCount: 40 }, + { referralId: 300n, teamCount: 30 }, + ], + }; + + const stats = TeamStatistics.reconstitute(props); + + expect(stats.id).toBe(1n); + expect(stats.userId).toBe(100n); + expect(stats.directReferralCount).toBe(5); + expect(stats.totalTeamCount).toBe(100); + expect(stats.leaderboardScore).toBe(60); + }); + }); + + describe('addDirectReferral', () => { + it('should increment direct referral count', () => { + const stats = TeamStatistics.create(100n); + + stats.addDirectReferral(200n); + expect(stats.directReferralCount).toBe(1); + + stats.addDirectReferral(300n); + expect(stats.directReferralCount).toBe(2); + }); + }); + + describe('addPersonalPlanting', () => { + it('should add personal planting count', () => { + const stats = TeamStatistics.create(100n); + + stats.addPersonalPlanting(10, '110000', '110100'); + + expect(stats.personalPlantingCount).toBe(10); + expect(stats.totalTeamCount).toBe(10); + }); + + it('should emit TeamStatisticsUpdatedEvent', () => { + const stats = TeamStatistics.create(100n); + stats.addPersonalPlanting(10, '110000', '110100'); + + expect(stats.domainEvents.length).toBe(1); + expect(stats.domainEvents[0]).toBeInstanceOf(TeamStatisticsUpdatedEvent); + }); + + it('should update province/city distribution', () => { + const stats = TeamStatistics.create(100n); + + stats.addPersonalPlanting(10, '110000', '110100'); + stats.addPersonalPlanting(5, '110000', '110100'); + + expect(stats.provinceCityDistribution.getCityTotal('110000', '110100')).toBe(15); + }); + }); + + describe('addTeamPlanting', () => { + it('should add team planting count', () => { + const stats = TeamStatistics.create(100n); + + stats.addTeamPlanting(20, '110000', '110100', 200n); + + expect(stats.teamPlantingCount).toBe(20); + expect(stats.totalTeamCount).toBe(20); + }); + + it('should track direct referral team count', () => { + const stats = TeamStatistics.create(100n); + stats.addDirectReferral(200n); + stats.addDirectReferral(300n); + + stats.addTeamPlanting(30, '110000', '110100', 200n); + stats.addTeamPlanting(20, '110000', '110100', 300n); + + const directStats = stats.getDirectReferralStats(); + expect(directStats.get(200n)).toBe(30); + expect(directStats.get(300n)).toBe(20); + }); + + it('should recalculate leaderboard score', () => { + const stats = TeamStatistics.create(100n); + stats.addDirectReferral(200n); + stats.addDirectReferral(300n); + + stats.addTeamPlanting(50, '110000', '110100', 200n); + stats.addTeamPlanting(30, '110000', '110100', 300n); + + // Total = 80, max direct = 50, score = 30 + expect(stats.leaderboardScore).toBe(30); + }); + }); + + describe('getDirectReferralStats', () => { + it('should return copy of direct referral stats', () => { + const stats = TeamStatistics.create(100n); + stats.addDirectReferral(200n); + stats.addTeamPlanting(30, '110000', '110100', 200n); + + const directStats = stats.getDirectReferralStats(); + directStats.set(999n, 100); // Modify copy + + // Original should not be affected + expect(stats.getDirectReferralStats().has(999n)).toBe(false); + }); + }); + + describe('clearDomainEvents', () => { + it('should clear domain events', () => { + const stats = TeamStatistics.create(100n); + stats.addPersonalPlanting(10, '110000', '110100'); + expect(stats.domainEvents.length).toBe(1); + + stats.clearDomainEvents(); + expect(stats.domainEvents.length).toBe(0); + }); + }); + + describe('toPersistence', () => { + it('should convert to persistence format', () => { + const stats = TeamStatistics.create(100n); + stats.addDirectReferral(200n); + stats.addTeamPlanting(30, '110000', '110100', 200n); + + const data = stats.toPersistence(); + + expect(data.userId).toBe(100n); + expect(data.directReferralCount).toBe(1); + expect(data.totalTeamCount).toBe(30); + expect(data.teamPlantingCount).toBe(30); + expect(data.directReferrals).toContainEqual({ referralId: 200n, teamCount: 30 }); + }); + }); +}); diff --git a/backend/services/referral-service/test/domain/services/leaderboard-calculation.service.spec.ts b/backend/services/referral-service/test/domain/services/leaderboard-calculation.service.spec.ts new file mode 100644 index 00000000..6ea1d076 --- /dev/null +++ b/backend/services/referral-service/test/domain/services/leaderboard-calculation.service.spec.ts @@ -0,0 +1,85 @@ +import { LeaderboardCalculationService } from '../../../src/domain/services/leaderboard-calculation.service'; +import { LeaderboardScore } from '../../../src/domain/value-objects'; + +describe('LeaderboardCalculationService', () => { + let service: LeaderboardCalculationService; + + beforeEach(() => { + service = new LeaderboardCalculationService(); + }); + + describe('calculateScore', () => { + it('should calculate score correctly', () => { + const stats = [ + { referralId: 1n, teamCount: 30 }, + { referralId: 2n, teamCount: 40 }, + { referralId: 3n, teamCount: 30 }, + ]; + const score = service.calculateScore(100, stats); + + expect(score.totalTeamCount).toBe(100); + expect(score.maxDirectTeamCount).toBe(40); + expect(score.score).toBe(60); + }); + + it('should return zero score for empty teams', () => { + const score = service.calculateScore(0, []); + + expect(score.score).toBe(0); + }); + }); + + describe('updateScoreOnPlanting', () => { + it('should update score when planting added to direct referral team', () => { + const currentScore = LeaderboardScore.calculate(100, [30, 40, 30]); + const stats = [ + { referralId: 1n, teamCount: 30 }, + { referralId: 2n, teamCount: 40 }, + { referralId: 3n, teamCount: 30 }, + ]; + + const newScore = service.updateScoreOnPlanting(currentScore, 10, 1n, stats); + + // New total = 110, new max = 40 (unchanged), score = 70 + expect(newScore.totalTeamCount).toBe(110); + expect(newScore.score).toBe(70); + }); + + it('should update score when max team increases', () => { + const currentScore = LeaderboardScore.calculate(100, [30, 40, 30]); + const stats = [ + { referralId: 1n, teamCount: 30 }, + { referralId: 2n, teamCount: 40 }, + { referralId: 3n, teamCount: 30 }, + ]; + + const newScore = service.updateScoreOnPlanting(currentScore, 20, 2n, stats); + + // New total = 120, new max = 60, score = 60 + expect(newScore.totalTeamCount).toBe(120); + expect(newScore.maxDirectTeamCount).toBe(60); + expect(newScore.score).toBe(60); + }); + }); + + describe('compareRank', () => { + it('should correctly compare for ranking', () => { + const scoreA = LeaderboardScore.calculate(100, [40, 30, 30]); // score = 60 + const scoreB = LeaderboardScore.calculate(80, [30, 30, 20]); // score = 50 + + expect(service.compareRank(scoreA, scoreB)).toBeLessThan(0); // A ranks higher + }); + }); + + describe('validateScore', () => { + it('should return true for valid score', () => { + const score = LeaderboardScore.calculate(100, [40, 30, 30]); + expect(service.validateScore(score, 100, 40)).toBe(true); + }); + + it('should return false for invalid score', () => { + const score = LeaderboardScore.calculate(100, [40, 30, 30]); + expect(service.validateScore(score, 100, 50)).toBe(false); + }); + }); +}); diff --git a/backend/services/referral-service/test/domain/services/referral-chain.service.spec.ts b/backend/services/referral-service/test/domain/services/referral-chain.service.spec.ts new file mode 100644 index 00000000..2ec1afe2 --- /dev/null +++ b/backend/services/referral-service/test/domain/services/referral-chain.service.spec.ts @@ -0,0 +1,64 @@ +import { ReferralChainService } from '../../../src/domain/services/referral-chain.service'; + +describe('ReferralChainService', () => { + let service: ReferralChainService; + + beforeEach(() => { + service = new ReferralChainService(); + }); + + describe('buildChain', () => { + it('should build empty chain without referrer', () => { + const chain = service.buildChain(null); + expect(chain.chain).toEqual([]); + }); + + it('should build chain with referrer', () => { + const chain = service.buildChain(100n, [200n, 300n]); + expect(chain.chain).toEqual([100n, 200n, 300n]); + }); + }); + + describe('validateChain', () => { + it('should return true for valid chain', () => { + const chain = [100n, 200n, 300n]; + expect(service.validateChain(chain, 400n)).toBe(true); + }); + + it('should return false if user already in chain (circular)', () => { + const chain = [100n, 200n, 300n]; + expect(service.validateChain(chain, 200n)).toBe(false); + }); + + it('should return false if chain exceeds max depth', () => { + const chain = Array.from({ length: 10 }, (_, i) => BigInt(i + 1)); + expect(service.validateChain(chain, 100n)).toBe(false); + }); + }); + + describe('getAncestorsForUpdate', () => { + it('should return all ancestors when no limit', () => { + const chain = [100n, 200n, 300n]; + expect(service.getAncestorsForUpdate(chain)).toEqual([100n, 200n, 300n]); + }); + + it('should return limited ancestors when limit specified', () => { + const chain = [100n, 200n, 300n, 400n, 500n]; + expect(service.getAncestorsForUpdate(chain, 3)).toEqual([100n, 200n, 300n]); + }); + }); + + describe('getLevelInChain', () => { + it('should return correct level for ancestor', () => { + const chain = [100n, 200n, 300n]; + expect(service.getLevelInChain(chain, 100n)).toBe(0); + expect(service.getLevelInChain(chain, 200n)).toBe(1); + expect(service.getLevelInChain(chain, 300n)).toBe(2); + }); + + it('should return -1 if ancestor not in chain', () => { + const chain = [100n, 200n, 300n]; + expect(service.getLevelInChain(chain, 999n)).toBe(-1); + }); + }); +}); diff --git a/backend/services/referral-service/test/domain/value-objects/leaderboard-score.vo.spec.ts b/backend/services/referral-service/test/domain/value-objects/leaderboard-score.vo.spec.ts new file mode 100644 index 00000000..bd4c28b8 --- /dev/null +++ b/backend/services/referral-service/test/domain/value-objects/leaderboard-score.vo.spec.ts @@ -0,0 +1,79 @@ +import { LeaderboardScore } from '../../../src/domain/value-objects/leaderboard-score.vo'; + +describe('LeaderboardScore Value Object', () => { + describe('calculate', () => { + it('should calculate score correctly with single team', () => { + const score = LeaderboardScore.calculate(100, [100]); + expect(score.totalTeamCount).toBe(100); + expect(score.maxDirectTeamCount).toBe(100); + expect(score.score).toBe(0); // 100 - 100 = 0 + }); + + it('should calculate score correctly with multiple teams', () => { + const score = LeaderboardScore.calculate(100, [30, 40, 30]); + expect(score.totalTeamCount).toBe(100); + expect(score.maxDirectTeamCount).toBe(40); + expect(score.score).toBe(60); // 100 - 40 = 60 + }); + + it('should calculate score correctly with no teams', () => { + const score = LeaderboardScore.calculate(0, []); + expect(score.totalTeamCount).toBe(0); + expect(score.maxDirectTeamCount).toBe(0); + expect(score.score).toBe(0); + }); + + it('should not return negative score', () => { + const score = LeaderboardScore.calculate(50, [80]); + expect(score.score).toBe(0); + }); + + it('should encourage balanced teams', () => { + // 不均衡: 100 total, max 80 -> score = 20 + const unbalanced = LeaderboardScore.calculate(100, [80, 10, 10]); + // 均衡: 100 total, max 34 -> score = 66 + const balanced = LeaderboardScore.calculate(100, [34, 33, 33]); + + expect(balanced.score).toBeGreaterThan(unbalanced.score); + }); + }); + + describe('zero', () => { + it('should create zero score', () => { + const score = LeaderboardScore.zero(); + expect(score.totalTeamCount).toBe(0); + expect(score.maxDirectTeamCount).toBe(0); + expect(score.score).toBe(0); + }); + }); + + describe('recalculate', () => { + it('should recalculate with new values', () => { + const initial = LeaderboardScore.calculate(50, [30, 20]); + const updated = initial.recalculate(100, [50, 50]); + + expect(updated.totalTeamCount).toBe(100); + expect(updated.maxDirectTeamCount).toBe(50); + expect(updated.score).toBe(50); + }); + }); + + describe('compareTo', () => { + it('should compare scores for ranking (descending)', () => { + const scoreA = LeaderboardScore.calculate(100, [30, 30, 40]); + const scoreB = LeaderboardScore.calculate(80, [20, 20, 40]); + + // scoreA.score = 60, scoreB.score = 40 + // compareTo returns other.score - this.score for descending order + expect(scoreA.compareTo(scoreB)).toBeLessThan(0); // A ranks higher + expect(scoreB.compareTo(scoreA)).toBeGreaterThan(0); // B ranks lower + }); + + it('should return 0 for equal scores', () => { + const scoreA = LeaderboardScore.calculate(100, [50, 50]); + const scoreB = LeaderboardScore.calculate(100, [50, 50]); + + expect(scoreA.compareTo(scoreB)).toBe(0); + }); + }); +}); diff --git a/backend/services/referral-service/test/domain/value-objects/province-city-distribution.vo.spec.ts b/backend/services/referral-service/test/domain/value-objects/province-city-distribution.vo.spec.ts new file mode 100644 index 00000000..5c5d9332 --- /dev/null +++ b/backend/services/referral-service/test/domain/value-objects/province-city-distribution.vo.spec.ts @@ -0,0 +1,125 @@ +import { ProvinceCityDistribution } from '../../../src/domain/value-objects/province-city-distribution.vo'; + +describe('ProvinceCityDistribution Value Object', () => { + describe('empty', () => { + it('should create empty distribution', () => { + const dist = ProvinceCityDistribution.empty(); + expect(dist.getTotal()).toBe(0); + expect(dist.getAll()).toEqual([]); + }); + }); + + describe('fromJson', () => { + it('should create distribution from JSON', () => { + const json = { + '110000': { '110100': 10, '110200': 5 }, + '120000': { '120100': 8 }, + }; + const dist = ProvinceCityDistribution.fromJson(json); + + expect(dist.getProvinceTotal('110000')).toBe(15); + expect(dist.getProvinceTotal('120000')).toBe(8); + expect(dist.getCityTotal('110000', '110100')).toBe(10); + }); + + it('should return empty distribution for null', () => { + const dist = ProvinceCityDistribution.fromJson(null); + expect(dist.getTotal()).toBe(0); + }); + }); + + describe('add', () => { + it('should add to existing city', () => { + let dist = ProvinceCityDistribution.empty(); + dist = dist.add('110000', '110100', 5); + dist = dist.add('110000', '110100', 3); + + expect(dist.getCityTotal('110000', '110100')).toBe(8); + }); + + it('should add new province and city', () => { + let dist = ProvinceCityDistribution.empty(); + dist = dist.add('110000', '110100', 5); + dist = dist.add('120000', '120100', 3); + + expect(dist.getProvinceTotal('110000')).toBe(5); + expect(dist.getProvinceTotal('120000')).toBe(3); + }); + + it('should be immutable', () => { + const dist1 = ProvinceCityDistribution.empty(); + const dist2 = dist1.add('110000', '110100', 5); + + expect(dist1.getTotal()).toBe(0); + expect(dist2.getTotal()).toBe(5); + }); + }); + + describe('getProvinceTotal', () => { + it('should return province total', () => { + let dist = ProvinceCityDistribution.empty(); + dist = dist.add('110000', '110100', 5); + dist = dist.add('110000', '110200', 3); + + expect(dist.getProvinceTotal('110000')).toBe(8); + }); + + it('should return 0 for non-existent province', () => { + const dist = ProvinceCityDistribution.empty(); + expect(dist.getProvinceTotal('999999')).toBe(0); + }); + }); + + describe('getCityTotal', () => { + it('should return city total', () => { + let dist = ProvinceCityDistribution.empty(); + dist = dist.add('110000', '110100', 5); + + expect(dist.getCityTotal('110000', '110100')).toBe(5); + }); + + it('should return 0 for non-existent city', () => { + const dist = ProvinceCityDistribution.empty(); + expect(dist.getCityTotal('110000', '110100')).toBe(0); + }); + }); + + describe('getAll', () => { + it('should return all province/city counts', () => { + let dist = ProvinceCityDistribution.empty(); + dist = dist.add('110000', '110100', 5); + dist = dist.add('110000', '110200', 3); + dist = dist.add('120000', '120100', 8); + + const all = dist.getAll(); + expect(all.length).toBe(3); + expect(all).toContainEqual({ provinceCode: '110000', cityCode: '110100', count: 5 }); + expect(all).toContainEqual({ provinceCode: '110000', cityCode: '110200', count: 3 }); + expect(all).toContainEqual({ provinceCode: '120000', cityCode: '120100', count: 8 }); + }); + }); + + describe('getTotal', () => { + it('should return total count', () => { + let dist = ProvinceCityDistribution.empty(); + dist = dist.add('110000', '110100', 5); + dist = dist.add('110000', '110200', 3); + dist = dist.add('120000', '120100', 8); + + expect(dist.getTotal()).toBe(16); + }); + }); + + describe('toJson', () => { + it('should convert to JSON format', () => { + let dist = ProvinceCityDistribution.empty(); + dist = dist.add('110000', '110100', 5); + dist = dist.add('110000', '110200', 3); + + const json = dist.toJson(); + expect(json).toEqual({ + '110000': { '110100': 5, '110200': 3 }, + }); + }); + }); +}); diff --git a/backend/services/referral-service/test/domain/value-objects/referral-chain.vo.spec.ts b/backend/services/referral-service/test/domain/value-objects/referral-chain.vo.spec.ts new file mode 100644 index 00000000..feec38d0 --- /dev/null +++ b/backend/services/referral-service/test/domain/value-objects/referral-chain.vo.spec.ts @@ -0,0 +1,92 @@ +import { ReferralChain } from '../../../src/domain/value-objects/referral-chain.vo'; + +describe('ReferralChain Value Object', () => { + describe('create', () => { + it('should create empty chain when no referrer', () => { + const chain = ReferralChain.create(null); + expect(chain.chain).toEqual([]); + expect(chain.depth).toBe(0); + }); + + it('should create chain with referrer', () => { + const chain = ReferralChain.create(100n); + expect(chain.chain).toEqual([100n]); + expect(chain.depth).toBe(1); + }); + + it('should create chain with parent chain', () => { + const parentChain = [200n, 300n, 400n]; + const chain = ReferralChain.create(100n, parentChain); + expect(chain.chain).toEqual([100n, 200n, 300n, 400n]); + expect(chain.depth).toBe(4); + }); + + it('should truncate chain to MAX_DEPTH', () => { + const parentChain = [2n, 3n, 4n, 5n, 6n, 7n, 8n, 9n, 10n, 11n]; + const chain = ReferralChain.create(1n, parentChain); + expect(chain.chain.length).toBe(10); + expect(chain.chain[0]).toBe(1n); + expect(chain.chain[9]).toBe(10n); + }); + }); + + describe('fromArray', () => { + it('should create chain from array', () => { + const chain = ReferralChain.fromArray([1n, 2n, 3n]); + expect(chain.chain).toEqual([1n, 2n, 3n]); + }); + + it('should throw error if array exceeds MAX_DEPTH', () => { + const longArray = Array.from({ length: 11 }, (_, i) => BigInt(i + 1)); + expect(() => ReferralChain.fromArray(longArray)).toThrow('推荐链深度不能超过 10'); + }); + }); + + describe('empty', () => { + it('should create empty chain', () => { + const chain = ReferralChain.empty(); + expect(chain.chain).toEqual([]); + expect(chain.depth).toBe(0); + }); + }); + + describe('directReferrer', () => { + it('should return direct referrer', () => { + const chain = ReferralChain.create(100n, [200n, 300n]); + expect(chain.directReferrer).toBe(100n); + }); + + it('should return null for empty chain', () => { + const chain = ReferralChain.empty(); + expect(chain.directReferrer).toBeNull(); + }); + }); + + describe('getReferrerAtLevel', () => { + it('should return referrer at specified level', () => { + const chain = ReferralChain.fromArray([100n, 200n, 300n]); + expect(chain.getReferrerAtLevel(0)).toBe(100n); + expect(chain.getReferrerAtLevel(1)).toBe(200n); + expect(chain.getReferrerAtLevel(2)).toBe(300n); + }); + + it('should return null for out of bounds level', () => { + const chain = ReferralChain.fromArray([100n, 200n]); + expect(chain.getReferrerAtLevel(5)).toBeNull(); + }); + }); + + describe('getAllAncestors', () => { + it('should return all ancestors', () => { + const chain = ReferralChain.fromArray([100n, 200n, 300n]); + expect(chain.getAllAncestors()).toEqual([100n, 200n, 300n]); + }); + + it('should return copy of array', () => { + const chain = ReferralChain.fromArray([100n, 200n]); + const ancestors = chain.getAllAncestors(); + ancestors.push(999n); + expect(chain.getAllAncestors()).toEqual([100n, 200n]); + }); + }); +}); diff --git a/backend/services/referral-service/test/domain/value-objects/referral-code.vo.spec.ts b/backend/services/referral-service/test/domain/value-objects/referral-code.vo.spec.ts new file mode 100644 index 00000000..3de0fdec --- /dev/null +++ b/backend/services/referral-service/test/domain/value-objects/referral-code.vo.spec.ts @@ -0,0 +1,64 @@ +import { ReferralCode } from '../../../src/domain/value-objects/referral-code.vo'; + +describe('ReferralCode Value Object', () => { + describe('create', () => { + it('should create ReferralCode from valid string', () => { + const code = ReferralCode.create('RWATEST123'); + expect(code.value).toBe('RWATEST123'); + }); + + it('should convert lowercase to uppercase', () => { + const code = ReferralCode.create('rwatest123'); + expect(code.value).toBe('RWATEST123'); + }); + + it('should throw error for too short code', () => { + expect(() => ReferralCode.create('ABC')).toThrow('推荐码长度必须在6-20个字符之间'); + }); + + it('should throw error for too long code', () => { + expect(() => ReferralCode.create('A'.repeat(21))).toThrow('推荐码长度必须在6-20个字符之间'); + }); + + it('should throw error for invalid characters', () => { + expect(() => ReferralCode.create('RWA-TEST')).toThrow('推荐码只能包含大写字母和数字'); + }); + }); + + describe('generate', () => { + it('should generate referral code with RWA prefix', () => { + const code = ReferralCode.generate(123456789n); + expect(code.value).toMatch(/^RWA/); + }); + + it('should generate valid referral code', () => { + const code = ReferralCode.generate(123456789n); + expect(code.value.length).toBeGreaterThanOrEqual(6); + expect(code.value.length).toBeLessThanOrEqual(20); + expect(code.value).toMatch(/^[A-Z0-9]+$/); + }); + + it('should generate different codes for same user (due to random component)', () => { + const code1 = ReferralCode.generate(123n); + const code2 = ReferralCode.generate(123n); + // Note: there's a small chance they could be the same, but very unlikely + // This test verifies the random component is working + expect(code1.value).toBeDefined(); + expect(code2.value).toBeDefined(); + }); + }); + + describe('equals', () => { + it('should return true for equal codes', () => { + const code1 = ReferralCode.create('RWATEST123'); + const code2 = ReferralCode.create('RWATEST123'); + expect(code1.equals(code2)).toBe(true); + }); + + it('should return false for different codes', () => { + const code1 = ReferralCode.create('RWATEST123'); + const code2 = ReferralCode.create('RWATEST456'); + expect(code1.equals(code2)).toBe(false); + }); + }); +}); diff --git a/backend/services/referral-service/test/domain/value-objects/user-id.vo.spec.ts b/backend/services/referral-service/test/domain/value-objects/user-id.vo.spec.ts new file mode 100644 index 00000000..2c871e25 --- /dev/null +++ b/backend/services/referral-service/test/domain/value-objects/user-id.vo.spec.ts @@ -0,0 +1,49 @@ +import { UserId } from '../../../src/domain/value-objects/user-id.vo'; + +describe('UserId Value Object', () => { + describe('create', () => { + it('should create UserId from bigint', () => { + const userId = UserId.create(123n); + expect(userId.value).toBe(123n); + }); + + it('should create UserId from string', () => { + const userId = UserId.create('456'); + expect(userId.value).toBe(456n); + }); + + it('should create UserId from number', () => { + const userId = UserId.create(789); + expect(userId.value).toBe(789n); + }); + + it('should throw error for zero value', () => { + expect(() => UserId.create(0n)).toThrow('用户ID必须大于0'); + }); + + it('should throw error for negative value', () => { + expect(() => UserId.create(-1n)).toThrow('用户ID必须大于0'); + }); + }); + + describe('equals', () => { + it('should return true for equal values', () => { + const userId1 = UserId.create(123n); + const userId2 = UserId.create(123n); + expect(userId1.equals(userId2)).toBe(true); + }); + + it('should return false for different values', () => { + const userId1 = UserId.create(123n); + const userId2 = UserId.create(456n); + expect(userId1.equals(userId2)).toBe(false); + }); + }); + + describe('toString', () => { + it('should return string representation', () => { + const userId = UserId.create(123n); + expect(userId.toString()).toBe('123'); + }); + }); +}); diff --git a/backend/services/referral-service/test/e2e/app.e2e-spec.ts b/backend/services/referral-service/test/e2e/app.e2e-spec.ts new file mode 100644 index 00000000..7db4685f --- /dev/null +++ b/backend/services/referral-service/test/e2e/app.e2e-spec.ts @@ -0,0 +1,369 @@ +import { Test, TestingModule } from '@nestjs/testing'; +import { INestApplication, ValidationPipe } from '@nestjs/common'; +import * as request from 'supertest'; +import { AppModule } from '../../src/app.module'; +import { PrismaService } from '../../src/infrastructure/database/prisma.service'; +import { KafkaService } from '../../src/infrastructure/messaging/kafka.service'; +import { RedisService } from '../../src/infrastructure/cache/redis.service'; +import * as jwt from 'jsonwebtoken'; + +// Mock services to avoid real connections in E2E tests +class MockPrismaService { + async onModuleInit() {} + async onModuleDestroy() {} + $connect = jest.fn(); + $disconnect = jest.fn(); + referralRelationship = { + upsert: jest.fn(), + findUnique: jest.fn(), + findMany: jest.fn(), + count: jest.fn(), + }; + teamStatistics = { + upsert: jest.fn(), + create: jest.fn(), + findUnique: jest.fn(), + findMany: jest.fn(), + count: jest.fn(), + update: jest.fn(), + }; + directReferral = { + upsert: jest.fn(), + findMany: jest.fn(), + }; + $transaction = jest.fn((fn) => fn(this)); +} + +class MockKafkaService { + async onModuleInit() {} + async onModuleDestroy() {} + publish = jest.fn(); + publishBatch = jest.fn(); + subscribe = jest.fn(); +} + +class MockRedisService { + async onModuleInit() {} + async onModuleDestroy() {} + get = jest.fn().mockResolvedValue(null); + set = jest.fn(); + del = jest.fn(); + zadd = jest.fn(); + zrevrank = jest.fn().mockResolvedValue(null); + zrevrange = jest.fn().mockResolvedValue([]); + zrevrangeWithScores = jest.fn().mockResolvedValue([]); + zincrby = jest.fn(); +} + +describe('Referral Service (E2E)', () => { + let app: INestApplication; + let mockPrisma: MockPrismaService; + const JWT_SECRET = 'test-jwt-secret-for-e2e-tests'; + + const generateToken = (userId: bigint) => { + return jwt.sign( + { sub: `user-${userId}`, userId: userId.toString(), type: 'access' }, + JWT_SECRET, + { expiresIn: '1h' }, + ); + }; + + beforeAll(async () => { + mockPrisma = new MockPrismaService(); + + const moduleFixture: TestingModule = await Test.createTestingModule({ + imports: [AppModule], + }) + .overrideProvider(PrismaService) + .useValue(mockPrisma) + .overrideProvider(KafkaService) + .useClass(MockKafkaService) + .overrideProvider(RedisService) + .useClass(MockRedisService) + .compile(); + + app = moduleFixture.createNestApplication(); + app.useGlobalPipes( + new ValidationPipe({ + whitelist: true, + forbidNonWhitelisted: true, + transform: true, + transformOptions: { enableImplicitConversion: true }, + }), + ); + app.setGlobalPrefix('api/v1'); + + await app.init(); + }); + + afterAll(async () => { + await app.close(); + }); + + beforeEach(() => { + jest.clearAllMocks(); + }); + + describe('Health Endpoints', () => { + it('GET /api/v1/health - should return health status', () => { + return request(app.getHttpServer()) + .get('/api/v1/health') + .expect(200) + .expect((res) => { + expect(res.body.status).toBe('ok'); + expect(res.body.service).toBe('referral-service'); + }); + }); + + it('GET /api/v1/health/ready - should return ready status', () => { + return request(app.getHttpServer()) + .get('/api/v1/health/ready') + .expect(200) + .expect((res) => { + expect(res.body.status).toBe('ready'); + }); + }); + + it('GET /api/v1/health/live - should return alive status', () => { + return request(app.getHttpServer()) + .get('/api/v1/health/live') + .expect(200) + .expect((res) => { + expect(res.body.status).toBe('alive'); + }); + }); + }); + + describe('Referral Endpoints', () => { + describe('GET /api/v1/referral/validate/:code', () => { + it('should return valid=true for existing referral code', async () => { + mockPrisma.referralRelationship.findUnique.mockResolvedValueOnce({ + id: 1n, + userId: 100n, + myReferralCode: 'RWATEST123', + referrerId: null, + ancestorPath: [], + createdAt: new Date(), + updatedAt: new Date(), + }); + + return request(app.getHttpServer()) + .get('/api/v1/referral/validate/RWATEST123') + .expect(200) + .expect((res) => { + expect(res.body.valid).toBe(true); + expect(res.body.referrerId).toBe('100'); + }); + }); + + it('should return valid=false for non-existing referral code', async () => { + mockPrisma.referralRelationship.findUnique.mockResolvedValueOnce(null); + + return request(app.getHttpServer()) + .get('/api/v1/referral/validate/NONEXISTENT') + .expect(200) + .expect((res) => { + expect(res.body.valid).toBe(false); + }); + }); + }); + + describe('POST /api/v1/referral/validate', () => { + it('should validate referral code via POST', async () => { + mockPrisma.referralRelationship.findUnique.mockResolvedValueOnce({ + id: 1n, + userId: 100n, + myReferralCode: 'RWATEST123', + referrerId: null, + ancestorPath: [], + createdAt: new Date(), + updatedAt: new Date(), + }); + + return request(app.getHttpServer()) + .post('/api/v1/referral/validate') + .send({ code: 'RWATEST123' }) + .expect(200) + .expect((res) => { + expect(res.body.valid).toBe(true); + }); + }); + + it('should reject invalid code format', () => { + return request(app.getHttpServer()) + .post('/api/v1/referral/validate') + .send({ code: 'abc' }) // Too short + .expect(400); + }); + }); + + describe('GET /api/v1/referral/me (Protected)', () => { + it('should return 401 without token', () => { + return request(app.getHttpServer()).get('/api/v1/referral/me').expect(401); + }); + + it('should return user referral info with valid token', async () => { + const userId = 100n; + const token = generateToken(userId); + + mockPrisma.referralRelationship.findUnique.mockResolvedValueOnce({ + id: 1n, + userId, + myReferralCode: 'RWATEST123', + referrerId: null, + ancestorPath: [], + createdAt: new Date(), + updatedAt: new Date(), + }); + + mockPrisma.teamStatistics.findUnique.mockResolvedValueOnce({ + id: 1n, + userId, + directReferralCount: 5, + totalTeamCount: 100, + selfPlantingCount: 10, + totalTeamPlantingCount: 90, + effectivePlantingCountForRanking: 60, + maxSingleTeamPlantingCount: 40, + provinceCityDistribution: {}, + lastCalcAt: new Date(), + createdAt: new Date(), + updatedAt: new Date(), + }); + + mockPrisma.directReferral.findMany.mockResolvedValueOnce([]); + + return request(app.getHttpServer()) + .get('/api/v1/referral/me') + .set('Authorization', `Bearer ${token}`) + .expect(200) + .expect((res) => { + expect(res.body.userId).toBe('100'); + expect(res.body.referralCode).toBe('RWATEST123'); + expect(res.body.directReferralCount).toBe(5); + }); + }); + }); + }); + + describe('Leaderboard Endpoints', () => { + describe('GET /api/v1/leaderboard', () => { + it('should return leaderboard entries', async () => { + mockPrisma.teamStatistics.findMany.mockResolvedValueOnce([ + { + userId: 100n, + effectivePlantingCountForRanking: 100, + totalTeamPlantingCount: 150, + directReferralCount: 10, + }, + { + userId: 101n, + effectivePlantingCountForRanking: 80, + totalTeamPlantingCount: 100, + directReferralCount: 5, + }, + ]); + + return request(app.getHttpServer()) + .get('/api/v1/leaderboard') + .expect(200) + .expect((res) => { + expect(res.body.entries).toBeDefined(); + expect(res.body.entries.length).toBe(2); + expect(res.body.entries[0].rank).toBe(1); + expect(res.body.entries[0].score).toBe(100); + }); + }); + + it('should support pagination', async () => { + mockPrisma.teamStatistics.findMany.mockResolvedValueOnce([]); + + return request(app.getHttpServer()) + .get('/api/v1/leaderboard?limit=10&offset=20') + .expect(200) + .expect((res) => { + expect(res.body.entries).toBeDefined(); + }); + }); + }); + + describe('GET /api/v1/leaderboard/me (Protected)', () => { + it('should return user rank with valid token', async () => { + const userId = 100n; + const token = generateToken(userId); + + mockPrisma.teamStatistics.findUnique.mockResolvedValueOnce({ + effectivePlantingCountForRanking: 50, + }); + mockPrisma.teamStatistics.count.mockResolvedValueOnce(5); + mockPrisma.teamStatistics.findMany.mockResolvedValueOnce([ + { userId: 100n, effectivePlantingCountForRanking: 50 }, + ]); + + return request(app.getHttpServer()) + .get('/api/v1/leaderboard/me') + .set('Authorization', `Bearer ${token}`) + .expect(200) + .expect((res) => { + expect(res.body.userId).toBe('100'); + expect(res.body.rank).toBeDefined(); + }); + }); + }); + }); + + describe('Team Statistics Endpoints', () => { + describe('GET /api/v1/team-statistics/me/distribution (Protected)', () => { + it('should return province/city distribution', async () => { + const userId = 100n; + const token = generateToken(userId); + + mockPrisma.teamStatistics.findUnique.mockResolvedValueOnce({ + id: 1n, + userId, + directReferralCount: 5, + totalTeamCount: 100, + selfPlantingCount: 10, + totalTeamPlantingCount: 90, + effectivePlantingCountForRanking: 60, + maxSingleTeamPlantingCount: 40, + provinceCityDistribution: { + '110000': { '110100': 50, '110200': 30 }, + '120000': { '120100': 20 }, + }, + lastCalcAt: new Date(), + createdAt: new Date(), + updatedAt: new Date(), + }); + + mockPrisma.directReferral.findMany.mockResolvedValueOnce([]); + + return request(app.getHttpServer()) + .get('/api/v1/team-statistics/me/distribution') + .set('Authorization', `Bearer ${token}`) + .expect(200) + .expect((res) => { + expect(res.body.provinces).toBeDefined(); + expect(res.body.totalCount).toBeDefined(); + }); + }); + }); + }); + + describe('Error Handling', () => { + it('should return 404 for non-existent routes', () => { + return request(app.getHttpServer()).get('/api/v1/nonexistent').expect(404); + }); + + it('should return 401 for protected routes without auth', () => { + return request(app.getHttpServer()).get('/api/v1/referral/me').expect(401); + }); + + it('should return 401 for invalid token', () => { + return request(app.getHttpServer()) + .get('/api/v1/referral/me') + .set('Authorization', 'Bearer invalid-token') + .expect(401); + }); + }); +}); diff --git a/backend/services/referral-service/test/integration/mocks/prisma.mock.ts b/backend/services/referral-service/test/integration/mocks/prisma.mock.ts new file mode 100644 index 00000000..142ffcff --- /dev/null +++ b/backend/services/referral-service/test/integration/mocks/prisma.mock.ts @@ -0,0 +1,282 @@ +/** + * Prisma Mock for Integration Tests + * 不依赖真实数据库的模拟实现 + */ + +type MockReferralRelationship = { + id: bigint; + userId: bigint; + referrerId: bigint | null; + rootUserId: bigint | null; + myReferralCode: string; + usedReferralCode: string | null; + ancestorPath: bigint[]; + depth: number; + directReferralCount: number; + activeDirectCount: number; + createdAt: Date; + updatedAt: Date; +}; + +type MockTeamStatistics = { + id: bigint; + userId: bigint; + directReferralCount: number; + totalTeamCount: number; + selfPlantingCount: number; + totalTeamPlantingCount: number; + effectivePlantingCountForRanking: number; + maxSingleTeamPlantingCount: number; + provinceCityDistribution: Record>; + lastCalcAt: Date | null; + createdAt: Date; + updatedAt: Date; +}; + +type MockDirectReferral = { + id: bigint; + referrerId: bigint; + referralId: bigint; + referralSequence: bigint; + teamPlantingCount: number; + createdAt: Date; + updatedAt: Date; +}; + +export class MockPrismaService { + private _referralRelationshipsStore: Map = new Map(); + private _teamStatisticsStore: Map = new Map(); + private _directReferralsStore: Map = new Map(); + private idCounter = 1n; + + // ReferralRelationship operations + referralRelationship = { + upsert: async (args: { + where: { userId: bigint }; + create: Partial; + update: Partial; + }) => { + const existing = this._referralRelationshipsStore.get(args.where.userId); + if (existing) { + const updated = { ...existing, ...args.update, updatedAt: new Date() }; + this._referralRelationshipsStore.set(args.where.userId, updated); + return updated; + } + const created: MockReferralRelationship = { + id: this.idCounter++, + userId: args.where.userId, + referrerId: args.create.referrerId ?? null, + rootUserId: args.create.rootUserId ?? null, + myReferralCode: args.create.myReferralCode!, + usedReferralCode: args.create.usedReferralCode ?? null, + ancestorPath: args.create.ancestorPath ?? [], + depth: args.create.depth ?? 0, + directReferralCount: 0, + activeDirectCount: 0, + createdAt: new Date(), + updatedAt: new Date(), + }; + this._referralRelationshipsStore.set(args.where.userId, created); + return created; + }, + + findUnique: async (args: { where: { userId?: bigint; myReferralCode?: string }; select?: unknown }) => { + if (args.where.userId) { + return this._referralRelationshipsStore.get(args.where.userId) ?? null; + } + if (args.where.myReferralCode) { + for (const r of this._referralRelationshipsStore.values()) { + if (r.myReferralCode === args.where.myReferralCode) return r; + } + } + return null; + }, + + findMany: async (args: { where?: { referrerId?: bigint }; orderBy?: unknown }) => { + const results: MockReferralRelationship[] = []; + for (const r of this._referralRelationshipsStore.values()) { + if (!args.where?.referrerId || r.referrerId === args.where.referrerId) { + results.push(r); + } + } + return results; + }, + + count: async (args: { where: { myReferralCode?: string; userId?: bigint } }) => { + let count = 0; + for (const r of this._referralRelationshipsStore.values()) { + if (args.where.myReferralCode && r.myReferralCode === args.where.myReferralCode) count++; + if (args.where.userId && r.userId === args.where.userId) count++; + } + return count; + }, + }; + + // TeamStatistics operations + teamStatistics = { + upsert: async (args: { + where: { userId: bigint }; + create: Partial; + update: Partial; + }) => { + const existing = this._teamStatisticsStore.get(args.where.userId); + if (existing) { + const updated = { ...existing, ...args.update, updatedAt: new Date() }; + this._teamStatisticsStore.set(args.where.userId, updated); + return updated; + } + const created: MockTeamStatistics = { + id: this.idCounter++, + userId: args.where.userId, + directReferralCount: args.create.directReferralCount ?? 0, + totalTeamCount: args.create.totalTeamCount ?? 0, + selfPlantingCount: args.create.selfPlantingCount ?? 0, + totalTeamPlantingCount: args.create.totalTeamPlantingCount ?? 0, + effectivePlantingCountForRanking: args.create.effectivePlantingCountForRanking ?? 0, + maxSingleTeamPlantingCount: args.create.maxSingleTeamPlantingCount ?? 0, + provinceCityDistribution: (args.create.provinceCityDistribution as Record>) ?? {}, + lastCalcAt: args.create.lastCalcAt ?? new Date(), + createdAt: new Date(), + updatedAt: new Date(), + }; + this._teamStatisticsStore.set(args.where.userId, created); + return created; + }, + + create: async (args: { data: Partial }) => { + const created: MockTeamStatistics = { + id: this.idCounter++, + userId: args.data.userId!, + directReferralCount: args.data.directReferralCount ?? 0, + totalTeamCount: args.data.totalTeamCount ?? 0, + selfPlantingCount: args.data.selfPlantingCount ?? 0, + totalTeamPlantingCount: args.data.totalTeamPlantingCount ?? 0, + effectivePlantingCountForRanking: args.data.effectivePlantingCountForRanking ?? 0, + maxSingleTeamPlantingCount: args.data.maxSingleTeamPlantingCount ?? 0, + provinceCityDistribution: (args.data.provinceCityDistribution as Record>) ?? {}, + lastCalcAt: args.data.lastCalcAt ?? new Date(), + createdAt: new Date(), + updatedAt: new Date(), + }; + this._teamStatisticsStore.set(created.userId, created); + return created; + }, + + findUnique: async (args: { where: { userId: bigint } }) => { + return this._teamStatisticsStore.get(args.where.userId) ?? null; + }, + + findMany: async (args: { + where?: { userId?: { in: bigint[] }; effectivePlantingCountForRanking?: { gt: number } }; + orderBy?: unknown[]; + skip?: number; + take?: number; + select?: unknown; + }) => { + let results = Array.from(this._teamStatisticsStore.values()); + + if (args.where?.userId?.in) { + results = results.filter((s) => args.where!.userId!.in.includes(s.userId)); + } + if (args.where?.effectivePlantingCountForRanking?.gt !== undefined) { + results = results.filter( + (s) => s.effectivePlantingCountForRanking > args.where!.effectivePlantingCountForRanking!.gt, + ); + } + + // Sort by score descending + results.sort((a, b) => b.effectivePlantingCountForRanking - a.effectivePlantingCountForRanking); + + if (args.skip) results = results.slice(args.skip); + if (args.take) results = results.slice(0, args.take); + + return results; + }, + + count: async (args: { where: { effectivePlantingCountForRanking?: { gt: number } } }) => { + let count = 0; + for (const s of this._teamStatisticsStore.values()) { + if ( + args.where.effectivePlantingCountForRanking?.gt !== undefined && + s.effectivePlantingCountForRanking > args.where.effectivePlantingCountForRanking.gt + ) { + count++; + } + } + return count; + }, + + update: async (args: { where: { userId: bigint }; data: Partial }) => { + const existing = this._teamStatisticsStore.get(args.where.userId); + if (!existing) throw new Error('Not found'); + const updated = { ...existing, ...args.data, updatedAt: new Date() }; + this._teamStatisticsStore.set(args.where.userId, updated); + return updated; + }, + }; + + // DirectReferral operations + directReferral = { + upsert: async (args: { + where: { uk_referrer_referral: { referrerId: bigint; referralId: bigint } }; + create: Partial; + update: Partial | { teamPlantingCount: { increment: number } }; + }) => { + const key = `${args.where.uk_referrer_referral.referrerId}-${args.where.uk_referrer_referral.referralId}`; + const existing = this._directReferralsStore.get(key); + + if (existing) { + let teamPlantingCount = existing.teamPlantingCount; + if ('teamPlantingCount' in args.update) { + if (typeof args.update.teamPlantingCount === 'object' && 'increment' in args.update.teamPlantingCount) { + teamPlantingCount += args.update.teamPlantingCount.increment; + } else { + teamPlantingCount = args.update.teamPlantingCount as number; + } + } + const updated = { ...existing, teamPlantingCount, updatedAt: new Date() }; + this._directReferralsStore.set(key, updated); + return updated; + } + + const created: MockDirectReferral = { + id: this.idCounter++, + referrerId: args.where.uk_referrer_referral.referrerId, + referralId: args.where.uk_referrer_referral.referralId, + referralSequence: args.create.referralSequence!, + teamPlantingCount: args.create.teamPlantingCount ?? 0, + createdAt: new Date(), + updatedAt: new Date(), + }; + this._directReferralsStore.set(key, created); + return created; + }, + + findMany: async (args: { where: { referrerId: bigint }; select?: unknown }) => { + const results: MockDirectReferral[] = []; + for (const dr of this._directReferralsStore.values()) { + if (dr.referrerId === args.where.referrerId) { + results.push(dr); + } + } + return results; + }, + }; + + // Transaction support + $transaction = async (fn: (tx: MockPrismaService) => Promise): Promise => { + return fn(this); + }; + + // Cleanup for tests + $cleanup() { + this._referralRelationshipsStore.clear(); + this._teamStatisticsStore.clear(); + this._directReferralsStore.clear(); + this.idCounter = 1n; + } + + // Lifecycle methods + async onModuleInit() {} + async onModuleDestroy() {} +} diff --git a/backend/services/referral-service/test/integration/repositories/referral-relationship.repository.integration.spec.ts b/backend/services/referral-service/test/integration/repositories/referral-relationship.repository.integration.spec.ts new file mode 100644 index 00000000..f607f732 --- /dev/null +++ b/backend/services/referral-service/test/integration/repositories/referral-relationship.repository.integration.spec.ts @@ -0,0 +1,173 @@ +import { Test, TestingModule } from '@nestjs/testing'; +import { ReferralRelationshipRepository } from '../../../src/infrastructure/repositories/referral-relationship.repository'; +import { PrismaService } from '../../../src/infrastructure/database/prisma.service'; +import { MockPrismaService } from '../mocks/prisma.mock'; +import { ReferralRelationship } from '../../../src/domain'; + +describe('ReferralRelationshipRepository (Integration)', () => { + let repository: ReferralRelationshipRepository; + let mockPrisma: MockPrismaService; + + beforeEach(async () => { + mockPrisma = new MockPrismaService(); + + const module: TestingModule = await Test.createTestingModule({ + providers: [ + ReferralRelationshipRepository, + { + provide: PrismaService, + useValue: mockPrisma, + }, + ], + }).compile(); + + repository = module.get(ReferralRelationshipRepository); + }); + + afterEach(() => { + mockPrisma.$cleanup(); + }); + + describe('save', () => { + it('should save a new referral relationship', async () => { + const relationship = ReferralRelationship.create(100n, null); + const saved = await repository.save(relationship); + + expect(saved).toBeDefined(); + expect(saved.userId).toBe(100n); + expect(saved.referralCode).toBeDefined(); + expect(saved.referralChain).toEqual([]); + }); + + it('should save referral relationship with referrer', async () => { + // First create the referrer + const referrer = ReferralRelationship.create(50n, null); + await repository.save(referrer); + + // Then create with referrer + const relationship = ReferralRelationship.create(100n, 50n, []); + const saved = await repository.save(relationship); + + expect(saved.userId).toBe(100n); + expect(saved.referrerId).toBe(50n); + expect(saved.referralChain).toContain(50n); + }); + + it('should update existing referral relationship', async () => { + const relationship = ReferralRelationship.create(100n, null); + await repository.save(relationship); + + // Save again should update + const updated = await repository.save(relationship); + expect(updated.userId).toBe(100n); + }); + }); + + describe('findByUserId', () => { + it('should find relationship by user ID', async () => { + const relationship = ReferralRelationship.create(100n, null); + await repository.save(relationship); + + const found = await repository.findByUserId(100n); + + expect(found).toBeDefined(); + expect(found!.userId).toBe(100n); + }); + + it('should return null for non-existent user', async () => { + const found = await repository.findByUserId(999n); + expect(found).toBeNull(); + }); + }); + + describe('findByReferralCode', () => { + it('should find relationship by referral code', async () => { + const relationship = ReferralRelationship.create(100n, null); + const saved = await repository.save(relationship); + + const found = await repository.findByReferralCode(saved.referralCode); + + expect(found).toBeDefined(); + expect(found!.userId).toBe(100n); + }); + + it('should return null for non-existent code', async () => { + const found = await repository.findByReferralCode('NONEXISTENT'); + expect(found).toBeNull(); + }); + }); + + describe('findDirectReferrals', () => { + it('should find all direct referrals', async () => { + // Create referrer + const referrer = ReferralRelationship.create(50n, null); + await repository.save(referrer); + + // Create direct referrals + const ref1 = ReferralRelationship.create(100n, 50n, []); + const ref2 = ReferralRelationship.create(101n, 50n, []); + await repository.save(ref1); + await repository.save(ref2); + + const directReferrals = await repository.findDirectReferrals(50n); + + expect(directReferrals.length).toBe(2); + expect(directReferrals.map((r) => r.userId)).toContain(100n); + expect(directReferrals.map((r) => r.userId)).toContain(101n); + }); + + it('should return empty array for user with no referrals', async () => { + const referrer = ReferralRelationship.create(50n, null); + await repository.save(referrer); + + const directReferrals = await repository.findDirectReferrals(50n); + expect(directReferrals.length).toBe(0); + }); + }); + + describe('existsByReferralCode', () => { + it('should return true for existing code', async () => { + const relationship = ReferralRelationship.create(100n, null); + const saved = await repository.save(relationship); + + const exists = await repository.existsByReferralCode(saved.referralCode); + expect(exists).toBe(true); + }); + + it('should return false for non-existent code', async () => { + const exists = await repository.existsByReferralCode('NONEXISTENT'); + expect(exists).toBe(false); + }); + }); + + describe('existsByUserId', () => { + it('should return true for existing user', async () => { + const relationship = ReferralRelationship.create(100n, null); + await repository.save(relationship); + + const exists = await repository.existsByUserId(100n); + expect(exists).toBe(true); + }); + + it('should return false for non-existent user', async () => { + const exists = await repository.existsByUserId(999n); + expect(exists).toBe(false); + }); + }); + + describe('getReferralChain', () => { + it('should return referral chain', async () => { + const parentChain = [200n, 300n]; + const relationship = ReferralRelationship.create(100n, 50n, parentChain); + await repository.save(relationship); + + const chain = await repository.getReferralChain(100n); + expect(chain).toContain(50n); + }); + + it('should return empty array for non-existent user', async () => { + const chain = await repository.getReferralChain(999n); + expect(chain).toEqual([]); + }); + }); +}); diff --git a/backend/services/referral-service/test/integration/repositories/team-statistics.repository.integration.spec.ts b/backend/services/referral-service/test/integration/repositories/team-statistics.repository.integration.spec.ts new file mode 100644 index 00000000..b2cbc074 --- /dev/null +++ b/backend/services/referral-service/test/integration/repositories/team-statistics.repository.integration.spec.ts @@ -0,0 +1,174 @@ +import { Test, TestingModule } from '@nestjs/testing'; +import { TeamStatisticsRepository } from '../../../src/infrastructure/repositories/team-statistics.repository'; +import { PrismaService } from '../../../src/infrastructure/database/prisma.service'; +import { MockPrismaService } from '../mocks/prisma.mock'; +import { TeamStatistics } from '../../../src/domain'; + +describe('TeamStatisticsRepository (Integration)', () => { + let repository: TeamStatisticsRepository; + let mockPrisma: MockPrismaService; + + beforeEach(async () => { + mockPrisma = new MockPrismaService(); + + const module: TestingModule = await Test.createTestingModule({ + providers: [ + TeamStatisticsRepository, + { + provide: PrismaService, + useValue: mockPrisma, + }, + ], + }).compile(); + + repository = module.get(TeamStatisticsRepository); + }); + + afterEach(() => { + mockPrisma.$cleanup(); + }); + + describe('create', () => { + it('should create new team statistics', async () => { + const stats = await repository.create(100n); + + expect(stats).toBeDefined(); + expect(stats.userId).toBe(100n); + expect(stats.directReferralCount).toBe(0); + expect(stats.totalTeamCount).toBe(0); + expect(stats.leaderboardScore).toBe(0); + }); + }); + + describe('save', () => { + it('should save team statistics', async () => { + const stats = TeamStatistics.create(100n); + stats.addPersonalPlanting(10, '110000', '110100'); + + const saved = await repository.save(stats); + + expect(saved).toBeDefined(); + expect(saved.userId).toBe(100n); + expect(saved.personalPlantingCount).toBe(10); + }); + + it('should update existing statistics', async () => { + // Create initial + const stats = await repository.create(100n); + + // Create new instance and add planting + const updated = TeamStatistics.create(100n); + updated.addPersonalPlanting(20, '110000', '110100'); + + const saved = await repository.save(updated); + expect(saved.personalPlantingCount).toBe(20); + }); + }); + + describe('findByUserId', () => { + it('should find statistics by user ID', async () => { + await repository.create(100n); + + const found = await repository.findByUserId(100n); + + expect(found).toBeDefined(); + expect(found!.userId).toBe(100n); + }); + + it('should return null for non-existent user', async () => { + const found = await repository.findByUserId(999n); + expect(found).toBeNull(); + }); + }); + + describe('findByUserIds', () => { + it('should find statistics for multiple users', async () => { + await repository.create(100n); + await repository.create(101n); + await repository.create(102n); + + const found = await repository.findByUserIds([100n, 101n]); + + expect(found.length).toBe(2); + expect(found.map((s) => s.userId)).toContain(100n); + expect(found.map((s) => s.userId)).toContain(101n); + }); + }); + + describe('getLeaderboard', () => { + it('should return leaderboard sorted by score', async () => { + // Create users with different scores + const stats1 = TeamStatistics.create(100n); + stats1.addPersonalPlanting(50, '110000', '110100'); + await repository.save(stats1); + + const stats2 = TeamStatistics.create(101n); + stats2.addPersonalPlanting(100, '110000', '110100'); + await repository.save(stats2); + + const stats3 = TeamStatistics.create(102n); + stats3.addPersonalPlanting(30, '110000', '110100'); + await repository.save(stats3); + + const leaderboard = await repository.getLeaderboard({ limit: 10 }); + + expect(leaderboard.length).toBe(3); + // Should be sorted by score descending + expect(leaderboard[0].userId).toBe(101n); + expect(leaderboard[0].rank).toBe(1); + expect(leaderboard[1].userId).toBe(100n); + expect(leaderboard[1].rank).toBe(2); + }); + + it('should respect limit and offset', async () => { + for (let i = 0; i < 10; i++) { + const stats = TeamStatistics.create(BigInt(100 + i)); + stats.addPersonalPlanting(10 + i, '110000', '110100'); + await repository.save(stats); + } + + const leaderboard = await repository.getLeaderboard({ limit: 5, offset: 2 }); + + expect(leaderboard.length).toBe(5); + expect(leaderboard[0].rank).toBe(3); // 1-indexed, offset by 2 + }); + }); + + describe('getUserRank', () => { + it('should return correct rank', async () => { + const stats1 = TeamStatistics.create(100n); + stats1.addPersonalPlanting(100, '110000', '110100'); + await repository.save(stats1); + + const stats2 = TeamStatistics.create(101n); + stats2.addPersonalPlanting(50, '110000', '110100'); + await repository.save(stats2); + + const rank = await repository.getUserRank(101n); + expect(rank).toBe(2); + }); + + it('should return null for non-existent user', async () => { + const rank = await repository.getUserRank(999n); + expect(rank).toBeNull(); + }); + }); + + describe('batchUpdateTeamCounts', () => { + it('should batch update team counts for multiple users', async () => { + await repository.create(100n); + await repository.create(101n); + + await repository.batchUpdateTeamCounts([ + { userId: 100n, countDelta: 10, provinceCode: '110000', cityCode: '110100' }, + { userId: 101n, countDelta: 20, provinceCode: '120000', cityCode: '120100' }, + ]); + + const stats100 = await repository.findByUserId(100n); + const stats101 = await repository.findByUserId(101n); + + expect(stats100!.teamPlantingCount).toBe(10); + expect(stats101!.teamPlantingCount).toBe(20); + }); + }); +}); diff --git a/backend/services/referral-service/test/integration/services/referral.service.integration.spec.ts b/backend/services/referral-service/test/integration/services/referral.service.integration.spec.ts new file mode 100644 index 00000000..f0242db0 --- /dev/null +++ b/backend/services/referral-service/test/integration/services/referral.service.integration.spec.ts @@ -0,0 +1,225 @@ +import { Test, TestingModule } from '@nestjs/testing'; +import { ReferralService } from '../../../src/application/services/referral.service'; +import { + REFERRAL_RELATIONSHIP_REPOSITORY, + TEAM_STATISTICS_REPOSITORY, + ReferralChainService, + ReferralRelationship, +} from '../../../src/domain'; +import { EventPublisherService, LeaderboardCacheService } from '../../../src/infrastructure'; +import { CreateReferralRelationshipCommand } from '../../../src/application/commands'; +import { GetUserReferralInfoQuery, GetDirectReferralsQuery } from '../../../src/application/queries'; + +// Mock repository that works with domain aggregates +class MockReferralRepository { + private data = new Map(); + private codeToUserId = new Map(); + + async save(relationship: ReferralRelationship): Promise { + this.data.set(relationship.userId, relationship); + this.codeToUserId.set(relationship.referralCode, relationship.userId); + return relationship; + } + + async findByUserId(userId: bigint): Promise { + return this.data.get(userId) ?? null; + } + + async findByReferralCode(code: string): Promise { + const userId = this.codeToUserId.get(code); + if (!userId) return null; + return this.findByUserId(userId); + } + + async findDirectReferrals(userId: bigint): Promise { + const results: ReferralRelationship[] = []; + for (const record of this.data.values()) { + if (record.referrerId === userId) { + results.push(record); + } + } + return results; + } + + async existsByReferralCode(code: string): Promise { + return this.codeToUserId.has(code); + } + + async existsByUserId(userId: bigint): Promise { + return this.data.has(userId); + } + + async getReferralChain(userId: bigint): Promise { + const record = this.data.get(userId); + return record?.referralChain ?? []; + } +} + +class MockTeamStatsRepository { + private data = new Map void; getDirectReferralStats: () => Map }>(); + + async create(userId: bigint) { + const stats = { + userId, + directReferralCount: 0, + totalTeamCount: 0, + personalPlantingCount: 0, + teamPlantingCount: 0, + leaderboardScore: 0, + addDirectReferral: function(id: bigint) { this.directReferralCount++; }, + getDirectReferralStats: () => new Map(), + }; + this.data.set(userId, stats); + return stats; + } + + async save(stats: { userId: bigint }) { + const existing = this.data.get(stats.userId); + if (existing) { + Object.assign(existing, stats); + } + return existing; + } + + async findByUserId(userId: bigint) { + return this.data.get(userId) ?? null; + } +} + +class MockEventPublisher { + async publishDomainEvents() {} + async publishEvent() {} +} + +class MockLeaderboardCache { + async getUserRank() { return 1; } + async updateScore() {} +} + +describe('ReferralService (Integration)', () => { + let service: ReferralService; + let referralRepo: MockReferralRepository; + let teamStatsRepo: MockTeamStatsRepository; + + beforeEach(async () => { + referralRepo = new MockReferralRepository(); + teamStatsRepo = new MockTeamStatsRepository(); + + const module: TestingModule = await Test.createTestingModule({ + providers: [ + ReferralService, + ReferralChainService, + { + provide: REFERRAL_RELATIONSHIP_REPOSITORY, + useValue: referralRepo, + }, + { + provide: TEAM_STATISTICS_REPOSITORY, + useValue: teamStatsRepo, + }, + { + provide: EventPublisherService, + useClass: MockEventPublisher, + }, + { + provide: LeaderboardCacheService, + useClass: MockLeaderboardCache, + }, + ], + }).compile(); + + service = module.get(ReferralService); + }); + + describe('createReferralRelationship', () => { + it('should create referral relationship without referrer', async () => { + const command = new CreateReferralRelationshipCommand(100n, null); + const result = await service.createReferralRelationship(command); + + expect(result).toBeDefined(); + expect(result.referralCode).toBeDefined(); + expect(result.referralCode.length).toBeGreaterThanOrEqual(6); + }); + + it('should create referral relationship with valid referrer code', async () => { + // First create a referrer + const referrerCommand = new CreateReferralRelationshipCommand(50n, null); + const referrerResult = await service.createReferralRelationship(referrerCommand); + + // Then create with referrer code + const command = new CreateReferralRelationshipCommand(100n, referrerResult.referralCode); + const result = await service.createReferralRelationship(command); + + expect(result).toBeDefined(); + expect(result.referralCode).toBeDefined(); + }); + + it('should throw error if user already has referral relationship', async () => { + const command = new CreateReferralRelationshipCommand(100n, null); + await service.createReferralRelationship(command); + + await expect(service.createReferralRelationship(command)).rejects.toThrow('用户已存在推荐关系'); + }); + + it('should throw error for invalid referral code', async () => { + const command = new CreateReferralRelationshipCommand(100n, 'INVALID_CODE'); + + await expect(service.createReferralRelationship(command)).rejects.toThrow('推荐码不存在'); + }); + }); + + describe('getUserReferralInfo', () => { + it('should return user referral info', async () => { + // Create user first + const createCommand = new CreateReferralRelationshipCommand(100n, null); + await service.createReferralRelationship(createCommand); + + const query = new GetUserReferralInfoQuery(100n); + const result = await service.getUserReferralInfo(query); + + expect(result).toBeDefined(); + expect(result.userId).toBe('100'); + expect(result.referralCode).toBeDefined(); + expect(result.directReferralCount).toBe(0); + }); + + it('should throw error for non-existent user', async () => { + const query = new GetUserReferralInfoQuery(999n); + + await expect(service.getUserReferralInfo(query)).rejects.toThrow('用户推荐关系不存在'); + }); + }); + + describe('getDirectReferrals', () => { + it('should return direct referrals list', async () => { + // Create referrer + const referrerCommand = new CreateReferralRelationshipCommand(50n, null); + const referrerResult = await service.createReferralRelationship(referrerCommand); + + // Create direct referrals + await service.createReferralRelationship(new CreateReferralRelationshipCommand(100n, referrerResult.referralCode)); + await service.createReferralRelationship(new CreateReferralRelationshipCommand(101n, referrerResult.referralCode)); + + const query = new GetDirectReferralsQuery(50n); + const result = await service.getDirectReferrals(query); + + expect(result.referrals.length).toBe(2); + expect(result.total).toBe(2); + }); + }); + + describe('validateReferralCode', () => { + it('should return true for valid code', async () => { + const createCommand = new CreateReferralRelationshipCommand(100n, null); + const { referralCode } = await service.createReferralRelationship(createCommand); + + const isValid = await service.validateReferralCode(referralCode); + expect(isValid).toBe(true); + }); + + it('should return false for invalid code', async () => { + const isValid = await service.validateReferralCode('INVALID'); + expect(isValid).toBe(false); + }); + }); +}); diff --git a/backend/services/referral-service/test/jest-e2e.json b/backend/services/referral-service/test/jest-e2e.json new file mode 100644 index 00000000..e8345203 --- /dev/null +++ b/backend/services/referral-service/test/jest-e2e.json @@ -0,0 +1,14 @@ +{ + "moduleFileExtensions": ["js", "json", "ts"], + "rootDir": "..", + "testEnvironment": "node", + "testRegex": ".e2e-spec.ts$", + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "moduleNameMapper": { + "^@/(.*)$": "/src/$1" + }, + "setupFilesAfterEnv": ["/test/setup-e2e.ts"], + "testTimeout": 60000 +} diff --git a/backend/services/referral-service/test/jest-integration.json b/backend/services/referral-service/test/jest-integration.json new file mode 100644 index 00000000..34bba64f --- /dev/null +++ b/backend/services/referral-service/test/jest-integration.json @@ -0,0 +1,14 @@ +{ + "moduleFileExtensions": ["js", "json", "ts"], + "rootDir": "..", + "testEnvironment": "node", + "testRegex": ".integration.spec.ts$", + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "moduleNameMapper": { + "^@/(.*)$": "/src/$1" + }, + "setupFilesAfterEnv": ["/test/setup-integration.ts"], + "testTimeout": 30000 +} diff --git a/backend/services/referral-service/test/setup-e2e.ts b/backend/services/referral-service/test/setup-e2e.ts new file mode 100644 index 00000000..f6a01fd2 --- /dev/null +++ b/backend/services/referral-service/test/setup-e2e.ts @@ -0,0 +1,22 @@ +/** + * E2E Test Setup + * 端到端测试环境设置 + */ + +// 增加测试超时时间 +jest.setTimeout(60000); + +// 设置测试环境变量 +process.env.NODE_ENV = 'test'; +process.env.DATABASE_URL = process.env.DATABASE_URL || 'postgresql://postgres:postgres@localhost:5432/referral_test?schema=public'; +process.env.REDIS_HOST = process.env.REDIS_HOST || 'localhost'; +process.env.REDIS_PORT = process.env.REDIS_PORT || '6379'; +process.env.KAFKA_BROKERS = process.env.KAFKA_BROKERS || 'localhost:9092'; +process.env.JWT_SECRET = 'test-jwt-secret-for-e2e-tests'; +process.env.PORT = '3099'; // 使用不同的端口避免冲突 + +// 全局清理 +afterAll(async () => { + // 等待所有异步操作完成 + await new Promise((resolve) => setTimeout(resolve, 1000)); +}); diff --git a/backend/services/referral-service/test/setup-integration.ts b/backend/services/referral-service/test/setup-integration.ts new file mode 100644 index 00000000..0d40b73c --- /dev/null +++ b/backend/services/referral-service/test/setup-integration.ts @@ -0,0 +1,21 @@ +/** + * Integration Test Setup + * 集成测试环境设置 + */ + +// 增加测试超时时间 +jest.setTimeout(30000); + +// 设置测试环境变量 +process.env.NODE_ENV = 'test'; +process.env.DATABASE_URL = process.env.DATABASE_URL || 'postgresql://postgres:postgres@localhost:5432/referral_test?schema=public'; +process.env.REDIS_HOST = process.env.REDIS_HOST || 'localhost'; +process.env.REDIS_PORT = process.env.REDIS_PORT || '6379'; +process.env.KAFKA_BROKERS = process.env.KAFKA_BROKERS || 'localhost:9092'; +process.env.JWT_SECRET = 'test-jwt-secret-for-integration-tests'; + +// 全局清理 +afterAll(async () => { + // 等待所有异步操作完成 + await new Promise((resolve) => setTimeout(resolve, 500)); +}); diff --git a/backend/services/referral-service/tsconfig.build.json b/backend/services/referral-service/tsconfig.build.json new file mode 100644 index 00000000..64f86c6b --- /dev/null +++ b/backend/services/referral-service/tsconfig.build.json @@ -0,0 +1,4 @@ +{ + "extends": "./tsconfig.json", + "exclude": ["node_modules", "test", "dist", "**/*spec.ts"] +} diff --git a/backend/services/referral-service/tsconfig.json b/backend/services/referral-service/tsconfig.json index e69de29b..a66de921 100644 --- a/backend/services/referral-service/tsconfig.json +++ b/backend/services/referral-service/tsconfig.json @@ -0,0 +1,24 @@ +{ + "compilerOptions": { + "module": "commonjs", + "declaration": true, + "removeComments": true, + "emitDecoratorMetadata": true, + "experimentalDecorators": true, + "allowSyntheticDefaultImports": true, + "target": "ES2021", + "sourceMap": true, + "outDir": "./dist", + "baseUrl": "./src", + "paths": { + "@/*": ["./*"] + }, + "incremental": true, + "skipLibCheck": true, + "strictNullChecks": true, + "noImplicitAny": true, + "strictBindCallApply": true, + "forceConsistentCasingInFileNames": true, + "noFallthroughCasesInSwitch": true + } +}